annotate src/main/webapp/imageServer/resources/js/jquery-ui-1.10.4/ui/jquery.ui.autocomplete.js @ 7:764f47286679

(none)
author jurzua
date Wed, 29 Oct 2014 14:28:34 +0000
parents
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
7
jurzua
parents:
diff changeset
1 /*!
jurzua
parents:
diff changeset
2 * jQuery UI Autocomplete 1.10.4
jurzua
parents:
diff changeset
3 * http://jqueryui.com
jurzua
parents:
diff changeset
4 *
jurzua
parents:
diff changeset
5 * Copyright 2014 jQuery Foundation and other contributors
jurzua
parents:
diff changeset
6 * Released under the MIT license.
jurzua
parents:
diff changeset
7 * http://jquery.org/license
jurzua
parents:
diff changeset
8 *
jurzua
parents:
diff changeset
9 * http://api.jqueryui.com/autocomplete/
jurzua
parents:
diff changeset
10 *
jurzua
parents:
diff changeset
11 * Depends:
jurzua
parents:
diff changeset
12 * jquery.ui.core.js
jurzua
parents:
diff changeset
13 * jquery.ui.widget.js
jurzua
parents:
diff changeset
14 * jquery.ui.position.js
jurzua
parents:
diff changeset
15 * jquery.ui.menu.js
jurzua
parents:
diff changeset
16 */
jurzua
parents:
diff changeset
17 (function( $, undefined ) {
jurzua
parents:
diff changeset
18
jurzua
parents:
diff changeset
19 $.widget( "ui.autocomplete", {
jurzua
parents:
diff changeset
20 version: "1.10.4",
jurzua
parents:
diff changeset
21 defaultElement: "<input>",
jurzua
parents:
diff changeset
22 options: {
jurzua
parents:
diff changeset
23 appendTo: null,
jurzua
parents:
diff changeset
24 autoFocus: false,
jurzua
parents:
diff changeset
25 delay: 300,
jurzua
parents:
diff changeset
26 minLength: 1,
jurzua
parents:
diff changeset
27 position: {
jurzua
parents:
diff changeset
28 my: "left top",
jurzua
parents:
diff changeset
29 at: "left bottom",
jurzua
parents:
diff changeset
30 collision: "none"
jurzua
parents:
diff changeset
31 },
jurzua
parents:
diff changeset
32 source: null,
jurzua
parents:
diff changeset
33
jurzua
parents:
diff changeset
34 // callbacks
jurzua
parents:
diff changeset
35 change: null,
jurzua
parents:
diff changeset
36 close: null,
jurzua
parents:
diff changeset
37 focus: null,
jurzua
parents:
diff changeset
38 open: null,
jurzua
parents:
diff changeset
39 response: null,
jurzua
parents:
diff changeset
40 search: null,
jurzua
parents:
diff changeset
41 select: null
jurzua
parents:
diff changeset
42 },
jurzua
parents:
diff changeset
43
jurzua
parents:
diff changeset
44 requestIndex: 0,
jurzua
parents:
diff changeset
45 pending: 0,
jurzua
parents:
diff changeset
46
jurzua
parents:
diff changeset
47 _create: function() {
jurzua
parents:
diff changeset
48 // Some browsers only repeat keydown events, not keypress events,
jurzua
parents:
diff changeset
49 // so we use the suppressKeyPress flag to determine if we've already
jurzua
parents:
diff changeset
50 // handled the keydown event. #7269
jurzua
parents:
diff changeset
51 // Unfortunately the code for & in keypress is the same as the up arrow,
jurzua
parents:
diff changeset
52 // so we use the suppressKeyPressRepeat flag to avoid handling keypress
jurzua
parents:
diff changeset
53 // events when we know the keydown event was used to modify the
jurzua
parents:
diff changeset
54 // search term. #7799
jurzua
parents:
diff changeset
55 var suppressKeyPress, suppressKeyPressRepeat, suppressInput,
jurzua
parents:
diff changeset
56 nodeName = this.element[0].nodeName.toLowerCase(),
jurzua
parents:
diff changeset
57 isTextarea = nodeName === "textarea",
jurzua
parents:
diff changeset
58 isInput = nodeName === "input";
jurzua
parents:
diff changeset
59
jurzua
parents:
diff changeset
60 this.isMultiLine =
jurzua
parents:
diff changeset
61 // Textareas are always multi-line
jurzua
parents:
diff changeset
62 isTextarea ? true :
jurzua
parents:
diff changeset
63 // Inputs are always single-line, even if inside a contentEditable element
jurzua
parents:
diff changeset
64 // IE also treats inputs as contentEditable
jurzua
parents:
diff changeset
65 isInput ? false :
jurzua
parents:
diff changeset
66 // All other element types are determined by whether or not they're contentEditable
jurzua
parents:
diff changeset
67 this.element.prop( "isContentEditable" );
jurzua
parents:
diff changeset
68
jurzua
parents:
diff changeset
69 this.valueMethod = this.element[ isTextarea || isInput ? "val" : "text" ];
jurzua
parents:
diff changeset
70 this.isNewMenu = true;
jurzua
parents:
diff changeset
71
jurzua
parents:
diff changeset
72 this.element
jurzua
parents:
diff changeset
73 .addClass( "ui-autocomplete-input" )
jurzua
parents:
diff changeset
74 .attr( "autocomplete", "off" );
jurzua
parents:
diff changeset
75
jurzua
parents:
diff changeset
76 this._on( this.element, {
jurzua
parents:
diff changeset
77 keydown: function( event ) {
jurzua
parents:
diff changeset
78 if ( this.element.prop( "readOnly" ) ) {
jurzua
parents:
diff changeset
79 suppressKeyPress = true;
jurzua
parents:
diff changeset
80 suppressInput = true;
jurzua
parents:
diff changeset
81 suppressKeyPressRepeat = true;
jurzua
parents:
diff changeset
82 return;
jurzua
parents:
diff changeset
83 }
jurzua
parents:
diff changeset
84
jurzua
parents:
diff changeset
85 suppressKeyPress = false;
jurzua
parents:
diff changeset
86 suppressInput = false;
jurzua
parents:
diff changeset
87 suppressKeyPressRepeat = false;
jurzua
parents:
diff changeset
88 var keyCode = $.ui.keyCode;
jurzua
parents:
diff changeset
89 switch( event.keyCode ) {
jurzua
parents:
diff changeset
90 case keyCode.PAGE_UP:
jurzua
parents:
diff changeset
91 suppressKeyPress = true;
jurzua
parents:
diff changeset
92 this._move( "previousPage", event );
jurzua
parents:
diff changeset
93 break;
jurzua
parents:
diff changeset
94 case keyCode.PAGE_DOWN:
jurzua
parents:
diff changeset
95 suppressKeyPress = true;
jurzua
parents:
diff changeset
96 this._move( "nextPage", event );
jurzua
parents:
diff changeset
97 break;
jurzua
parents:
diff changeset
98 case keyCode.UP:
jurzua
parents:
diff changeset
99 suppressKeyPress = true;
jurzua
parents:
diff changeset
100 this._keyEvent( "previous", event );
jurzua
parents:
diff changeset
101 break;
jurzua
parents:
diff changeset
102 case keyCode.DOWN:
jurzua
parents:
diff changeset
103 suppressKeyPress = true;
jurzua
parents:
diff changeset
104 this._keyEvent( "next", event );
jurzua
parents:
diff changeset
105 break;
jurzua
parents:
diff changeset
106 case keyCode.ENTER:
jurzua
parents:
diff changeset
107 case keyCode.NUMPAD_ENTER:
jurzua
parents:
diff changeset
108 // when menu is open and has focus
jurzua
parents:
diff changeset
109 if ( this.menu.active ) {
jurzua
parents:
diff changeset
110 // #6055 - Opera still allows the keypress to occur
jurzua
parents:
diff changeset
111 // which causes forms to submit
jurzua
parents:
diff changeset
112 suppressKeyPress = true;
jurzua
parents:
diff changeset
113 event.preventDefault();
jurzua
parents:
diff changeset
114 this.menu.select( event );
jurzua
parents:
diff changeset
115 }
jurzua
parents:
diff changeset
116 break;
jurzua
parents:
diff changeset
117 case keyCode.TAB:
jurzua
parents:
diff changeset
118 if ( this.menu.active ) {
jurzua
parents:
diff changeset
119 this.menu.select( event );
jurzua
parents:
diff changeset
120 }
jurzua
parents:
diff changeset
121 break;
jurzua
parents:
diff changeset
122 case keyCode.ESCAPE:
jurzua
parents:
diff changeset
123 if ( this.menu.element.is( ":visible" ) ) {
jurzua
parents:
diff changeset
124 this._value( this.term );
jurzua
parents:
diff changeset
125 this.close( event );
jurzua
parents:
diff changeset
126 // Different browsers have different default behavior for escape
jurzua
parents:
diff changeset
127 // Single press can mean undo or clear
jurzua
parents:
diff changeset
128 // Double press in IE means clear the whole form
jurzua
parents:
diff changeset
129 event.preventDefault();
jurzua
parents:
diff changeset
130 }
jurzua
parents:
diff changeset
131 break;
jurzua
parents:
diff changeset
132 default:
jurzua
parents:
diff changeset
133 suppressKeyPressRepeat = true;
jurzua
parents:
diff changeset
134 // search timeout should be triggered before the input value is changed
jurzua
parents:
diff changeset
135 this._searchTimeout( event );
jurzua
parents:
diff changeset
136 break;
jurzua
parents:
diff changeset
137 }
jurzua
parents:
diff changeset
138 },
jurzua
parents:
diff changeset
139 keypress: function( event ) {
jurzua
parents:
diff changeset
140 if ( suppressKeyPress ) {
jurzua
parents:
diff changeset
141 suppressKeyPress = false;
jurzua
parents:
diff changeset
142 if ( !this.isMultiLine || this.menu.element.is( ":visible" ) ) {
jurzua
parents:
diff changeset
143 event.preventDefault();
jurzua
parents:
diff changeset
144 }
jurzua
parents:
diff changeset
145 return;
jurzua
parents:
diff changeset
146 }
jurzua
parents:
diff changeset
147 if ( suppressKeyPressRepeat ) {
jurzua
parents:
diff changeset
148 return;
jurzua
parents:
diff changeset
149 }
jurzua
parents:
diff changeset
150
jurzua
parents:
diff changeset
151 // replicate some key handlers to allow them to repeat in Firefox and Opera
jurzua
parents:
diff changeset
152 var keyCode = $.ui.keyCode;
jurzua
parents:
diff changeset
153 switch( event.keyCode ) {
jurzua
parents:
diff changeset
154 case keyCode.PAGE_UP:
jurzua
parents:
diff changeset
155 this._move( "previousPage", event );
jurzua
parents:
diff changeset
156 break;
jurzua
parents:
diff changeset
157 case keyCode.PAGE_DOWN:
jurzua
parents:
diff changeset
158 this._move( "nextPage", event );
jurzua
parents:
diff changeset
159 break;
jurzua
parents:
diff changeset
160 case keyCode.UP:
jurzua
parents:
diff changeset
161 this._keyEvent( "previous", event );
jurzua
parents:
diff changeset
162 break;
jurzua
parents:
diff changeset
163 case keyCode.DOWN:
jurzua
parents:
diff changeset
164 this._keyEvent( "next", event );
jurzua
parents:
diff changeset
165 break;
jurzua
parents:
diff changeset
166 }
jurzua
parents:
diff changeset
167 },
jurzua
parents:
diff changeset
168 input: function( event ) {
jurzua
parents:
diff changeset
169 if ( suppressInput ) {
jurzua
parents:
diff changeset
170 suppressInput = false;
jurzua
parents:
diff changeset
171 event.preventDefault();
jurzua
parents:
diff changeset
172 return;
jurzua
parents:
diff changeset
173 }
jurzua
parents:
diff changeset
174 this._searchTimeout( event );
jurzua
parents:
diff changeset
175 },
jurzua
parents:
diff changeset
176 focus: function() {
jurzua
parents:
diff changeset
177 this.selectedItem = null;
jurzua
parents:
diff changeset
178 this.previous = this._value();
jurzua
parents:
diff changeset
179 },
jurzua
parents:
diff changeset
180 blur: function( event ) {
jurzua
parents:
diff changeset
181 if ( this.cancelBlur ) {
jurzua
parents:
diff changeset
182 delete this.cancelBlur;
jurzua
parents:
diff changeset
183 return;
jurzua
parents:
diff changeset
184 }
jurzua
parents:
diff changeset
185
jurzua
parents:
diff changeset
186 clearTimeout( this.searching );
jurzua
parents:
diff changeset
187 this.close( event );
jurzua
parents:
diff changeset
188 this._change( event );
jurzua
parents:
diff changeset
189 }
jurzua
parents:
diff changeset
190 });
jurzua
parents:
diff changeset
191
jurzua
parents:
diff changeset
192 this._initSource();
jurzua
parents:
diff changeset
193 this.menu = $( "<ul>" )
jurzua
parents:
diff changeset
194 .addClass( "ui-autocomplete ui-front" )
jurzua
parents:
diff changeset
195 .appendTo( this._appendTo() )
jurzua
parents:
diff changeset
196 .menu({
jurzua
parents:
diff changeset
197 // disable ARIA support, the live region takes care of that
jurzua
parents:
diff changeset
198 role: null
jurzua
parents:
diff changeset
199 })
jurzua
parents:
diff changeset
200 .hide()
jurzua
parents:
diff changeset
201 .data( "ui-menu" );
jurzua
parents:
diff changeset
202
jurzua
parents:
diff changeset
203 this._on( this.menu.element, {
jurzua
parents:
diff changeset
204 mousedown: function( event ) {
jurzua
parents:
diff changeset
205 // prevent moving focus out of the text field
jurzua
parents:
diff changeset
206 event.preventDefault();
jurzua
parents:
diff changeset
207
jurzua
parents:
diff changeset
208 // IE doesn't prevent moving focus even with event.preventDefault()
jurzua
parents:
diff changeset
209 // so we set a flag to know when we should ignore the blur event
jurzua
parents:
diff changeset
210 this.cancelBlur = true;
jurzua
parents:
diff changeset
211 this._delay(function() {
jurzua
parents:
diff changeset
212 delete this.cancelBlur;
jurzua
parents:
diff changeset
213 });
jurzua
parents:
diff changeset
214
jurzua
parents:
diff changeset
215 // clicking on the scrollbar causes focus to shift to the body
jurzua
parents:
diff changeset
216 // but we can't detect a mouseup or a click immediately afterward
jurzua
parents:
diff changeset
217 // so we have to track the next mousedown and close the menu if
jurzua
parents:
diff changeset
218 // the user clicks somewhere outside of the autocomplete
jurzua
parents:
diff changeset
219 var menuElement = this.menu.element[ 0 ];
jurzua
parents:
diff changeset
220 if ( !$( event.target ).closest( ".ui-menu-item" ).length ) {
jurzua
parents:
diff changeset
221 this._delay(function() {
jurzua
parents:
diff changeset
222 var that = this;
jurzua
parents:
diff changeset
223 this.document.one( "mousedown", function( event ) {
jurzua
parents:
diff changeset
224 if ( event.target !== that.element[ 0 ] &&
jurzua
parents:
diff changeset
225 event.target !== menuElement &&
jurzua
parents:
diff changeset
226 !$.contains( menuElement, event.target ) ) {
jurzua
parents:
diff changeset
227 that.close();
jurzua
parents:
diff changeset
228 }
jurzua
parents:
diff changeset
229 });
jurzua
parents:
diff changeset
230 });
jurzua
parents:
diff changeset
231 }
jurzua
parents:
diff changeset
232 },
jurzua
parents:
diff changeset
233 menufocus: function( event, ui ) {
jurzua
parents:
diff changeset
234 // support: Firefox
jurzua
parents:
diff changeset
235 // Prevent accidental activation of menu items in Firefox (#7024 #9118)
jurzua
parents:
diff changeset
236 if ( this.isNewMenu ) {
jurzua
parents:
diff changeset
237 this.isNewMenu = false;
jurzua
parents:
diff changeset
238 if ( event.originalEvent && /^mouse/.test( event.originalEvent.type ) ) {
jurzua
parents:
diff changeset
239 this.menu.blur();
jurzua
parents:
diff changeset
240
jurzua
parents:
diff changeset
241 this.document.one( "mousemove", function() {
jurzua
parents:
diff changeset
242 $( event.target ).trigger( event.originalEvent );
jurzua
parents:
diff changeset
243 });
jurzua
parents:
diff changeset
244
jurzua
parents:
diff changeset
245 return;
jurzua
parents:
diff changeset
246 }
jurzua
parents:
diff changeset
247 }
jurzua
parents:
diff changeset
248
jurzua
parents:
diff changeset
249 var item = ui.item.data( "ui-autocomplete-item" );
jurzua
parents:
diff changeset
250 if ( false !== this._trigger( "focus", event, { item: item } ) ) {
jurzua
parents:
diff changeset
251 // use value to match what will end up in the input, if it was a key event
jurzua
parents:
diff changeset
252 if ( event.originalEvent && /^key/.test( event.originalEvent.type ) ) {
jurzua
parents:
diff changeset
253 this._value( item.value );
jurzua
parents:
diff changeset
254 }
jurzua
parents:
diff changeset
255 } else {
jurzua
parents:
diff changeset
256 // Normally the input is populated with the item's value as the
jurzua
parents:
diff changeset
257 // menu is navigated, causing screen readers to notice a change and
jurzua
parents:
diff changeset
258 // announce the item. Since the focus event was canceled, this doesn't
jurzua
parents:
diff changeset
259 // happen, so we update the live region so that screen readers can
jurzua
parents:
diff changeset
260 // still notice the change and announce it.
jurzua
parents:
diff changeset
261 this.liveRegion.text( item.value );
jurzua
parents:
diff changeset
262 }
jurzua
parents:
diff changeset
263 },
jurzua
parents:
diff changeset
264 menuselect: function( event, ui ) {
jurzua
parents:
diff changeset
265 var item = ui.item.data( "ui-autocomplete-item" ),
jurzua
parents:
diff changeset
266 previous = this.previous;
jurzua
parents:
diff changeset
267
jurzua
parents:
diff changeset
268 // only trigger when focus was lost (click on menu)
jurzua
parents:
diff changeset
269 if ( this.element[0] !== this.document[0].activeElement ) {
jurzua
parents:
diff changeset
270 this.element.focus();
jurzua
parents:
diff changeset
271 this.previous = previous;
jurzua
parents:
diff changeset
272 // #6109 - IE triggers two focus events and the second
jurzua
parents:
diff changeset
273 // is asynchronous, so we need to reset the previous
jurzua
parents:
diff changeset
274 // term synchronously and asynchronously :-(
jurzua
parents:
diff changeset
275 this._delay(function() {
jurzua
parents:
diff changeset
276 this.previous = previous;
jurzua
parents:
diff changeset
277 this.selectedItem = item;
jurzua
parents:
diff changeset
278 });
jurzua
parents:
diff changeset
279 }
jurzua
parents:
diff changeset
280
jurzua
parents:
diff changeset
281 if ( false !== this._trigger( "select", event, { item: item } ) ) {
jurzua
parents:
diff changeset
282 this._value( item.value );
jurzua
parents:
diff changeset
283 }
jurzua
parents:
diff changeset
284 // reset the term after the select event
jurzua
parents:
diff changeset
285 // this allows custom select handling to work properly
jurzua
parents:
diff changeset
286 this.term = this._value();
jurzua
parents:
diff changeset
287
jurzua
parents:
diff changeset
288 this.close( event );
jurzua
parents:
diff changeset
289 this.selectedItem = item;
jurzua
parents:
diff changeset
290 }
jurzua
parents:
diff changeset
291 });
jurzua
parents:
diff changeset
292
jurzua
parents:
diff changeset
293 this.liveRegion = $( "<span>", {
jurzua
parents:
diff changeset
294 role: "status",
jurzua
parents:
diff changeset
295 "aria-live": "polite"
jurzua
parents:
diff changeset
296 })
jurzua
parents:
diff changeset
297 .addClass( "ui-helper-hidden-accessible" )
jurzua
parents:
diff changeset
298 .insertBefore( this.element );
jurzua
parents:
diff changeset
299
jurzua
parents:
diff changeset
300 // turning off autocomplete prevents the browser from remembering the
jurzua
parents:
diff changeset
301 // value when navigating through history, so we re-enable autocomplete
jurzua
parents:
diff changeset
302 // if the page is unloaded before the widget is destroyed. #7790
jurzua
parents:
diff changeset
303 this._on( this.window, {
jurzua
parents:
diff changeset
304 beforeunload: function() {
jurzua
parents:
diff changeset
305 this.element.removeAttr( "autocomplete" );
jurzua
parents:
diff changeset
306 }
jurzua
parents:
diff changeset
307 });
jurzua
parents:
diff changeset
308 },
jurzua
parents:
diff changeset
309
jurzua
parents:
diff changeset
310 _destroy: function() {
jurzua
parents:
diff changeset
311 clearTimeout( this.searching );
jurzua
parents:
diff changeset
312 this.element
jurzua
parents:
diff changeset
313 .removeClass( "ui-autocomplete-input" )
jurzua
parents:
diff changeset
314 .removeAttr( "autocomplete" );
jurzua
parents:
diff changeset
315 this.menu.element.remove();
jurzua
parents:
diff changeset
316 this.liveRegion.remove();
jurzua
parents:
diff changeset
317 },
jurzua
parents:
diff changeset
318
jurzua
parents:
diff changeset
319 _setOption: function( key, value ) {
jurzua
parents:
diff changeset
320 this._super( key, value );
jurzua
parents:
diff changeset
321 if ( key === "source" ) {
jurzua
parents:
diff changeset
322 this._initSource();
jurzua
parents:
diff changeset
323 }
jurzua
parents:
diff changeset
324 if ( key === "appendTo" ) {
jurzua
parents:
diff changeset
325 this.menu.element.appendTo( this._appendTo() );
jurzua
parents:
diff changeset
326 }
jurzua
parents:
diff changeset
327 if ( key === "disabled" && value && this.xhr ) {
jurzua
parents:
diff changeset
328 this.xhr.abort();
jurzua
parents:
diff changeset
329 }
jurzua
parents:
diff changeset
330 },
jurzua
parents:
diff changeset
331
jurzua
parents:
diff changeset
332 _appendTo: function() {
jurzua
parents:
diff changeset
333 var element = this.options.appendTo;
jurzua
parents:
diff changeset
334
jurzua
parents:
diff changeset
335 if ( element ) {
jurzua
parents:
diff changeset
336 element = element.jquery || element.nodeType ?
jurzua
parents:
diff changeset
337 $( element ) :
jurzua
parents:
diff changeset
338 this.document.find( element ).eq( 0 );
jurzua
parents:
diff changeset
339 }
jurzua
parents:
diff changeset
340
jurzua
parents:
diff changeset
341 if ( !element ) {
jurzua
parents:
diff changeset
342 element = this.element.closest( ".ui-front" );
jurzua
parents:
diff changeset
343 }
jurzua
parents:
diff changeset
344
jurzua
parents:
diff changeset
345 if ( !element.length ) {
jurzua
parents:
diff changeset
346 element = this.document[0].body;
jurzua
parents:
diff changeset
347 }
jurzua
parents:
diff changeset
348
jurzua
parents:
diff changeset
349 return element;
jurzua
parents:
diff changeset
350 },
jurzua
parents:
diff changeset
351
jurzua
parents:
diff changeset
352 _initSource: function() {
jurzua
parents:
diff changeset
353 var array, url,
jurzua
parents:
diff changeset
354 that = this;
jurzua
parents:
diff changeset
355 if ( $.isArray(this.options.source) ) {
jurzua
parents:
diff changeset
356 array = this.options.source;
jurzua
parents:
diff changeset
357 this.source = function( request, response ) {
jurzua
parents:
diff changeset
358 response( $.ui.autocomplete.filter( array, request.term ) );
jurzua
parents:
diff changeset
359 };
jurzua
parents:
diff changeset
360 } else if ( typeof this.options.source === "string" ) {
jurzua
parents:
diff changeset
361 url = this.options.source;
jurzua
parents:
diff changeset
362 this.source = function( request, response ) {
jurzua
parents:
diff changeset
363 if ( that.xhr ) {
jurzua
parents:
diff changeset
364 that.xhr.abort();
jurzua
parents:
diff changeset
365 }
jurzua
parents:
diff changeset
366 that.xhr = $.ajax({
jurzua
parents:
diff changeset
367 url: url,
jurzua
parents:
diff changeset
368 data: request,
jurzua
parents:
diff changeset
369 dataType: "json",
jurzua
parents:
diff changeset
370 success: function( data ) {
jurzua
parents:
diff changeset
371 response( data );
jurzua
parents:
diff changeset
372 },
jurzua
parents:
diff changeset
373 error: function() {
jurzua
parents:
diff changeset
374 response( [] );
jurzua
parents:
diff changeset
375 }
jurzua
parents:
diff changeset
376 });
jurzua
parents:
diff changeset
377 };
jurzua
parents:
diff changeset
378 } else {
jurzua
parents:
diff changeset
379 this.source = this.options.source;
jurzua
parents:
diff changeset
380 }
jurzua
parents:
diff changeset
381 },
jurzua
parents:
diff changeset
382
jurzua
parents:
diff changeset
383 _searchTimeout: function( event ) {
jurzua
parents:
diff changeset
384 clearTimeout( this.searching );
jurzua
parents:
diff changeset
385 this.searching = this._delay(function() {
jurzua
parents:
diff changeset
386 // only search if the value has changed
jurzua
parents:
diff changeset
387 if ( this.term !== this._value() ) {
jurzua
parents:
diff changeset
388 this.selectedItem = null;
jurzua
parents:
diff changeset
389 this.search( null, event );
jurzua
parents:
diff changeset
390 }
jurzua
parents:
diff changeset
391 }, this.options.delay );
jurzua
parents:
diff changeset
392 },
jurzua
parents:
diff changeset
393
jurzua
parents:
diff changeset
394 search: function( value, event ) {
jurzua
parents:
diff changeset
395 value = value != null ? value : this._value();
jurzua
parents:
diff changeset
396
jurzua
parents:
diff changeset
397 // always save the actual value, not the one passed as an argument
jurzua
parents:
diff changeset
398 this.term = this._value();
jurzua
parents:
diff changeset
399
jurzua
parents:
diff changeset
400 if ( value.length < this.options.minLength ) {
jurzua
parents:
diff changeset
401 return this.close( event );
jurzua
parents:
diff changeset
402 }
jurzua
parents:
diff changeset
403
jurzua
parents:
diff changeset
404 if ( this._trigger( "search", event ) === false ) {
jurzua
parents:
diff changeset
405 return;
jurzua
parents:
diff changeset
406 }
jurzua
parents:
diff changeset
407
jurzua
parents:
diff changeset
408 return this._search( value );
jurzua
parents:
diff changeset
409 },
jurzua
parents:
diff changeset
410
jurzua
parents:
diff changeset
411 _search: function( value ) {
jurzua
parents:
diff changeset
412 this.pending++;
jurzua
parents:
diff changeset
413 this.element.addClass( "ui-autocomplete-loading" );
jurzua
parents:
diff changeset
414 this.cancelSearch = false;
jurzua
parents:
diff changeset
415
jurzua
parents:
diff changeset
416 this.source( { term: value }, this._response() );
jurzua
parents:
diff changeset
417 },
jurzua
parents:
diff changeset
418
jurzua
parents:
diff changeset
419 _response: function() {
jurzua
parents:
diff changeset
420 var index = ++this.requestIndex;
jurzua
parents:
diff changeset
421
jurzua
parents:
diff changeset
422 return $.proxy(function( content ) {
jurzua
parents:
diff changeset
423 if ( index === this.requestIndex ) {
jurzua
parents:
diff changeset
424 this.__response( content );
jurzua
parents:
diff changeset
425 }
jurzua
parents:
diff changeset
426
jurzua
parents:
diff changeset
427 this.pending--;
jurzua
parents:
diff changeset
428 if ( !this.pending ) {
jurzua
parents:
diff changeset
429 this.element.removeClass( "ui-autocomplete-loading" );
jurzua
parents:
diff changeset
430 }
jurzua
parents:
diff changeset
431 }, this );
jurzua
parents:
diff changeset
432 },
jurzua
parents:
diff changeset
433
jurzua
parents:
diff changeset
434 __response: function( content ) {
jurzua
parents:
diff changeset
435 if ( content ) {
jurzua
parents:
diff changeset
436 content = this._normalize( content );
jurzua
parents:
diff changeset
437 }
jurzua
parents:
diff changeset
438 this._trigger( "response", null, { content: content } );
jurzua
parents:
diff changeset
439 if ( !this.options.disabled && content && content.length && !this.cancelSearch ) {
jurzua
parents:
diff changeset
440 this._suggest( content );
jurzua
parents:
diff changeset
441 this._trigger( "open" );
jurzua
parents:
diff changeset
442 } else {
jurzua
parents:
diff changeset
443 // use ._close() instead of .close() so we don't cancel future searches
jurzua
parents:
diff changeset
444 this._close();
jurzua
parents:
diff changeset
445 }
jurzua
parents:
diff changeset
446 },
jurzua
parents:
diff changeset
447
jurzua
parents:
diff changeset
448 close: function( event ) {
jurzua
parents:
diff changeset
449 this.cancelSearch = true;
jurzua
parents:
diff changeset
450 this._close( event );
jurzua
parents:
diff changeset
451 },
jurzua
parents:
diff changeset
452
jurzua
parents:
diff changeset
453 _close: function( event ) {
jurzua
parents:
diff changeset
454 if ( this.menu.element.is( ":visible" ) ) {
jurzua
parents:
diff changeset
455 this.menu.element.hide();
jurzua
parents:
diff changeset
456 this.menu.blur();
jurzua
parents:
diff changeset
457 this.isNewMenu = true;
jurzua
parents:
diff changeset
458 this._trigger( "close", event );
jurzua
parents:
diff changeset
459 }
jurzua
parents:
diff changeset
460 },
jurzua
parents:
diff changeset
461
jurzua
parents:
diff changeset
462 _change: function( event ) {
jurzua
parents:
diff changeset
463 if ( this.previous !== this._value() ) {
jurzua
parents:
diff changeset
464 this._trigger( "change", event, { item: this.selectedItem } );
jurzua
parents:
diff changeset
465 }
jurzua
parents:
diff changeset
466 },
jurzua
parents:
diff changeset
467
jurzua
parents:
diff changeset
468 _normalize: function( items ) {
jurzua
parents:
diff changeset
469 // assume all items have the right format when the first item is complete
jurzua
parents:
diff changeset
470 if ( items.length && items[0].label && items[0].value ) {
jurzua
parents:
diff changeset
471 return items;
jurzua
parents:
diff changeset
472 }
jurzua
parents:
diff changeset
473 return $.map( items, function( item ) {
jurzua
parents:
diff changeset
474 if ( typeof item === "string" ) {
jurzua
parents:
diff changeset
475 return {
jurzua
parents:
diff changeset
476 label: item,
jurzua
parents:
diff changeset
477 value: item
jurzua
parents:
diff changeset
478 };
jurzua
parents:
diff changeset
479 }
jurzua
parents:
diff changeset
480 return $.extend({
jurzua
parents:
diff changeset
481 label: item.label || item.value,
jurzua
parents:
diff changeset
482 value: item.value || item.label
jurzua
parents:
diff changeset
483 }, item );
jurzua
parents:
diff changeset
484 });
jurzua
parents:
diff changeset
485 },
jurzua
parents:
diff changeset
486
jurzua
parents:
diff changeset
487 _suggest: function( items ) {
jurzua
parents:
diff changeset
488 var ul = this.menu.element.empty();
jurzua
parents:
diff changeset
489 this._renderMenu( ul, items );
jurzua
parents:
diff changeset
490 this.isNewMenu = true;
jurzua
parents:
diff changeset
491 this.menu.refresh();
jurzua
parents:
diff changeset
492
jurzua
parents:
diff changeset
493 // size and position menu
jurzua
parents:
diff changeset
494 ul.show();
jurzua
parents:
diff changeset
495 this._resizeMenu();
jurzua
parents:
diff changeset
496 ul.position( $.extend({
jurzua
parents:
diff changeset
497 of: this.element
jurzua
parents:
diff changeset
498 }, this.options.position ));
jurzua
parents:
diff changeset
499
jurzua
parents:
diff changeset
500 if ( this.options.autoFocus ) {
jurzua
parents:
diff changeset
501 this.menu.next();
jurzua
parents:
diff changeset
502 }
jurzua
parents:
diff changeset
503 },
jurzua
parents:
diff changeset
504
jurzua
parents:
diff changeset
505 _resizeMenu: function() {
jurzua
parents:
diff changeset
506 var ul = this.menu.element;
jurzua
parents:
diff changeset
507 ul.outerWidth( Math.max(
jurzua
parents:
diff changeset
508 // Firefox wraps long text (possibly a rounding bug)
jurzua
parents:
diff changeset
509 // so we add 1px to avoid the wrapping (#7513)
jurzua
parents:
diff changeset
510 ul.width( "" ).outerWidth() + 1,
jurzua
parents:
diff changeset
511 this.element.outerWidth()
jurzua
parents:
diff changeset
512 ) );
jurzua
parents:
diff changeset
513 },
jurzua
parents:
diff changeset
514
jurzua
parents:
diff changeset
515 _renderMenu: function( ul, items ) {
jurzua
parents:
diff changeset
516 var that = this;
jurzua
parents:
diff changeset
517 $.each( items, function( index, item ) {
jurzua
parents:
diff changeset
518 that._renderItemData( ul, item );
jurzua
parents:
diff changeset
519 });
jurzua
parents:
diff changeset
520 },
jurzua
parents:
diff changeset
521
jurzua
parents:
diff changeset
522 _renderItemData: function( ul, item ) {
jurzua
parents:
diff changeset
523 return this._renderItem( ul, item ).data( "ui-autocomplete-item", item );
jurzua
parents:
diff changeset
524 },
jurzua
parents:
diff changeset
525
jurzua
parents:
diff changeset
526 _renderItem: function( ul, item ) {
jurzua
parents:
diff changeset
527 return $( "<li>" )
jurzua
parents:
diff changeset
528 .append( $( "<a>" ).text( item.label ) )
jurzua
parents:
diff changeset
529 .appendTo( ul );
jurzua
parents:
diff changeset
530 },
jurzua
parents:
diff changeset
531
jurzua
parents:
diff changeset
532 _move: function( direction, event ) {
jurzua
parents:
diff changeset
533 if ( !this.menu.element.is( ":visible" ) ) {
jurzua
parents:
diff changeset
534 this.search( null, event );
jurzua
parents:
diff changeset
535 return;
jurzua
parents:
diff changeset
536 }
jurzua
parents:
diff changeset
537 if ( this.menu.isFirstItem() && /^previous/.test( direction ) ||
jurzua
parents:
diff changeset
538 this.menu.isLastItem() && /^next/.test( direction ) ) {
jurzua
parents:
diff changeset
539 this._value( this.term );
jurzua
parents:
diff changeset
540 this.menu.blur();
jurzua
parents:
diff changeset
541 return;
jurzua
parents:
diff changeset
542 }
jurzua
parents:
diff changeset
543 this.menu[ direction ]( event );
jurzua
parents:
diff changeset
544 },
jurzua
parents:
diff changeset
545
jurzua
parents:
diff changeset
546 widget: function() {
jurzua
parents:
diff changeset
547 return this.menu.element;
jurzua
parents:
diff changeset
548 },
jurzua
parents:
diff changeset
549
jurzua
parents:
diff changeset
550 _value: function() {
jurzua
parents:
diff changeset
551 return this.valueMethod.apply( this.element, arguments );
jurzua
parents:
diff changeset
552 },
jurzua
parents:
diff changeset
553
jurzua
parents:
diff changeset
554 _keyEvent: function( keyEvent, event ) {
jurzua
parents:
diff changeset
555 if ( !this.isMultiLine || this.menu.element.is( ":visible" ) ) {
jurzua
parents:
diff changeset
556 this._move( keyEvent, event );
jurzua
parents:
diff changeset
557
jurzua
parents:
diff changeset
558 // prevents moving cursor to beginning/end of the text field in some browsers
jurzua
parents:
diff changeset
559 event.preventDefault();
jurzua
parents:
diff changeset
560 }
jurzua
parents:
diff changeset
561 }
jurzua
parents:
diff changeset
562 });
jurzua
parents:
diff changeset
563
jurzua
parents:
diff changeset
564 $.extend( $.ui.autocomplete, {
jurzua
parents:
diff changeset
565 escapeRegex: function( value ) {
jurzua
parents:
diff changeset
566 return value.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g, "\\$&");
jurzua
parents:
diff changeset
567 },
jurzua
parents:
diff changeset
568 filter: function(array, term) {
jurzua
parents:
diff changeset
569 var matcher = new RegExp( $.ui.autocomplete.escapeRegex(term), "i" );
jurzua
parents:
diff changeset
570 return $.grep( array, function(value) {
jurzua
parents:
diff changeset
571 return matcher.test( value.label || value.value || value );
jurzua
parents:
diff changeset
572 });
jurzua
parents:
diff changeset
573 }
jurzua
parents:
diff changeset
574 });
jurzua
parents:
diff changeset
575
jurzua
parents:
diff changeset
576
jurzua
parents:
diff changeset
577 // live region extension, adding a `messages` option
jurzua
parents:
diff changeset
578 // NOTE: This is an experimental API. We are still investigating
jurzua
parents:
diff changeset
579 // a full solution for string manipulation and internationalization.
jurzua
parents:
diff changeset
580 $.widget( "ui.autocomplete", $.ui.autocomplete, {
jurzua
parents:
diff changeset
581 options: {
jurzua
parents:
diff changeset
582 messages: {
jurzua
parents:
diff changeset
583 noResults: "No search results.",
jurzua
parents:
diff changeset
584 results: function( amount ) {
jurzua
parents:
diff changeset
585 return amount + ( amount > 1 ? " results are" : " result is" ) +
jurzua
parents:
diff changeset
586 " available, use up and down arrow keys to navigate.";
jurzua
parents:
diff changeset
587 }
jurzua
parents:
diff changeset
588 }
jurzua
parents:
diff changeset
589 },
jurzua
parents:
diff changeset
590
jurzua
parents:
diff changeset
591 __response: function( content ) {
jurzua
parents:
diff changeset
592 var message;
jurzua
parents:
diff changeset
593 this._superApply( arguments );
jurzua
parents:
diff changeset
594 if ( this.options.disabled || this.cancelSearch ) {
jurzua
parents:
diff changeset
595 return;
jurzua
parents:
diff changeset
596 }
jurzua
parents:
diff changeset
597 if ( content && content.length ) {
jurzua
parents:
diff changeset
598 message = this.options.messages.results( content.length );
jurzua
parents:
diff changeset
599 } else {
jurzua
parents:
diff changeset
600 message = this.options.messages.noResults;
jurzua
parents:
diff changeset
601 }
jurzua
parents:
diff changeset
602 this.liveRegion.text( message );
jurzua
parents:
diff changeset
603 }
jurzua
parents:
diff changeset
604 });
jurzua
parents:
diff changeset
605
jurzua
parents:
diff changeset
606 }( jQuery ));