/* * typeahead.js * https://github.com/twitter/typeahead * Copyright 2013 Twitter, Inc. and other contributors; Licensed MIT */ var TypeaheadView = (function() { var html = { wrapper: '', hint: '', dropdown: '' }, css = { wrapper: { position: 'relative', display: 'inline-block' }, hint: { position: 'absolute', top: '0', left: '0', borderColor: 'transparent', boxShadow: 'none' }, query: { position: 'relative', verticalAlign: 'top', backgroundColor: 'transparent' }, dropdown: { position: 'absolute', top: '100%', left: '0', // TODO: should this be configurable? zIndex: '100', display: 'none' } }; // ie specific styling if (utils.isMsie()) { // ie6-8 (and 9?) doesn't fire hover and click events for elements with // transparent backgrounds, for a workaround, use 1x1 transparent gif utils.mixin(css.query, { backgroundImage: 'url()' }); } // ie7 and under specific styling if (utils.isMsie() && utils.isMsie() <= 7) { utils.mixin(css.wrapper, { display: 'inline', zoom: '1' }); // if someone can tell me why this is necessary to align // the hint with the query in ie7, i'll send you $5 - @JakeHarding utils.mixin(css.query, { marginTop: '-1px' }); } // constructor // ----------- function TypeaheadView(o) { var $menu, $input, $hint; utils.bindAll(this); this.$node = buildDomStructure(o.input); this.datasets = o.datasets; this.dir = null; this.eventBus = o.eventBus; $menu = this.$node.find('.tt-dropdown-menu'); $input = this.$node.find('.tt-query'); $hint = this.$node.find('.tt-hint'); this.dropdownView = new DropdownView({ menu: $menu }) .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({ input: $input, hint: $hint }) .on('focused', this._openDropdown) .on('blured', this._closeDropdown) .on('blured', this._setInputValueToQuery) .on('blured', this._checkMismatch) .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); } utils.mixin(TypeaheadView.prototype, EventTarget, { // private methods // --------------- _managePreventDefault: function(e) { var $e = e.data, hint, inputValue, preventDefault = false; switch (e.type) { 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(); }, _setLanguageDirection: function() { var dir = this.inputView.getLanguageDirection(); if (dir !== this.dir) { this.dir = dir; this.$node.css('direction', dir); this.dropdownView.setLanguageDirection(dir); } }, _updateHint: function() { var suggestion = this.dropdownView.getFirstSuggestion(), hint = suggestion ? suggestion.value : null, dropdownIsVisible = this.dropdownView.isVisible(), inputHasOverflow = this.inputView.isOverflow(), inputValue, query, escapedQuery, beginsWithQuery, match; if (hint && dropdownIsVisible && !inputHasOverflow) { inputValue = this.inputView.getInputValue(); query = inputValue .replace(/\s{2,}/g, ' ') // condense whitespace .replace(/^\s+/g, ''); // strip leading whitespace escapedQuery = utils.escapeRegExChars(query); beginsWithQuery = new RegExp('^(?:' + escapedQuery + ')(.*$)', 'i'); match = beginsWithQuery.exec(hint); this.inputView.setHintValue(inputValue + (match ? match[1] : '')); } }, _clearHint: function() { this.inputView.setHintValue(''); }, _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() { if (this.inputView.$input.is(":focus")) { this.dropdownView.open(); } }, _closeDropdown: function(e) { this.dropdownView[e.type === 'blured' ? 'closeUnlessMouseIsOverDropdown' : 'close'](); }, _moveDropdownCursor: function(e) { var $e = e.data; if (!$e.shiftKey && !$e.ctrlKey && !$e.metaKey) { this.dropdownView[e.type === 'upKeyed' ? 'moveCursorUp' : 'moveCursorDown'](); } }, _handleSelection: function(e) { var byClick = e.type === 'suggestionSelected', suggestion = byClick ? e.data : this.dropdownView.getSuggestionUnderCursor(); if (suggestion) { this.inputView.setInputValue(suggestion.value); // if triggered by click, ensure the query input still has focus // if triggered by keypress, prevent default browser behavior // which is most likely the submission of a form // note: e.data is the jquery event byClick ? this.inputView.focus() : e.data.preventDefault(); // focus is not a synchronous event in ie, so we deal with it byClick && utils.isMsie() ? utils.defer(this.dropdownView.close) : this.dropdownView.close(); this.eventBus.trigger('selected', suggestion.datum, suggestion.dataset); } }, _checkMismatch: function() { var that = this, query = this.inputView.getQuery(), hint = this.inputView.getHintValue(); if (utils.isBlankString(query)) { this.eventBus.trigger('mismatched'); return; } if (hint !== "") return; // TODO check cache only, otherwise only works with single datasets utils.each(this.datasets, function(i, dataset) { dataset.getSuggestions(query, function(suggestions) { var matches = suggestions.length; var selectedSuggestion; utils.each(suggestions, function(i, suggestion) { if (suggestion.value == query) { selectedSuggestion = suggestion; } }); if (selectedSuggestion) { that.eventBus.trigger( 'matched', selectedSuggestion.datum, selectedSuggestion.dataset ); } else { that.eventBus.trigger('mismatched'); } }); }); }, _getSuggestions: function() { var that = this, query = this.inputView.getQuery(); if (utils.isBlankString(query)) { return; } utils.each(this.datasets, function(i, dataset) { dataset.getSuggestions(query, function(suggestions) { // only render the suggestions if the query hasn't changed if (query === that.inputView.getQuery()) { that.dropdownView.renderSuggestions(dataset, suggestions); } }); }); }, _autocomplete: function(e) { var isCursorAtEnd, ignoreEvent, query, hint, suggestion; if (e.type === 'rightKeyed' || e.type === 'leftKeyed') { isCursorAtEnd = this.inputView.isCursorAtEnd(); ignoreEvent = this.inputView.getLanguageDirection() === 'ltr' ? e.type === 'leftKeyed' : e.type === 'rightKeyed'; if (!isCursorAtEnd || ignoreEvent) { return; } } 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) { this.eventBus.trigger(e.type); }, // public methods // -------------- destroy: function() { this.inputView.destroy(); this.dropdownView.destroy(); destroyDomStructure(this.$node); this.$node = null; }, setQuery: function(query) { this.inputView.setQuery(query); this.inputView.setInputValue(query); this._clearHint(); this._clearSuggestions(); this._getSuggestions(); } }); return TypeaheadView; function buildDomStructure(input) { var $wrapper = $(html.wrapper), $dropdown = $(html.dropdown), $input = $(input), $hint = $(html.hint); $wrapper = $wrapper.css(css.wrapper); $dropdown = $dropdown.css(css.dropdown); $hint .css(css.hint) // copy background styles from query input to hint input .css({ backgroundAttachment: $input.css('background-attachment'), backgroundClip: $input.css('background-clip'), backgroundColor: $input.css('background-color'), backgroundImage: $input.css('background-image'), backgroundOrigin: $input.css('background-origin'), backgroundPosition: $input.css('background-position'), backgroundRepeat: $input.css('background-repeat'), backgroundSize: $input.css('background-size') }); // store the original values of the attrs that get modified // so modifications can be reverted on destroy $input.data('ttAttrs', { dir: $input.attr('dir'), autocomplete: $input.attr('autocomplete'), spellcheck: $input.attr('spellcheck'), style: $input.attr('style') }); $input .addClass('tt-query') .attr({ autocomplete: 'off', spellcheck: false }) .css(css.query); // ie7 does not like it when dir is set to auto, // it does not like it one bit try { !$input.attr('dir') && $input.attr('dir', 'auto'); } catch (e) {} return $input .wrap($wrapper) .parent() .prepend($hint) .append($dropdown); } function destroyDomStructure($node) { var $input = $node.find('.tt-query'); // need to remove attrs that weren't previously defined and // revert attrs that originally had a value utils.each($input.data('ttAttrs'), function(key, val) { utils.isUndefined(val) ? $input.removeAttr(key) : $input.attr(key, val); }); $input .detach() .removeData('ttAttrs') .removeClass('tt-query') .insertAfter($node); $node.remove(); } })();