typeahead.js 由 twitter 开发的快速功能强大的自动完成插件

发布于 2020-04-05 17:08:34 字数 17903 浏览 1378 评论 0

typeahead.js 是实现预输入的强大灵活的 JavaScript 库,为构建强大的类型提供了坚实的基础。Typeahead.js 库包含了两个组件:推荐引擎 Bloodhound 以及 UI 视图 typeahead 。

推荐引擎负责对给定的查询计算推荐结果,UI 视图渲染推荐并处理 DOM 交互。这两者都可分开使用,但是,结合使用可以提供丰富的预输入的体验。

目录结构

typeahead的项目的目录结构如下:

  • dist/ : 项目分发包的目录
  • doc/ : 文档目录
  • src/ : 源码目录
  • test/ : 测试目录
  • .jshintrc : ??
  • .travis.yml : travis-ci服务, github上的开源项目使用的持续构建服务
  • bower.json : bower工具的配置文件
  • CHANGELOG.md : 更改日志
  • composer.json : 项目的组成信息
  • Gruntfile.js : 统一项目管理
  • karma.conf.js : 自动化单元测试
  • package.json : 类似Gemfile的,项目的包依赖
  • typeahead.js.jquery.json : 子项目包依赖
  • typeahead项目中样例在其 gh-pages分支中。

Node小知识: Jasmine做单元测试,Karma自动化完成单元测试,Grunt启动Karma统一项目管理,Yeoman最后封装成一个项目原型模板,npm做nodejs的包依赖管理,bower做javascript的包依赖管理

推荐引擎负责对给定的查询计算推荐结果,UI视图渲染推荐并处理DOM交互。这两者都可分开使用,但是,结合使用可以提供丰富的预输入的体验。以下,分别介绍两者。

Bloodhound

Bloodhound 是 typeahead.js 的推荐引擎,健壮且灵活,而且还提供诸如预取,智能缓存,快速查找,远程数据回填。下面,从特性和使用方面分别介绍,使用则从API、选项、预取等方面介绍。

其特性如下:

  • 可使用硬编码数据
  • 初始化是预取数据,从而减少推荐时延
  • 智能使用本地存储从而减少网络请求
  • 从远程源中回填建议项
  • 对远程请求的速率限制并缓存网络请求,从而减轻负载

具体的使用,按API,选项,数据集,定制事件和 Look and Feel介绍。

API

  • new Bloodhound(options)是构造函数,其接受哈希选项作为唯一的参数。
var engine = new Bloodhound({
  name: 'animals',
  local: [{ val: 'dog' }, { val: 'pig' }, { val: 'moose' }], // 本地数据源
  remote: 'http://example.com/animals?q=%QUERY',       // 远程数据源
  datumTokenizer: function(d) {
  return Bloodhound.tokenizers.whitespace(d.val);
  },
  queryTokenizer: Bloodhound.tokenizers.whitespace
});
  • Bloodhound#initialize(reinitialize)启动了推荐引擎。其启动过程包括处理有 local以及 prefetch提供的数据。初始化完成前,其他方法均不起作用; 初始化完成后,返回一个 jQuery promise
var promise = engine.initialize();

promise
.done(function() { console.log('success!'); })
.fail(function() { console.log('err!'); });

After the initial call of initialize, how subsequent invocations of the method behave depends on the reinitializeargument. If reinitializeis false, the method will not execute the initialization logic and will just return the same jQuery promise returned by the initial invocation. If reinitializeis truthy, the method will behave as if it were being called for the first time.

initialize调用之后,后续调用方法的行为依赖于 reinitialize参数。如果 reinitialize为false,函数不执行初始化逻辑,并仅仅返回和最初调用相同的jQuery promise。 如果 reinitialize为真,初始化就重新执行。

var promise1 = engine.initialize();
var promise2 = engine.initialize();
var promise3 = engine.initialize(true);

promise1 === promise2; // 相等性判断
promise3 !== promise1 && promise3 !== promise2;
  • Bloodhound#add(datums)获取一个数据参数datums,其中的数据将被添加到搜索索引中,从而增强推荐引擎。
engine.add([{ val: 'one' }, { val: 'two' }]);
  • Bloodhound#clear()从搜索引擎中移除所有推荐。
engine.clear();
  • Bloodhound#clearPrefetchCache()如果使用了 prefetch选项,数据将会缓存到本地存储中,从而减少不必要的网络请求。 clearPrefetchCache提供了可编程移除缓存的方法。
engine.clearPrefetchCache();
  • Bloodhound#clearRemoteCache()如果使用了 remote选项,Bloodhound将会缓存最近10条响应,从而提供最佳的用户体验。 clearRemoteCache提供编程清除缓存的方法。
engine.clearRemoteCache();
  • Bloodhound.noConflict()返回Bloodhound构造器的引用,并返回 window.Bloodhound的先前的值。可以用来避免命名冲突。
var Dachshund = Bloodhound.noConflict();
  • Bloodhound#get(query, cb)query计算一组建议。 cb为之前提到datums。 cb总是与在客户端可用的建议一起同步调用。如果客户端的不够,将会考虑 remotecb也可以异步的方式,混合着客户端和 remote端数据进行调用。注: 例子中显示, cb为函数。
bloodhound.get(myQuery, function(suggestions) {
  suggestions.each(function(suggestion) { console.log(suggestion); });
});

Options

当实例化一个Bloodhund推荐引擎时,存在如下的选项可配置:

  • datumTokenizer– 将datum转换为字符串token的数组的函数,其签名为 (datum)必须
  • queryTokenizer– 将查询转换为字符串token的数组的函数,其签名为 (query)必须
  • limitBloodhound#get方法返回的最大的建议数目. 如果不够,就用远程的数据回填。默认为5个。
  • dupDetector– 如果设置该选项,其值为带有 (remoteMatch, localMatch)选项的冗余检测函数,即数据中存在冗余,则返回为true,否则返回为false。如果不设置,则不执行冗余检测。
  • sorter– 用来为给定查询所匹配的数据进行排序。
  • local– datums数组或返回datums数组的函数。
  • prefetch– 可以是包含datums数组的JSON文件的URL,更多信息,参考[prefetch]哈希选项。
  • remote– 当 localprefetch提供的数据不充分,或需要更多的可配置性时,用来从远程获取建议的URL。

Prefetch

在初始化过程中,预取并处理数据。如果浏览器支持本地存储,处理数据将被缓存,从而避免后续页面加载中额外的网络请求。

警告 尽管小数据集可以不用预取缓存,并且预取不意味着包含整个数据集。相反的,需将其看作推荐的一级缓存。如果不谨记在心,可能会触及本地缓存限制。

当配置 prefetch时,如下选项亦可用:

  • url– 链接到包含 datums 数组的 JSON 文件的 URL。 必须
  • cacheKey– 存储在本地的数据的key。默认为 url的值
  • ttl– 预取的数据缓存在本地存储中的时间(毫秒),默认为 86400000(1天)
  • thumbprint– 用作预取数据的数据指纹。如果和本地缓存的数据不匹配,数据将会被重新获取
  • filter– 将响应体转换成datums的数组的函数,其签名为 filter(parsedResponse),返回值为datums数组。
  • ajax– 传递给 jQuery.ajax的ajax设置

Remote

远程数据仅在 localprefetch提供的数据不充分时,才会使用。为了避免到远程端过于繁琐的请求数,请求被限速了。

当配置 remote时,如下的选项是可用的:

  • url– 当 localprefetch的数据不足时,发出请求的URL。 必须
  • wildcard- url中的模式,当发出请求时,将会被替换为用户的查询。默认为 %QUERY
  • replace– 用来覆盖请求URL的函数, 其方法签名为 replace(url, query),返回值为一合法的URL。如果设置,将不对URL执行任何替换?
  • rateLimitBy– 该方法用来实时显示网络请求数目。其值为 debouncethrottle,默认为 debounce
  • rateLimitWaitrateLimitBy所使用的时间间隔(毫秒),默认为 300
  • filter– 将响应体转换为数组datums的函数,其方法签名为 filter(parsedResponse),返回值为datums数组
  • ajax– 传给 jQuery.ajaxajax设置对象

Datums

Datums are JavaScript objects that hydrate the pool of possible suggestions. Bloodhound doesn’t expect datums to contain any specific properties as any operations performed on datums are done using functions defined by the user i.e.datumTokenizer, dupDetector, and sorter.

Tokens

The algorithm used by bloodhounds for providing suggestions for a given query is token-based. When Bloodhound#getis called, it tokenizes queryusing queryTokenizerand then invokes cbwith all of the datums that contain those tokens.

For a quick example, if a datum was tokenized into the following set of tokens…

['typeahead.js', 'typeahead', 'autocomplete', 'javascript'];

…it would be a valid match for queries such as:

  • typehead
  • typehead.js
  • autoco
  • java type

typeahead

typeahead.js的UI组件是一个jQuery插件,其负责渲染建议并处理DOM交互。其特性如下:

  • 在用户输入时,显示建议
  • 将顶层推荐显示为hint(例如,背景文字)
  • 支持可定制的模板
  • 同RTL语言和输入编辑器工作良好
  • 高亮匹配的查询
  • 触发定制事件

规范

为了最大程度的利用用户现有的关于typeahead.js的知识,typeahead.js UI的行为是仿照谷歌的搜索框。下面的伪代码,介绍UI界面如何处理相关事件的。

预输入需要考虑的事件有: 聚焦是失焦,值的改变,方向键,Tab键,Esc键

输入控件聚焦

activate typeahead  //激活typeahead

输入控件失去焦点

deactivate typeahead //失效typeahead
close dropdown menu  //关闭下拉菜单
remove hint      //移除hint
clear suggestions from dropdown menu //清理下拉菜单中的推荐

输入控件中的值的改变

IF query satisfies minLength requirement THEN  //查询满足最小长度需求
  request suggestions for new query      //为新的查询请求推荐

  IF suggestions are available THEN  
  render suggestions in dropdown menu
  open dropdown menu 
  update hint
  ELSE
  close dropdown menu 
  clear suggestions from dropdown menu
  remove hint
  ENDIF
ELSE
  close dropdown menu 
  clear suggestions from dropdown menu
  remove hint
ENDIF

按下Up键

IF dropdown menu is open THEN
  move dropdown menu cursor up 1 suggestion
ELSE
  request suggestions for current query

  IF suggestions are available THEN
  render suggestions in dropdown menu
  open dropdown menu 
  update hint
  ENDIF
ENDIF

按下Down键

IF dropdown menu is open THEN
  move dropdown menu cursor down 1 suggestion
ELSE
  request suggestions for current query

  IF suggestions are available THEN
  render suggestions in dropdown menu
  open dropdown menu 
  update hint
  ENDIF
ENDIF

左箭头按下

IF detected query language direction is right-to-left THEN
  IF hint is being shown THEN
  IF text cursor is at end of query THEN
    autocomplete query to hint
  ENDIF
  ENDIF
ENDIF

右箭头按下

IF detected query language direction is left-to-right THEN
  IF hint is being shown THEN
  IF text cursor is at the end of the query THEN
    autocomplete query to hint
  ENDIF
  ENDIF
ENDIF

按Tab键

IF dropdown menu cursor is on suggestion THEN
  close dropdown menu
  update query to display key of suggestion
  remove hint
ELSIF hint is being shown THEN
  autocomplete query to hint
ENDIF

Enter按键

IF dropdown menu cursor is on suggestion THEN
  close dropdown menu
  update query to display key of suggestion
  remove hint
  prevent default browser action e.g. form submit
ENDIF

Esc按键

close dropdown menu
remove hint

点击推荐项

update query to display key of suggestion
close dropdown menu
remove hint

API

  • jQuery#typeahead(options, [\*datasets])

Turns any input[type="text"]element into a typeahead. optionsis an options hash that’s used to configure the typeahead to your liking. Refer to Options for more info regarding the available configs. Subsequent arguments (*datasets), are individual option hashes for datasets. For more details regarding datasets, refer to Datasets .

将任何 input[type="text"]元素转换成typehead。 options参数hash将typeahead配置为你所喜欢的, *datasets参数是配置数据集的独立选项hash。更多关于选项的信息参考下文。

$('.typeahead').typeahead({
  minLength: 3,
  highlight: true,
},
{
  name: 'my-dataset',
  source: mySource
});
  • jQuery#typeahead('destroy')移除typeahead功能,并将 input元素的状态重置为原始状态。
$('.typeahead').typeahead('destroy');
  • jQuery#typeahead('open')打开typeahead下拉菜单。 注意,打开菜单不意味着菜单可见。仅当其打开并存在内容时,菜单才可见。
$('.typeahead').typeahead('open');
  • jQuery#typeahead('close')关闭typeahead的下拉菜单。
$('.typeahead').typeahead('close');
  • jQuery#typeahead('val')返回typeahead的当前值,该值为用户输入到 input元素中的文本。
var myVal = $('.typeahead').typeahead('val');
  • jQuery#typeahead('val', val)设置typeahead的值,要来替代 jQuery#val函数。
$('.typeahead').typeahead('val', myVal);
  • jQuery.fn.typeahead.noConflict()返回typeahead插件的引用,并将 jQuery.fn.typeahead重置为先前值。这可以用来避免命名冲突。
var typeahead = jQuery.fn.typeahead.noConflict();
jQuery.fn._typeahead = typeahead;

选项(Options)

当初始化typeahead时,存在如下的可配置的选项:

  • highlight– 设置为 true时,当建议渲染时,在文本节点中,匹配查询模式的文字将被带有 tt-highlightclass的 strong元素包裹。默认设置为 false
  • hint– 设置为 false时,typeahead 将不会显示hint。默认为 true.
  • minLength– 推荐引擎开始渲染所需要的最小字符。默认为 1.

数据集(Datasets)

typeahead可以由一个或多个数据集组成。但用户修改typeahead的值时,每个数据集都会尝试渲染为新的查询渲染值。

大多数情况下,一个数据集足够了。只有在需要在下拉菜单中,以某些分类关系分组渲染推荐时,才需要使用多个数据源。例如,在 twitter.com中,搜索预输入将结果分组为相关搜索,趋势,账户 - 这就需要使用多个数据集。

数据集可以通过如下的选项进行配置:

  • source– 推荐的数据源支持。值为带有 (query, cb)签名的函数。该函数将会计为 query计算推荐集,然后以计算的推荐集调用 cb。函数 cb的调用可以是同步的,也可以是异步的。Bloodhound推荐引擎可在这里使用,更多参考[Bloodhound Integration]。 必须
  • name– 数据集的名字。该名字可以被追加到 tt-dataset-,从而形成包含DOM元素的类名。只能由下划线、-,字母和数字组成。默认为随机数。
  • displayKey– 对于一个给定的推荐对象,决定其的字符串表示,并将会在某个输入控件选择后使用。其值可以是关键字符串,或者是将推荐对象转换为string的函数。默认为 value
  • templates– 渲染数据集使用的哈希模板。注意:预编译的模板是将javascript对象作为第一参数,并返回为HTML字符串。
    • empty– 当给定查询推荐数为0时,渲染 empty中的内容。 empty的值可以是HTML字符串或预编译模板。如果是预编译模板,其内容中将包含 query
    • footer– 数据集底部渲染的内容,可为HTML字符串或预编译模板。如果是预编译模板,其中包含 queryisEmpty
    • header– 数据集头部渲染的内容,可为HTML字符串或预编译模板。如果是预编译模板,其中包含 queryisEmpty
    • suggestion– 用来渲染单个推荐。其值必须是预编译模板。其中包含关联的建议对象。默认为将 displayKey包装在 p标签中: <p></p>

Custom Events

typeahead组件触发了如下的定制的事件:

  • typeahead:opened– 当typeahead的下拉菜单打开时触发。
  • typeahead:closed– 当typeahead的下拉菜单关闭时触发。
  • typeahead:cursorchanged– 但下拉菜单的光标移动到另一个推荐时,触发事件。事件处理器将接受三个参数: jQuery事件对象、推荐对象以及推荐对象所属的数据集名。
  • typeahead:selected– 当下拉菜单被渲染时触发。事件处理器接受三个参数: jQuery事件对象,推荐对象以及推荐对象所属的数据集名。
  • typeahead:autocompleted– 当查询自动补全时触发。自动补全意味着将 hint改变为查询,事件器将调用3个参数: jQuery事件对象、推荐对象以及推荐对象所属的数据集名。

All custom events are triggered on the element initialized as a typeahead.

所有的定制事件将会在元素初始化为typeahead时触发。

Look and Feel

下面是虚构的mustache模板,用来描述typeahead下拉菜单DOM元素结构。 注意, header, footer, suggestion以及 empty都是有datase提供的模板。

<span class="tt-dropdown-menu">
  <div class="tt-dataset-">
    }
    <span class="tt-suggestions">
      <div class="tt-suggestion">}</div>
      }
    </span>
    }
  </div>
  
</span>

当用户在 .tt-suggestion 的 a 链接上移动鼠标或键盘时,将会追加样式 tt-cursor。可使用该样式类标识位于光标下的推荐。

Bloodhound Integration

由于数据集期望 source 属性是一个函数,所以不能直接将 Bloodhound 推荐引擎传递进来。相反的需要使用推荐引擎的 typeahead 适配器 ttAdapter

var engine = new Bloodhound({ /* options */ });
engine.initialize();
$('.typeahead').typeahead(null, {
  displayKey: myDisplayKey // if not set, will default to 'value',
  source: engine.ttAdapter()
});

相关链接

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

JSmiles

生命进入颠沛而奔忙的本质状态,并将以不断告别和相遇的陈旧方式继续下去。

文章
评论
84963 人气
更多

推荐作者

微信用户

文章 0 评论 0

小情绪

文章 0 评论 0

ゞ记忆︶ㄣ

文章 0 评论 0

笨死的猪

文章 0 评论 0

彭明超

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文