Commit 69f0e5be by Alexander Makarov

Merge pull request #2329 from tonydspaniard/1094-modal-generator-autocomplete-fix

1904 modal generator autocomplete fix
parents d438913e 333630ae
...@@ -5,6 +5,7 @@ Yii Framework 2 gii extension Change Log ...@@ -5,6 +5,7 @@ Yii Framework 2 gii extension Change Log
---------------------------- ----------------------------
- Bug #1405: fixed disambiguation of relation names generated by gii (qiangxue) - Bug #1405: fixed disambiguation of relation names generated by gii (qiangxue)
- Bug #1904: Fixed autocomplete to work with underscore inputs "_" (tonydspaniard)
- Bug #2298: Fixed the bug that Gii controller generator did not allow digit in the controller ID (qiangxue) - Bug #2298: Fixed the bug that Gii controller generator did not allow digit in the controller ID (qiangxue)
- Bug: fixed controller in crud template to avoid returning query in findModel() (cebe) - Bug: fixed controller in crud template to avoid returning query in findModel() (cebe)
- Enh #1624: generate rules for unique indexes (lucianobaraglia) - Enh #1624: generate rules for unique indexes (lucianobaraglia)
......
...@@ -71,6 +71,15 @@ yii.gii = (function ($) { ...@@ -71,6 +71,15 @@ yii.gii = (function ($) {
}; };
return { return {
autocomplete: function (counter, data) {
var datum = new Bloodhound({
datumTokenizer: function(d){return Bloodhound.tokenizers.whitespace(d.word);},
queryTokenizer: Bloodhound.tokenizers.whitespace,
local: data
});
datum.initialize();
jQuery('.typeahead-'+counter).typeahead(null,{displayKey: 'word', source: datum.ttAdapter()});
},
init: function () { init: function () {
initHintBlocks(); initHintBlocks();
initStickyInputs(); initStickyInputs();
......
/*! /*!
* typeahead.js 0.9.3 * typeahead.js 0.10.0
* https://github.com/twitter/typeahead * https://github.com/twitter/typeahead.js
* Copyright 2013 Twitter, Inc. and other contributors; Licensed MIT * Copyright 2013 Twitter, Inc. and other contributors; Licensed MIT
*/ */
(function($) { (function($) {
var VERSION = "0.9.3"; var _ = {
var utils = {
isMsie: function() { isMsie: function() {
var match = /(msie) ([\w.]+)/i.exec(navigator.userAgent); return /(msie|trident)/i.test(navigator.userAgent) ? navigator.userAgent.match(/(msie |rv:)(\d+(.\d+)?)/i)[2] : false;
return match ? parseInt(match[2], 10) : false;
}, },
isBlankString: function(str) { isBlankString: function(str) {
return !str || /^\s*$/.test(str); return !str || /^\s*$/.test(str);
...@@ -30,21 +28,12 @@ ...@@ -30,21 +28,12 @@
return typeof obj === "undefined"; return typeof obj === "undefined";
}, },
bind: $.proxy, bind: $.proxy,
bindAll: function(obj) { each: function(collection, cb) {
var val; $.each(collection, reverseArgs);
for (var key in obj) { function reverseArgs(index, value) {
$.isFunction(val = obj[key]) && (obj[key] = $.proxy(val, obj)); return cb(value, index);
} }
}, },
indexOf: function(haystack, needle) {
for (var i = 0; i < haystack.length; i++) {
if (haystack[i] === needle) {
return i;
}
}
return -1;
},
each: $.each,
map: $.map, map: $.map,
filter: $.grep, filter: $.grep,
every: function(obj, test) { every: function(obj, test) {
...@@ -78,6 +67,12 @@ ...@@ -78,6 +67,12 @@
return counter++; return counter++;
}; };
}(), }(),
templatify: function templatify(obj) {
return $.isFunction(obj) ? obj : template;
function template() {
return String(obj);
}
},
defer: function(fn) { defer: function(fn) {
setTimeout(fn, 0); setTimeout(fn, 0);
}, },
...@@ -123,69 +118,69 @@ ...@@ -123,69 +118,69 @@
return result; return result;
}; };
}, },
tokenizeQuery: function(str) {
return $.trim(str).toLowerCase().split(/[\s]+/);
},
tokenizeText: function(str) {
return $.trim(str).toLowerCase().split(/[\s\-_]+/);
},
getProtocol: function() {
return location.protocol;
},
noop: function() {} noop: function() {}
}; };
var EventTarget = function() { var VERSION = "0.10.0";
var eventSplitter = /\s+/; var LruCache = function(root, undefined) {
return { function LruCache(maxSize) {
on: function(events, callback) { this.maxSize = maxSize || 100;
var event; this.size = 0;
if (!callback) { this.hash = {};
return this; this.list = new List();
} }
this._callbacks = this._callbacks || {}; _.mixin(LruCache.prototype, {
events = events.split(eventSplitter); set: function set(key, val) {
while (event = events.shift()) { var tailItem = this.list.tail, node;
this._callbacks[event] = this._callbacks[event] || []; if (this.size >= this.maxSize) {
this._callbacks[event].push(callback); this.list.remove(tailItem);
delete this.hash[tailItem.key];
}
if (node = this.hash[key]) {
node.val = val;
this.list.moveToFront(node);
} else {
node = new Node(key, val);
this.list.add(node);
this.hash[key] = node;
this.size++;
} }
return this;
}, },
trigger: function(events, data) { get: function get(key) {
var event, callbacks; var node = this.hash[key];
if (!this._callbacks) { if (node) {
return this; this.list.moveToFront(node);
} return node.val;
events = events.split(eventSplitter);
while (event = events.shift()) {
if (callbacks = this._callbacks[event]) {
for (var i = 0; i < callbacks.length; i += 1) {
callbacks[i].call(this, {
type: event,
data: data
});
} }
} }
});
function List() {
this.head = this.tail = null;
} }
return this; _.mixin(List.prototype, {
} add: function add(node) {
}; if (this.head) {
}(); node.next = this.head;
var EventBus = function() { this.head.prev = node;
var namespace = "typeahead:";
function EventBus(o) {
if (!o || !o.el) {
$.error("EventBus initialized without el");
}
this.$el = $(o.el);
} }
utils.mixin(EventBus.prototype, { this.head = node;
trigger: function(type) { this.tail = this.tail || node;
var args = [].slice.call(arguments, 1); },
this.$el.trigger(namespace + type, args); remove: function remove(node) {
node.prev ? node.prev.next = node.next : this.head = node.next;
node.next ? node.next.prev = node.prev : this.tail = node.prev;
},
moveToFront: function(node) {
this.remove(node);
this.add(node);
} }
}); });
return EventBus; function Node(key, val) {
}(); this.key = key;
this.val = val;
this.prev = this.next = null;
}
return LruCache;
}(this);
var PersistentStorage = function() { var PersistentStorage = function() {
var ls, methods; var ls, methods;
try { try {
...@@ -215,7 +210,7 @@ ...@@ -215,7 +210,7 @@
return decode(ls.getItem(this._prefix(key))); return decode(ls.getItem(this._prefix(key)));
}, },
set: function(key, val, ttl) { set: function(key, val, ttl) {
if (utils.isNumber(ttl)) { if (_.isNumber(ttl)) {
ls.setItem(this._ttlKey(key), encode(now() + ttl)); ls.setItem(this._ttlKey(key), encode(now() + ttl));
} else { } else {
ls.removeItem(this._ttlKey(key)); ls.removeItem(this._ttlKey(key));
...@@ -241,330 +236,630 @@ ...@@ -241,330 +236,630 @@
}, },
isExpired: function(key) { isExpired: function(key) {
var ttl = decode(ls.getItem(this._ttlKey(key))); var ttl = decode(ls.getItem(this._ttlKey(key)));
return utils.isNumber(ttl) && now() > ttl ? true : false; return _.isNumber(ttl) && now() > ttl ? true : false;
} }
}; };
} else { } else {
methods = { methods = {
get: utils.noop, get: _.noop,
set: utils.noop, set: _.noop,
remove: utils.noop, remove: _.noop,
clear: utils.noop, clear: _.noop,
isExpired: utils.noop isExpired: _.noop
}; };
} }
utils.mixin(PersistentStorage.prototype, methods); _.mixin(PersistentStorage.prototype, methods);
return PersistentStorage; return PersistentStorage;
function now() { function now() {
return new Date().getTime(); return new Date().getTime();
} }
function encode(val) { function encode(val) {
return JSON.stringify(utils.isUndefined(val) ? null : val); return JSON.stringify(_.isUndefined(val) ? null : val);
} }
function decode(val) { function decode(val) {
return JSON.parse(val); return JSON.parse(val);
} }
}(); }();
var RequestCache = function() {
function RequestCache(o) {
utils.bindAll(this);
o = o || {};
this.sizeLimit = o.sizeLimit || 10;
this.cache = {};
this.cachedKeysByAge = [];
}
utils.mixin(RequestCache.prototype, {
get: function(url) {
return this.cache[url];
},
set: function(url, resp) {
var requestToEvict;
if (this.cachedKeysByAge.length === this.sizeLimit) {
requestToEvict = this.cachedKeysByAge.shift();
delete this.cache[requestToEvict];
}
this.cache[url] = resp;
this.cachedKeysByAge.push(url);
}
});
return RequestCache;
}();
var Transport = function() { var Transport = function() {
var pendingRequestsCount = 0, pendingRequests = {}, maxPendingRequests, requestCache; var pendingRequestsCount = 0, pendingRequests = {}, maxPendingRequests = 6, requestCache = new LruCache(10);
function Transport(o) { function Transport(o) {
utils.bindAll(this); o = o || {};
o = utils.isString(o) ? { this._send = o.send ? callbackToDeferred(o.send) : $.ajax;
url: o this._get = o.rateLimiter ? o.rateLimiter(this._get) : this._get;
} : o;
requestCache = requestCache || new RequestCache();
maxPendingRequests = utils.isNumber(o.maxParallelRequests) ? o.maxParallelRequests : maxPendingRequests || 6;
this.url = o.url;
this.wildcard = o.wildcard || "%QUERY";
this.filter = o.filter;
this.replace = o.replace;
this.ajaxSettings = {
type: "get",
cache: o.cache,
timeout: o.timeout,
dataType: o.dataType || "json",
beforeSend: o.beforeSend
};
this._get = (/^throttle$/i.test(o.rateLimitFn) ? utils.throttle : utils.debounce)(this._get, o.rateLimitWait || 300);
} }
utils.mixin(Transport.prototype, { Transport.setMaxPendingRequests = function setMaxPendingRequests(num) {
_get: function(url, cb) { maxPendingRequests = num;
var that = this; };
if (belowPendingRequestsThreshold()) { Transport.resetCache = function clearCache() {
this._sendRequest(url).done(done); requestCache = new LruCache(10);
};
_.mixin(Transport.prototype, {
_get: function(url, o, cb) {
var that = this, jqXhr;
if (jqXhr = pendingRequests[url]) {
jqXhr.done(done);
} else if (pendingRequestsCount < maxPendingRequests) {
pendingRequestsCount++;
pendingRequests[url] = this._send(url, o).done(done).always(always);
} else { } else {
this.onDeckRequestArgs = [].slice.call(arguments, 0); this.onDeckRequestArgs = [].slice.call(arguments, 0);
} }
function done(resp) { function done(resp) {
var data = that.filter ? that.filter(resp) : resp; cb && cb(resp);
cb && cb(data);
requestCache.set(url, resp); requestCache.set(url, resp);
} }
},
_sendRequest: function(url) {
var that = this, jqXhr = pendingRequests[url];
if (!jqXhr) {
incrementPendingRequests();
jqXhr = pendingRequests[url] = $.ajax(url, this.ajaxSettings).always(always);
}
return jqXhr;
function always() { function always() {
decrementPendingRequests(); pendingRequestsCount--;
pendingRequests[url] = null; delete pendingRequests[url];
if (that.onDeckRequestArgs) { if (that.onDeckRequestArgs) {
that._get.apply(that, that.onDeckRequestArgs); that._get.apply(that, that.onDeckRequestArgs);
that.onDeckRequestArgs = null; that.onDeckRequestArgs = null;
} }
} }
}, },
get: function(query, cb) { get: function(url, o, cb) {
var that = this, encodedQuery = encodeURIComponent(query || ""), url, resp; var that = this, resp;
cb = cb || utils.noop; if (_.isFunction(o)) {
url = this.replace ? this.replace(this.url, encodedQuery) : this.url.replace(this.wildcard, encodedQuery); cb = o;
o = {};
}
if (resp = requestCache.get(url)) { if (resp = requestCache.get(url)) {
utils.defer(function() { _.defer(function() {
cb(that.filter ? that.filter(resp) : resp); cb && cb(resp);
}); });
} else { } else {
this._get(url, cb); this._get(url, o, cb);
} }
return !!resp; return !!resp;
} }
}); });
return Transport; return Transport;
function incrementPendingRequests() { function callbackToDeferred(fn) {
pendingRequestsCount++; return function customSendWrapper(url, o) {
var deferred = $.Deferred();
fn(url, o, onSuccess, onError);
return deferred;
function onSuccess(resp) {
_.defer(function() {
deferred.resolve(resp);
});
} }
function decrementPendingRequests() { function onError(err) {
pendingRequestsCount--; _.defer(function() {
deferred.reject(err);
});
} }
function belowPendingRequestsThreshold() { };
return pendingRequestsCount < maxPendingRequests;
} }
}(); }();
var Dataset = function() { var SearchIndex = function() {
var keys = { function SearchIndex(o) {
thumbprint: "thumbprint", o = o || {};
protocol: "protocol", if (!o.datumTokenizer || !o.queryTokenizer) {
itemHash: "itemHash", $.error("datumTokenizer and queryTokenizer are both required");
adjacencyList: "adjacencyList"
};
function Dataset(o) {
utils.bindAll(this);
if (utils.isString(o.template) && !o.engine) {
$.error("no template engine specified");
} }
if (!o.local && !o.prefetch && !o.remote) { this.datumTokenizer = o.datumTokenizer;
$.error("one of local, prefetch, or remote is required"); this.queryTokenizer = o.queryTokenizer;
this.datums = [];
this.trie = newNode();
}
_.mixin(SearchIndex.prototype, {
bootstrap: function bootstrap(o) {
this.datums = o.datums;
this.trie = o.trie;
},
add: function(data) {
var that = this;
data = _.isArray(data) ? data : [ data ];
_.each(data, function(datum) {
var id, tokens;
id = that.datums.push(datum) - 1;
tokens = normalizeTokens(that.datumTokenizer(datum));
_.each(tokens, function(token) {
var node, chars, ch, ids;
node = that.trie;
chars = token.split("");
while (ch = chars.shift()) {
node = node.children[ch] || (node.children[ch] = newNode());
node.ids.push(id);
} }
this.name = o.name || utils.getUniqueId();
this.limit = o.limit || 5;
this.minLength = o.minLength || 1;
this.header = o.header;
this.footer = o.footer;
this.valueKey = o.valueKey || "value";
this.template = compileTemplate(o.template, o.engine, this.valueKey);
this.local = o.local;
this.prefetch = o.prefetch;
this.remote = o.remote;
this.itemHash = {};
this.adjacencyList = {};
this.storage = o.name ? new PersistentStorage(o.name) : null;
}
utils.mixin(Dataset.prototype, {
_processLocalData: function(data) {
this._mergeProcessedData(this._processData(data));
},
_loadPrefetchData: function(o) {
var that = this, thumbprint = VERSION + (o.thumbprint || ""), storedThumbprint, storedProtocol, storedItemHash, storedAdjacencyList, isExpired, deferred;
if (this.storage) {
storedThumbprint = this.storage.get(keys.thumbprint);
storedProtocol = this.storage.get(keys.protocol);
storedItemHash = this.storage.get(keys.itemHash);
storedAdjacencyList = this.storage.get(keys.adjacencyList);
}
isExpired = storedThumbprint !== thumbprint || storedProtocol !== utils.getProtocol();
o = utils.isString(o) ? {
url: o
} : o;
o.ttl = utils.isNumber(o.ttl) ? o.ttl : 24 * 60 * 60 * 1e3;
if (storedItemHash && storedAdjacencyList && !isExpired) {
this._mergeProcessedData({
itemHash: storedItemHash,
adjacencyList: storedAdjacencyList
}); });
deferred = $.Deferred().resolve(); });
} else { },
deferred = $.getJSON(o.url).done(processPrefetchData); get: function get(query) {
var that = this, tokens, matches;
tokens = normalizeTokens(this.queryTokenizer(query));
_.each(tokens, function(token) {
var node, chars, ch, ids;
if (matches && matches.length === 0) {
return false;
} }
return deferred; node = that.trie;
function processPrefetchData(data) { chars = token.split("");
var filteredData = o.filter ? o.filter(data) : data, processedData = that._processData(filteredData), itemHash = processedData.itemHash, adjacencyList = processedData.adjacencyList; while (node && (ch = chars.shift())) {
if (that.storage) { node = node.children[ch];
that.storage.set(keys.itemHash, itemHash, o.ttl);
that.storage.set(keys.adjacencyList, adjacencyList, o.ttl);
that.storage.set(keys.thumbprint, thumbprint, o.ttl);
that.storage.set(keys.protocol, utils.getProtocol(), o.ttl);
} }
that._mergeProcessedData(processedData); if (node && chars.length === 0) {
ids = node.ids.slice(0);
matches = matches ? getIntersection(matches, ids) : ids;
} else {
matches = [];
return false;
} }
});
return matches ? _.map(unique(matches), function(id) {
return that.datums[id];
}) : [];
}, },
_transformDatum: function(datum) { serialize: function serialize() {
var value = utils.isString(datum) ? datum : datum[this.valueKey], tokens = datum.tokens || utils.tokenizeText(value), item = { return {
value: value, datums: this.datums,
tokens: tokens trie: this.trie
}; };
if (utils.isString(datum)) {
item.datum = {};
item.datum[this.valueKey] = datum;
} else {
item.datum = datum;
} }
item.tokens = utils.filter(item.tokens, function(token) {
return !utils.isBlankString(token);
}); });
item.tokens = utils.map(item.tokens, function(token) { return SearchIndex;
return token.toLowerCase(); function normalizeTokens(tokens) {
}); tokens = _.filter(tokens, function(token) {
return item; return !!token;
},
_processData: function(data) {
var that = this, itemHash = {}, adjacencyList = {};
utils.each(data, function(i, datum) {
var item = that._transformDatum(datum), id = utils.getUniqueId(item.value);
itemHash[id] = item;
utils.each(item.tokens, function(i, token) {
var character = token.charAt(0), adjacency = adjacencyList[character] || (adjacencyList[character] = [ id ]);
!~utils.indexOf(adjacency, id) && adjacency.push(id);
}); });
tokens = _.map(tokens, function(token) {
return token.toLowerCase();
}); });
return tokens;
}
function newNode() {
return { return {
itemHash: itemHash, ids: [],
adjacencyList: adjacencyList children: {}
}; };
}, }
_mergeProcessedData: function(processedData) { function unique(array) {
var that = this; var seen = {}, uniques = [];
utils.mixin(this.itemHash, processedData.itemHash); for (var i = 0; i < array.length; i++) {
utils.each(processedData.adjacencyList, function(character, adjacency) { if (!seen[array[i]]) {
var masterAdjacency = that.adjacencyList[character]; seen[array[i]] = true;
that.adjacencyList[character] = masterAdjacency ? masterAdjacency.concat(adjacency) : adjacency; uniques.push(array[i]);
}
}
return uniques;
}
function getIntersection(arrayA, arrayB) {
var ai = 0, bi = 0, intersection = [];
arrayA = arrayA.sort(compare);
arrayB = arrayB.sort(compare);
while (ai < arrayA.length && bi < arrayB.length) {
if (arrayA[ai] < arrayB[bi]) {
ai++;
} else if (arrayA[ai] > arrayB[bi]) {
bi++;
} else {
intersection.push(arrayA[ai]);
ai++;
bi++;
}
}
return intersection;
function compare(a, b) {
return a - b;
}
}
}();
var oParser = function() {
return {
local: getLocal,
prefetch: getPrefetch,
remote: getRemote
};
function getLocal(o) {
return o.local || null;
}
function getPrefetch(o) {
var prefetch, defaults;
defaults = {
url: null,
thumbprint: "",
ttl: 24 * 60 * 60 * 1e3,
filter: null,
ajax: {}
};
if (prefetch = o.prefetch || null) {
prefetch = _.isString(prefetch) ? {
url: prefetch
} : prefetch;
prefetch = _.mixin(defaults, prefetch);
prefetch.thumbprint = VERSION + prefetch.thumbprint;
prefetch.ajax.method = prefetch.ajax.method || "get";
prefetch.ajax.dataType = prefetch.ajax.dataType || "json";
!prefetch.url && $.error("prefetch requires url to be set");
}
return prefetch;
}
function getRemote(o) {
var remote, defaults;
defaults = {
url: null,
wildcard: "%QUERY",
replace: null,
rateLimitBy: "debounce",
rateLimitWait: 300,
send: null,
filter: null,
ajax: {}
};
if (remote = o.remote || null) {
remote = _.isString(remote) ? {
url: remote
} : remote;
remote = _.mixin(defaults, remote);
remote.rateLimiter = /^throttle$/i.test(remote.rateLimitBy) ? byThrottle(remote.rateLimitWait) : byDebounce(remote.rateLimitWait);
remote.ajax.method = remote.ajax.method || "get";
remote.ajax.dataType = remote.ajax.dataType || "json";
delete remote.rateLimitBy;
delete remote.rateLimitWait;
!remote.url && $.error("remote requires url to be set");
}
return remote;
function byDebounce(wait) {
return function(fn) {
return _.debounce(fn, wait);
};
}
function byThrottle(wait) {
return function(fn) {
return _.throttle(fn, wait);
};
}
}
}();
var Bloodhound = window.Bloodhound = function() {
var keys;
keys = {
data: "data",
protocol: "protocol",
thumbprint: "thumbprint"
};
function Bloodhound(o) {
if (!o || !o.local && !o.prefetch && !o.remote) {
$.error("one of local, prefetch, or remote is required");
}
this.limit = o.limit || 5;
this.sorter = o.sorter || noSort;
this.dupDetector = o.dupDetector || ignoreDuplicates;
this.local = oParser.local(o);
this.prefetch = oParser.prefetch(o);
this.remote = oParser.remote(o);
this.cacheKey = this.prefetch ? this.prefetch.cacheKey || this.prefetch.url : null;
this.index = new SearchIndex({
datumTokenizer: o.datumTokenizer,
queryTokenizer: o.queryTokenizer
}); });
this.storage = this.cacheKey ? new PersistentStorage(this.cacheKey) : null;
}
Bloodhound.tokenizers = {
whitespace: function whitespaceTokenizer(s) {
return s.split(/\s+/);
}, },
_getLocalSuggestions: function(terms) { nonword: function nonwordTokenizer(s) {
var that = this, firstChars = [], lists = [], shortestList, suggestions = []; return s.split(/\W+/);
utils.each(terms, function(i, term) { }
var firstChar = term.charAt(0); };
!~utils.indexOf(firstChars, firstChar) && firstChars.push(firstChar); _.mixin(Bloodhound.prototype, {
}); _loadPrefetch: function loadPrefetch(o) {
utils.each(firstChars, function(i, firstChar) { var that = this, serialized, deferred;
var list = that.adjacencyList[firstChar]; if (serialized = this._readFromStorage(o.thumbprint)) {
if (!list) { this.index.bootstrap(serialized);
return false; deferred = $.Deferred().resolve();
} else {
deferred = $.ajax(o.url, o.ajax).done(handlePrefetchResponse);
} }
lists.push(list); return deferred;
if (!shortestList || list.length < shortestList.length) { function handlePrefetchResponse(resp) {
shortestList = list; var filtered;
filtered = o.filter ? o.filter(resp) : resp;
that.add(filtered);
that._saveToStorage(that.index.serialize(), o.thumbprint, o.ttl);
} }
}); },
if (lists.length < firstChars.length) { _getFromRemote: function getFromRemote(query, cb) {
return []; var that = this, url, uriEncodedQuery;
query = query || "";
uriEncodedQuery = encodeURIComponent(query);
url = this.remote.replace ? this.remote.replace(this.remote.url, query) : this.remote.url.replace(this.remote.wildcard, uriEncodedQuery);
return this.transport.get(url, this.remote.ajax, handleRemoteResponse);
function handleRemoteResponse(resp) {
var filtered = that.remote.filter ? that.remote.filter(resp) : resp;
cb(filtered);
} }
utils.each(shortestList, function(i, id) { },
var item = that.itemHash[id], isCandidate, isMatch; _saveToStorage: function saveToStorage(data, thumbprint, ttl) {
isCandidate = utils.every(lists, function(list) { if (this.storage) {
return ~utils.indexOf(list, id); this.storage.set(keys.data, data, ttl);
}); this.storage.set(keys.protocol, location.protocol, ttl);
isMatch = isCandidate && utils.every(terms, function(term) { this.storage.set(keys.thumbprint, thumbprint, ttl);
return utils.some(item.tokens, function(token) { }
return token.indexOf(term) === 0; },
}); _readFromStorage: function readFromStorage(thumbprint) {
var stored = {};
if (this.storage) {
stored.data = this.storage.get(keys.data);
stored.protocol = this.storage.get(keys.protocol);
stored.thumbprint = this.storage.get(keys.thumbprint);
}
isExpired = stored.thumbprint !== thumbprint || stored.protocol !== location.protocol;
return stored.data && !isExpired ? stored.data : null;
},
initialize: function initialize() {
var that = this, deferred;
deferred = this.prefetch ? this._loadPrefetch(this.prefetch) : $.Deferred().resolve();
this.local && deferred.done(addLocalToIndex);
this.transport = this.remote ? new Transport(this.remote) : null;
this.initialize = function initialize() {
return deferred.promise();
};
return deferred.promise();
function addLocalToIndex() {
that.add(that.local);
}
},
add: function add(data) {
this.index.add(data);
},
get: function get(query, cb) {
var that = this, matches, cacheHit = false;
matches = this.index.get(query).sort(this.sorter).slice(0, this.limit);
if (matches.length < this.limit && this.transport) {
cacheHit = this._getFromRemote(query, returnRemoteMatches);
}
!cacheHit && cb && cb(matches);
function returnRemoteMatches(remoteMatches) {
var matchesWithBackfill = matches.slice(0);
_.each(remoteMatches, function(remoteMatch) {
var isDuplicate;
isDuplicate = _.some(matchesWithBackfill, function(match) {
return that.dupDetector(remoteMatch, match);
}); });
isMatch && suggestions.push(item); !isDuplicate && matchesWithBackfill.push(remoteMatch);
return matchesWithBackfill.length < that.limit;
}); });
return suggestions; cb && cb(matchesWithBackfill.sort(that.sorter));
}
}, },
initialize: function() { ttAdapter: function ttAdapter() {
var deferred; return _.bind(this.get, this);
this.local && this._processLocalData(this.local); }
this.transport = this.remote ? new Transport(this.remote) : null; });
deferred = this.prefetch ? this._loadPrefetchData(this.prefetch) : $.Deferred().resolve(); return Bloodhound;
this.local = this.prefetch = this.remote = null; function noSort() {
this.initialize = function() { return 0;
return deferred; }
function ignoreDuplicates() {
return false;
}
}();
var html = {
wrapper: '<span class="twitter-typeahead"></span>',
dropdown: '<span class="tt-dropdown-menu"></span>',
dataset: '<div class="tt-dataset-%CLASS%"></div>',
suggestions: '<span class="tt-suggestions"></span>',
suggestion: '<div class="tt-suggestion">%BODY%</div>'
}; };
return deferred; var css = {
wrapper: {
position: "relative",
display: "inline-block"
}, },
getSuggestions: function(query, cb) { hint: {
var that = this, terms, suggestions, cacheHit = false; position: "absolute",
if (query.length < this.minLength) { top: "0",
return; left: "0",
borderColor: "transparent",
boxShadow: "none"
},
input: {
position: "relative",
verticalAlign: "top",
backgroundColor: "transparent"
},
inputWithNoHint: {
position: "relative",
verticalAlign: "top"
},
dropdown: {
position: "absolute",
top: "100%",
left: "0",
zIndex: "100",
display: "none"
},
suggestions: {
display: "block"
},
suggestion: {
whiteSpace: "nowrap",
cursor: "pointer"
},
suggestionChild: {
whiteSpace: "normal"
},
ltr: {
left: "0",
right: "auto"
},
rtl: {
left: "auto",
right: " 0"
} }
terms = utils.tokenizeQuery(query); };
suggestions = this._getLocalSuggestions(terms).slice(0, this.limit); if (_.isMsie()) {
if (suggestions.length < this.limit && this.transport) { _.mixin(css.input, {
cacheHit = this.transport.get(query, processRemoteData); backgroundImage: "url()"
}
!cacheHit && cb && cb(suggestions);
function processRemoteData(data) {
suggestions = suggestions.slice(0);
utils.each(data, function(i, datum) {
var item = that._transformDatum(datum), isDuplicate;
isDuplicate = utils.some(suggestions, function(suggestion) {
return item.value === suggestion.value;
}); });
!isDuplicate && suggestions.push(item); }
return suggestions.length < that.limit; if (_.isMsie() && _.isMsie() <= 7) {
_.mixin(css.input, {
marginTop: "-1px"
}); });
cb && cb(suggestions);
} }
var EventBus = function() {
var namespace = "typeahead:";
function EventBus(o) {
if (!o || !o.el) {
$.error("EventBus initialized without el");
}
this.$el = $(o.el);
}
_.mixin(EventBus.prototype, {
trigger: function(type) {
var args = [].slice.call(arguments, 1);
this.$el.trigger(namespace + type, args);
} }
}); });
return Dataset; return EventBus;
function compileTemplate(template, engine, valueKey) { }();
var renderFn, compiledTemplate; var EventEmitter = function() {
if (utils.isFunction(template)) { var splitter = /\s+/, nextTick = getNextTick();
renderFn = template; return {
} else if (utils.isString(template)) { onSync: onSync,
compiledTemplate = engine.compile(template); onAsync: onAsync,
renderFn = utils.bind(compiledTemplate.render, compiledTemplate); off: off,
trigger: trigger
};
function on(method, types, cb, context) {
var type;
if (!cb) {
return this;
}
types = types.split(splitter);
cb = context ? bindContext(cb, context) : cb;
this._callbacks = this._callbacks || {};
while (type = types.shift()) {
this._callbacks[type] = this._callbacks[type] || {
sync: [],
async: []
};
this._callbacks[type][method].push(cb);
}
return this;
}
function onAsync(types, cb, context) {
return on.call(this, "async", types, cb, context);
}
function onSync(types, cb, context) {
return on.call(this, "sync", types, cb, context);
}
function off(types) {
var type;
if (!this._callbacks) {
return this;
}
types = types.split(splitter);
while (type = types.shift()) {
delete this._callbacks[type];
}
return this;
}
function trigger(types) {
var that = this, type, callbacks, args, syncFlush, asyncFlush;
if (!this._callbacks) {
return this;
}
types = types.split(splitter);
args = [].slice.call(arguments, 1);
while ((type = types.shift()) && (callbacks = this._callbacks[type])) {
syncFlush = getFlush(callbacks.sync, this, [ type ].concat(args));
asyncFlush = getFlush(callbacks.async, this, [ type ].concat(args));
syncFlush() && nextTick(asyncFlush);
}
return this;
}
function getFlush(callbacks, context, args) {
return flush;
function flush() {
var cancelled;
for (var i = 0; !cancelled && i < callbacks.length; i += 1) {
cancelled = callbacks[i].apply(context, args) === false;
}
return !cancelled;
}
}
function getNextTick() {
var nextTickFn, messageChannel;
if (window.setImmediate) {
nextTickFn = function nextTickSetImmediate(fn) {
setImmediate(function() {
fn();
});
};
} else { } else {
renderFn = function(context) { nextTickFn = function nextTickSetTimeout(fn) {
return "<p>" + context[valueKey] + "</p>"; setTimeout(function() {
fn();
}, 0);
}; };
} }
return renderFn; return nextTickFn;
}
function bindContext(fn, context) {
return fn.bind ? fn.bind(context) : function() {
fn.apply(context, [].slice.call(arguments, 0));
};
} }
}(); }();
var InputView = function() { var highlight = function(doc) {
function InputView(o) { var defaults = {
var that = this; node: null,
utils.bindAll(this); pattern: null,
this.specialKeyCodeMap = { tagName: "strong",
className: null,
wordsOnly: false,
caseSensitive: false
};
return function hightlight(o) {
var regex;
o = _.mixin({}, defaults, o);
if (!o.node || !o.pattern) {
return;
}
o.pattern = _.isArray(o.pattern) ? o.pattern : [ o.pattern ];
regex = getRegex(o.pattern, o.caseSensitive, o.wordsOnly);
traverse(o.node, hightlightTextNode);
function hightlightTextNode(textNode) {
var match, patternNode;
if (match = regex.exec(textNode.data)) {
wrapperNode = doc.createElement(o.tagName);
o.className && (wrapperNode.className = o.className);
patternNode = textNode.splitText(match.index);
patternNode.splitText(match[0].length);
wrapperNode.appendChild(patternNode.cloneNode(true));
textNode.parentNode.replaceChild(wrapperNode, patternNode);
}
return !!match;
}
function traverse(el, hightlightTextNode) {
var childNode, TEXT_NODE_TYPE = 3;
for (var i = 0; i < el.childNodes.length; i++) {
childNode = el.childNodes[i];
if (childNode.nodeType === TEXT_NODE_TYPE) {
i += hightlightTextNode(childNode) ? 1 : 0;
} else {
traverse(childNode, hightlightTextNode);
}
}
}
};
function getRegex(patterns, caseSensitive, wordsOnly) {
var escapedPatterns = [], regexStr;
for (var i = 0; i < patterns.length; i++) {
escapedPatterns.push(_.escapeRegExChars(patterns[i]));
}
regexStr = wordsOnly ? "\\b(" + escapedPatterns.join("|") + ")\\b" : "(" + escapedPatterns.join("|") + ")";
return caseSensitive ? new RegExp(regexStr) : new RegExp(regexStr, "i");
}
}(window.document);
var Input = function() {
var specialKeyCodeMap;
specialKeyCodeMap = {
9: "tab", 9: "tab",
27: "esc", 27: "esc",
37: "left", 37: "left",
...@@ -573,84 +868,141 @@ ...@@ -573,84 +868,141 @@
38: "up", 38: "up",
40: "down" 40: "down"
}; };
function Input(o) {
var that = this, onBlur, onFocus, onKeydown, onInput;
o = o || {};
if (!o.input) {
$.error("input is missing");
}
onBlur = _.bind(this._onBlur, this);
onFocus = _.bind(this._onFocus, this);
onKeydown = _.bind(this._onKeydown, this);
onInput = _.bind(this._onInput, this);
this.$hint = $(o.hint); this.$hint = $(o.hint);
this.$input = $(o.input).on("blur.tt", this._handleBlur).on("focus.tt", this._handleFocus).on("keydown.tt", this._handleSpecialKeyEvent); this.$input = $(o.input).on("blur.tt", onBlur).on("focus.tt", onFocus).on("keydown.tt", onKeydown);
if (!utils.isMsie()) { if (this.$hint.length === 0) {
this.$input.on("input.tt", this._compareQueryToInputValue); this.setHintValue = this.getHintValue = this.clearHint = _.noop;
}
if (!_.isMsie()) {
this.$input.on("input.tt", onInput);
} else { } else {
this.$input.on("keydown.tt keypress.tt cut.tt paste.tt", function($e) { this.$input.on("keydown.tt keypress.tt cut.tt paste.tt", function($e) {
if (that.specialKeyCodeMap[$e.which || $e.keyCode]) { if (specialKeyCodeMap[$e.which || $e.keyCode]) {
return; return;
} }
utils.defer(that._compareQueryToInputValue); _.defer(_.bind(that._onInput, that, $e));
}); });
} }
this.query = this.$input.val(); this.query = this.$input.val();
this.$overflowHelper = buildOverflowHelper(this.$input); this.$overflowHelper = buildOverflowHelper(this.$input);
} }
utils.mixin(InputView.prototype, EventTarget, { Input.normalizeQuery = function(str) {
_handleFocus: function() { return (str || "").replace(/^\s*/g, "").replace(/\s{2,}/g, " ");
};
_.mixin(Input.prototype, EventEmitter, {
_onBlur: function onBlur($e) {
this.resetInputValue();
this.trigger("blurred");
},
_onFocus: function onFocus($e) {
this.trigger("focused"); this.trigger("focused");
}, },
_handleBlur: function() { _onKeydown: function onKeydown($e) {
this.trigger("blured"); var keyName = specialKeyCodeMap[$e.which || $e.keyCode];
this._managePreventDefault(keyName, $e);
if (keyName && this._shouldTrigger(keyName, $e)) {
this.trigger(keyName + "Keyed", $e);
}
}, },
_handleSpecialKeyEvent: function($e) { _onInput: function onInput($e) {
var keyName = this.specialKeyCodeMap[$e.which || $e.keyCode]; this._checkInputValue();
keyName && this.trigger(keyName + "Keyed", $e);
}, },
_compareQueryToInputValue: function() { _managePreventDefault: function managePreventDefault(keyName, $e) {
var inputValue = this.getInputValue(), isSameQuery = compareQueries(this.query, inputValue), isSameQueryExceptWhitespace = isSameQuery ? this.query.length !== inputValue.length : false; var preventDefault, hintValue, inputValue;
if (isSameQueryExceptWhitespace) { switch (keyName) {
this.trigger("whitespaceChanged", { case "tab":
value: this.query hintValue = this.getHintValue();
}); inputValue = this.getInputValue();
} else if (!isSameQuery) { preventDefault = hintValue && hintValue !== inputValue && !withModifier($e);
this.trigger("queryChanged", { break;
value: this.query = inputValue
}); case "up":
case "down":
preventDefault = !withModifier($e);
break;
default:
preventDefault = false;
} }
preventDefault && $e.preventDefault();
}, },
destroy: function() { _shouldTrigger: function shouldTrigger(keyName, $e) {
this.$hint.off(".tt"); var trigger;
this.$input.off(".tt"); switch (keyName) {
this.$hint = this.$input = this.$overflowHelper = null; case "tab":
trigger = !withModifier($e);
break;
default:
trigger = true;
}
return trigger;
},
_checkInputValue: function checkInputValue() {
var inputValue, areEquivalent, hasDifferentWhitespace;
inputValue = this.getInputValue();
areEquivalent = areQueriesEquivalent(inputValue, this.query);
hasDifferentWhitespace = areEquivalent ? this.query.length !== inputValue.length : false;
if (!areEquivalent) {
this.trigger("queryChanged", this.query = inputValue);
} else if (hasDifferentWhitespace) {
this.trigger("whitespaceChanged", this.query);
}
}, },
focus: function() { focus: function focus() {
this.$input.focus(); this.$input.focus();
}, },
blur: function() { blur: function blur() {
this.$input.blur(); this.$input.blur();
}, },
getQuery: function() { getQuery: function getQuery() {
return this.query; return this.query;
}, },
setQuery: function(query) { setQuery: function setQuery(query) {
this.query = query; this.query = query;
}, },
getInputValue: function() { getInputValue: function getInputValue() {
return this.$input.val(); return this.$input.val();
}, },
setInputValue: function(value, silent) { setInputValue: function setInputValue(value, silent) {
this.$input.val(value); this.$input.val(value);
!silent && this._compareQueryToInputValue(); !silent && this._checkInputValue();
}, },
getHintValue: function() { getHintValue: function getHintValue() {
return this.$hint.val(); return this.$hint.val();
}, },
setHintValue: function(value) { setHintValue: function setHintValue(value) {
this.$hint.val(value); this.$hint.val(value);
}, },
getLanguageDirection: function() { resetInputValue: function resetInputValue() {
this.$input.val(this.query);
},
clearHint: function clearHint() {
this.$hint.val("");
},
getLanguageDirection: function getLanguageDirection() {
return (this.$input.css("direction") || "ltr").toLowerCase(); return (this.$input.css("direction") || "ltr").toLowerCase();
}, },
isOverflow: function() { hasOverflow: function hasOverflow() {
var constraint = this.$input.width() - 2;
this.$overflowHelper.text(this.getInputValue()); this.$overflowHelper.text(this.getInputValue());
return this.$overflowHelper.width() > this.$input.width(); return this.$overflowHelper.width() >= constraint;
}, },
isCursorAtEnd: function() { isCursorAtEnd: function() {
var valueLength = this.$input.val().length, selectionStart = this.$input[0].selectionStart, range; var valueLength, selectionStart, range;
if (utils.isNumber(selectionStart)) { valueLength = this.$input.val().length;
selectionStart = this.$input[0].selectionStart;
if (_.isNumber(selectionStart)) {
return selectionStart === valueLength; return selectionStart === valueLength;
} else if (document.selection) { } else if (document.selection) {
range = document.selection.createRange(); range = document.selection.createRange();
...@@ -658,13 +1010,17 @@ ...@@ -658,13 +1010,17 @@
return valueLength === range.text.length; return valueLength === range.text.length;
} }
return true; return true;
},
destroy: function destroy() {
this.$hint.off(".tt");
this.$input.off(".tt");
this.$hint = this.$input = this.$overflowHelper = null;
} }
}); });
return InputView; return Input;
function buildOverflowHelper($input) { function buildOverflowHelper($input) {
return $("<span></span>").css({ return $('<pre aria-hidden="true"></pre>').css({
position: "absolute", position: "absolute",
left: "-9999px",
visibility: "hidden", visibility: "hidden",
whiteSpace: "nowrap", whiteSpace: "nowrap",
fontFamily: $input.css("font-family"), fontFamily: $input.css("font-family"),
...@@ -679,452 +1035,597 @@ ...@@ -679,452 +1035,597 @@
textTransform: $input.css("text-transform") textTransform: $input.css("text-transform")
}).insertAfter($input); }).insertAfter($input);
} }
function compareQueries(a, b) { function areQueriesEquivalent(a, b) {
a = (a || "").replace(/^\s*/g, "").replace(/\s{2,}/g, " "); return Input.normalizeQuery(a) === Input.normalizeQuery(b);
b = (b || "").replace(/^\s*/g, "").replace(/\s{2,}/g, " "); }
return a === b; function withModifier($e) {
return $e.altKey || $e.ctrlKey || $e.metaKey || $e.shiftKey;
} }
}(); }();
var DropdownView = function() { var Dataset = function() {
var html = { var datasetKey = "ttDataset", valueKey = "ttValue", datumKey = "ttDatum";
suggestionsList: '<span class="tt-suggestions"></span>' function Dataset(o) {
}, css = { o = o || {};
suggestionsList: { o.templates = o.templates || {};
display: "block" if (!o.source) {
$.error("missing source");
}
this.query = null;
this.highlight = !!o.highlight;
this.name = o.name || _.getUniqueId();
this.source = o.source;
this.valueKey = o.displayKey || "value";
this.templates = getTemplates(o.templates, this.valueKey);
this.$el = $(html.dataset.replace("%CLASS%", this.name));
}
Dataset.extractDatasetName = function extractDatasetName(el) {
return $(el).data(datasetKey);
};
Dataset.extractValue = function extractDatum(el) {
return $(el).data(valueKey);
};
Dataset.extractDatum = function extractDatum(el) {
return $(el).data(datumKey);
};
_.mixin(Dataset.prototype, EventEmitter, {
_render: function render(query, suggestions) {
if (!this.$el) {
return;
}
var that = this, hasSuggestions;
this.$el.empty();
hasSuggestions = suggestions && suggestions.length;
if (!hasSuggestions && this.templates.empty) {
this.$el.html(getEmptyHtml()).prepend(that.templates.header ? getHeaderHtml() : null).append(that.templates.footer ? getFooterHtml() : null);
} else if (hasSuggestions) {
this.$el.html(getSuggestionsHtml()).prepend(that.templates.header ? getHeaderHtml() : null).append(that.templates.footer ? getFooterHtml() : null);
}
this.trigger("rendered");
function getEmptyHtml() {
return that.templates.empty({
query: query
});
}
function getSuggestionsHtml() {
var $suggestions;
$suggestions = $(html.suggestions).css(css.suggestions).append(_.map(suggestions, getSuggestionNode));
that.highlight && highlight({
node: $suggestions[0],
pattern: query
});
return $suggestions;
function getSuggestionNode(suggestion) {
var $el, innerHtml, outerHtml;
innerHtml = that.templates.suggestion(suggestion);
outerHtml = html.suggestion.replace("%BODY%", innerHtml);
$el = $(outerHtml).data(datasetKey, that.name).data(valueKey, suggestion[that.valueKey]).data(datumKey, suggestion);
$el.children().each(function() {
$(this).css(css.suggestionChild);
});
return $el;
}
}
function getHeaderHtml() {
return that.templates.header({
query: query,
isEmpty: !hasSuggestions
});
}
function getFooterHtml() {
return that.templates.footer({
query: query,
isEmpty: !hasSuggestions
});
}
}, },
suggestion: { getRoot: function getRoot() {
whiteSpace: "nowrap", return this.$el;
cursor: "pointer"
}, },
suggestionChild: { update: function update(query) {
whiteSpace: "normal" var that = this;
this.query = query;
this.source(query, renderIfQueryIsSame);
function renderIfQueryIsSame(suggestions) {
query === that.query && that._render(query, suggestions);
}
},
clear: function clear() {
this._render(this.query || "");
},
isEmpty: function isEmpty() {
return this.$el.is(":empty");
},
destroy: function destroy() {
this.$el = null;
} }
});
return Dataset;
function getTemplates(templates, valueKey) {
return {
empty: templates.empty && _.templatify(templates.empty),
header: templates.header && _.templatify(templates.header),
footer: templates.footer && _.templatify(templates.footer),
suggestion: templates.suggestion || suggestionTemplate
}; };
function DropdownView(o) { function suggestionTemplate(context) {
utils.bindAll(this); return "<p>" + context[valueKey] + "</p>";
}
}
}();
var Dropdown = function() {
function Dropdown(o) {
var that = this, onMouseEnter, onMouseLeave, onSuggestionClick, onSuggestionMouseEnter, onSuggestionMouseLeave;
o = o || {};
if (!o.menu) {
$.error("menu is required");
}
this.isOpen = false; this.isOpen = false;
this.isEmpty = true; this.isEmpty = true;
this.isMouseOverDropdown = false; this.isMouseOverDropdown = false;
this.$menu = $(o.menu).on("mouseenter.tt", this._handleMouseenter).on("mouseleave.tt", this._handleMouseleave).on("click.tt", ".tt-suggestion", this._handleSelection).on("mouseover.tt", ".tt-suggestion", this._handleMouseover); this.datasets = _.map(o.datasets, initializeDataset);
onMouseEnter = _.bind(this._onMouseEnter, this);
onMouseLeave = _.bind(this._onMouseLeave, this);
onSuggestionClick = _.bind(this._onSuggestionClick, this);
onSuggestionMouseEnter = _.bind(this._onSuggestionMouseEnter, this);
onSuggestionMouseLeave = _.bind(this._onSuggestionMouseLeave, this);
this.$menu = $(o.menu).on("mouseenter.tt", onMouseEnter).on("mouseleave.tt", onMouseLeave).on("click.tt", ".tt-suggestion", onSuggestionClick).on("mouseenter.tt", ".tt-suggestion", onSuggestionMouseEnter).on("mouseleave.tt", ".tt-suggestion", onSuggestionMouseLeave);
_.each(this.datasets, function(dataset) {
that.$menu.append(dataset.getRoot());
dataset.onSync("rendered", that._onRendered, that);
});
} }
utils.mixin(DropdownView.prototype, EventTarget, { _.mixin(Dropdown.prototype, EventEmitter, {
_handleMouseenter: function() { _onMouseEnter: function onMouseEnter($e) {
this.isMouseOverDropdown = true; this.isMouseOverDropdown = true;
}, },
_handleMouseleave: function() { _onMouseLeave: function onMouseLeave($e) {
this.isMouseOverDropdown = false; this.isMouseOverDropdown = false;
}, },
_handleMouseover: function($e) { _onSuggestionClick: function onSuggestionClick($e) {
var $suggestion = $($e.currentTarget); this.trigger("suggestionClicked", $($e.currentTarget));
this._getSuggestions().removeClass("tt-is-under-cursor");
$suggestion.addClass("tt-is-under-cursor");
}, },
_handleSelection: function($e) { _onSuggestionMouseEnter: function onSuggestionMouseEnter($e) {
var $suggestion = $($e.currentTarget); this._removeCursor();
this.trigger("suggestionSelected", extractSuggestion($suggestion)); this._setCursor($($e.currentTarget), true);
}, },
_show: function() { _onSuggestionMouseLeave: function onSuggestionMouseLeave($e) {
this.$menu.css("display", "block"); this._removeCursor();
},
_onRendered: function onRendered() {
this.isEmpty = _.every(this.datasets, isDatasetEmpty);
this.isEmpty ? this._hide() : this.isOpen && this._show();
this.trigger("datasetRendered");
function isDatasetEmpty(dataset) {
return dataset.isEmpty();
}
}, },
_hide: function() { _hide: function() {
this.$menu.hide(); this.$menu.hide();
}, },
_moveCursor: function(increment) { _show: function() {
var $suggestions, $cur, nextIndex, $underCursor; this.$menu.css("display", "block");
if (!this.isVisible()) { },
_getSuggestions: function getSuggestions() {
return this.$menu.find(".tt-suggestion");
},
_getCursor: function getCursor() {
return this.$menu.find(".tt-cursor").first();
},
_setCursor: function setCursor($el, silent) {
$el.first().addClass("tt-cursor");
!silent && this.trigger("cursorMoved");
},
_removeCursor: function removeCursor() {
this._getCursor().removeClass("tt-cursor");
},
_moveCursor: function moveCursor(increment) {
var $suggestions, $oldCursor, newCursorIndex, $newCursor;
if (!this.isOpen) {
return; return;
} }
$oldCursor = this._getCursor();
$suggestions = this._getSuggestions(); $suggestions = this._getSuggestions();
$cur = $suggestions.filter(".tt-is-under-cursor"); this._removeCursor();
$cur.removeClass("tt-is-under-cursor"); newCursorIndex = $suggestions.index($oldCursor) + increment;
nextIndex = $suggestions.index($cur) + increment; newCursorIndex = (newCursorIndex + 1) % ($suggestions.length + 1) - 1;
nextIndex = (nextIndex + 1) % ($suggestions.length + 1) - 1; if (newCursorIndex === -1) {
if (nextIndex === -1) {
this.trigger("cursorRemoved"); this.trigger("cursorRemoved");
return; return;
} else if (nextIndex < -1) { } else if (newCursorIndex < -1) {
nextIndex = $suggestions.length - 1; newCursorIndex = $suggestions.length - 1;
} }
$underCursor = $suggestions.eq(nextIndex).addClass("tt-is-under-cursor"); this._setCursor($newCursor = $suggestions.eq(newCursorIndex));
this._ensureVisibility($underCursor); this._ensureVisible($newCursor);
this.trigger("cursorMoved", extractSuggestion($underCursor)); },
}, _ensureVisible: function ensureVisible($el) {
_getSuggestions: function() { var elTop, elBottom, menuScrollTop, menuHeight;
return this.$menu.find(".tt-suggestions > .tt-suggestion"); elTop = $el.position().top;
}, elBottom = elTop + $el.outerHeight(true);
_ensureVisibility: function($el) { menuScrollTop = this.$menu.scrollTop();
var menuHeight = this.$menu.height() + parseInt(this.$menu.css("paddingTop"), 10) + parseInt(this.$menu.css("paddingBottom"), 10), menuScrollTop = this.$menu.scrollTop(), elTop = $el.position().top, elBottom = elTop + $el.outerHeight(true); menuHeight = this.$menu.height() + parseInt(this.$menu.css("paddingTop"), 10) + parseInt(this.$menu.css("paddingBottom"), 10);
if (elTop < 0) { if (elTop < 0) {
this.$menu.scrollTop(menuScrollTop + elTop); this.$menu.scrollTop(menuScrollTop + elTop);
} else if (menuHeight < elBottom) { } else if (menuHeight < elBottom) {
this.$menu.scrollTop(menuScrollTop + (elBottom - menuHeight)); this.$menu.scrollTop(menuScrollTop + (elBottom - menuHeight));
} }
}, },
destroy: function() { close: function close() {
this.$menu.off(".tt");
this.$menu = null;
},
isVisible: function() {
return this.isOpen && !this.isEmpty;
},
closeUnlessMouseIsOverDropdown: function() {
if (!this.isMouseOverDropdown) {
this.close();
}
},
close: function() {
if (this.isOpen) { if (this.isOpen) {
this.isOpen = false; this.isOpen = this.isMouseOverDropdown = false;
this.isMouseOverDropdown = false; this._removeCursor();
this._hide(); this._hide();
this.$menu.find(".tt-suggestions > .tt-suggestion").removeClass("tt-is-under-cursor");
this.trigger("closed"); this.trigger("closed");
} }
}, },
open: function() { open: function open() {
if (!this.isOpen) { if (!this.isOpen) {
this.isOpen = true; this.isOpen = true;
!this.isEmpty && this._show(); !this.isEmpty && this._show();
this.trigger("opened"); this.trigger("opened");
} }
}, },
setLanguageDirection: function(dir) { setLanguageDirection: function setLanguageDirection(dir) {
var ltrCss = { this.$menu.css(dir === "ltr" ? css.ltr : css.rtl);
left: "0",
right: "auto"
}, rtlCss = {
left: "auto",
right: " 0"
};
dir === "ltr" ? this.$menu.css(ltrCss) : this.$menu.css(rtlCss);
}, },
moveCursorUp: function() { moveCursorUp: function moveCursorUp() {
this._moveCursor(-1); this._moveCursor(-1);
}, },
moveCursorDown: function() { moveCursorDown: function moveCursorDown() {
this._moveCursor(+1); this._moveCursor(+1);
}, },
getSuggestionUnderCursor: function() { getDatumForSuggestion: function getDatumForSuggestion($el) {
var $suggestion = this._getSuggestions().filter(".tt-is-under-cursor").first(); var datum = null;
return $suggestion.length > 0 ? extractSuggestion($suggestion) : null; if ($el.length) {
}, datum = {
getFirstSuggestion: function() { raw: Dataset.extractDatum($el),
var $suggestion = this._getSuggestions().first(); value: Dataset.extractValue($el),
return $suggestion.length > 0 ? extractSuggestion($suggestion) : null; datasetName: Dataset.extractDatasetName($el)
}, };
renderSuggestions: function(dataset, suggestions) {
var datasetClassName = "tt-dataset-" + dataset.name, wrapper = '<div class="tt-suggestion">%body</div>', compiledHtml, $suggestionsList, $dataset = this.$menu.find("." + datasetClassName), elBuilder, fragment, $el;
if ($dataset.length === 0) {
$suggestionsList = $(html.suggestionsList).css(css.suggestionsList);
$dataset = $("<div></div>").addClass(datasetClassName).append(dataset.header).append($suggestionsList).append(dataset.footer).appendTo(this.$menu);
}
if (suggestions.length > 0) {
this.isEmpty = false;
this.isOpen && this._show();
elBuilder = document.createElement("div");
fragment = document.createDocumentFragment();
utils.each(suggestions, function(i, suggestion) {
suggestion.dataset = dataset.name;
compiledHtml = dataset.template(suggestion.datum);
elBuilder.innerHTML = wrapper.replace("%body", compiledHtml);
$el = $(elBuilder.firstChild).css(css.suggestion).data("suggestion", suggestion);
$el.children().each(function() {
$(this).css(css.suggestionChild);
});
fragment.appendChild($el[0]);
});
$dataset.show().find(".tt-suggestions").html(fragment);
} else {
this.clearSuggestions(dataset.name);
} }
this.trigger("suggestionsRendered"); return datum;
}, },
clearSuggestions: function(datasetName) { getDatumForCursor: function getDatumForCursor() {
var $datasets = datasetName ? this.$menu.find(".tt-dataset-" + datasetName) : this.$menu.find('[class^="tt-dataset-"]'), $suggestions = $datasets.find(".tt-suggestions"); return this.getDatumForSuggestion(this._getCursor().first());
$datasets.hide(); },
$suggestions.empty(); getDatumForTopSuggestion: function getDatumForTopSuggestion() {
if (this._getSuggestions().length === 0) { return this.getDatumForSuggestion(this._getSuggestions().first());
this.isEmpty = true; },
this._hide(); update: function update(query) {
} _.each(this.datasets, updateDataset);
} function updateDataset(dataset) {
}); dataset.update(query);
return DropdownView;
function extractSuggestion($el) {
return $el.data("suggestion");
} }
}();
var TypeaheadView = function() {
var html = {
wrapper: '<span class="twitter-typeahead"></span>',
hint: '<input class="tt-hint" type="text" autocomplete="off" spellcheck="off" disabled>',
dropdown: '<span class="tt-dropdown-menu"></span>'
}, css = {
wrapper: {
position: "relative",
display: "inline-block"
}, },
hint: { empty: function empty() {
position: "absolute", _.each(this.datasets, clearDataset);
top: "0", function clearDataset(dataset) {
left: "0", dataset.clear();
borderColor: "transparent", }
boxShadow: "none"
}, },
query: { isVisible: function isVisible() {
position: "relative", return this.isOpen && !this.isEmpty;
verticalAlign: "top",
backgroundColor: "transparent"
}, },
dropdown: { destroy: function destroy() {
position: "absolute", this.$menu.off(".tt");
top: "100%", this.$menu = null;
left: "0", _.each(this.datasets, destroyDataset);
zIndex: "100", function destroyDataset(dataset) {
display: "none" dataset.destroy();
} }
};
if (utils.isMsie()) {
utils.mixin(css.query, {
backgroundImage: "url()"
});
} }
if (utils.isMsie() && utils.isMsie() <= 7) {
utils.mixin(css.wrapper, {
display: "inline",
zoom: "1"
});
utils.mixin(css.query, {
marginTop: "-1px"
}); });
return Dropdown;
function initializeDataset(oDataset) {
return new Dataset(oDataset);
} }
function TypeaheadView(o) { }();
var $menu, $input, $hint; var Typeahead = function() {
utils.bindAll(this); var attrsKey = "ttAttrs";
this.$node = buildDomStructure(o.input); function Typeahead(o) {
this.datasets = o.datasets; var $menu, $input, $hint, datasets;
this.dir = null; o = o || {};
this.eventBus = o.eventBus; if (!o.input) {
$.error("missing input");
}
this.autoselect = !!o.autoselect;
this.minLength = _.isNumber(o.minLength) ? o.minLength : 1;
this.$node = buildDomStructure(o.input, o.withHint);
$menu = this.$node.find(".tt-dropdown-menu"); $menu = this.$node.find(".tt-dropdown-menu");
$input = this.$node.find(".tt-query"); $input = this.$node.find(".tt-input");
$hint = this.$node.find(".tt-hint"); $hint = this.$node.find(".tt-hint");
this.dropdownView = new DropdownView({ this.eventBus = o.eventBus || new EventBus({
menu: $menu el: $input
}).on("suggestionSelected", this._handleSelection).on("cursorMoved", this._clearHint).on("cursorMoved", this._setInputValueToSuggestionUnderCursor).on("cursorRemoved", this._setInputValueToQuery).on("cursorRemoved", this._updateHint).on("suggestionsRendered", this._updateHint).on("opened", this._updateHint).on("closed", this._clearHint).on("opened closed", this._propagateEvent); });
this.inputView = new InputView({ this.dropdown = new Dropdown({
menu: $menu,
datasets: o.datasets
}).onSync("suggestionClicked", this._onSuggestionClicked, this).onSync("cursorMoved", this._onCursorMoved, this).onSync("cursorRemoved", this._onCursorRemoved, this).onSync("opened", this._onOpened, this).onSync("closed", this._onClosed, this).onAsync("datasetRendered", this._onDatasetRendered, this);
this.input = new Input({
input: $input, input: $input,
hint: $hint hint: $hint
}).on("focused", this._openDropdown).on("blured", this._closeDropdown).on("blured", this._setInputValueToQuery).on("enterKeyed tabKeyed", this._handleSelection).on("queryChanged", this._clearHint).on("queryChanged", this._clearSuggestions).on("queryChanged", this._getSuggestions).on("whitespaceChanged", this._updateHint).on("queryChanged whitespaceChanged", this._openDropdown).on("queryChanged whitespaceChanged", this._setLanguageDirection).on("escKeyed", this._closeDropdown).on("escKeyed", this._setInputValueToQuery).on("tabKeyed upKeyed downKeyed", this._managePreventDefault).on("upKeyed downKeyed", this._moveDropdownCursor).on("upKeyed downKeyed", this._openDropdown).on("tabKeyed leftKeyed rightKeyed", this._autocomplete); }).onSync("focused", this._onFocused, this).onSync("blurred", this._onBlurred, this).onSync("enterKeyed", this._onEnterKeyed, this).onSync("tabKeyed", this._onTabKeyed, this).onSync("escKeyed", this._onEscKeyed, this).onSync("upKeyed", this._onUpKeyed, this).onSync("downKeyed", this._onDownKeyed, this).onSync("leftKeyed", this._onLeftKeyed, this).onSync("rightKeyed", this._onRightKeyed, this).onSync("queryChanged", this._onQueryChanged, this).onSync("whitespaceChanged", this._onWhitespaceChanged, this);
} $menu.on("mousedown.tt", function($e) {
utils.mixin(TypeaheadView.prototype, EventTarget, { if (_.isMsie() && _.isMsie() < 9) {
_managePreventDefault: function(e) { $input[0].onbeforedeactivate = function() {
var $e = e.data, hint, inputValue, preventDefault = false; window.event.returnValue = false;
switch (e.type) { $input[0].onbeforedeactivate = null;
case "tabKeyed": };
hint = this.inputView.getHintValue();
inputValue = this.inputView.getInputValue();
preventDefault = hint && hint !== inputValue;
break;
case "upKeyed":
case "downKeyed":
preventDefault = !$e.shiftKey && !$e.ctrlKey && !$e.metaKey;
break;
} }
preventDefault && $e.preventDefault(); $e.preventDefault();
}, });
_setLanguageDirection: function() {
var dir = this.inputView.getLanguageDirection();
if (dir !== this.dir) {
this.dir = dir;
this.$node.css("direction", dir);
this.dropdownView.setLanguageDirection(dir);
} }
}, _.mixin(Typeahead.prototype, {
_updateHint: function() { _onSuggestionClicked: function onSuggestionClicked(type, $el) {
var suggestion = this.dropdownView.getFirstSuggestion(), hint = suggestion ? suggestion.value : null, dropdownIsVisible = this.dropdownView.isVisible(), inputHasOverflow = this.inputView.isOverflow(), inputValue, query, escapedQuery, beginsWithQuery, match; var datum;
if (hint && dropdownIsVisible && !inputHasOverflow) { if (datum = this.dropdown.getDatumForSuggestion($el)) {
inputValue = this.inputView.getInputValue(); this._select(datum);
query = inputValue.replace(/\s{2,}/g, " ").replace(/^\s+/g, ""); }
escapedQuery = utils.escapeRegExChars(query); },
beginsWithQuery = new RegExp("^(?:" + escapedQuery + ")(.*$)", "i"); _onCursorMoved: function onCursorMoved() {
match = beginsWithQuery.exec(hint); var datum = this.dropdown.getDatumForCursor();
this.inputView.setHintValue(inputValue + (match ? match[1] : "")); this.input.clearHint();
this.input.setInputValue(datum.value, true);
this.eventBus.trigger("cursorchanged", datum.raw, datum.datasetName);
},
_onCursorRemoved: function onCursorRemoved() {
this.input.resetInputValue();
this._updateHint();
},
_onDatasetRendered: function onDatasetRendered() {
this._updateHint();
},
_onOpened: function onOpened() {
this._updateHint();
this.eventBus.trigger("opened");
},
_onClosed: function onClosed() {
this.input.clearHint();
this.eventBus.trigger("closed");
},
_onFocused: function onFocused() {
this.dropdown.open();
},
_onBlurred: function onBlurred() {
!this.dropdown.isMouseOverDropdown && this.dropdown.close();
},
_onEnterKeyed: function onEnterKeyed(type, $e) {
var cursorDatum, topSuggestionDatum;
cursorDatum = this.dropdown.getDatumForCursor();
topSuggestionDatum = this.dropdown.getDatumForTopSuggestion();
if (cursorDatum) {
this._select(cursorDatum);
$e.preventDefault();
} else if (this.autoselect && topSuggestionDatum) {
this._select(topSuggestionDatum);
$e.preventDefault();
}
},
_onTabKeyed: function onTabKeyed(type, $e) {
var datum;
if (datum = this.dropdown.getDatumForCursor()) {
this._select(datum);
$e.preventDefault();
} else {
this._autocomplete();
} }
}, },
_clearHint: function() { _onEscKeyed: function onEscKeyed() {
this.inputView.setHintValue(""); this.dropdown.close();
}, this.input.resetInputValue();
_clearSuggestions: function() {
this.dropdownView.clearSuggestions();
},
_setInputValueToQuery: function() {
this.inputView.setInputValue(this.inputView.getQuery());
},
_setInputValueToSuggestionUnderCursor: function(e) {
var suggestion = e.data;
this.inputView.setInputValue(suggestion.value, true);
},
_openDropdown: function() {
this.dropdownView.open();
},
_closeDropdown: function(e) {
this.dropdownView[e.type === "blured" ? "closeUnlessMouseIsOverDropdown" : "close"]();
}, },
_moveDropdownCursor: function(e) { _onUpKeyed: function onUpKeyed() {
var $e = e.data; var query = this.input.getQuery();
if (!$e.shiftKey && !$e.ctrlKey && !$e.metaKey) { if (!this.dropdown.isOpen && query.length >= this.minLength) {
this.dropdownView[e.type === "upKeyed" ? "moveCursorUp" : "moveCursorDown"](); this.dropdown.update(query);
} }
this.dropdown.open();
this.dropdown.moveCursorUp();
}, },
_handleSelection: function(e) { _onDownKeyed: function onDownKeyed() {
var byClick = e.type === "suggestionSelected", suggestion = byClick ? e.data : this.dropdownView.getSuggestionUnderCursor(); var query = this.input.getQuery();
if (suggestion) { if (!this.dropdown.isOpen && query.length >= this.minLength) {
this.inputView.setInputValue(suggestion.value); this.dropdown.update(query);
byClick ? this.inputView.focus() : e.data.preventDefault();
byClick && utils.isMsie() ? utils.defer(this.dropdownView.close) : this.dropdownView.close();
this.eventBus.trigger("selected", suggestion.datum, suggestion.dataset);
} }
this.dropdown.open();
this.dropdown.moveCursorDown();
}, },
_getSuggestions: function() { _onLeftKeyed: function onLeftKeyed() {
var that = this, query = this.inputView.getQuery(); this.dir === "rtl" && this._autocomplete();
if (utils.isBlankString(query)) {
return;
}
utils.each(this.datasets, function(i, dataset) {
dataset.getSuggestions(query, function(suggestions) {
if (query === that.inputView.getQuery()) {
that.dropdownView.renderSuggestions(dataset, suggestions);
}
});
});
}, },
_autocomplete: function(e) { _onRightKeyed: function onRightKeyed() {
var isCursorAtEnd, ignoreEvent, query, hint, suggestion; this.dir === "ltr" && this._autocomplete();
if (e.type === "rightKeyed" || e.type === "leftKeyed") { },
isCursorAtEnd = this.inputView.isCursorAtEnd(); _onQueryChanged: function onQueryChanged(e, query) {
ignoreEvent = this.inputView.getLanguageDirection() === "ltr" ? e.type === "leftKeyed" : e.type === "rightKeyed"; this.input.clearHint();
if (!isCursorAtEnd || ignoreEvent) { this.dropdown.empty();
return; query.length >= this.minLength && this.dropdown.update(query);
} this.dropdown.open();
} this._setLanguageDirection();
query = this.inputView.getQuery();
hint = this.inputView.getHintValue();
if (hint !== "" && query !== hint) {
suggestion = this.dropdownView.getFirstSuggestion();
this.inputView.setInputValue(suggestion.value);
this.eventBus.trigger("autocompleted", suggestion.datum, suggestion.dataset);
}
}, },
_propagateEvent: function(e) { _onWhitespaceChanged: function onWhitespaceChanged() {
this.eventBus.trigger(e.type); this._updateHint();
this.dropdown.open();
}, },
destroy: function() { _setLanguageDirection: function setLanguageDirection() {
this.inputView.destroy(); var dir;
this.dropdownView.destroy(); if (this.dir !== (dir = this.input.getLanguageDirection())) {
this.dir = dir;
this.$node.css("direction", dir);
this.dropdown.setLanguageDirection(dir);
}
},
_updateHint: function updateHint() {
var datum, inputValue, query, escapedQuery, frontMatchRegEx, match;
datum = this.dropdown.getDatumForTopSuggestion();
if (datum && this.dropdown.isVisible() && !this.input.hasOverflow()) {
inputValue = this.input.getInputValue();
query = Input.normalizeQuery(inputValue);
escapedQuery = _.escapeRegExChars(query);
frontMatchRegEx = new RegExp("^(?:" + escapedQuery + ")(.*$)", "i");
match = frontMatchRegEx.exec(datum.value);
this.input.setHintValue(inputValue + (match ? match[1] : ""));
}
},
_autocomplete: function autocomplete() {
var hint, query, datum;
hint = this.input.getHintValue();
query = this.input.getQuery();
if (hint && query !== hint && this.input.isCursorAtEnd()) {
datum = this.dropdown.getDatumForTopSuggestion();
datum && this.input.setInputValue(datum.value);
this.eventBus.trigger("autocompleted", datum.raw, datum.datasetName);
}
},
_select: function select(datum) {
this.input.clearHint();
this.input.setQuery(datum.value);
this.input.setInputValue(datum.value, true);
this.dropdown.empty();
this._setLanguageDirection();
_.defer(_.bind(this.dropdown.close, this.dropdown));
this.eventBus.trigger("selected", datum.raw, datum.datasetName);
},
open: function open() {
this.dropdown.open();
},
close: function close() {
this.dropdown.close();
},
getQuery: function getQuery() {
return this.input.getQuery();
},
setQuery: function setQuery(val) {
this.input.setInputValue(val);
},
destroy: function destroy() {
this.input.destroy();
this.dropdown.destroy();
destroyDomStructure(this.$node); destroyDomStructure(this.$node);
this.$node = null; this.$node = null;
},
setQuery: function(query) {
this.inputView.setQuery(query);
this.inputView.setInputValue(query);
this._clearHint();
this._clearSuggestions();
this._getSuggestions();
} }
}); });
return TypeaheadView; return Typeahead;
function buildDomStructure(input) { function buildDomStructure(input, withHint) {
var $wrapper = $(html.wrapper), $dropdown = $(html.dropdown), $input = $(input), $hint = $(html.hint); var $input, $wrapper, $dropdown, $hint;
$wrapper = $wrapper.css(css.wrapper); $input = $(input);
$dropdown = $dropdown.css(css.dropdown); $wrapper = $(html.wrapper).css(css.wrapper);
$hint.css(css.hint).css({ $dropdown = $(html.dropdown).css(css.dropdown);
backgroundAttachment: $input.css("background-attachment"), $hint = $input.clone().css(css.hint).css(getBackgroundStyles($input));
backgroundClip: $input.css("background-clip"), $hint.removeData().addClass("tt-hint").removeAttr("id name placeholder").prop("disabled", true).attr({
backgroundColor: $input.css("background-color"), autocomplete: "off",
backgroundImage: $input.css("background-image"), spellcheck: "false"
backgroundOrigin: $input.css("background-origin"),
backgroundPosition: $input.css("background-position"),
backgroundRepeat: $input.css("background-repeat"),
backgroundSize: $input.css("background-size")
}); });
$input.data("ttAttrs", { $input.data(attrsKey, {
dir: $input.attr("dir"), dir: $input.attr("dir"),
autocomplete: $input.attr("autocomplete"), autocomplete: $input.attr("autocomplete"),
spellcheck: $input.attr("spellcheck"), spellcheck: $input.attr("spellcheck"),
style: $input.attr("style") style: $input.attr("style")
}); });
$input.addClass("tt-query").attr({ $input.addClass("tt-input").attr({
autocomplete: "off", autocomplete: "off",
spellcheck: false spellcheck: false
}).css(css.query); }).css(withHint ? css.input : css.inputWithNoHint);
try { try {
!$input.attr("dir") && $input.attr("dir", "auto"); !$input.attr("dir") && $input.attr("dir", "auto");
} catch (e) {} } catch (e) {}
return $input.wrap($wrapper).parent().prepend($hint).append($dropdown); return $input.wrap($wrapper).parent().prepend(withHint ? $hint : null).append($dropdown);
}
function getBackgroundStyles($el) {
return {
backgroundAttachment: $el.css("background-attachment"),
backgroundClip: $el.css("background-clip"),
backgroundColor: $el.css("background-color"),
backgroundImage: $el.css("background-image"),
backgroundOrigin: $el.css("background-origin"),
backgroundPosition: $el.css("background-position"),
backgroundRepeat: $el.css("background-repeat"),
backgroundSize: $el.css("background-size")
};
} }
function destroyDomStructure($node) { function destroyDomStructure($node) {
var $input = $node.find(".tt-query"); var $input = $node.find(".tt-input");
utils.each($input.data("ttAttrs"), function(key, val) { _.each($input.data(attrsKey), function(val, key) {
utils.isUndefined(val) ? $input.removeAttr(key) : $input.attr(key, val); _.isUndefined(val) ? $input.removeAttr(key) : $input.attr(key, val);
}); });
$input.detach().removeData("ttAttrs").removeClass("tt-query").insertAfter($node); $input.detach().removeData(attrsKey).removeClass("tt-input").insertAfter($node);
$node.remove(); $node.remove();
} }
}(); }();
(function() { (function() {
var cache = {}, viewKey = "ttView", methods; var typeaheadKey, methods;
typeaheadKey = "ttTypeahead";
methods = { methods = {
initialize: function(datasetDefs) { initialize: function initialize(o) {
var datasets; var datasets = [].slice.call(arguments, 1);
datasetDefs = utils.isArray(datasetDefs) ? datasetDefs : [ datasetDefs ]; o = o || {};
if (datasetDefs.length === 0) { return this.each(attach);
$.error("no datasets provided"); function attach() {
} var $input = $(this), eventBus, typeahead;
datasets = utils.map(datasetDefs, function(o) { _.each(datasets, function(d) {
var dataset = cache[o.name] ? cache[o.name] : new Dataset(o); d.highlight = !!o.highlight;
if (o.name) {
cache[o.name] = dataset;
}
return dataset;
});
return this.each(initialize);
function initialize() {
var $input = $(this), deferreds, eventBus = new EventBus({
el: $input
});
deferreds = utils.map(datasets, function(dataset) {
return dataset.initialize();
}); });
$input.data(viewKey, new TypeaheadView({ typeahead = new Typeahead({
input: $input, input: $input,
eventBus: eventBus = new EventBus({ eventBus: eventBus = new EventBus({
el: $input el: $input
}), }),
withHint: _.isUndefined(o.hint) ? true : !!o.hint,
minLength: o.minLength,
autoselect: o.autoselect,
datasets: datasets datasets: datasets
}));
$.when.apply($, deferreds).always(function() {
utils.defer(function() {
eventBus.trigger("initialized");
}); });
$input.data(typeaheadKey, typeahead);
function trigger(eventName) {
return function() {
_.defer(function() {
eventBus.trigger(eventName);
}); });
};
}
}
},
open: function open() {
return this.each(openTypeahead);
function openTypeahead() {
var $input = $(this), typeahead;
if (typeahead = $input.data(typeaheadKey)) {
typeahead.open();
}
} }
}, },
destroy: function() { close: function close() {
return this.each(destroy); return this.each(closeTypeahead);
function destroy() { function closeTypeahead() {
var $this = $(this), view = $this.data(viewKey); var $input = $(this), typeahead;
if (view) { if (typeahead = $input.data(typeaheadKey)) {
view.destroy(); typeahead.close();
$this.removeData(viewKey);
} }
} }
}, },
setQuery: function(query) { val: function val(newVal) {
return this.each(setQuery); return _.isString(newVal) ? this.each(setQuery) : this.map(getQuery).get();
function setQuery() { function setQuery() {
var view = $(this).data(viewKey); var $input = $(this), typeahead;
view && view.setQuery(query); if (typeahead = $input.data(typeaheadKey)) {
typeahead.setQuery(newVal);
}
}
function getQuery() {
var $input = $(this), typeahead, query;
if (typeahead = $input.data(typeaheadKey)) {
query = typeahead.getQuery();
}
return query;
}
},
destroy: function destroy() {
return this.each(unattach);
function unattach() {
var $input = $(this), typeahead;
if (typeahead = $input.data(typeaheadKey)) {
typeahead.destroy();
$input.removeData(typeaheadKey);
}
} }
} }
}; };
......
...@@ -63,7 +63,10 @@ class ActiveField extends \yii\widgets\ActiveField ...@@ -63,7 +63,10 @@ class ActiveField extends \yii\widgets\ActiveField
{ {
static $counter = 0; static $counter = 0;
$this->inputOptions['class'] .= ' typeahead-' . (++$counter); $this->inputOptions['class'] .= ' typeahead-' . (++$counter);
$this->form->getView()->registerJs("jQuery('.typeahead-{$counter}').typeahead({local: " . Json::encode($data) . "});"); foreach ($data as &$item) {
$item = array('word' => $item);
}
$this->form->getView()->registerJs("yii.gii.autocomplete($counter, " . Json::encode($data) . ");");
return $this; return $this;
} }
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment