Benjamin Renard commited on 2014-07-20 15:04:14
Showing 4 changed files, with 1877 additions and 0 deletions.
| ... | ... |
@@ -0,0 +1,1782 @@ |
| 1 |
+/*! |
|
| 2 |
+ * typeahead.js 0.10.4 |
|
| 3 |
+ * https://github.com/twitter/typeahead.js |
|
| 4 |
+ * Copyright 2013-2014 Twitter, Inc. and other contributors; Licensed MIT |
|
| 5 |
+ */ |
|
| 6 |
+ |
|
| 7 |
+(function($) {
|
|
| 8 |
+ var _ = function() {
|
|
| 9 |
+ "use strict"; |
|
| 10 |
+ return {
|
|
| 11 |
+ isMsie: function() {
|
|
| 12 |
+ return /(msie|trident)/i.test(navigator.userAgent) ? navigator.userAgent.match(/(msie |rv:)(\d+(.\d+)?)/i)[2] : false; |
|
| 13 |
+ }, |
|
| 14 |
+ isBlankString: function(str) {
|
|
| 15 |
+ return !str || /^\s*$/.test(str); |
|
| 16 |
+ }, |
|
| 17 |
+ escapeRegExChars: function(str) {
|
|
| 18 |
+ return str.replace(/[\-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g, "\\$&");
|
|
| 19 |
+ }, |
|
| 20 |
+ isString: function(obj) {
|
|
| 21 |
+ return typeof obj === "string"; |
|
| 22 |
+ }, |
|
| 23 |
+ isNumber: function(obj) {
|
|
| 24 |
+ return typeof obj === "number"; |
|
| 25 |
+ }, |
|
| 26 |
+ isArray: $.isArray, |
|
| 27 |
+ isFunction: $.isFunction, |
|
| 28 |
+ isObject: $.isPlainObject, |
|
| 29 |
+ isUndefined: function(obj) {
|
|
| 30 |
+ return typeof obj === "undefined"; |
|
| 31 |
+ }, |
|
| 32 |
+ toStr: function toStr(s) {
|
|
| 33 |
+ return _.isUndefined(s) || s === null ? "" : s + ""; |
|
| 34 |
+ }, |
|
| 35 |
+ bind: $.proxy, |
|
| 36 |
+ each: function(collection, cb) {
|
|
| 37 |
+ $.each(collection, reverseArgs); |
|
| 38 |
+ function reverseArgs(index, value) {
|
|
| 39 |
+ return cb(value, index); |
|
| 40 |
+ } |
|
| 41 |
+ }, |
|
| 42 |
+ map: $.map, |
|
| 43 |
+ filter: $.grep, |
|
| 44 |
+ every: function(obj, test) {
|
|
| 45 |
+ var result = true; |
|
| 46 |
+ if (!obj) {
|
|
| 47 |
+ return result; |
|
| 48 |
+ } |
|
| 49 |
+ $.each(obj, function(key, val) {
|
|
| 50 |
+ if (!(result = test.call(null, val, key, obj))) {
|
|
| 51 |
+ return false; |
|
| 52 |
+ } |
|
| 53 |
+ }); |
|
| 54 |
+ return !!result; |
|
| 55 |
+ }, |
|
| 56 |
+ some: function(obj, test) {
|
|
| 57 |
+ var result = false; |
|
| 58 |
+ if (!obj) {
|
|
| 59 |
+ return result; |
|
| 60 |
+ } |
|
| 61 |
+ $.each(obj, function(key, val) {
|
|
| 62 |
+ if (result = test.call(null, val, key, obj)) {
|
|
| 63 |
+ return false; |
|
| 64 |
+ } |
|
| 65 |
+ }); |
|
| 66 |
+ return !!result; |
|
| 67 |
+ }, |
|
| 68 |
+ mixin: $.extend, |
|
| 69 |
+ getUniqueId: function() {
|
|
| 70 |
+ var counter = 0; |
|
| 71 |
+ return function() {
|
|
| 72 |
+ return counter++; |
|
| 73 |
+ }; |
|
| 74 |
+ }(), |
|
| 75 |
+ templatify: function templatify(obj) {
|
|
| 76 |
+ return $.isFunction(obj) ? obj : template; |
|
| 77 |
+ function template() {
|
|
| 78 |
+ return String(obj); |
|
| 79 |
+ } |
|
| 80 |
+ }, |
|
| 81 |
+ defer: function(fn) {
|
|
| 82 |
+ setTimeout(fn, 0); |
|
| 83 |
+ }, |
|
| 84 |
+ debounce: function(func, wait, immediate) {
|
|
| 85 |
+ var timeout, result; |
|
| 86 |
+ return function() {
|
|
| 87 |
+ var context = this, args = arguments, later, callNow; |
|
| 88 |
+ later = function() {
|
|
| 89 |
+ timeout = null; |
|
| 90 |
+ if (!immediate) {
|
|
| 91 |
+ result = func.apply(context, args); |
|
| 92 |
+ } |
|
| 93 |
+ }; |
|
| 94 |
+ callNow = immediate && !timeout; |
|
| 95 |
+ clearTimeout(timeout); |
|
| 96 |
+ timeout = setTimeout(later, wait); |
|
| 97 |
+ if (callNow) {
|
|
| 98 |
+ result = func.apply(context, args); |
|
| 99 |
+ } |
|
| 100 |
+ return result; |
|
| 101 |
+ }; |
|
| 102 |
+ }, |
|
| 103 |
+ throttle: function(func, wait) {
|
|
| 104 |
+ var context, args, timeout, result, previous, later; |
|
| 105 |
+ previous = 0; |
|
| 106 |
+ later = function() {
|
|
| 107 |
+ previous = new Date(); |
|
| 108 |
+ timeout = null; |
|
| 109 |
+ result = func.apply(context, args); |
|
| 110 |
+ }; |
|
| 111 |
+ return function() {
|
|
| 112 |
+ var now = new Date(), remaining = wait - (now - previous); |
|
| 113 |
+ context = this; |
|
| 114 |
+ args = arguments; |
|
| 115 |
+ if (remaining <= 0) {
|
|
| 116 |
+ clearTimeout(timeout); |
|
| 117 |
+ timeout = null; |
|
| 118 |
+ previous = now; |
|
| 119 |
+ result = func.apply(context, args); |
|
| 120 |
+ } else if (!timeout) {
|
|
| 121 |
+ timeout = setTimeout(later, remaining); |
|
| 122 |
+ } |
|
| 123 |
+ return result; |
|
| 124 |
+ }; |
|
| 125 |
+ }, |
|
| 126 |
+ noop: function() {}
|
|
| 127 |
+ }; |
|
| 128 |
+ }(); |
|
| 129 |
+ var VERSION = "0.10.4"; |
|
| 130 |
+ var tokenizers = function() {
|
|
| 131 |
+ "use strict"; |
|
| 132 |
+ return {
|
|
| 133 |
+ nonword: nonword, |
|
| 134 |
+ whitespace: whitespace, |
|
| 135 |
+ obj: {
|
|
| 136 |
+ nonword: getObjTokenizer(nonword), |
|
| 137 |
+ whitespace: getObjTokenizer(whitespace) |
|
| 138 |
+ } |
|
| 139 |
+ }; |
|
| 140 |
+ function whitespace(str) {
|
|
| 141 |
+ str = _.toStr(str); |
|
| 142 |
+ return str ? str.split(/\s+/) : []; |
|
| 143 |
+ } |
|
| 144 |
+ function nonword(str) {
|
|
| 145 |
+ str = _.toStr(str); |
|
| 146 |
+ return str ? str.split(/\W+/) : []; |
|
| 147 |
+ } |
|
| 148 |
+ function getObjTokenizer(tokenizer) {
|
|
| 149 |
+ return function setKey() {
|
|
| 150 |
+ var args = [].slice.call(arguments, 0); |
|
| 151 |
+ return function tokenize(o) {
|
|
| 152 |
+ var tokens = []; |
|
| 153 |
+ _.each(args, function(k) {
|
|
| 154 |
+ tokens = tokens.concat(tokenizer(_.toStr(o[k]))); |
|
| 155 |
+ }); |
|
| 156 |
+ return tokens; |
|
| 157 |
+ }; |
|
| 158 |
+ }; |
|
| 159 |
+ } |
|
| 160 |
+ }(); |
|
| 161 |
+ var LruCache = function() {
|
|
| 162 |
+ "use strict"; |
|
| 163 |
+ function LruCache(maxSize) {
|
|
| 164 |
+ this.maxSize = _.isNumber(maxSize) ? maxSize : 100; |
|
| 165 |
+ this.reset(); |
|
| 166 |
+ if (this.maxSize <= 0) {
|
|
| 167 |
+ this.set = this.get = $.noop; |
|
| 168 |
+ } |
|
| 169 |
+ } |
|
| 170 |
+ _.mixin(LruCache.prototype, {
|
|
| 171 |
+ set: function set(key, val) {
|
|
| 172 |
+ var tailItem = this.list.tail, node; |
|
| 173 |
+ if (this.size >= this.maxSize) {
|
|
| 174 |
+ this.list.remove(tailItem); |
|
| 175 |
+ delete this.hash[tailItem.key]; |
|
| 176 |
+ } |
|
| 177 |
+ if (node = this.hash[key]) {
|
|
| 178 |
+ node.val = val; |
|
| 179 |
+ this.list.moveToFront(node); |
|
| 180 |
+ } else {
|
|
| 181 |
+ node = new Node(key, val); |
|
| 182 |
+ this.list.add(node); |
|
| 183 |
+ this.hash[key] = node; |
|
| 184 |
+ this.size++; |
|
| 185 |
+ } |
|
| 186 |
+ }, |
|
| 187 |
+ get: function get(key) {
|
|
| 188 |
+ var node = this.hash[key]; |
|
| 189 |
+ if (node) {
|
|
| 190 |
+ this.list.moveToFront(node); |
|
| 191 |
+ return node.val; |
|
| 192 |
+ } |
|
| 193 |
+ }, |
|
| 194 |
+ reset: function reset() {
|
|
| 195 |
+ this.size = 0; |
|
| 196 |
+ this.hash = {};
|
|
| 197 |
+ this.list = new List(); |
|
| 198 |
+ } |
|
| 199 |
+ }); |
|
| 200 |
+ function List() {
|
|
| 201 |
+ this.head = this.tail = null; |
|
| 202 |
+ } |
|
| 203 |
+ _.mixin(List.prototype, {
|
|
| 204 |
+ add: function add(node) {
|
|
| 205 |
+ if (this.head) {
|
|
| 206 |
+ node.next = this.head; |
|
| 207 |
+ this.head.prev = node; |
|
| 208 |
+ } |
|
| 209 |
+ this.head = node; |
|
| 210 |
+ this.tail = this.tail || node; |
|
| 211 |
+ }, |
|
| 212 |
+ remove: function remove(node) {
|
|
| 213 |
+ node.prev ? node.prev.next = node.next : this.head = node.next; |
|
| 214 |
+ node.next ? node.next.prev = node.prev : this.tail = node.prev; |
|
| 215 |
+ }, |
|
| 216 |
+ moveToFront: function(node) {
|
|
| 217 |
+ this.remove(node); |
|
| 218 |
+ this.add(node); |
|
| 219 |
+ } |
|
| 220 |
+ }); |
|
| 221 |
+ function Node(key, val) {
|
|
| 222 |
+ this.key = key; |
|
| 223 |
+ this.val = val; |
|
| 224 |
+ this.prev = this.next = null; |
|
| 225 |
+ } |
|
| 226 |
+ return LruCache; |
|
| 227 |
+ }(); |
|
| 228 |
+ var PersistentStorage = function() {
|
|
| 229 |
+ "use strict"; |
|
| 230 |
+ var ls, methods; |
|
| 231 |
+ try {
|
|
| 232 |
+ ls = window.localStorage; |
|
| 233 |
+ ls.setItem("~~~", "!");
|
|
| 234 |
+ ls.removeItem("~~~");
|
|
| 235 |
+ } catch (err) {
|
|
| 236 |
+ ls = null; |
|
| 237 |
+ } |
|
| 238 |
+ function PersistentStorage(namespace) {
|
|
| 239 |
+ this.prefix = [ "__", namespace, "__" ].join("");
|
|
| 240 |
+ this.ttlKey = "__ttl__"; |
|
| 241 |
+ this.keyMatcher = new RegExp("^" + _.escapeRegExChars(this.prefix));
|
|
| 242 |
+ } |
|
| 243 |
+ if (ls && window.JSON) {
|
|
| 244 |
+ methods = {
|
|
| 245 |
+ _prefix: function(key) {
|
|
| 246 |
+ return this.prefix + key; |
|
| 247 |
+ }, |
|
| 248 |
+ _ttlKey: function(key) {
|
|
| 249 |
+ return this._prefix(key) + this.ttlKey; |
|
| 250 |
+ }, |
|
| 251 |
+ get: function(key) {
|
|
| 252 |
+ if (this.isExpired(key)) {
|
|
| 253 |
+ this.remove(key); |
|
| 254 |
+ } |
|
| 255 |
+ return decode(ls.getItem(this._prefix(key))); |
|
| 256 |
+ }, |
|
| 257 |
+ set: function(key, val, ttl) {
|
|
| 258 |
+ if (_.isNumber(ttl)) {
|
|
| 259 |
+ ls.setItem(this._ttlKey(key), encode(now() + ttl)); |
|
| 260 |
+ } else {
|
|
| 261 |
+ ls.removeItem(this._ttlKey(key)); |
|
| 262 |
+ } |
|
| 263 |
+ return ls.setItem(this._prefix(key), encode(val)); |
|
| 264 |
+ }, |
|
| 265 |
+ remove: function(key) {
|
|
| 266 |
+ ls.removeItem(this._ttlKey(key)); |
|
| 267 |
+ ls.removeItem(this._prefix(key)); |
|
| 268 |
+ return this; |
|
| 269 |
+ }, |
|
| 270 |
+ clear: function() {
|
|
| 271 |
+ var i, key, keys = [], len = ls.length; |
|
| 272 |
+ for (i = 0; i < len; i++) {
|
|
| 273 |
+ if ((key = ls.key(i)).match(this.keyMatcher)) {
|
|
| 274 |
+ keys.push(key.replace(this.keyMatcher, "")); |
|
| 275 |
+ } |
|
| 276 |
+ } |
|
| 277 |
+ for (i = keys.length; i--; ) {
|
|
| 278 |
+ this.remove(keys[i]); |
|
| 279 |
+ } |
|
| 280 |
+ return this; |
|
| 281 |
+ }, |
|
| 282 |
+ isExpired: function(key) {
|
|
| 283 |
+ var ttl = decode(ls.getItem(this._ttlKey(key))); |
|
| 284 |
+ return _.isNumber(ttl) && now() > ttl ? true : false; |
|
| 285 |
+ } |
|
| 286 |
+ }; |
|
| 287 |
+ } else {
|
|
| 288 |
+ methods = {
|
|
| 289 |
+ get: _.noop, |
|
| 290 |
+ set: _.noop, |
|
| 291 |
+ remove: _.noop, |
|
| 292 |
+ clear: _.noop, |
|
| 293 |
+ isExpired: _.noop |
|
| 294 |
+ }; |
|
| 295 |
+ } |
|
| 296 |
+ _.mixin(PersistentStorage.prototype, methods); |
|
| 297 |
+ return PersistentStorage; |
|
| 298 |
+ function now() {
|
|
| 299 |
+ return new Date().getTime(); |
|
| 300 |
+ } |
|
| 301 |
+ function encode(val) {
|
|
| 302 |
+ return JSON.stringify(_.isUndefined(val) ? null : val); |
|
| 303 |
+ } |
|
| 304 |
+ function decode(val) {
|
|
| 305 |
+ return JSON.parse(val); |
|
| 306 |
+ } |
|
| 307 |
+ }(); |
|
| 308 |
+ var Transport = function() {
|
|
| 309 |
+ "use strict"; |
|
| 310 |
+ var pendingRequestsCount = 0, pendingRequests = {}, maxPendingRequests = 6, sharedCache = new LruCache(10);
|
|
| 311 |
+ function Transport(o) {
|
|
| 312 |
+ o = o || {};
|
|
| 313 |
+ this.cancelled = false; |
|
| 314 |
+ this.lastUrl = null; |
|
| 315 |
+ this._send = o.transport ? callbackToDeferred(o.transport) : $.ajax; |
|
| 316 |
+ this._get = o.rateLimiter ? o.rateLimiter(this._get) : this._get; |
|
| 317 |
+ this._cache = o.cache === false ? new LruCache(0) : sharedCache; |
|
| 318 |
+ } |
|
| 319 |
+ Transport.setMaxPendingRequests = function setMaxPendingRequests(num) {
|
|
| 320 |
+ maxPendingRequests = num; |
|
| 321 |
+ }; |
|
| 322 |
+ Transport.resetCache = function resetCache() {
|
|
| 323 |
+ sharedCache.reset(); |
|
| 324 |
+ }; |
|
| 325 |
+ _.mixin(Transport.prototype, {
|
|
| 326 |
+ _get: function(url, o, cb) {
|
|
| 327 |
+ var that = this, jqXhr; |
|
| 328 |
+ if (this.cancelled || url !== this.lastUrl) {
|
|
| 329 |
+ return; |
|
| 330 |
+ } |
|
| 331 |
+ if (jqXhr = pendingRequests[url]) {
|
|
| 332 |
+ jqXhr.done(done).fail(fail); |
|
| 333 |
+ } else if (pendingRequestsCount < maxPendingRequests) {
|
|
| 334 |
+ pendingRequestsCount++; |
|
| 335 |
+ pendingRequests[url] = this._send(url, o).done(done).fail(fail).always(always); |
|
| 336 |
+ } else {
|
|
| 337 |
+ this.onDeckRequestArgs = [].slice.call(arguments, 0); |
|
| 338 |
+ } |
|
| 339 |
+ function done(resp) {
|
|
| 340 |
+ cb && cb(null, resp); |
|
| 341 |
+ that._cache.set(url, resp); |
|
| 342 |
+ } |
|
| 343 |
+ function fail() {
|
|
| 344 |
+ cb && cb(true); |
|
| 345 |
+ } |
|
| 346 |
+ function always() {
|
|
| 347 |
+ pendingRequestsCount--; |
|
| 348 |
+ delete pendingRequests[url]; |
|
| 349 |
+ if (that.onDeckRequestArgs) {
|
|
| 350 |
+ that._get.apply(that, that.onDeckRequestArgs); |
|
| 351 |
+ that.onDeckRequestArgs = null; |
|
| 352 |
+ } |
|
| 353 |
+ } |
|
| 354 |
+ }, |
|
| 355 |
+ get: function(url, o, cb) {
|
|
| 356 |
+ var resp; |
|
| 357 |
+ if (_.isFunction(o)) {
|
|
| 358 |
+ cb = o; |
|
| 359 |
+ o = {};
|
|
| 360 |
+ } |
|
| 361 |
+ this.cancelled = false; |
|
| 362 |
+ this.lastUrl = url; |
|
| 363 |
+ if (resp = this._cache.get(url)) {
|
|
| 364 |
+ _.defer(function() {
|
|
| 365 |
+ cb && cb(null, resp); |
|
| 366 |
+ }); |
|
| 367 |
+ } else {
|
|
| 368 |
+ this._get(url, o, cb); |
|
| 369 |
+ } |
|
| 370 |
+ return !!resp; |
|
| 371 |
+ }, |
|
| 372 |
+ cancel: function() {
|
|
| 373 |
+ this.cancelled = true; |
|
| 374 |
+ } |
|
| 375 |
+ }); |
|
| 376 |
+ return Transport; |
|
| 377 |
+ function callbackToDeferred(fn) {
|
|
| 378 |
+ return function customSendWrapper(url, o) {
|
|
| 379 |
+ var deferred = $.Deferred(); |
|
| 380 |
+ fn(url, o, onSuccess, onError); |
|
| 381 |
+ return deferred; |
|
| 382 |
+ function onSuccess(resp) {
|
|
| 383 |
+ _.defer(function() {
|
|
| 384 |
+ deferred.resolve(resp); |
|
| 385 |
+ }); |
|
| 386 |
+ } |
|
| 387 |
+ function onError(err) {
|
|
| 388 |
+ _.defer(function() {
|
|
| 389 |
+ deferred.reject(err); |
|
| 390 |
+ }); |
|
| 391 |
+ } |
|
| 392 |
+ }; |
|
| 393 |
+ } |
|
| 394 |
+ }(); |
|
| 395 |
+ var SearchIndex = function() {
|
|
| 396 |
+ "use strict"; |
|
| 397 |
+ function SearchIndex(o) {
|
|
| 398 |
+ o = o || {};
|
|
| 399 |
+ if (!o.datumTokenizer || !o.queryTokenizer) {
|
|
| 400 |
+ $.error("datumTokenizer and queryTokenizer are both required");
|
|
| 401 |
+ } |
|
| 402 |
+ this.datumTokenizer = o.datumTokenizer; |
|
| 403 |
+ this.queryTokenizer = o.queryTokenizer; |
|
| 404 |
+ this.reset(); |
|
| 405 |
+ } |
|
| 406 |
+ _.mixin(SearchIndex.prototype, {
|
|
| 407 |
+ bootstrap: function bootstrap(o) {
|
|
| 408 |
+ this.datums = o.datums; |
|
| 409 |
+ this.trie = o.trie; |
|
| 410 |
+ }, |
|
| 411 |
+ add: function(data) {
|
|
| 412 |
+ var that = this; |
|
| 413 |
+ data = _.isArray(data) ? data : [ data ]; |
|
| 414 |
+ _.each(data, function(datum) {
|
|
| 415 |
+ var id, tokens; |
|
| 416 |
+ id = that.datums.push(datum) - 1; |
|
| 417 |
+ tokens = normalizeTokens(that.datumTokenizer(datum)); |
|
| 418 |
+ _.each(tokens, function(token) {
|
|
| 419 |
+ var node, chars, ch; |
|
| 420 |
+ node = that.trie; |
|
| 421 |
+ chars = token.split("");
|
|
| 422 |
+ while (ch = chars.shift()) {
|
|
| 423 |
+ node = node.children[ch] || (node.children[ch] = newNode()); |
|
| 424 |
+ node.ids.push(id); |
|
| 425 |
+ } |
|
| 426 |
+ }); |
|
| 427 |
+ }); |
|
| 428 |
+ }, |
|
| 429 |
+ get: function get(query) {
|
|
| 430 |
+ var that = this, tokens, matches; |
|
| 431 |
+ tokens = normalizeTokens(this.queryTokenizer(query)); |
|
| 432 |
+ _.each(tokens, function(token) {
|
|
| 433 |
+ var node, chars, ch, ids; |
|
| 434 |
+ if (matches && matches.length === 0) {
|
|
| 435 |
+ return false; |
|
| 436 |
+ } |
|
| 437 |
+ node = that.trie; |
|
| 438 |
+ chars = token.split("");
|
|
| 439 |
+ while (node && (ch = chars.shift())) {
|
|
| 440 |
+ node = node.children[ch]; |
|
| 441 |
+ } |
|
| 442 |
+ if (node && chars.length === 0) {
|
|
| 443 |
+ ids = node.ids.slice(0); |
|
| 444 |
+ matches = matches ? getIntersection(matches, ids) : ids; |
|
| 445 |
+ } else {
|
|
| 446 |
+ matches = []; |
|
| 447 |
+ return false; |
|
| 448 |
+ } |
|
| 449 |
+ }); |
|
| 450 |
+ return matches ? _.map(unique(matches), function(id) {
|
|
| 451 |
+ return that.datums[id]; |
|
| 452 |
+ }) : []; |
|
| 453 |
+ }, |
|
| 454 |
+ reset: function reset() {
|
|
| 455 |
+ this.datums = []; |
|
| 456 |
+ this.trie = newNode(); |
|
| 457 |
+ }, |
|
| 458 |
+ serialize: function serialize() {
|
|
| 459 |
+ return {
|
|
| 460 |
+ datums: this.datums, |
|
| 461 |
+ trie: this.trie |
|
| 462 |
+ }; |
|
| 463 |
+ } |
|
| 464 |
+ }); |
|
| 465 |
+ return SearchIndex; |
|
| 466 |
+ function normalizeTokens(tokens) {
|
|
| 467 |
+ tokens = _.filter(tokens, function(token) {
|
|
| 468 |
+ return !!token; |
|
| 469 |
+ }); |
|
| 470 |
+ tokens = _.map(tokens, function(token) {
|
|
| 471 |
+ return token.toLowerCase(); |
|
| 472 |
+ }); |
|
| 473 |
+ return tokens; |
|
| 474 |
+ } |
|
| 475 |
+ function newNode() {
|
|
| 476 |
+ return {
|
|
| 477 |
+ ids: [], |
|
| 478 |
+ children: {}
|
|
| 479 |
+ }; |
|
| 480 |
+ } |
|
| 481 |
+ function unique(array) {
|
|
| 482 |
+ var seen = {}, uniques = [];
|
|
| 483 |
+ for (var i = 0, len = array.length; i < len; i++) {
|
|
| 484 |
+ if (!seen[array[i]]) {
|
|
| 485 |
+ seen[array[i]] = true; |
|
| 486 |
+ uniques.push(array[i]); |
|
| 487 |
+ } |
|
| 488 |
+ } |
|
| 489 |
+ return uniques; |
|
| 490 |
+ } |
|
| 491 |
+ function getIntersection(arrayA, arrayB) {
|
|
| 492 |
+ var ai = 0, bi = 0, intersection = []; |
|
| 493 |
+ arrayA = arrayA.sort(compare); |
|
| 494 |
+ arrayB = arrayB.sort(compare); |
|
| 495 |
+ var lenArrayA = arrayA.length, lenArrayB = arrayB.length; |
|
| 496 |
+ while (ai < lenArrayA && bi < lenArrayB) {
|
|
| 497 |
+ if (arrayA[ai] < arrayB[bi]) {
|
|
| 498 |
+ ai++; |
|
| 499 |
+ } else if (arrayA[ai] > arrayB[bi]) {
|
|
| 500 |
+ bi++; |
|
| 501 |
+ } else {
|
|
| 502 |
+ intersection.push(arrayA[ai]); |
|
| 503 |
+ ai++; |
|
| 504 |
+ bi++; |
|
| 505 |
+ } |
|
| 506 |
+ } |
|
| 507 |
+ return intersection; |
|
| 508 |
+ function compare(a, b) {
|
|
| 509 |
+ return a - b; |
|
| 510 |
+ } |
|
| 511 |
+ } |
|
| 512 |
+ }(); |
|
| 513 |
+ var oParser = function() {
|
|
| 514 |
+ "use strict"; |
|
| 515 |
+ return {
|
|
| 516 |
+ local: getLocal, |
|
| 517 |
+ prefetch: getPrefetch, |
|
| 518 |
+ remote: getRemote |
|
| 519 |
+ }; |
|
| 520 |
+ function getLocal(o) {
|
|
| 521 |
+ return o.local || null; |
|
| 522 |
+ } |
|
| 523 |
+ function getPrefetch(o) {
|
|
| 524 |
+ var prefetch, defaults; |
|
| 525 |
+ defaults = {
|
|
| 526 |
+ url: null, |
|
| 527 |
+ thumbprint: "", |
|
| 528 |
+ ttl: 24 * 60 * 60 * 1e3, |
|
| 529 |
+ filter: null, |
|
| 530 |
+ ajax: {}
|
|
| 531 |
+ }; |
|
| 532 |
+ if (prefetch = o.prefetch || null) {
|
|
| 533 |
+ prefetch = _.isString(prefetch) ? {
|
|
| 534 |
+ url: prefetch |
|
| 535 |
+ } : prefetch; |
|
| 536 |
+ prefetch = _.mixin(defaults, prefetch); |
|
| 537 |
+ prefetch.thumbprint = VERSION + prefetch.thumbprint; |
|
| 538 |
+ prefetch.ajax.type = prefetch.ajax.type || "GET"; |
|
| 539 |
+ prefetch.ajax.dataType = prefetch.ajax.dataType || "json"; |
|
| 540 |
+ !prefetch.url && $.error("prefetch requires url to be set");
|
|
| 541 |
+ } |
|
| 542 |
+ return prefetch; |
|
| 543 |
+ } |
|
| 544 |
+ function getRemote(o) {
|
|
| 545 |
+ var remote, defaults; |
|
| 546 |
+ defaults = {
|
|
| 547 |
+ url: null, |
|
| 548 |
+ cache: true, |
|
| 549 |
+ wildcard: "%QUERY", |
|
| 550 |
+ replace: null, |
|
| 551 |
+ rateLimitBy: "debounce", |
|
| 552 |
+ rateLimitWait: 300, |
|
| 553 |
+ send: null, |
|
| 554 |
+ filter: null, |
|
| 555 |
+ ajax: {}
|
|
| 556 |
+ }; |
|
| 557 |
+ if (remote = o.remote || null) {
|
|
| 558 |
+ remote = _.isString(remote) ? {
|
|
| 559 |
+ url: remote |
|
| 560 |
+ } : remote; |
|
| 561 |
+ remote = _.mixin(defaults, remote); |
|
| 562 |
+ remote.rateLimiter = /^throttle$/i.test(remote.rateLimitBy) ? byThrottle(remote.rateLimitWait) : byDebounce(remote.rateLimitWait); |
|
| 563 |
+ remote.ajax.type = remote.ajax.type || "GET"; |
|
| 564 |
+ remote.ajax.dataType = remote.ajax.dataType || "json"; |
|
| 565 |
+ delete remote.rateLimitBy; |
|
| 566 |
+ delete remote.rateLimitWait; |
|
| 567 |
+ !remote.url && $.error("remote requires url to be set");
|
|
| 568 |
+ } |
|
| 569 |
+ return remote; |
|
| 570 |
+ function byDebounce(wait) {
|
|
| 571 |
+ return function(fn) {
|
|
| 572 |
+ return _.debounce(fn, wait); |
|
| 573 |
+ }; |
|
| 574 |
+ } |
|
| 575 |
+ function byThrottle(wait) {
|
|
| 576 |
+ return function(fn) {
|
|
| 577 |
+ return _.throttle(fn, wait); |
|
| 578 |
+ }; |
|
| 579 |
+ } |
|
| 580 |
+ } |
|
| 581 |
+ }(); |
|
| 582 |
+ (function(root) {
|
|
| 583 |
+ "use strict"; |
|
| 584 |
+ var old, keys; |
|
| 585 |
+ old = root.Bloodhound; |
|
| 586 |
+ keys = {
|
|
| 587 |
+ data: "data", |
|
| 588 |
+ protocol: "protocol", |
|
| 589 |
+ thumbprint: "thumbprint" |
|
| 590 |
+ }; |
|
| 591 |
+ root.Bloodhound = Bloodhound; |
|
| 592 |
+ function Bloodhound(o) {
|
|
| 593 |
+ if (!o || !o.local && !o.prefetch && !o.remote) {
|
|
| 594 |
+ $.error("one of local, prefetch, or remote is required");
|
|
| 595 |
+ } |
|
| 596 |
+ this.limit = o.limit || 5; |
|
| 597 |
+ this.sorter = getSorter(o.sorter); |
|
| 598 |
+ this.dupDetector = o.dupDetector || ignoreDuplicates; |
|
| 599 |
+ this.local = oParser.local(o); |
|
| 600 |
+ this.prefetch = oParser.prefetch(o); |
|
| 601 |
+ this.remote = oParser.remote(o); |
|
| 602 |
+ this.cacheKey = this.prefetch ? this.prefetch.cacheKey || this.prefetch.url : null; |
|
| 603 |
+ this.index = new SearchIndex({
|
|
| 604 |
+ datumTokenizer: o.datumTokenizer, |
|
| 605 |
+ queryTokenizer: o.queryTokenizer |
|
| 606 |
+ }); |
|
| 607 |
+ this.storage = this.cacheKey ? new PersistentStorage(this.cacheKey) : null; |
|
| 608 |
+ } |
|
| 609 |
+ Bloodhound.noConflict = function noConflict() {
|
|
| 610 |
+ root.Bloodhound = old; |
|
| 611 |
+ return Bloodhound; |
|
| 612 |
+ }; |
|
| 613 |
+ Bloodhound.tokenizers = tokenizers; |
|
| 614 |
+ _.mixin(Bloodhound.prototype, {
|
|
| 615 |
+ _loadPrefetch: function loadPrefetch(o) {
|
|
| 616 |
+ var that = this, serialized, deferred; |
|
| 617 |
+ if (serialized = this._readFromStorage(o.thumbprint)) {
|
|
| 618 |
+ this.index.bootstrap(serialized); |
|
| 619 |
+ deferred = $.Deferred().resolve(); |
|
| 620 |
+ } else {
|
|
| 621 |
+ deferred = $.ajax(o.url, o.ajax).done(handlePrefetchResponse); |
|
| 622 |
+ } |
|
| 623 |
+ return deferred; |
|
| 624 |
+ function handlePrefetchResponse(resp) {
|
|
| 625 |
+ that.clear(); |
|
| 626 |
+ that.add(o.filter ? o.filter(resp) : resp); |
|
| 627 |
+ that._saveToStorage(that.index.serialize(), o.thumbprint, o.ttl); |
|
| 628 |
+ } |
|
| 629 |
+ }, |
|
| 630 |
+ _getFromRemote: function getFromRemote(query, cb) {
|
|
| 631 |
+ var that = this, url, uriEncodedQuery; |
|
| 632 |
+ if (!this.transport) {
|
|
| 633 |
+ return; |
|
| 634 |
+ } |
|
| 635 |
+ query = query || ""; |
|
| 636 |
+ uriEncodedQuery = encodeURIComponent(query); |
|
| 637 |
+ url = this.remote.replace ? this.remote.replace(this.remote.url, query) : this.remote.url.replace(this.remote.wildcard, uriEncodedQuery); |
|
| 638 |
+ return this.transport.get(url, this.remote.ajax, handleRemoteResponse); |
|
| 639 |
+ function handleRemoteResponse(err, resp) {
|
|
| 640 |
+ err ? cb([]) : cb(that.remote.filter ? that.remote.filter(resp) : resp); |
|
| 641 |
+ } |
|
| 642 |
+ }, |
|
| 643 |
+ _cancelLastRemoteRequest: function cancelLastRemoteRequest() {
|
|
| 644 |
+ this.transport && this.transport.cancel(); |
|
| 645 |
+ }, |
|
| 646 |
+ _saveToStorage: function saveToStorage(data, thumbprint, ttl) {
|
|
| 647 |
+ if (this.storage) {
|
|
| 648 |
+ this.storage.set(keys.data, data, ttl); |
|
| 649 |
+ this.storage.set(keys.protocol, location.protocol, ttl); |
|
| 650 |
+ this.storage.set(keys.thumbprint, thumbprint, ttl); |
|
| 651 |
+ } |
|
| 652 |
+ }, |
|
| 653 |
+ _readFromStorage: function readFromStorage(thumbprint) {
|
|
| 654 |
+ var stored = {}, isExpired;
|
|
| 655 |
+ if (this.storage) {
|
|
| 656 |
+ stored.data = this.storage.get(keys.data); |
|
| 657 |
+ stored.protocol = this.storage.get(keys.protocol); |
|
| 658 |
+ stored.thumbprint = this.storage.get(keys.thumbprint); |
|
| 659 |
+ } |
|
| 660 |
+ isExpired = stored.thumbprint !== thumbprint || stored.protocol !== location.protocol; |
|
| 661 |
+ return stored.data && !isExpired ? stored.data : null; |
|
| 662 |
+ }, |
|
| 663 |
+ _initialize: function initialize() {
|
|
| 664 |
+ var that = this, local = this.local, deferred; |
|
| 665 |
+ deferred = this.prefetch ? this._loadPrefetch(this.prefetch) : $.Deferred().resolve(); |
|
| 666 |
+ local && deferred.done(addLocalToIndex); |
|
| 667 |
+ this.transport = this.remote ? new Transport(this.remote) : null; |
|
| 668 |
+ return this.initPromise = deferred.promise(); |
|
| 669 |
+ function addLocalToIndex() {
|
|
| 670 |
+ that.add(_.isFunction(local) ? local() : local); |
|
| 671 |
+ } |
|
| 672 |
+ }, |
|
| 673 |
+ initialize: function initialize(force) {
|
|
| 674 |
+ return !this.initPromise || force ? this._initialize() : this.initPromise; |
|
| 675 |
+ }, |
|
| 676 |
+ add: function add(data) {
|
|
| 677 |
+ this.index.add(data); |
|
| 678 |
+ }, |
|
| 679 |
+ get: function get(query, cb) {
|
|
| 680 |
+ var that = this, matches = [], cacheHit = false; |
|
| 681 |
+ matches = this.index.get(query); |
|
| 682 |
+ matches = this.sorter(matches).slice(0, this.limit); |
|
| 683 |
+ matches.length < this.limit ? cacheHit = this._getFromRemote(query, returnRemoteMatches) : this._cancelLastRemoteRequest(); |
|
| 684 |
+ if (!cacheHit) {
|
|
| 685 |
+ (matches.length > 0 || !this.transport) && cb && cb(matches); |
|
| 686 |
+ } |
|
| 687 |
+ function returnRemoteMatches(remoteMatches) {
|
|
| 688 |
+ var matchesWithBackfill = matches.slice(0); |
|
| 689 |
+ _.each(remoteMatches, function(remoteMatch) {
|
|
| 690 |
+ var isDuplicate; |
|
| 691 |
+ isDuplicate = _.some(matchesWithBackfill, function(match) {
|
|
| 692 |
+ return that.dupDetector(remoteMatch, match); |
|
| 693 |
+ }); |
|
| 694 |
+ !isDuplicate && matchesWithBackfill.push(remoteMatch); |
|
| 695 |
+ return matchesWithBackfill.length < that.limit; |
|
| 696 |
+ }); |
|
| 697 |
+ cb && cb(that.sorter(matchesWithBackfill)); |
|
| 698 |
+ } |
|
| 699 |
+ }, |
|
| 700 |
+ clear: function clear() {
|
|
| 701 |
+ this.index.reset(); |
|
| 702 |
+ }, |
|
| 703 |
+ clearPrefetchCache: function clearPrefetchCache() {
|
|
| 704 |
+ this.storage && this.storage.clear(); |
|
| 705 |
+ }, |
|
| 706 |
+ clearRemoteCache: function clearRemoteCache() {
|
|
| 707 |
+ this.transport && Transport.resetCache(); |
|
| 708 |
+ }, |
|
| 709 |
+ ttAdapter: function ttAdapter() {
|
|
| 710 |
+ return _.bind(this.get, this); |
|
| 711 |
+ } |
|
| 712 |
+ }); |
|
| 713 |
+ return Bloodhound; |
|
| 714 |
+ function getSorter(sortFn) {
|
|
| 715 |
+ return _.isFunction(sortFn) ? sort : noSort; |
|
| 716 |
+ function sort(array) {
|
|
| 717 |
+ return array.sort(sortFn); |
|
| 718 |
+ } |
|
| 719 |
+ function noSort(array) {
|
|
| 720 |
+ return array; |
|
| 721 |
+ } |
|
| 722 |
+ } |
|
| 723 |
+ function ignoreDuplicates() {
|
|
| 724 |
+ return false; |
|
| 725 |
+ } |
|
| 726 |
+ })(this); |
|
| 727 |
+ var html = function() {
|
|
| 728 |
+ return {
|
|
| 729 |
+ wrapper: '<span class="twitter-typeahead"></span>', |
|
| 730 |
+ dropdown: '<span class="tt-dropdown-menu"></span>', |
|
| 731 |
+ dataset: '<div class="tt-dataset-%CLASS%"></div>', |
|
| 732 |
+ suggestions: '<span class="tt-suggestions"></span>', |
|
| 733 |
+ suggestion: '<div class="tt-suggestion"></div>' |
|
| 734 |
+ }; |
|
| 735 |
+ }(); |
|
| 736 |
+ var css = function() {
|
|
| 737 |
+ "use strict"; |
|
| 738 |
+ var css = {
|
|
| 739 |
+ wrapper: {
|
|
| 740 |
+ position: "relative", |
|
| 741 |
+ display: "inline-block" |
|
| 742 |
+ }, |
|
| 743 |
+ hint: {
|
|
| 744 |
+ position: "absolute", |
|
| 745 |
+ top: "0", |
|
| 746 |
+ left: "0", |
|
| 747 |
+ borderColor: "transparent", |
|
| 748 |
+ boxShadow: "none", |
|
| 749 |
+ opacity: "1" |
|
| 750 |
+ }, |
|
| 751 |
+ input: {
|
|
| 752 |
+ position: "relative", |
|
| 753 |
+ verticalAlign: "top", |
|
| 754 |
+ backgroundColor: "transparent" |
|
| 755 |
+ }, |
|
| 756 |
+ inputWithNoHint: {
|
|
| 757 |
+ position: "relative", |
|
| 758 |
+ verticalAlign: "top" |
|
| 759 |
+ }, |
|
| 760 |
+ dropdown: {
|
|
| 761 |
+ position: "absolute", |
|
| 762 |
+ top: "100%", |
|
| 763 |
+ left: "0", |
|
| 764 |
+ zIndex: "100", |
|
| 765 |
+ display: "none" |
|
| 766 |
+ }, |
|
| 767 |
+ suggestions: {
|
|
| 768 |
+ display: "block" |
|
| 769 |
+ }, |
|
| 770 |
+ suggestion: {
|
|
| 771 |
+ whiteSpace: "nowrap", |
|
| 772 |
+ cursor: "pointer" |
|
| 773 |
+ }, |
|
| 774 |
+ suggestionChild: {
|
|
| 775 |
+ whiteSpace: "normal" |
|
| 776 |
+ }, |
|
| 777 |
+ ltr: {
|
|
| 778 |
+ left: "0", |
|
| 779 |
+ right: "auto" |
|
| 780 |
+ }, |
|
| 781 |
+ rtl: {
|
|
| 782 |
+ left: "auto", |
|
| 783 |
+ right: " 0" |
|
| 784 |
+ } |
|
| 785 |
+ }; |
|
| 786 |
+ if (_.isMsie()) {
|
|
| 787 |
+ _.mixin(css.input, {
|
|
| 788 |
+ backgroundImage: "url()" |
|
| 789 |
+ }); |
|
| 790 |
+ } |
|
| 791 |
+ if (_.isMsie() && _.isMsie() <= 7) {
|
|
| 792 |
+ _.mixin(css.input, {
|
|
| 793 |
+ marginTop: "-1px" |
|
| 794 |
+ }); |
|
| 795 |
+ } |
|
| 796 |
+ return css; |
|
| 797 |
+ }(); |
|
| 798 |
+ var EventBus = function() {
|
|
| 799 |
+ "use strict"; |
|
| 800 |
+ var namespace = "typeahead:"; |
|
| 801 |
+ function EventBus(o) {
|
|
| 802 |
+ if (!o || !o.el) {
|
|
| 803 |
+ $.error("EventBus initialized without el");
|
|
| 804 |
+ } |
|
| 805 |
+ this.$el = $(o.el); |
|
| 806 |
+ } |
|
| 807 |
+ _.mixin(EventBus.prototype, {
|
|
| 808 |
+ trigger: function(type) {
|
|
| 809 |
+ var args = [].slice.call(arguments, 1); |
|
| 810 |
+ this.$el.trigger(namespace + type, args); |
|
| 811 |
+ } |
|
| 812 |
+ }); |
|
| 813 |
+ return EventBus; |
|
| 814 |
+ }(); |
|
| 815 |
+ var EventEmitter = function() {
|
|
| 816 |
+ "use strict"; |
|
| 817 |
+ var splitter = /\s+/, nextTick = getNextTick(); |
|
| 818 |
+ return {
|
|
| 819 |
+ onSync: onSync, |
|
| 820 |
+ onAsync: onAsync, |
|
| 821 |
+ off: off, |
|
| 822 |
+ trigger: trigger |
|
| 823 |
+ }; |
|
| 824 |
+ function on(method, types, cb, context) {
|
|
| 825 |
+ var type; |
|
| 826 |
+ if (!cb) {
|
|
| 827 |
+ return this; |
|
| 828 |
+ } |
|
| 829 |
+ types = types.split(splitter); |
|
| 830 |
+ cb = context ? bindContext(cb, context) : cb; |
|
| 831 |
+ this._callbacks = this._callbacks || {};
|
|
| 832 |
+ while (type = types.shift()) {
|
|
| 833 |
+ this._callbacks[type] = this._callbacks[type] || {
|
|
| 834 |
+ sync: [], |
|
| 835 |
+ async: [] |
|
| 836 |
+ }; |
|
| 837 |
+ this._callbacks[type][method].push(cb); |
|
| 838 |
+ } |
|
| 839 |
+ return this; |
|
| 840 |
+ } |
|
| 841 |
+ function onAsync(types, cb, context) {
|
|
| 842 |
+ return on.call(this, "async", types, cb, context); |
|
| 843 |
+ } |
|
| 844 |
+ function onSync(types, cb, context) {
|
|
| 845 |
+ return on.call(this, "sync", types, cb, context); |
|
| 846 |
+ } |
|
| 847 |
+ function off(types) {
|
|
| 848 |
+ var type; |
|
| 849 |
+ if (!this._callbacks) {
|
|
| 850 |
+ return this; |
|
| 851 |
+ } |
|
| 852 |
+ types = types.split(splitter); |
|
| 853 |
+ while (type = types.shift()) {
|
|
| 854 |
+ delete this._callbacks[type]; |
|
| 855 |
+ } |
|
| 856 |
+ return this; |
|
| 857 |
+ } |
|
| 858 |
+ function trigger(types) {
|
|
| 859 |
+ var type, callbacks, args, syncFlush, asyncFlush; |
|
| 860 |
+ if (!this._callbacks) {
|
|
| 861 |
+ return this; |
|
| 862 |
+ } |
|
| 863 |
+ types = types.split(splitter); |
|
| 864 |
+ args = [].slice.call(arguments, 1); |
|
| 865 |
+ while ((type = types.shift()) && (callbacks = this._callbacks[type])) {
|
|
| 866 |
+ syncFlush = getFlush(callbacks.sync, this, [ type ].concat(args)); |
|
| 867 |
+ asyncFlush = getFlush(callbacks.async, this, [ type ].concat(args)); |
|
| 868 |
+ syncFlush() && nextTick(asyncFlush); |
|
| 869 |
+ } |
|
| 870 |
+ return this; |
|
| 871 |
+ } |
|
| 872 |
+ function getFlush(callbacks, context, args) {
|
|
| 873 |
+ return flush; |
|
| 874 |
+ function flush() {
|
|
| 875 |
+ var cancelled; |
|
| 876 |
+ for (var i = 0, len = callbacks.length; !cancelled && i < len; i += 1) {
|
|
| 877 |
+ cancelled = callbacks[i].apply(context, args) === false; |
|
| 878 |
+ } |
|
| 879 |
+ return !cancelled; |
|
| 880 |
+ } |
|
| 881 |
+ } |
|
| 882 |
+ function getNextTick() {
|
|
| 883 |
+ var nextTickFn; |
|
| 884 |
+ if (window.setImmediate) {
|
|
| 885 |
+ nextTickFn = function nextTickSetImmediate(fn) {
|
|
| 886 |
+ setImmediate(function() {
|
|
| 887 |
+ fn(); |
|
| 888 |
+ }); |
|
| 889 |
+ }; |
|
| 890 |
+ } else {
|
|
| 891 |
+ nextTickFn = function nextTickSetTimeout(fn) {
|
|
| 892 |
+ setTimeout(function() {
|
|
| 893 |
+ fn(); |
|
| 894 |
+ }, 0); |
|
| 895 |
+ }; |
|
| 896 |
+ } |
|
| 897 |
+ return nextTickFn; |
|
| 898 |
+ } |
|
| 899 |
+ function bindContext(fn, context) {
|
|
| 900 |
+ return fn.bind ? fn.bind(context) : function() {
|
|
| 901 |
+ fn.apply(context, [].slice.call(arguments, 0)); |
|
| 902 |
+ }; |
|
| 903 |
+ } |
|
| 904 |
+ }(); |
|
| 905 |
+ var highlight = function(doc) {
|
|
| 906 |
+ "use strict"; |
|
| 907 |
+ var defaults = {
|
|
| 908 |
+ node: null, |
|
| 909 |
+ pattern: null, |
|
| 910 |
+ tagName: "strong", |
|
| 911 |
+ className: null, |
|
| 912 |
+ wordsOnly: false, |
|
| 913 |
+ caseSensitive: false |
|
| 914 |
+ }; |
|
| 915 |
+ return function hightlight(o) {
|
|
| 916 |
+ var regex; |
|
| 917 |
+ o = _.mixin({}, defaults, o);
|
|
| 918 |
+ if (!o.node || !o.pattern) {
|
|
| 919 |
+ return; |
|
| 920 |
+ } |
|
| 921 |
+ o.pattern = _.isArray(o.pattern) ? o.pattern : [ o.pattern ]; |
|
| 922 |
+ regex = getRegex(o.pattern, o.caseSensitive, o.wordsOnly); |
|
| 923 |
+ traverse(o.node, hightlightTextNode); |
|
| 924 |
+ function hightlightTextNode(textNode) {
|
|
| 925 |
+ var match, patternNode, wrapperNode; |
|
| 926 |
+ if (match = regex.exec(textNode.data)) {
|
|
| 927 |
+ wrapperNode = doc.createElement(o.tagName); |
|
| 928 |
+ o.className && (wrapperNode.className = o.className); |
|
| 929 |
+ patternNode = textNode.splitText(match.index); |
|
| 930 |
+ patternNode.splitText(match[0].length); |
|
| 931 |
+ wrapperNode.appendChild(patternNode.cloneNode(true)); |
|
| 932 |
+ textNode.parentNode.replaceChild(wrapperNode, patternNode); |
|
| 933 |
+ } |
|
| 934 |
+ return !!match; |
|
| 935 |
+ } |
|
| 936 |
+ function traverse(el, hightlightTextNode) {
|
|
| 937 |
+ var childNode, TEXT_NODE_TYPE = 3; |
|
| 938 |
+ for (var i = 0; i < el.childNodes.length; i++) {
|
|
| 939 |
+ childNode = el.childNodes[i]; |
|
| 940 |
+ if (childNode.nodeType === TEXT_NODE_TYPE) {
|
|
| 941 |
+ i += hightlightTextNode(childNode) ? 1 : 0; |
|
| 942 |
+ } else {
|
|
| 943 |
+ traverse(childNode, hightlightTextNode); |
|
| 944 |
+ } |
|
| 945 |
+ } |
|
| 946 |
+ } |
|
| 947 |
+ }; |
|
| 948 |
+ function getRegex(patterns, caseSensitive, wordsOnly) {
|
|
| 949 |
+ var escapedPatterns = [], regexStr; |
|
| 950 |
+ for (var i = 0, len = patterns.length; i < len; i++) {
|
|
| 951 |
+ escapedPatterns.push(_.escapeRegExChars(patterns[i])); |
|
| 952 |
+ } |
|
| 953 |
+ regexStr = wordsOnly ? "\\b(" + escapedPatterns.join("|") + ")\\b" : "(" + escapedPatterns.join("|") + ")";
|
|
| 954 |
+ return caseSensitive ? new RegExp(regexStr) : new RegExp(regexStr, "i"); |
|
| 955 |
+ } |
|
| 956 |
+ }(window.document); |
|
| 957 |
+ var Input = function() {
|
|
| 958 |
+ "use strict"; |
|
| 959 |
+ var specialKeyCodeMap; |
|
| 960 |
+ specialKeyCodeMap = {
|
|
| 961 |
+ 9: "tab", |
|
| 962 |
+ 27: "esc", |
|
| 963 |
+ 37: "left", |
|
| 964 |
+ 39: "right", |
|
| 965 |
+ 13: "enter", |
|
| 966 |
+ 38: "up", |
|
| 967 |
+ 40: "down" |
|
| 968 |
+ }; |
|
| 969 |
+ function Input(o) {
|
|
| 970 |
+ var that = this, onBlur, onFocus, onKeydown, onInput; |
|
| 971 |
+ o = o || {};
|
|
| 972 |
+ if (!o.input) {
|
|
| 973 |
+ $.error("input is missing");
|
|
| 974 |
+ } |
|
| 975 |
+ onBlur = _.bind(this._onBlur, this); |
|
| 976 |
+ onFocus = _.bind(this._onFocus, this); |
|
| 977 |
+ onKeydown = _.bind(this._onKeydown, this); |
|
| 978 |
+ onInput = _.bind(this._onInput, this); |
|
| 979 |
+ this.$hint = $(o.hint); |
|
| 980 |
+ this.$input = $(o.input).on("blur.tt", onBlur).on("focus.tt", onFocus).on("keydown.tt", onKeydown);
|
|
| 981 |
+ if (this.$hint.length === 0) {
|
|
| 982 |
+ this.setHint = this.getHint = this.clearHint = this.clearHintIfInvalid = _.noop; |
|
| 983 |
+ } |
|
| 984 |
+ if (!_.isMsie()) {
|
|
| 985 |
+ this.$input.on("input.tt", onInput);
|
|
| 986 |
+ } else {
|
|
| 987 |
+ this.$input.on("keydown.tt keypress.tt cut.tt paste.tt", function($e) {
|
|
| 988 |
+ if (specialKeyCodeMap[$e.which || $e.keyCode]) {
|
|
| 989 |
+ return; |
|
| 990 |
+ } |
|
| 991 |
+ _.defer(_.bind(that._onInput, that, $e)); |
|
| 992 |
+ }); |
|
| 993 |
+ } |
|
| 994 |
+ this.query = this.$input.val(); |
|
| 995 |
+ this.$overflowHelper = buildOverflowHelper(this.$input); |
|
| 996 |
+ } |
|
| 997 |
+ Input.normalizeQuery = function(str) {
|
|
| 998 |
+ return (str || "").replace(/^\s*/g, "").replace(/\s{2,}/g, " ");
|
|
| 999 |
+ }; |
|
| 1000 |
+ _.mixin(Input.prototype, EventEmitter, {
|
|
| 1001 |
+ _onBlur: function onBlur() {
|
|
| 1002 |
+ this.resetInputValue(); |
|
| 1003 |
+ this.trigger("blurred");
|
|
| 1004 |
+ }, |
|
| 1005 |
+ _onFocus: function onFocus() {
|
|
| 1006 |
+ this.trigger("focused");
|
|
| 1007 |
+ }, |
|
| 1008 |
+ _onKeydown: function onKeydown($e) {
|
|
| 1009 |
+ var keyName = specialKeyCodeMap[$e.which || $e.keyCode]; |
|
| 1010 |
+ this._managePreventDefault(keyName, $e); |
|
| 1011 |
+ if (keyName && this._shouldTrigger(keyName, $e)) {
|
|
| 1012 |
+ this.trigger(keyName + "Keyed", $e); |
|
| 1013 |
+ } |
|
| 1014 |
+ }, |
|
| 1015 |
+ _onInput: function onInput() {
|
|
| 1016 |
+ this._checkInputValue(); |
|
| 1017 |
+ }, |
|
| 1018 |
+ _managePreventDefault: function managePreventDefault(keyName, $e) {
|
|
| 1019 |
+ var preventDefault, hintValue, inputValue; |
|
| 1020 |
+ switch (keyName) {
|
|
| 1021 |
+ case "tab": |
|
| 1022 |
+ hintValue = this.getHint(); |
|
| 1023 |
+ inputValue = this.getInputValue(); |
|
| 1024 |
+ preventDefault = hintValue && hintValue !== inputValue && !withModifier($e); |
|
| 1025 |
+ break; |
|
| 1026 |
+ |
|
| 1027 |
+ case "up": |
|
| 1028 |
+ case "down": |
|
| 1029 |
+ preventDefault = !withModifier($e); |
|
| 1030 |
+ break; |
|
| 1031 |
+ |
|
| 1032 |
+ default: |
|
| 1033 |
+ preventDefault = false; |
|
| 1034 |
+ } |
|
| 1035 |
+ preventDefault && $e.preventDefault(); |
|
| 1036 |
+ }, |
|
| 1037 |
+ _shouldTrigger: function shouldTrigger(keyName, $e) {
|
|
| 1038 |
+ var trigger; |
|
| 1039 |
+ switch (keyName) {
|
|
| 1040 |
+ case "tab": |
|
| 1041 |
+ trigger = !withModifier($e); |
|
| 1042 |
+ break; |
|
| 1043 |
+ |
|
| 1044 |
+ default: |
|
| 1045 |
+ trigger = true; |
|
| 1046 |
+ } |
|
| 1047 |
+ return trigger; |
|
| 1048 |
+ }, |
|
| 1049 |
+ _checkInputValue: function checkInputValue() {
|
|
| 1050 |
+ var inputValue, areEquivalent, hasDifferentWhitespace; |
|
| 1051 |
+ inputValue = this.getInputValue(); |
|
| 1052 |
+ areEquivalent = areQueriesEquivalent(inputValue, this.query); |
|
| 1053 |
+ hasDifferentWhitespace = areEquivalent ? this.query.length !== inputValue.length : false; |
|
| 1054 |
+ this.query = inputValue; |
|
| 1055 |
+ if (!areEquivalent) {
|
|
| 1056 |
+ this.trigger("queryChanged", this.query);
|
|
| 1057 |
+ } else if (hasDifferentWhitespace) {
|
|
| 1058 |
+ this.trigger("whitespaceChanged", this.query);
|
|
| 1059 |
+ } |
|
| 1060 |
+ }, |
|
| 1061 |
+ focus: function focus() {
|
|
| 1062 |
+ this.$input.focus(); |
|
| 1063 |
+ }, |
|
| 1064 |
+ blur: function blur() {
|
|
| 1065 |
+ this.$input.blur(); |
|
| 1066 |
+ }, |
|
| 1067 |
+ getQuery: function getQuery() {
|
|
| 1068 |
+ return this.query; |
|
| 1069 |
+ }, |
|
| 1070 |
+ setQuery: function setQuery(query) {
|
|
| 1071 |
+ this.query = query; |
|
| 1072 |
+ }, |
|
| 1073 |
+ getInputValue: function getInputValue() {
|
|
| 1074 |
+ return this.$input.val(); |
|
| 1075 |
+ }, |
|
| 1076 |
+ setInputValue: function setInputValue(value, silent) {
|
|
| 1077 |
+ this.$input.val(value); |
|
| 1078 |
+ silent ? this.clearHint() : this._checkInputValue(); |
|
| 1079 |
+ }, |
|
| 1080 |
+ resetInputValue: function resetInputValue() {
|
|
| 1081 |
+ this.setInputValue(this.query, true); |
|
| 1082 |
+ }, |
|
| 1083 |
+ getHint: function getHint() {
|
|
| 1084 |
+ return this.$hint.val(); |
|
| 1085 |
+ }, |
|
| 1086 |
+ setHint: function setHint(value) {
|
|
| 1087 |
+ this.$hint.val(value); |
|
| 1088 |
+ }, |
|
| 1089 |
+ clearHint: function clearHint() {
|
|
| 1090 |
+ this.setHint("");
|
|
| 1091 |
+ }, |
|
| 1092 |
+ clearHintIfInvalid: function clearHintIfInvalid() {
|
|
| 1093 |
+ var val, hint, valIsPrefixOfHint, isValid; |
|
| 1094 |
+ val = this.getInputValue(); |
|
| 1095 |
+ hint = this.getHint(); |
|
| 1096 |
+ valIsPrefixOfHint = val !== hint && hint.indexOf(val) === 0; |
|
| 1097 |
+ isValid = val !== "" && valIsPrefixOfHint && !this.hasOverflow(); |
|
| 1098 |
+ !isValid && this.clearHint(); |
|
| 1099 |
+ }, |
|
| 1100 |
+ getLanguageDirection: function getLanguageDirection() {
|
|
| 1101 |
+ return (this.$input.css("direction") || "ltr").toLowerCase();
|
|
| 1102 |
+ }, |
|
| 1103 |
+ hasOverflow: function hasOverflow() {
|
|
| 1104 |
+ var constraint = this.$input.width() - 2; |
|
| 1105 |
+ this.$overflowHelper.text(this.getInputValue()); |
|
| 1106 |
+ return this.$overflowHelper.width() >= constraint; |
|
| 1107 |
+ }, |
|
| 1108 |
+ isCursorAtEnd: function() {
|
|
| 1109 |
+ var valueLength, selectionStart, range; |
|
| 1110 |
+ valueLength = this.$input.val().length; |
|
| 1111 |
+ selectionStart = this.$input[0].selectionStart; |
|
| 1112 |
+ if (_.isNumber(selectionStart)) {
|
|
| 1113 |
+ return selectionStart === valueLength; |
|
| 1114 |
+ } else if (document.selection) {
|
|
| 1115 |
+ range = document.selection.createRange(); |
|
| 1116 |
+ range.moveStart("character", -valueLength);
|
|
| 1117 |
+ return valueLength === range.text.length; |
|
| 1118 |
+ } |
|
| 1119 |
+ return true; |
|
| 1120 |
+ }, |
|
| 1121 |
+ destroy: function destroy() {
|
|
| 1122 |
+ this.$hint.off(".tt");
|
|
| 1123 |
+ this.$input.off(".tt");
|
|
| 1124 |
+ this.$hint = this.$input = this.$overflowHelper = null; |
|
| 1125 |
+ } |
|
| 1126 |
+ }); |
|
| 1127 |
+ return Input; |
|
| 1128 |
+ function buildOverflowHelper($input) {
|
|
| 1129 |
+ return $('<pre aria-hidden="true"></pre>').css({
|
|
| 1130 |
+ position: "absolute", |
|
| 1131 |
+ visibility: "hidden", |
|
| 1132 |
+ whiteSpace: "pre", |
|
| 1133 |
+ fontFamily: $input.css("font-family"),
|
|
| 1134 |
+ fontSize: $input.css("font-size"),
|
|
| 1135 |
+ fontStyle: $input.css("font-style"),
|
|
| 1136 |
+ fontVariant: $input.css("font-variant"),
|
|
| 1137 |
+ fontWeight: $input.css("font-weight"),
|
|
| 1138 |
+ wordSpacing: $input.css("word-spacing"),
|
|
| 1139 |
+ letterSpacing: $input.css("letter-spacing"),
|
|
| 1140 |
+ textIndent: $input.css("text-indent"),
|
|
| 1141 |
+ textRendering: $input.css("text-rendering"),
|
|
| 1142 |
+ textTransform: $input.css("text-transform")
|
|
| 1143 |
+ }).insertAfter($input); |
|
| 1144 |
+ } |
|
| 1145 |
+ function areQueriesEquivalent(a, b) {
|
|
| 1146 |
+ return Input.normalizeQuery(a) === Input.normalizeQuery(b); |
|
| 1147 |
+ } |
|
| 1148 |
+ function withModifier($e) {
|
|
| 1149 |
+ return $e.altKey || $e.ctrlKey || $e.metaKey || $e.shiftKey; |
|
| 1150 |
+ } |
|
| 1151 |
+ }(); |
|
| 1152 |
+ var Dataset = function() {
|
|
| 1153 |
+ "use strict"; |
|
| 1154 |
+ var datasetKey = "ttDataset", valueKey = "ttValue", datumKey = "ttDatum"; |
|
| 1155 |
+ function Dataset(o) {
|
|
| 1156 |
+ o = o || {};
|
|
| 1157 |
+ o.templates = o.templates || {};
|
|
| 1158 |
+ if (!o.source) {
|
|
| 1159 |
+ $.error("missing source");
|
|
| 1160 |
+ } |
|
| 1161 |
+ if (o.name && !isValidName(o.name)) {
|
|
| 1162 |
+ $.error("invalid dataset name: " + o.name);
|
|
| 1163 |
+ } |
|
| 1164 |
+ this.query = null; |
|
| 1165 |
+ this.highlight = !!o.highlight; |
|
| 1166 |
+ this.name = o.name || _.getUniqueId(); |
|
| 1167 |
+ this.source = o.source; |
|
| 1168 |
+ this.displayFn = getDisplayFn(o.display || o.displayKey); |
|
| 1169 |
+ this.templates = getTemplates(o.templates, this.displayFn); |
|
| 1170 |
+ this.$el = $(html.dataset.replace("%CLASS%", this.name));
|
|
| 1171 |
+ } |
|
| 1172 |
+ Dataset.extractDatasetName = function extractDatasetName(el) {
|
|
| 1173 |
+ return $(el).data(datasetKey); |
|
| 1174 |
+ }; |
|
| 1175 |
+ Dataset.extractValue = function extractDatum(el) {
|
|
| 1176 |
+ return $(el).data(valueKey); |
|
| 1177 |
+ }; |
|
| 1178 |
+ Dataset.extractDatum = function extractDatum(el) {
|
|
| 1179 |
+ return $(el).data(datumKey); |
|
| 1180 |
+ }; |
|
| 1181 |
+ _.mixin(Dataset.prototype, EventEmitter, {
|
|
| 1182 |
+ _render: function render(query, suggestions) {
|
|
| 1183 |
+ if (!this.$el) {
|
|
| 1184 |
+ return; |
|
| 1185 |
+ } |
|
| 1186 |
+ var that = this, hasSuggestions; |
|
| 1187 |
+ this.$el.empty(); |
|
| 1188 |
+ hasSuggestions = suggestions && suggestions.length; |
|
| 1189 |
+ if (!hasSuggestions && this.templates.empty) {
|
|
| 1190 |
+ this.$el.html(getEmptyHtml()).prepend(that.templates.header ? getHeaderHtml() : null).append(that.templates.footer ? getFooterHtml() : null); |
|
| 1191 |
+ } else if (hasSuggestions) {
|
|
| 1192 |
+ this.$el.html(getSuggestionsHtml()).prepend(that.templates.header ? getHeaderHtml() : null).append(that.templates.footer ? getFooterHtml() : null); |
|
| 1193 |
+ } |
|
| 1194 |
+ this.trigger("rendered");
|
|
| 1195 |
+ function getEmptyHtml() {
|
|
| 1196 |
+ return that.templates.empty({
|
|
| 1197 |
+ query: query, |
|
| 1198 |
+ isEmpty: true |
|
| 1199 |
+ }); |
|
| 1200 |
+ } |
|
| 1201 |
+ function getSuggestionsHtml() {
|
|
| 1202 |
+ var $suggestions, nodes; |
|
| 1203 |
+ $suggestions = $(html.suggestions).css(css.suggestions); |
|
| 1204 |
+ nodes = _.map(suggestions, getSuggestionNode); |
|
| 1205 |
+ $suggestions.append.apply($suggestions, nodes); |
|
| 1206 |
+ that.highlight && highlight({
|
|
| 1207 |
+ className: "tt-highlight", |
|
| 1208 |
+ node: $suggestions[0], |
|
| 1209 |
+ pattern: query |
|
| 1210 |
+ }); |
|
| 1211 |
+ return $suggestions; |
|
| 1212 |
+ function getSuggestionNode(suggestion) {
|
|
| 1213 |
+ var $el; |
|
| 1214 |
+ $el = $(html.suggestion).append(that.templates.suggestion(suggestion)).data(datasetKey, that.name).data(valueKey, that.displayFn(suggestion)).data(datumKey, suggestion); |
|
| 1215 |
+ $el.children().each(function() {
|
|
| 1216 |
+ $(this).css(css.suggestionChild); |
|
| 1217 |
+ }); |
|
| 1218 |
+ return $el; |
|
| 1219 |
+ } |
|
| 1220 |
+ } |
|
| 1221 |
+ function getHeaderHtml() {
|
|
| 1222 |
+ return that.templates.header({
|
|
| 1223 |
+ query: query, |
|
| 1224 |
+ isEmpty: !hasSuggestions |
|
| 1225 |
+ }); |
|
| 1226 |
+ } |
|
| 1227 |
+ function getFooterHtml() {
|
|
| 1228 |
+ return that.templates.footer({
|
|
| 1229 |
+ query: query, |
|
| 1230 |
+ isEmpty: !hasSuggestions |
|
| 1231 |
+ }); |
|
| 1232 |
+ } |
|
| 1233 |
+ }, |
|
| 1234 |
+ getRoot: function getRoot() {
|
|
| 1235 |
+ return this.$el; |
|
| 1236 |
+ }, |
|
| 1237 |
+ update: function update(query) {
|
|
| 1238 |
+ var that = this; |
|
| 1239 |
+ this.query = query; |
|
| 1240 |
+ this.canceled = false; |
|
| 1241 |
+ this.source(query, render); |
|
| 1242 |
+ function render(suggestions) {
|
|
| 1243 |
+ if (!that.canceled && query === that.query) {
|
|
| 1244 |
+ that._render(query, suggestions); |
|
| 1245 |
+ } |
|
| 1246 |
+ } |
|
| 1247 |
+ }, |
|
| 1248 |
+ cancel: function cancel() {
|
|
| 1249 |
+ this.canceled = true; |
|
| 1250 |
+ }, |
|
| 1251 |
+ clear: function clear() {
|
|
| 1252 |
+ this.cancel(); |
|
| 1253 |
+ this.$el.empty(); |
|
| 1254 |
+ this.trigger("rendered");
|
|
| 1255 |
+ }, |
|
| 1256 |
+ isEmpty: function isEmpty() {
|
|
| 1257 |
+ return this.$el.is(":empty");
|
|
| 1258 |
+ }, |
|
| 1259 |
+ destroy: function destroy() {
|
|
| 1260 |
+ this.$el = null; |
|
| 1261 |
+ } |
|
| 1262 |
+ }); |
|
| 1263 |
+ return Dataset; |
|
| 1264 |
+ function getDisplayFn(display) {
|
|
| 1265 |
+ display = display || "value"; |
|
| 1266 |
+ return _.isFunction(display) ? display : displayFn; |
|
| 1267 |
+ function displayFn(obj) {
|
|
| 1268 |
+ return obj[display]; |
|
| 1269 |
+ } |
|
| 1270 |
+ } |
|
| 1271 |
+ function getTemplates(templates, displayFn) {
|
|
| 1272 |
+ return {
|
|
| 1273 |
+ empty: templates.empty && _.templatify(templates.empty), |
|
| 1274 |
+ header: templates.header && _.templatify(templates.header), |
|
| 1275 |
+ footer: templates.footer && _.templatify(templates.footer), |
|
| 1276 |
+ suggestion: templates.suggestion || suggestionTemplate |
|
| 1277 |
+ }; |
|
| 1278 |
+ function suggestionTemplate(context) {
|
|
| 1279 |
+ return "<p>" + displayFn(context) + "</p>"; |
|
| 1280 |
+ } |
|
| 1281 |
+ } |
|
| 1282 |
+ function isValidName(str) {
|
|
| 1283 |
+ return /^[_a-zA-Z0-9-]+$/.test(str); |
|
| 1284 |
+ } |
|
| 1285 |
+ }(); |
|
| 1286 |
+ var Dropdown = function() {
|
|
| 1287 |
+ "use strict"; |
|
| 1288 |
+ function Dropdown(o) {
|
|
| 1289 |
+ var that = this, onSuggestionClick, onSuggestionMouseEnter, onSuggestionMouseLeave; |
|
| 1290 |
+ o = o || {};
|
|
| 1291 |
+ if (!o.menu) {
|
|
| 1292 |
+ $.error("menu is required");
|
|
| 1293 |
+ } |
|
| 1294 |
+ this.isOpen = false; |
|
| 1295 |
+ this.isEmpty = true; |
|
| 1296 |
+ this.datasets = _.map(o.datasets, initializeDataset); |
|
| 1297 |
+ onSuggestionClick = _.bind(this._onSuggestionClick, this); |
|
| 1298 |
+ onSuggestionMouseEnter = _.bind(this._onSuggestionMouseEnter, this); |
|
| 1299 |
+ onSuggestionMouseLeave = _.bind(this._onSuggestionMouseLeave, this); |
|
| 1300 |
+ this.$menu = $(o.menu).on("click.tt", ".tt-suggestion", onSuggestionClick).on("mouseenter.tt", ".tt-suggestion", onSuggestionMouseEnter).on("mouseleave.tt", ".tt-suggestion", onSuggestionMouseLeave);
|
|
| 1301 |
+ _.each(this.datasets, function(dataset) {
|
|
| 1302 |
+ that.$menu.append(dataset.getRoot()); |
|
| 1303 |
+ dataset.onSync("rendered", that._onRendered, that);
|
|
| 1304 |
+ }); |
|
| 1305 |
+ } |
|
| 1306 |
+ _.mixin(Dropdown.prototype, EventEmitter, {
|
|
| 1307 |
+ _onSuggestionClick: function onSuggestionClick($e) {
|
|
| 1308 |
+ this.trigger("suggestionClicked", $($e.currentTarget));
|
|
| 1309 |
+ }, |
|
| 1310 |
+ _onSuggestionMouseEnter: function onSuggestionMouseEnter($e) {
|
|
| 1311 |
+ this._removeCursor(); |
|
| 1312 |
+ this._setCursor($($e.currentTarget), true); |
|
| 1313 |
+ }, |
|
| 1314 |
+ _onSuggestionMouseLeave: function onSuggestionMouseLeave() {
|
|
| 1315 |
+ this._removeCursor(); |
|
| 1316 |
+ }, |
|
| 1317 |
+ _onRendered: function onRendered() {
|
|
| 1318 |
+ this.isEmpty = _.every(this.datasets, isDatasetEmpty); |
|
| 1319 |
+ this.isEmpty ? this._hide() : this.isOpen && this._show(); |
|
| 1320 |
+ this.trigger("datasetRendered");
|
|
| 1321 |
+ function isDatasetEmpty(dataset) {
|
|
| 1322 |
+ return dataset.isEmpty(); |
|
| 1323 |
+ } |
|
| 1324 |
+ }, |
|
| 1325 |
+ _hide: function() {
|
|
| 1326 |
+ this.$menu.hide(); |
|
| 1327 |
+ }, |
|
| 1328 |
+ _show: function() {
|
|
| 1329 |
+ this.$menu.css("display", "block");
|
|
| 1330 |
+ }, |
|
| 1331 |
+ _getSuggestions: function getSuggestions() {
|
|
| 1332 |
+ return this.$menu.find(".tt-suggestion");
|
|
| 1333 |
+ }, |
|
| 1334 |
+ _getCursor: function getCursor() {
|
|
| 1335 |
+ return this.$menu.find(".tt-cursor").first();
|
|
| 1336 |
+ }, |
|
| 1337 |
+ _setCursor: function setCursor($el, silent) {
|
|
| 1338 |
+ $el.first().addClass("tt-cursor");
|
|
| 1339 |
+ !silent && this.trigger("cursorMoved");
|
|
| 1340 |
+ }, |
|
| 1341 |
+ _removeCursor: function removeCursor() {
|
|
| 1342 |
+ this._getCursor().removeClass("tt-cursor");
|
|
| 1343 |
+ }, |
|
| 1344 |
+ _moveCursor: function moveCursor(increment) {
|
|
| 1345 |
+ var $suggestions, $oldCursor, newCursorIndex, $newCursor; |
|
| 1346 |
+ if (!this.isOpen) {
|
|
| 1347 |
+ return; |
|
| 1348 |
+ } |
|
| 1349 |
+ $oldCursor = this._getCursor(); |
|
| 1350 |
+ $suggestions = this._getSuggestions(); |
|
| 1351 |
+ this._removeCursor(); |
|
| 1352 |
+ newCursorIndex = $suggestions.index($oldCursor) + increment; |
|
| 1353 |
+ newCursorIndex = (newCursorIndex + 1) % ($suggestions.length + 1) - 1; |
|
| 1354 |
+ if (newCursorIndex === -1) {
|
|
| 1355 |
+ this.trigger("cursorRemoved");
|
|
| 1356 |
+ return; |
|
| 1357 |
+ } else if (newCursorIndex < -1) {
|
|
| 1358 |
+ newCursorIndex = $suggestions.length - 1; |
|
| 1359 |
+ } |
|
| 1360 |
+ this._setCursor($newCursor = $suggestions.eq(newCursorIndex)); |
|
| 1361 |
+ this._ensureVisible($newCursor); |
|
| 1362 |
+ }, |
|
| 1363 |
+ _ensureVisible: function ensureVisible($el) {
|
|
| 1364 |
+ var elTop, elBottom, menuScrollTop, menuHeight; |
|
| 1365 |
+ elTop = $el.position().top; |
|
| 1366 |
+ elBottom = elTop + $el.outerHeight(true); |
|
| 1367 |
+ menuScrollTop = this.$menu.scrollTop(); |
|
| 1368 |
+ menuHeight = this.$menu.height() + parseInt(this.$menu.css("paddingTop"), 10) + parseInt(this.$menu.css("paddingBottom"), 10);
|
|
| 1369 |
+ if (elTop < 0) {
|
|
| 1370 |
+ this.$menu.scrollTop(menuScrollTop + elTop); |
|
| 1371 |
+ } else if (menuHeight < elBottom) {
|
|
| 1372 |
+ this.$menu.scrollTop(menuScrollTop + (elBottom - menuHeight)); |
|
| 1373 |
+ } |
|
| 1374 |
+ }, |
|
| 1375 |
+ close: function close() {
|
|
| 1376 |
+ if (this.isOpen) {
|
|
| 1377 |
+ this.isOpen = false; |
|
| 1378 |
+ this._removeCursor(); |
|
| 1379 |
+ this._hide(); |
|
| 1380 |
+ this.trigger("closed");
|
|
| 1381 |
+ } |
|
| 1382 |
+ }, |
|
| 1383 |
+ open: function open() {
|
|
| 1384 |
+ if (!this.isOpen) {
|
|
| 1385 |
+ this.isOpen = true; |
|
| 1386 |
+ !this.isEmpty && this._show(); |
|
| 1387 |
+ this.trigger("opened");
|
|
| 1388 |
+ } |
|
| 1389 |
+ }, |
|
| 1390 |
+ setLanguageDirection: function setLanguageDirection(dir) {
|
|
| 1391 |
+ this.$menu.css(dir === "ltr" ? css.ltr : css.rtl); |
|
| 1392 |
+ }, |
|
| 1393 |
+ moveCursorUp: function moveCursorUp() {
|
|
| 1394 |
+ this._moveCursor(-1); |
|
| 1395 |
+ }, |
|
| 1396 |
+ moveCursorDown: function moveCursorDown() {
|
|
| 1397 |
+ this._moveCursor(+1); |
|
| 1398 |
+ }, |
|
| 1399 |
+ getDatumForSuggestion: function getDatumForSuggestion($el) {
|
|
| 1400 |
+ var datum = null; |
|
| 1401 |
+ if ($el.length) {
|
|
| 1402 |
+ datum = {
|
|
| 1403 |
+ raw: Dataset.extractDatum($el), |
|
| 1404 |
+ value: Dataset.extractValue($el), |
|
| 1405 |
+ datasetName: Dataset.extractDatasetName($el) |
|
| 1406 |
+ }; |
|
| 1407 |
+ } |
|
| 1408 |
+ return datum; |
|
| 1409 |
+ }, |
|
| 1410 |
+ getDatumForCursor: function getDatumForCursor() {
|
|
| 1411 |
+ return this.getDatumForSuggestion(this._getCursor().first()); |
|
| 1412 |
+ }, |
|
| 1413 |
+ getDatumForTopSuggestion: function getDatumForTopSuggestion() {
|
|
| 1414 |
+ return this.getDatumForSuggestion(this._getSuggestions().first()); |
|
| 1415 |
+ }, |
|
| 1416 |
+ update: function update(query) {
|
|
| 1417 |
+ _.each(this.datasets, updateDataset); |
|
| 1418 |
+ function updateDataset(dataset) {
|
|
| 1419 |
+ dataset.update(query); |
|
| 1420 |
+ } |
|
| 1421 |
+ }, |
|
| 1422 |
+ empty: function empty() {
|
|
| 1423 |
+ _.each(this.datasets, clearDataset); |
|
| 1424 |
+ this.isEmpty = true; |
|
| 1425 |
+ function clearDataset(dataset) {
|
|
| 1426 |
+ dataset.clear(); |
|
| 1427 |
+ } |
|
| 1428 |
+ }, |
|
| 1429 |
+ isVisible: function isVisible() {
|
|
| 1430 |
+ return this.isOpen && !this.isEmpty; |
|
| 1431 |
+ }, |
|
| 1432 |
+ destroy: function destroy() {
|
|
| 1433 |
+ this.$menu.off(".tt");
|
|
| 1434 |
+ this.$menu = null; |
|
| 1435 |
+ _.each(this.datasets, destroyDataset); |
|
| 1436 |
+ function destroyDataset(dataset) {
|
|
| 1437 |
+ dataset.destroy(); |
|
| 1438 |
+ } |
|
| 1439 |
+ } |
|
| 1440 |
+ }); |
|
| 1441 |
+ return Dropdown; |
|
| 1442 |
+ function initializeDataset(oDataset) {
|
|
| 1443 |
+ return new Dataset(oDataset); |
|
| 1444 |
+ } |
|
| 1445 |
+ }(); |
|
| 1446 |
+ var Typeahead = function() {
|
|
| 1447 |
+ "use strict"; |
|
| 1448 |
+ var attrsKey = "ttAttrs"; |
|
| 1449 |
+ function Typeahead(o) {
|
|
| 1450 |
+ var $menu, $input, $hint; |
|
| 1451 |
+ o = o || {};
|
|
| 1452 |
+ if (!o.input) {
|
|
| 1453 |
+ $.error("missing input");
|
|
| 1454 |
+ } |
|
| 1455 |
+ this.isActivated = false; |
|
| 1456 |
+ this.autoselect = !!o.autoselect; |
|
| 1457 |
+ this.minLength = _.isNumber(o.minLength) ? o.minLength : 1; |
|
| 1458 |
+ this.$node = buildDom(o.input, o.withHint); |
|
| 1459 |
+ $menu = this.$node.find(".tt-dropdown-menu");
|
|
| 1460 |
+ $input = this.$node.find(".tt-input");
|
|
| 1461 |
+ $hint = this.$node.find(".tt-hint");
|
|
| 1462 |
+ $input.on("blur.tt", function($e) {
|
|
| 1463 |
+ var active, isActive, hasActive; |
|
| 1464 |
+ active = document.activeElement; |
|
| 1465 |
+ isActive = $menu.is(active); |
|
| 1466 |
+ hasActive = $menu.has(active).length > 0; |
|
| 1467 |
+ if (_.isMsie() && (isActive || hasActive)) {
|
|
| 1468 |
+ $e.preventDefault(); |
|
| 1469 |
+ $e.stopImmediatePropagation(); |
|
| 1470 |
+ _.defer(function() {
|
|
| 1471 |
+ $input.focus(); |
|
| 1472 |
+ }); |
|
| 1473 |
+ } |
|
| 1474 |
+ }); |
|
| 1475 |
+ $menu.on("mousedown.tt", function($e) {
|
|
| 1476 |
+ $e.preventDefault(); |
|
| 1477 |
+ }); |
|
| 1478 |
+ this.eventBus = o.eventBus || new EventBus({
|
|
| 1479 |
+ el: $input |
|
| 1480 |
+ }); |
|
| 1481 |
+ this.dropdown = new Dropdown({
|
|
| 1482 |
+ menu: $menu, |
|
| 1483 |
+ datasets: o.datasets |
|
| 1484 |
+ }).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);
|
|
| 1485 |
+ this.input = new Input({
|
|
| 1486 |
+ input: $input, |
|
| 1487 |
+ hint: $hint |
|
| 1488 |
+ }).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);
|
|
| 1489 |
+ this._setLanguageDirection(); |
|
| 1490 |
+ } |
|
| 1491 |
+ _.mixin(Typeahead.prototype, {
|
|
| 1492 |
+ _onSuggestionClicked: function onSuggestionClicked(type, $el) {
|
|
| 1493 |
+ var datum; |
|
| 1494 |
+ if (datum = this.dropdown.getDatumForSuggestion($el)) {
|
|
| 1495 |
+ this._select(datum); |
|
| 1496 |
+ } |
|
| 1497 |
+ }, |
|
| 1498 |
+ _onCursorMoved: function onCursorMoved() {
|
|
| 1499 |
+ var datum = this.dropdown.getDatumForCursor(); |
|
| 1500 |
+ this.input.setInputValue(datum.value, true); |
|
| 1501 |
+ this.eventBus.trigger("cursorchanged", datum.raw, datum.datasetName);
|
|
| 1502 |
+ }, |
|
| 1503 |
+ _onCursorRemoved: function onCursorRemoved() {
|
|
| 1504 |
+ this.input.resetInputValue(); |
|
| 1505 |
+ this._updateHint(); |
|
| 1506 |
+ }, |
|
| 1507 |
+ _onDatasetRendered: function onDatasetRendered() {
|
|
| 1508 |
+ this._updateHint(); |
|
| 1509 |
+ }, |
|
| 1510 |
+ _onOpened: function onOpened() {
|
|
| 1511 |
+ this._updateHint(); |
|
| 1512 |
+ this.eventBus.trigger("opened");
|
|
| 1513 |
+ }, |
|
| 1514 |
+ _onClosed: function onClosed() {
|
|
| 1515 |
+ this.input.clearHint(); |
|
| 1516 |
+ this.eventBus.trigger("closed");
|
|
| 1517 |
+ }, |
|
| 1518 |
+ _onFocused: function onFocused() {
|
|
| 1519 |
+ this.isActivated = true; |
|
| 1520 |
+ this.dropdown.open(); |
|
| 1521 |
+ }, |
|
| 1522 |
+ _onBlurred: function onBlurred() {
|
|
| 1523 |
+ this.isActivated = false; |
|
| 1524 |
+ this.dropdown.empty(); |
|
| 1525 |
+ this.dropdown.close(); |
|
| 1526 |
+ }, |
|
| 1527 |
+ _onEnterKeyed: function onEnterKeyed(type, $e) {
|
|
| 1528 |
+ var cursorDatum, topSuggestionDatum; |
|
| 1529 |
+ cursorDatum = this.dropdown.getDatumForCursor(); |
|
| 1530 |
+ topSuggestionDatum = this.dropdown.getDatumForTopSuggestion(); |
|
| 1531 |
+ if (cursorDatum) {
|
|
| 1532 |
+ this._select(cursorDatum); |
|
| 1533 |
+ $e.preventDefault(); |
|
| 1534 |
+ } else if (this.autoselect && topSuggestionDatum) {
|
|
| 1535 |
+ this._select(topSuggestionDatum); |
|
| 1536 |
+ $e.preventDefault(); |
|
| 1537 |
+ } |
|
| 1538 |
+ }, |
|
| 1539 |
+ _onTabKeyed: function onTabKeyed(type, $e) {
|
|
| 1540 |
+ var datum; |
|
| 1541 |
+ if (datum = this.dropdown.getDatumForCursor()) {
|
|
| 1542 |
+ this._select(datum); |
|
| 1543 |
+ $e.preventDefault(); |
|
| 1544 |
+ } else {
|
|
| 1545 |
+ this._autocomplete(true); |
|
| 1546 |
+ } |
|
| 1547 |
+ }, |
|
| 1548 |
+ _onEscKeyed: function onEscKeyed() {
|
|
| 1549 |
+ this.dropdown.close(); |
|
| 1550 |
+ this.input.resetInputValue(); |
|
| 1551 |
+ }, |
|
| 1552 |
+ _onUpKeyed: function onUpKeyed() {
|
|
| 1553 |
+ var query = this.input.getQuery(); |
|
| 1554 |
+ this.dropdown.isEmpty && query.length >= this.minLength ? this.dropdown.update(query) : this.dropdown.moveCursorUp(); |
|
| 1555 |
+ this.dropdown.open(); |
|
| 1556 |
+ }, |
|
| 1557 |
+ _onDownKeyed: function onDownKeyed() {
|
|
| 1558 |
+ var query = this.input.getQuery(); |
|
| 1559 |
+ this.dropdown.isEmpty && query.length >= this.minLength ? this.dropdown.update(query) : this.dropdown.moveCursorDown(); |
|
| 1560 |
+ this.dropdown.open(); |
|
| 1561 |
+ }, |
|
| 1562 |
+ _onLeftKeyed: function onLeftKeyed() {
|
|
| 1563 |
+ this.dir === "rtl" && this._autocomplete(); |
|
| 1564 |
+ }, |
|
| 1565 |
+ _onRightKeyed: function onRightKeyed() {
|
|
| 1566 |
+ this.dir === "ltr" && this._autocomplete(); |
|
| 1567 |
+ }, |
|
| 1568 |
+ _onQueryChanged: function onQueryChanged(e, query) {
|
|
| 1569 |
+ this.input.clearHintIfInvalid(); |
|
| 1570 |
+ query.length >= this.minLength ? this.dropdown.update(query) : this.dropdown.empty(); |
|
| 1571 |
+ this.dropdown.open(); |
|
| 1572 |
+ this._setLanguageDirection(); |
|
| 1573 |
+ }, |
|
| 1574 |
+ _onWhitespaceChanged: function onWhitespaceChanged() {
|
|
| 1575 |
+ this._updateHint(); |
|
| 1576 |
+ this.dropdown.open(); |
|
| 1577 |
+ }, |
|
| 1578 |
+ _setLanguageDirection: function setLanguageDirection() {
|
|
| 1579 |
+ var dir; |
|
| 1580 |
+ if (this.dir !== (dir = this.input.getLanguageDirection())) {
|
|
| 1581 |
+ this.dir = dir; |
|
| 1582 |
+ this.$node.css("direction", dir);
|
|
| 1583 |
+ this.dropdown.setLanguageDirection(dir); |
|
| 1584 |
+ } |
|
| 1585 |
+ }, |
|
| 1586 |
+ _updateHint: function updateHint() {
|
|
| 1587 |
+ var datum, val, query, escapedQuery, frontMatchRegEx, match; |
|
| 1588 |
+ datum = this.dropdown.getDatumForTopSuggestion(); |
|
| 1589 |
+ if (datum && this.dropdown.isVisible() && !this.input.hasOverflow()) {
|
|
| 1590 |
+ val = this.input.getInputValue(); |
|
| 1591 |
+ query = Input.normalizeQuery(val); |
|
| 1592 |
+ escapedQuery = _.escapeRegExChars(query); |
|
| 1593 |
+ frontMatchRegEx = new RegExp("^(?:" + escapedQuery + ")(.+$)", "i");
|
|
| 1594 |
+ match = frontMatchRegEx.exec(datum.value); |
|
| 1595 |
+ match ? this.input.setHint(val + match[1]) : this.input.clearHint(); |
|
| 1596 |
+ } else {
|
|
| 1597 |
+ this.input.clearHint(); |
|
| 1598 |
+ } |
|
| 1599 |
+ }, |
|
| 1600 |
+ _autocomplete: function autocomplete(laxCursor) {
|
|
| 1601 |
+ var hint, query, isCursorAtEnd, datum; |
|
| 1602 |
+ hint = this.input.getHint(); |
|
| 1603 |
+ query = this.input.getQuery(); |
|
| 1604 |
+ isCursorAtEnd = laxCursor || this.input.isCursorAtEnd(); |
|
| 1605 |
+ if (hint && query !== hint && isCursorAtEnd) {
|
|
| 1606 |
+ datum = this.dropdown.getDatumForTopSuggestion(); |
|
| 1607 |
+ datum && this.input.setInputValue(datum.value); |
|
| 1608 |
+ this.eventBus.trigger("autocompleted", datum.raw, datum.datasetName);
|
|
| 1609 |
+ } |
|
| 1610 |
+ }, |
|
| 1611 |
+ _select: function select(datum) {
|
|
| 1612 |
+ this.input.setQuery(datum.value); |
|
| 1613 |
+ this.input.setInputValue(datum.value, true); |
|
| 1614 |
+ this._setLanguageDirection(); |
|
| 1615 |
+ this.eventBus.trigger("selected", datum.raw, datum.datasetName);
|
|
| 1616 |
+ this.dropdown.close(); |
|
| 1617 |
+ _.defer(_.bind(this.dropdown.empty, this.dropdown)); |
|
| 1618 |
+ }, |
|
| 1619 |
+ open: function open() {
|
|
| 1620 |
+ this.dropdown.open(); |
|
| 1621 |
+ }, |
|
| 1622 |
+ close: function close() {
|
|
| 1623 |
+ this.dropdown.close(); |
|
| 1624 |
+ }, |
|
| 1625 |
+ setVal: function setVal(val) {
|
|
| 1626 |
+ val = _.toStr(val); |
|
| 1627 |
+ if (this.isActivated) {
|
|
| 1628 |
+ this.input.setInputValue(val); |
|
| 1629 |
+ } else {
|
|
| 1630 |
+ this.input.setQuery(val); |
|
| 1631 |
+ this.input.setInputValue(val, true); |
|
| 1632 |
+ } |
|
| 1633 |
+ this._setLanguageDirection(); |
|
| 1634 |
+ }, |
|
| 1635 |
+ getVal: function getVal() {
|
|
| 1636 |
+ return this.input.getQuery(); |
|
| 1637 |
+ }, |
|
| 1638 |
+ destroy: function destroy() {
|
|
| 1639 |
+ this.input.destroy(); |
|
| 1640 |
+ this.dropdown.destroy(); |
|
| 1641 |
+ destroyDomStructure(this.$node); |
|
| 1642 |
+ this.$node = null; |
|
| 1643 |
+ } |
|
| 1644 |
+ }); |
|
| 1645 |
+ return Typeahead; |
|
| 1646 |
+ function buildDom(input, withHint) {
|
|
| 1647 |
+ var $input, $wrapper, $dropdown, $hint; |
|
| 1648 |
+ $input = $(input); |
|
| 1649 |
+ $wrapper = $(html.wrapper).css(css.wrapper); |
|
| 1650 |
+ $dropdown = $(html.dropdown).css(css.dropdown); |
|
| 1651 |
+ $hint = $input.clone().css(css.hint).css(getBackgroundStyles($input)); |
|
| 1652 |
+ $hint.val("").removeData().addClass("tt-hint").removeAttr("id name placeholder required").prop("readonly", true).attr({
|
|
| 1653 |
+ autocomplete: "off", |
|
| 1654 |
+ spellcheck: "false", |
|
| 1655 |
+ tabindex: -1 |
|
| 1656 |
+ }); |
|
| 1657 |
+ $input.data(attrsKey, {
|
|
| 1658 |
+ dir: $input.attr("dir"),
|
|
| 1659 |
+ autocomplete: $input.attr("autocomplete"),
|
|
| 1660 |
+ spellcheck: $input.attr("spellcheck"),
|
|
| 1661 |
+ style: $input.attr("style")
|
|
| 1662 |
+ }); |
|
| 1663 |
+ $input.addClass("tt-input").attr({
|
|
| 1664 |
+ autocomplete: "off", |
|
| 1665 |
+ spellcheck: false |
|
| 1666 |
+ }).css(withHint ? css.input : css.inputWithNoHint); |
|
| 1667 |
+ try {
|
|
| 1668 |
+ !$input.attr("dir") && $input.attr("dir", "auto");
|
|
| 1669 |
+ } catch (e) {}
|
|
| 1670 |
+ return $input.wrap($wrapper).parent().prepend(withHint ? $hint : null).append($dropdown); |
|
| 1671 |
+ } |
|
| 1672 |
+ function getBackgroundStyles($el) {
|
|
| 1673 |
+ return {
|
|
| 1674 |
+ backgroundAttachment: $el.css("background-attachment"),
|
|
| 1675 |
+ backgroundClip: $el.css("background-clip"),
|
|
| 1676 |
+ backgroundColor: $el.css("background-color"),
|
|
| 1677 |
+ backgroundImage: $el.css("background-image"),
|
|
| 1678 |
+ backgroundOrigin: $el.css("background-origin"),
|
|
| 1679 |
+ backgroundPosition: $el.css("background-position"),
|
|
| 1680 |
+ backgroundRepeat: $el.css("background-repeat"),
|
|
| 1681 |
+ backgroundSize: $el.css("background-size")
|
|
| 1682 |
+ }; |
|
| 1683 |
+ } |
|
| 1684 |
+ function destroyDomStructure($node) {
|
|
| 1685 |
+ var $input = $node.find(".tt-input");
|
|
| 1686 |
+ _.each($input.data(attrsKey), function(val, key) {
|
|
| 1687 |
+ _.isUndefined(val) ? $input.removeAttr(key) : $input.attr(key, val); |
|
| 1688 |
+ }); |
|
| 1689 |
+ $input.detach().removeData(attrsKey).removeClass("tt-input").insertAfter($node);
|
|
| 1690 |
+ $node.remove(); |
|
| 1691 |
+ } |
|
| 1692 |
+ }(); |
|
| 1693 |
+ (function() {
|
|
| 1694 |
+ "use strict"; |
|
| 1695 |
+ var old, typeaheadKey, methods; |
|
| 1696 |
+ old = $.fn.typeahead; |
|
| 1697 |
+ typeaheadKey = "ttTypeahead"; |
|
| 1698 |
+ methods = {
|
|
| 1699 |
+ initialize: function initialize(o, datasets) {
|
|
| 1700 |
+ datasets = _.isArray(datasets) ? datasets : [].slice.call(arguments, 1); |
|
| 1701 |
+ o = o || {};
|
|
| 1702 |
+ return this.each(attach); |
|
| 1703 |
+ function attach() {
|
|
| 1704 |
+ var $input = $(this), eventBus, typeahead; |
|
| 1705 |
+ _.each(datasets, function(d) {
|
|
| 1706 |
+ d.highlight = !!o.highlight; |
|
| 1707 |
+ }); |
|
| 1708 |
+ typeahead = new Typeahead({
|
|
| 1709 |
+ input: $input, |
|
| 1710 |
+ eventBus: eventBus = new EventBus({
|
|
| 1711 |
+ el: $input |
|
| 1712 |
+ }), |
|
| 1713 |
+ withHint: _.isUndefined(o.hint) ? true : !!o.hint, |
|
| 1714 |
+ minLength: o.minLength, |
|
| 1715 |
+ autoselect: o.autoselect, |
|
| 1716 |
+ datasets: datasets |
|
| 1717 |
+ }); |
|
| 1718 |
+ $input.data(typeaheadKey, typeahead); |
|
| 1719 |
+ } |
|
| 1720 |
+ }, |
|
| 1721 |
+ open: function open() {
|
|
| 1722 |
+ return this.each(openTypeahead); |
|
| 1723 |
+ function openTypeahead() {
|
|
| 1724 |
+ var $input = $(this), typeahead; |
|
| 1725 |
+ if (typeahead = $input.data(typeaheadKey)) {
|
|
| 1726 |
+ typeahead.open(); |
|
| 1727 |
+ } |
|
| 1728 |
+ } |
|
| 1729 |
+ }, |
|
| 1730 |
+ close: function close() {
|
|
| 1731 |
+ return this.each(closeTypeahead); |
|
| 1732 |
+ function closeTypeahead() {
|
|
| 1733 |
+ var $input = $(this), typeahead; |
|
| 1734 |
+ if (typeahead = $input.data(typeaheadKey)) {
|
|
| 1735 |
+ typeahead.close(); |
|
| 1736 |
+ } |
|
| 1737 |
+ } |
|
| 1738 |
+ }, |
|
| 1739 |
+ val: function val(newVal) {
|
|
| 1740 |
+ return !arguments.length ? getVal(this.first()) : this.each(setVal); |
|
| 1741 |
+ function setVal() {
|
|
| 1742 |
+ var $input = $(this), typeahead; |
|
| 1743 |
+ if (typeahead = $input.data(typeaheadKey)) {
|
|
| 1744 |
+ typeahead.setVal(newVal); |
|
| 1745 |
+ } |
|
| 1746 |
+ } |
|
| 1747 |
+ function getVal($input) {
|
|
| 1748 |
+ var typeahead, query; |
|
| 1749 |
+ if (typeahead = $input.data(typeaheadKey)) {
|
|
| 1750 |
+ query = typeahead.getVal(); |
|
| 1751 |
+ } |
|
| 1752 |
+ return query; |
|
| 1753 |
+ } |
|
| 1754 |
+ }, |
|
| 1755 |
+ destroy: function destroy() {
|
|
| 1756 |
+ return this.each(unattach); |
|
| 1757 |
+ function unattach() {
|
|
| 1758 |
+ var $input = $(this), typeahead; |
|
| 1759 |
+ if (typeahead = $input.data(typeaheadKey)) {
|
|
| 1760 |
+ typeahead.destroy(); |
|
| 1761 |
+ $input.removeData(typeaheadKey); |
|
| 1762 |
+ } |
|
| 1763 |
+ } |
|
| 1764 |
+ } |
|
| 1765 |
+ }; |
|
| 1766 |
+ $.fn.typeahead = function(method) {
|
|
| 1767 |
+ var tts; |
|
| 1768 |
+ if (methods[method] && method !== "initialize") {
|
|
| 1769 |
+ tts = this.filter(function() {
|
|
| 1770 |
+ return !!$(this).data(typeaheadKey); |
|
| 1771 |
+ }); |
|
| 1772 |
+ return methods[method].apply(tts, [].slice.call(arguments, 1)); |
|
| 1773 |
+ } else {
|
|
| 1774 |
+ return methods.initialize.apply(this, arguments); |
|
| 1775 |
+ } |
|
| 1776 |
+ }; |
|
| 1777 |
+ $.fn.typeahead.noConflict = function noConflict() {
|
|
| 1778 |
+ $.fn.typeahead = old; |
|
| 1779 |
+ return this; |
|
| 1780 |
+ }; |
|
| 1781 |
+ })(); |
|
| 1782 |
+})(window.jQuery); |
|
| 0 | 1783 |
\ No newline at end of file |
| ... | ... |
@@ -399,9 +399,26 @@ on_show_add_contribution_modal=function(e) {
|
| 399 | 399 |
} |
| 400 | 400 |
$('#add_contribution_modal #add_contribution_category').html(cats);
|
| 401 | 401 |
$('#add_contribution_modal #add_contribution_category')[0].value=current_cat;
|
| 402 |
+ $('#add_contribution_modal #add_contribution_title').typeahead({
|
|
| 403 |
+ hint: true, |
|
| 404 |
+ highlight: true, |
|
| 405 |
+ minLength: 1 |
|
| 406 |
+ }, |
|
| 407 |
+ {
|
|
| 408 |
+ name: 'titles', |
|
| 409 |
+ displayKey: 'value', |
|
| 410 |
+ source: group.findContributionByTitleMatches() |
|
| 411 |
+ }); |
|
| 412 |
+ |
|
| 402 | 413 |
$('#add_contribution_modal #add_contribution_title').focus();
|
| 403 | 414 |
} |
| 404 | 415 |
|
| 416 |
+on_select_contribution_suggestion=function(event,choice,name) {
|
|
| 417 |
+ if (jQuery.type(choice['category'])=='string') {
|
|
| 418 |
+ $('#add_contribution_modal #add_contribution_category')[0].value=choice['category'];
|
|
| 419 |
+ } |
|
| 420 |
+} |
|
| 421 |
+ |
|
| 405 | 422 |
on_click_add_contribution_btn=function() {
|
| 406 | 423 |
$('#add_contribution_modal').data('group-uuid',$('#view-group').data('uuid'));
|
| 407 | 424 |
$('#add_contribution_modal #edit_uuid')[0].value='-1';
|
| ... | ... |
@@ -828,6 +845,7 @@ $( document ).ready( function() {
|
| 828 | 845 |
$("#add_contribution_modal").on('shown.bs.modal',on_show_add_contribution_modal);
|
| 829 | 846 |
$("#add_contribution_modal").on('hidden.bs.modal',on_close_add_contribution_modal);
|
| 830 | 847 |
$("#add_contribution_modal form").on('submit',on_valid_add_contribution_modal);
|
| 848 |
+ $('#add_contribution_modal #add_contribution_title').on('typeahead:selected', on_select_contribution_suggestion);
|
|
| 831 | 849 |
|
| 832 | 850 |
$("#display_balance_btn").bind('click',on_display_balance_btn_click);
|
| 833 | 851 |
|
| ... | ... |
@@ -306,6 +306,37 @@ function Group(uuid,name,data) {
|
| 306 | 306 |
); |
| 307 | 307 |
} |
| 308 | 308 |
|
| 309 |
+ this.findContributionByTitleMatches=function() {
|
|
| 310 |
+ var contributions=this.contributions; |
|
| 311 |
+ |
|
| 312 |
+ return function(q, cb) {
|
|
| 313 |
+ var matches, substrRegex; |
|
| 314 |
+ |
|
| 315 |
+ // an array that will be populated with substring matches |
|
| 316 |
+ matches = []; |
|
| 317 |
+ |
|
| 318 |
+ // regex used to determine if a string contains the substring `q` |
|
| 319 |
+ substrRegex = new RegExp(q, 'i'); |
|
| 320 |
+ |
|
| 321 |
+ var titles=[]; |
|
| 322 |
+ for (uuid in contributions) {
|
|
| 323 |
+ if (substrRegex.test(contributions[uuid].title)) {
|
|
| 324 |
+ var title=String(contributions[uuid].title).replace(/^\s+|\s+$/g, ''); |
|
| 325 |
+ if (titles.indexOf(title.toLowerCase())!=-1) {
|
|
| 326 |
+ continue; |
|
| 327 |
+ } |
|
| 328 |
+ titles.push(title.toLowerCase()); |
|
| 329 |
+ matches.push({
|
|
| 330 |
+ value: title, |
|
| 331 |
+ category: contributions[uuid].category |
|
| 332 |
+ }); |
|
| 333 |
+ } |
|
| 334 |
+ } |
|
| 335 |
+ |
|
| 336 |
+ cb(matches); |
|
| 337 |
+ }; |
|
| 338 |
+ } |
|
| 339 |
+ |
|
| 309 | 340 |
/* |
| 310 | 341 |
* Categories |
| 311 | 342 |
*/ |
| ... | ... |
@@ -71,6 +71,51 @@ span.cat-color {
|
| 71 | 71 |
list-style-type: none; |
| 72 | 72 |
padding: 0; |
| 73 | 73 |
} |
| 74 |
+ |
|
| 75 |
+/* |
|
| 76 |
+ * Typehead |
|
| 77 |
+ */ |
|
| 78 |
+.tt-dropdown-menu {
|
|
| 79 |
+ position: absolute; |
|
| 80 |
+ top: 100%; |
|
| 81 |
+ left: 0; |
|
| 82 |
+ z-index: 1000; |
|
| 83 |
+ display: none; |
|
| 84 |
+ float: left; |
|
| 85 |
+ min-width: 160px; |
|
| 86 |
+ padding: 5px 0; |
|
| 87 |
+ margin: 2px 0 0; |
|
| 88 |
+ list-style: none; |
|
| 89 |
+ font-size: 14px; |
|
| 90 |
+ background-color: #ffffff; |
|
| 91 |
+ border: 1px solid #cccccc; |
|
| 92 |
+ border: 1px solid rgba(0, 0, 0, 0.15); |
|
| 93 |
+ border-radius: 4px; |
|
| 94 |
+ -webkit-box-shadow: 0 6px 12px rgba(0, 0, 0, 0.175); |
|
| 95 |
+ box-shadow: 0 6px 12px rgba(0, 0, 0, 0.175); |
|
| 96 |
+ background-clip: padding-box; |
|
| 97 |
+} |
|
| 98 |
+.tt-suggestion > p {
|
|
| 99 |
+ display: block; |
|
| 100 |
+ padding: 3px 20px; |
|
| 101 |
+ clear: both; |
|
| 102 |
+ font-weight: normal; |
|
| 103 |
+ line-height: 1.428571429; |
|
| 104 |
+ color: #333333; |
|
| 105 |
+ white-space: nowrap; |
|
| 106 |
+} |
|
| 107 |
+.tt-suggestion > p:hover, |
|
| 108 |
+.tt-suggestion > p:focus, |
|
| 109 |
+.tt-suggestion.tt-cursor p {
|
|
| 110 |
+ color: #ffffff; |
|
| 111 |
+ text-decoration: none; |
|
| 112 |
+ outline: 0; |
|
| 113 |
+ background-color: #428bca; |
|
| 114 |
+} |
|
| 115 |
+ |
|
| 116 |
+.twitter-typeahead, .tt-hint {
|
|
| 117 |
+ width: 100%; |
|
| 118 |
+} |
|
| 74 | 119 |
</style> |
| 75 | 120 |
<body> |
| 76 | 121 |
<div class="navbar navbar-inverse navbar-fixed-top" role="navigation"> |
| ... | ... |
@@ -569,6 +614,7 @@ span.cat-color {
|
| 569 | 614 |
<script src="inc/lib/pickadate/picker.js"></script> |
| 570 | 615 |
<script src="inc/lib/pickadate/picker.date.js"></script> |
| 571 | 616 |
<script src="inc/lib/pickadate/legacy.js"></script> |
| 617 |
+ <script src="inc/lib/typeahead.bundle.js"></script> |
|
| 572 | 618 |
<script src="inc/lib/uuid.js"></script> |
| 573 | 619 |
<script src="inc/myco_objects.js"></script> |
| 574 | 620 |
<script src="inc/myco_confirm.js"></script> |
| 575 | 621 |