Statistics
| Branch: | Revision:

blinker / firefox.plugin / data / jquery-ui.js @ e75d6b65

History | View | Annotate | Download (459.566 KB)

1 a03cd52e Thies Pfeiffer
/*! jQuery UI - v1.11.4 - 2015-03-11
2
* http://jqueryui.com
3
* Includes: core.js, widget.js, mouse.js, position.js, accordion.js, autocomplete.js, button.js, datepicker.js, dialog.js, draggable.js, droppable.js, effect.js, effect-blind.js, effect-bounce.js, effect-clip.js, effect-drop.js, effect-explode.js, effect-fade.js, effect-fold.js, effect-highlight.js, effect-puff.js, effect-pulsate.js, effect-scale.js, effect-shake.js, effect-size.js, effect-slide.js, effect-transfer.js, menu.js, progressbar.js, resizable.js, selectable.js, selectmenu.js, slider.js, sortable.js, spinner.js, tabs.js, tooltip.js
4
* Copyright 2015 jQuery Foundation and other contributors; Licensed MIT */
5
6
(function( factory ) {
7
        if ( typeof define === "function" && define.amd ) {
8
9
                // AMD. Register as an anonymous module.
10
                define([ "jquery" ], factory );
11
        } else {
12
13
                // Browser globals
14
                factory( jQuery );
15
        }
16
}(function( $ ) {
17
/*!
18
 * jQuery UI Core 1.11.4
19
 * http://jqueryui.com
20
 *
21
 * Copyright jQuery Foundation and other contributors
22
 * Released under the MIT license.
23
 * http://jquery.org/license
24
 *
25
 * http://api.jqueryui.com/category/ui-core/
26
 */
27
28
29
// $.ui might exist from components with no dependencies, e.g., $.ui.position
30
$.ui = $.ui || {};
31
32
$.extend( $.ui, {
33
        version: "1.11.4",
34
35
        keyCode: {
36
                BACKSPACE: 8,
37
                COMMA: 188,
38
                DELETE: 46,
39
                DOWN: 40,
40
                END: 35,
41
                ENTER: 13,
42
                ESCAPE: 27,
43
                HOME: 36,
44
                LEFT: 37,
45
                PAGE_DOWN: 34,
46
                PAGE_UP: 33,
47
                PERIOD: 190,
48
                RIGHT: 39,
49
                SPACE: 32,
50
                TAB: 9,
51
                UP: 38
52
        }
53
});
54
55
// plugins
56
$.fn.extend({
57
        scrollParent: function( includeHidden ) {
58
                var position = this.css( "position" ),
59
                        excludeStaticParent = position === "absolute",
60
                        overflowRegex = includeHidden ? /(auto|scroll|hidden)/ : /(auto|scroll)/,
61
                        scrollParent = this.parents().filter( function() {
62
                                var parent = $( this );
63
                                if ( excludeStaticParent && parent.css( "position" ) === "static" ) {
64
                                        return false;
65
                                }
66
                                return overflowRegex.test( parent.css( "overflow" ) + parent.css( "overflow-y" ) + parent.css( "overflow-x" ) );
67
                        }).eq( 0 );
68
69
                return position === "fixed" || !scrollParent.length ? $( this[ 0 ].ownerDocument || document ) : scrollParent;
70
        },
71
72
        uniqueId: (function() {
73
                var uuid = 0;
74
75
                return function() {
76
                        return this.each(function() {
77
                                if ( !this.id ) {
78
                                        this.id = "ui-id-" + ( ++uuid );
79
                                }
80
                        });
81
                };
82
        })(),
83
84
        removeUniqueId: function() {
85
                return this.each(function() {
86
                        if ( /^ui-id-\d+$/.test( this.id ) ) {
87
                                $( this ).removeAttr( "id" );
88
                        }
89
                });
90
        }
91
});
92
93
// selectors
94
function focusable( element, isTabIndexNotNaN ) {
95
        var map, mapName, img,
96
                nodeName = element.nodeName.toLowerCase();
97
        if ( "area" === nodeName ) {
98
                map = element.parentNode;
99
                mapName = map.name;
100
                if ( !element.href || !mapName || map.nodeName.toLowerCase() !== "map" ) {
101
                        return false;
102
                }
103
                img = $( "img[usemap='#" + mapName + "']" )[ 0 ];
104
                return !!img && visible( img );
105
        }
106
        return ( /^(input|select|textarea|button|object)$/.test( nodeName ) ?
107
                !element.disabled :
108
                "a" === nodeName ?
109
                        element.href || isTabIndexNotNaN :
110
                        isTabIndexNotNaN) &&
111
                // the element and all of its ancestors must be visible
112
                visible( element );
113
}
114
115
function visible( element ) {
116
        return $.expr.filters.visible( element ) &&
117
                !$( element ).parents().addBack().filter(function() {
118
                        return $.css( this, "visibility" ) === "hidden";
119
                }).length;
120
}
121
122
$.extend( $.expr[ ":" ], {
123
        data: $.expr.createPseudo ?
124
                $.expr.createPseudo(function( dataName ) {
125
                        return function( elem ) {
126
                                return !!$.data( elem, dataName );
127
                        };
128
                }) :
129
                // support: jQuery <1.8
130
                function( elem, i, match ) {
131
                        return !!$.data( elem, match[ 3 ] );
132
                },
133
134
        focusable: function( element ) {
135
                return focusable( element, !isNaN( $.attr( element, "tabindex" ) ) );
136
        },
137
138
        tabbable: function( element ) {
139
                var tabIndex = $.attr( element, "tabindex" ),
140
                        isTabIndexNaN = isNaN( tabIndex );
141
                return ( isTabIndexNaN || tabIndex >= 0 ) && focusable( element, !isTabIndexNaN );
142
        }
143
});
144
145
// support: jQuery <1.8
146
if ( !$( "<a>" ).outerWidth( 1 ).jquery ) {
147
        $.each( [ "Width", "Height" ], function( i, name ) {
148
                var side = name === "Width" ? [ "Left", "Right" ] : [ "Top", "Bottom" ],
149
                        type = name.toLowerCase(),
150
                        orig = {
151
                                innerWidth: $.fn.innerWidth,
152
                                innerHeight: $.fn.innerHeight,
153
                                outerWidth: $.fn.outerWidth,
154
                                outerHeight: $.fn.outerHeight
155
                        };
156
157
                function reduce( elem, size, border, margin ) {
158
                        $.each( side, function() {
159
                                size -= parseFloat( $.css( elem, "padding" + this ) ) || 0;
160
                                if ( border ) {
161
                                        size -= parseFloat( $.css( elem, "border" + this + "Width" ) ) || 0;
162
                                }
163
                                if ( margin ) {
164
                                        size -= parseFloat( $.css( elem, "margin" + this ) ) || 0;
165
                                }
166
                        });
167
                        return size;
168
                }
169
170
                $.fn[ "inner" + name ] = function( size ) {
171
                        if ( size === undefined ) {
172
                                return orig[ "inner" + name ].call( this );
173
                        }
174
175
                        return this.each(function() {
176
                                $( this ).css( type, reduce( this, size ) + "px" );
177
                        });
178
                };
179
180
                $.fn[ "outer" + name] = function( size, margin ) {
181
                        if ( typeof size !== "number" ) {
182
                                return orig[ "outer" + name ].call( this, size );
183
                        }
184
185
                        return this.each(function() {
186
                                $( this).css( type, reduce( this, size, true, margin ) + "px" );
187
                        });
188
                };
189
        });
190
}
191
192
// support: jQuery <1.8
193
if ( !$.fn.addBack ) {
194
        $.fn.addBack = function( selector ) {
195
                return this.add( selector == null ?
196
                        this.prevObject : this.prevObject.filter( selector )
197
                );
198
        };
199
}
200
201
// support: jQuery 1.6.1, 1.6.2 (http://bugs.jquery.com/ticket/9413)
202
if ( $( "<a>" ).data( "a-b", "a" ).removeData( "a-b" ).data( "a-b" ) ) {
203
        $.fn.removeData = (function( removeData ) {
204
                return function( key ) {
205
                        if ( arguments.length ) {
206
                                return removeData.call( this, $.camelCase( key ) );
207
                        } else {
208
                                return removeData.call( this );
209
                        }
210
                };
211
        })( $.fn.removeData );
212
}
213
214
// deprecated
215
$.ui.ie = !!/msie [\w.]+/.exec( navigator.userAgent.toLowerCase() );
216
217
$.fn.extend({
218
        focus: (function( orig ) {
219
                return function( delay, fn ) {
220
                        return typeof delay === "number" ?
221
                                this.each(function() {
222
                                        var elem = this;
223
                                        setTimeout(function() {
224
                                                $( elem ).focus();
225
                                                if ( fn ) {
226
                                                        fn.call( elem );
227
                                                }
228
                                        }, delay );
229
                                }) :
230
                                orig.apply( this, arguments );
231
                };
232
        })( $.fn.focus ),
233
234
        disableSelection: (function() {
235
                var eventType = "onselectstart" in document.createElement( "div" ) ?
236
                        "selectstart" :
237
                        "mousedown";
238
239
                return function() {
240
                        return this.bind( eventType + ".ui-disableSelection", function( event ) {
241
                                event.preventDefault();
242
                        });
243
                };
244
        })(),
245
246
        enableSelection: function() {
247
                return this.unbind( ".ui-disableSelection" );
248
        },
249
250
        zIndex: function( zIndex ) {
251
                if ( zIndex !== undefined ) {
252
                        return this.css( "zIndex", zIndex );
253
                }
254
255
                if ( this.length ) {
256
                        var elem = $( this[ 0 ] ), position, value;
257
                        while ( elem.length && elem[ 0 ] !== document ) {
258
                                // Ignore z-index if position is set to a value where z-index is ignored by the browser
259
                                // This makes behavior of this function consistent across browsers
260
                                // WebKit always returns auto if the element is positioned
261
                                position = elem.css( "position" );
262
                                if ( position === "absolute" || position === "relative" || position === "fixed" ) {
263
                                        // IE returns 0 when zIndex is not specified
264
                                        // other browsers return a string
265
                                        // we ignore the case of nested elements with an explicit value of 0
266
                                        // <div style="z-index: -10;"><div style="z-index: 0;"></div></div>
267
                                        value = parseInt( elem.css( "zIndex" ), 10 );
268
                                        if ( !isNaN( value ) && value !== 0 ) {
269
                                                return value;
270
                                        }
271
                                }
272
                                elem = elem.parent();
273
                        }
274
                }
275
276
                return 0;
277
        }
278
});
279
280
// $.ui.plugin is deprecated. Use $.widget() extensions instead.
281
$.ui.plugin = {
282
        add: function( module, option, set ) {
283
                var i,
284
                        proto = $.ui[ module ].prototype;
285
                for ( i in set ) {
286
                        proto.plugins[ i ] = proto.plugins[ i ] || [];
287
                        proto.plugins[ i ].push( [ option, set[ i ] ] );
288
                }
289
        },
290
        call: function( instance, name, args, allowDisconnected ) {
291
                var i,
292
                        set = instance.plugins[ name ];
293
294
                if ( !set ) {
295
                        return;
296
                }
297
298
                if ( !allowDisconnected && ( !instance.element[ 0 ].parentNode || instance.element[ 0 ].parentNode.nodeType === 11 ) ) {
299
                        return;
300
                }
301
302
                for ( i = 0; i < set.length; i++ ) {
303
                        if ( instance.options[ set[ i ][ 0 ] ] ) {
304
                                set[ i ][ 1 ].apply( instance.element, args );
305
                        }
306
                }
307
        }
308
};
309
310
311
/*!
312
 * jQuery UI Widget 1.11.4
313
 * http://jqueryui.com
314
 *
315
 * Copyright jQuery Foundation and other contributors
316
 * Released under the MIT license.
317
 * http://jquery.org/license
318
 *
319
 * http://api.jqueryui.com/jQuery.widget/
320
 */
321
322
323
var widget_uuid = 0,
324
        widget_slice = Array.prototype.slice;
325
326
$.cleanData = (function( orig ) {
327
        return function( elems ) {
328
                var events, elem, i;
329
                for ( i = 0; (elem = elems[i]) != null; i++ ) {
330
                        try {
331
332
                                // Only trigger remove when necessary to save time
333
                                events = $._data( elem, "events" );
334
                                if ( events && events.remove ) {
335
                                        $( elem ).triggerHandler( "remove" );
336
                                }
337
338
                        // http://bugs.jquery.com/ticket/8235
339
                        } catch ( e ) {}
340
                }
341
                orig( elems );
342
        };
343
})( $.cleanData );
344
345
$.widget = function( name, base, prototype ) {
346
        var fullName, existingConstructor, constructor, basePrototype,
347
                // proxiedPrototype allows the provided prototype to remain unmodified
348
                // so that it can be used as a mixin for multiple widgets (#8876)
349
                proxiedPrototype = {},
350
                namespace = name.split( "." )[ 0 ];
351
352
        name = name.split( "." )[ 1 ];
353
        fullName = namespace + "-" + name;
354
355
        if ( !prototype ) {
356
                prototype = base;
357
                base = $.Widget;
358
        }
359
360
        // create selector for plugin
361
        $.expr[ ":" ][ fullName.toLowerCase() ] = function( elem ) {
362
                return !!$.data( elem, fullName );
363
        };
364
365
        $[ namespace ] = $[ namespace ] || {};
366
        existingConstructor = $[ namespace ][ name ];
367
        constructor = $[ namespace ][ name ] = function( options, element ) {
368
                // allow instantiation without "new" keyword
369
                if ( !this._createWidget ) {
370
                        return new constructor( options, element );
371
                }
372
373
                // allow instantiation without initializing for simple inheritance
374
                // must use "new" keyword (the code above always passes args)
375
                if ( arguments.length ) {
376
                        this._createWidget( options, element );
377
                }
378
        };
379
        // extend with the existing constructor to carry over any static properties
380
        $.extend( constructor, existingConstructor, {
381
                version: prototype.version,
382
                // copy the object used to create the prototype in case we need to
383
                // redefine the widget later
384
                _proto: $.extend( {}, prototype ),
385
                // track widgets that inherit from this widget in case this widget is
386
                // redefined after a widget inherits from it
387
                _childConstructors: []
388
        });
389
390
        basePrototype = new base();
391
        // we need to make the options hash a property directly on the new instance
392
        // otherwise we'll modify the options hash on the prototype that we're
393
        // inheriting from
394
        basePrototype.options = $.widget.extend( {}, basePrototype.options );
395
        $.each( prototype, function( prop, value ) {
396
                if ( !$.isFunction( value ) ) {
397
                        proxiedPrototype[ prop ] = value;
398
                        return;
399
                }
400
                proxiedPrototype[ prop ] = (function() {
401
                        var _super = function() {
402
                                        return base.prototype[ prop ].apply( this, arguments );
403
                                },
404
                                _superApply = function( args ) {
405
                                        return base.prototype[ prop ].apply( this, args );
406
                                };
407
                        return function() {
408
                                var __super = this._super,
409
                                        __superApply = this._superApply,
410
                                        returnValue;
411
412
                                this._super = _super;
413
                                this._superApply = _superApply;
414
415
                                returnValue = value.apply( this, arguments );
416
417
                                this._super = __super;
418
                                this._superApply = __superApply;
419
420
                                return returnValue;
421
                        };
422
                })();
423
        });
424
        constructor.prototype = $.widget.extend( basePrototype, {
425
                // TODO: remove support for widgetEventPrefix
426
                // always use the name + a colon as the prefix, e.g., draggable:start
427
                // don't prefix for widgets that aren't DOM-based
428
                widgetEventPrefix: existingConstructor ? (basePrototype.widgetEventPrefix || name) : name
429
        }, proxiedPrototype, {
430
                constructor: constructor,
431
                namespace: namespace,
432
                widgetName: name,
433
                widgetFullName: fullName
434
        });
435
436
        // If this widget is being redefined then we need to find all widgets that
437
        // are inheriting from it and redefine all of them so that they inherit from
438
        // the new version of this widget. We're essentially trying to replace one
439
        // level in the prototype chain.
440
        if ( existingConstructor ) {
441
                $.each( existingConstructor._childConstructors, function( i, child ) {
442
                        var childPrototype = child.prototype;
443
444
                        // redefine the child widget using the same prototype that was
445
                        // originally used, but inherit from the new version of the base
446
                        $.widget( childPrototype.namespace + "." + childPrototype.widgetName, constructor, child._proto );
447
                });
448
                // remove the list of existing child constructors from the old constructor
449
                // so the old child constructors can be garbage collected
450
                delete existingConstructor._childConstructors;
451
        } else {
452
                base._childConstructors.push( constructor );
453
        }
454
455
        $.widget.bridge( name, constructor );
456
457
        return constructor;
458
};
459
460
$.widget.extend = function( target ) {
461
        var input = widget_slice.call( arguments, 1 ),
462
                inputIndex = 0,
463
                inputLength = input.length,
464
                key,
465
                value;
466
        for ( ; inputIndex < inputLength; inputIndex++ ) {
467
                for ( key in input[ inputIndex ] ) {
468
                        value = input[ inputIndex ][ key ];
469
                        if ( input[ inputIndex ].hasOwnProperty( key ) && value !== undefined ) {
470
                                // Clone objects
471
                                if ( $.isPlainObject( value ) ) {
472
                                        target[ key ] = $.isPlainObject( target[ key ] ) ?
473
                                                $.widget.extend( {}, target[ key ], value ) :
474
                                                // Don't extend strings, arrays, etc. with objects
475
                                                $.widget.extend( {}, value );
476
                                // Copy everything else by reference
477
                                } else {
478
                                        target[ key ] = value;
479
                                }
480
                        }
481
                }
482
        }
483
        return target;
484
};
485
486
$.widget.bridge = function( name, object ) {
487
        var fullName = object.prototype.widgetFullName || name;
488
        $.fn[ name ] = function( options ) {
489
                var isMethodCall = typeof options === "string",
490
                        args = widget_slice.call( arguments, 1 ),
491
                        returnValue = this;
492
493
                if ( isMethodCall ) {
494
                        this.each(function() {
495
                                var methodValue,
496
                                        instance = $.data( this, fullName );
497
                                if ( options === "instance" ) {
498
                                        returnValue = instance;
499
                                        return false;
500
                                }
501
                                if ( !instance ) {
502
                                        return $.error( "cannot call methods on " + name + " prior to initialization; " +
503
                                                "attempted to call method '" + options + "'" );
504
                                }
505
                                if ( !$.isFunction( instance[options] ) || options.charAt( 0 ) === "_" ) {
506
                                        return $.error( "no such method '" + options + "' for " + name + " widget instance" );
507
                                }
508
                                methodValue = instance[ options ].apply( instance, args );
509
                                if ( methodValue !== instance && methodValue !== undefined ) {
510
                                        returnValue = methodValue && methodValue.jquery ?
511
                                                returnValue.pushStack( methodValue.get() ) :
512
                                                methodValue;
513
                                        return false;
514
                                }
515
                        });
516
                } else {
517
518
                        // Allow multiple hashes to be passed on init
519
                        if ( args.length ) {
520
                                options = $.widget.extend.apply( null, [ options ].concat(args) );
521
                        }
522
523
                        this.each(function() {
524
                                var instance = $.data( this, fullName );
525
                                if ( instance ) {
526
                                        instance.option( options || {} );
527
                                        if ( instance._init ) {
528
                                                instance._init();
529
                                        }
530
                                } else {
531
                                        $.data( this, fullName, new object( options, this ) );
532
                                }
533
                        });
534
                }
535
536
                return returnValue;
537
        };
538
};
539
540
$.Widget = function( /* options, element */ ) {};
541
$.Widget._childConstructors = [];
542
543
$.Widget.prototype = {
544
        widgetName: "widget",
545
        widgetEventPrefix: "",
546
        defaultElement: "<div>",
547
        options: {
548
                disabled: false,
549
550
                // callbacks
551
                create: null
552
        },
553
        _createWidget: function( options, element ) {
554
                element = $( element || this.defaultElement || this )[ 0 ];
555
                this.element = $( element );
556
                this.uuid = widget_uuid++;
557
                this.eventNamespace = "." + this.widgetName + this.uuid;
558
559
                this.bindings = $();
560
                this.hoverable = $();
561
                this.focusable = $();
562
563
                if ( element !== this ) {
564
                        $.data( element, this.widgetFullName, this );
565
                        this._on( true, this.element, {
566
                                remove: function( event ) {
567
                                        if ( event.target === element ) {
568
                                                this.destroy();
569
                                        }
570
                                }
571
                        });
572
                        this.document = $( element.style ?
573
                                // element within the document
574
                                element.ownerDocument :
575
                                // element is window or document
576
                                element.document || element );
577
                        this.window = $( this.document[0].defaultView || this.document[0].parentWindow );
578
                }
579
580
                this.options = $.widget.extend( {},
581
                        this.options,
582
                        this._getCreateOptions(),
583
                        options );
584
585
                this._create();
586
                this._trigger( "create", null, this._getCreateEventData() );
587
                this._init();
588
        },
589
        _getCreateOptions: $.noop,
590
        _getCreateEventData: $.noop,
591
        _create: $.noop,
592
        _init: $.noop,
593
594
        destroy: function() {
595
                this._destroy();
596
                // we can probably remove the unbind calls in 2.0
597
                // all event bindings should go through this._on()
598
                this.element
599
                        .unbind( this.eventNamespace )
600
                        .removeData( this.widgetFullName )
601
                        // support: jquery <1.6.3
602
                        // http://bugs.jquery.com/ticket/9413
603
                        .removeData( $.camelCase( this.widgetFullName ) );
604
                this.widget()
605
                        .unbind( this.eventNamespace )
606
                        .removeAttr( "aria-disabled" )
607
                        .removeClass(
608
                                this.widgetFullName + "-disabled " +
609
                                "ui-state-disabled" );
610
611
                // clean up events and states
612
                this.bindings.unbind( this.eventNamespace );
613
                this.hoverable.removeClass( "ui-state-hover" );
614
                this.focusable.removeClass( "ui-state-focus" );
615
        },
616
        _destroy: $.noop,
617
618
        widget: function() {
619
                return this.element;
620
        },
621
622
        option: function( key, value ) {
623
                var options = key,
624
                        parts,
625
                        curOption,
626
                        i;
627
628
                if ( arguments.length === 0 ) {
629
                        // don't return a reference to the internal hash
630
                        return $.widget.extend( {}, this.options );
631
                }
632
633
                if ( typeof key === "string" ) {
634
                        // handle nested keys, e.g., "foo.bar" => { foo: { bar: ___ } }
635
                        options = {};
636
                        parts = key.split( "." );
637
                        key = parts.shift();
638
                        if ( parts.length ) {
639
                                curOption = options[ key ] = $.widget.extend( {}, this.options[ key ] );
640
                                for ( i = 0; i < parts.length - 1; i++ ) {
641
                                        curOption[ parts[ i ] ] = curOption[ parts[ i ] ] || {};
642
                                        curOption = curOption[ parts[ i ] ];
643
                                }
644
                                key = parts.pop();
645
                                if ( arguments.length === 1 ) {
646
                                        return curOption[ key ] === undefined ? null : curOption[ key ];
647
                                }
648
                                curOption[ key ] = value;
649
                        } else {
650
                                if ( arguments.length === 1 ) {
651
                                        return this.options[ key ] === undefined ? null : this.options[ key ];
652
                                }
653
                                options[ key ] = value;
654
                        }
655
                }
656
657
                this._setOptions( options );
658
659
                return this;
660
        },
661
        _setOptions: function( options ) {
662
                var key;
663
664
                for ( key in options ) {
665
                        this._setOption( key, options[ key ] );
666
                }
667
668
                return this;
669
        },
670
        _setOption: function( key, value ) {
671
                this.options[ key ] = value;
672
673
                if ( key === "disabled" ) {
674
                        this.widget()
675
                                .toggleClass( this.widgetFullName + "-disabled", !!value );
676
677
                        // If the widget is becoming disabled, then nothing is interactive
678
                        if ( value ) {
679
                                this.hoverable.removeClass( "ui-state-hover" );
680
                                this.focusable.removeClass( "ui-state-focus" );
681
                        }
682
                }
683
684
                return this;
685
        },
686
687
        enable: function() {
688
                return this._setOptions({ disabled: false });
689
        },
690
        disable: function() {
691
                return this._setOptions({ disabled: true });
692
        },
693
694
        _on: function( suppressDisabledCheck, element, handlers ) {
695
                var delegateElement,
696
                        instance = this;
697
698
                // no suppressDisabledCheck flag, shuffle arguments
699
                if ( typeof suppressDisabledCheck !== "boolean" ) {
700
                        handlers = element;
701
                        element = suppressDisabledCheck;
702
                        suppressDisabledCheck = false;
703
                }
704
705
                // no element argument, shuffle and use this.element
706
                if ( !handlers ) {
707
                        handlers = element;
708
                        element = this.element;
709
                        delegateElement = this.widget();
710
                } else {
711
                        element = delegateElement = $( element );
712
                        this.bindings = this.bindings.add( element );
713
                }
714
715
                $.each( handlers, function( event, handler ) {
716
                        function handlerProxy() {
717
                                // allow widgets to customize the disabled handling
718
                                // - disabled as an array instead of boolean
719
                                // - disabled class as method for disabling individual parts
720
                                if ( !suppressDisabledCheck &&
721
                                                ( instance.options.disabled === true ||
722
                                                        $( this ).hasClass( "ui-state-disabled" ) ) ) {
723
                                        return;
724
                                }
725
                                return ( typeof handler === "string" ? instance[ handler ] : handler )
726
                                        .apply( instance, arguments );
727
                        }
728
729
                        // copy the guid so direct unbinding works
730
                        if ( typeof handler !== "string" ) {
731
                                handlerProxy.guid = handler.guid =
732
                                        handler.guid || handlerProxy.guid || $.guid++;
733
                        }
734
735
                        var match = event.match( /^([\w:-]*)\s*(.*)$/ ),
736
                                eventName = match[1] + instance.eventNamespace,
737
                                selector = match[2];
738
                        if ( selector ) {
739
                                delegateElement.delegate( selector, eventName, handlerProxy );
740
                        } else {
741
                                element.bind( eventName, handlerProxy );
742
                        }
743
                });
744
        },
745
746
        _off: function( element, eventName ) {
747
                eventName = (eventName || "").split( " " ).join( this.eventNamespace + " " ) +
748
                        this.eventNamespace;
749
                element.unbind( eventName ).undelegate( eventName );
750
751
                // Clear the stack to avoid memory leaks (#10056)
752
                this.bindings = $( this.bindings.not( element ).get() );
753
                this.focusable = $( this.focusable.not( element ).get() );
754
                this.hoverable = $( this.hoverable.not( element ).get() );
755
        },
756
757
        _delay: function( handler, delay ) {
758
                function handlerProxy() {
759
                        return ( typeof handler === "string" ? instance[ handler ] : handler )
760
                                .apply( instance, arguments );
761
                }
762
                var instance = this;
763
                return setTimeout( handlerProxy, delay || 0 );
764
        },
765
766
        _hoverable: function( element ) {
767
                this.hoverable = this.hoverable.add( element );
768
                this._on( element, {
769
                        mouseenter: function( event ) {
770
                                $( event.currentTarget ).addClass( "ui-state-hover" );
771
                        },
772
                        mouseleave: function( event ) {
773
                                $( event.currentTarget ).removeClass( "ui-state-hover" );
774
                        }
775
                });
776
        },
777
778
        _focusable: function( element ) {
779
                this.focusable = this.focusable.add( element );
780
                this._on( element, {
781
                        focusin: function( event ) {
782
                                $( event.currentTarget ).addClass( "ui-state-focus" );
783
                        },
784
                        focusout: function( event ) {
785
                                $( event.currentTarget ).removeClass( "ui-state-focus" );
786
                        }
787
                });
788
        },
789
790
        _trigger: function( type, event, data ) {
791
                var prop, orig,
792
                        callback = this.options[ type ];
793
794
                data = data || {};
795
                event = $.Event( event );
796
                event.type = ( type === this.widgetEventPrefix ?
797
                        type :
798
                        this.widgetEventPrefix + type ).toLowerCase();
799
                // the original event may come from any element
800
                // so we need to reset the target on the new event
801
                event.target = this.element[ 0 ];
802
803
                // copy original event properties over to the new event
804
                orig = event.originalEvent;
805
                if ( orig ) {
806
                        for ( prop in orig ) {
807
                                if ( !( prop in event ) ) {
808
                                        event[ prop ] = orig[ prop ];
809
                                }
810
                        }
811
                }
812
813
                this.element.trigger( event, data );
814
                return !( $.isFunction( callback ) &&
815
                        callback.apply( this.element[0], [ event ].concat( data ) ) === false ||
816
                        event.isDefaultPrevented() );
817
        }
818
};
819
820
$.each( { show: "fadeIn", hide: "fadeOut" }, function( method, defaultEffect ) {
821
        $.Widget.prototype[ "_" + method ] = function( element, options, callback ) {
822
                if ( typeof options === "string" ) {
823
                        options = { effect: options };
824
                }
825
                var hasOptions,
826
                        effectName = !options ?
827
                                method :
828
                                options === true || typeof options === "number" ?
829
                                        defaultEffect :
830
                                        options.effect || defaultEffect;
831
                options = options || {};
832
                if ( typeof options === "number" ) {
833
                        options = { duration: options };
834
                }
835
                hasOptions = !$.isEmptyObject( options );
836
                options.complete = callback;
837
                if ( options.delay ) {
838
                        element.delay( options.delay );
839
                }
840
                if ( hasOptions && $.effects && $.effects.effect[ effectName ] ) {
841
                        element[ method ]( options );
842
                } else if ( effectName !== method && element[ effectName ] ) {
843
                        element[ effectName ]( options.duration, options.easing, callback );
844
                } else {
845
                        element.queue(function( next ) {
846
                                $( this )[ method ]();
847
                                if ( callback ) {
848
                                        callback.call( element[ 0 ] );
849
                                }
850
                                next();
851
                        });
852
                }
853
        };
854
});
855
856
var widget = $.widget;
857
858
859
/*!
860
 * jQuery UI Mouse 1.11.4
861
 * http://jqueryui.com
862
 *
863
 * Copyright jQuery Foundation and other contributors
864
 * Released under the MIT license.
865
 * http://jquery.org/license
866
 *
867
 * http://api.jqueryui.com/mouse/
868
 */
869
870
871
var mouseHandled = false;
872
$( document ).mouseup( function() {
873
        mouseHandled = false;
874
});
875
876
var mouse = $.widget("ui.mouse", {
877
        version: "1.11.4",
878
        options: {
879
                cancel: "input,textarea,button,select,option",
880
                distance: 1,
881
                delay: 0
882
        },
883
        _mouseInit: function() {
884
                var that = this;
885
886
                this.element
887
                        .bind("mousedown." + this.widgetName, function(event) {
888
                                return that._mouseDown(event);
889
                        })
890
                        .bind("click." + this.widgetName, function(event) {
891
                                if (true === $.data(event.target, that.widgetName + ".preventClickEvent")) {
892
                                        $.removeData(event.target, that.widgetName + ".preventClickEvent");
893
                                        event.stopImmediatePropagation();
894
                                        return false;
895
                                }
896
                        });
897
898
                this.started = false;
899
        },
900
901
        // TODO: make sure destroying one instance of mouse doesn't mess with
902
        // other instances of mouse
903
        _mouseDestroy: function() {
904
                this.element.unbind("." + this.widgetName);
905
                if ( this._mouseMoveDelegate ) {
906
                        this.document
907
                                .unbind("mousemove." + this.widgetName, this._mouseMoveDelegate)
908
                                .unbind("mouseup." + this.widgetName, this._mouseUpDelegate);
909
                }
910
        },
911
912
        _mouseDown: function(event) {
913
                // don't let more than one widget handle mouseStart
914
                if ( mouseHandled ) {
915
                        return;
916
                }
917
918
                this._mouseMoved = false;
919
920
                // we may have missed mouseup (out of window)
921
                (this._mouseStarted && this._mouseUp(event));
922
923
                this._mouseDownEvent = event;
924
925
                var that = this,
926
                        btnIsLeft = (event.which === 1),
927
                        // event.target.nodeName works around a bug in IE 8 with
928
                        // disabled inputs (#7620)
929
                        elIsCancel = (typeof this.options.cancel === "string" && event.target.nodeName ? $(event.target).closest(this.options.cancel).length : false);
930
                if (!btnIsLeft || elIsCancel || !this._mouseCapture(event)) {
931
                        return true;
932
                }
933
934
                this.mouseDelayMet = !this.options.delay;
935
                if (!this.mouseDelayMet) {
936
                        this._mouseDelayTimer = setTimeout(function() {
937
                                that.mouseDelayMet = true;
938
                        }, this.options.delay);
939
                }
940
941
                if (this._mouseDistanceMet(event) && this._mouseDelayMet(event)) {
942
                        this._mouseStarted = (this._mouseStart(event) !== false);
943
                        if (!this._mouseStarted) {
944
                                event.preventDefault();
945
                                return true;
946
                        }
947
                }
948
949
                // Click event may never have fired (Gecko & Opera)
950
                if (true === $.data(event.target, this.widgetName + ".preventClickEvent")) {
951
                        $.removeData(event.target, this.widgetName + ".preventClickEvent");
952
                }
953
954
                // these delegates are required to keep context
955
                this._mouseMoveDelegate = function(event) {
956
                        return that._mouseMove(event);
957
                };
958
                this._mouseUpDelegate = function(event) {
959
                        return that._mouseUp(event);
960
                };
961
962
                this.document
963
                        .bind( "mousemove." + this.widgetName, this._mouseMoveDelegate )
964
                        .bind( "mouseup." + this.widgetName, this._mouseUpDelegate );
965
966
                event.preventDefault();
967
968
                mouseHandled = true;
969
                return true;
970
        },
971
972
        _mouseMove: function(event) {
973
                // Only check for mouseups outside the document if you've moved inside the document
974
                // at least once. This prevents the firing of mouseup in the case of IE<9, which will
975
                // fire a mousemove event if content is placed under the cursor. See #7778
976
                // Support: IE <9
977
                if ( this._mouseMoved ) {
978
                        // IE mouseup check - mouseup happened when mouse was out of window
979
                        if ($.ui.ie && ( !document.documentMode || document.documentMode < 9 ) && !event.button) {
980
                                return this._mouseUp(event);
981
982
                        // Iframe mouseup check - mouseup occurred in another document
983
                        } else if ( !event.which ) {
984
                                return this._mouseUp( event );
985
                        }
986
                }
987
988
                if ( event.which || event.button ) {
989
                        this._mouseMoved = true;
990
                }
991
992
                if (this._mouseStarted) {
993
                        this._mouseDrag(event);
994
                        return event.preventDefault();
995
                }
996
997
                if (this._mouseDistanceMet(event) && this._mouseDelayMet(event)) {
998
                        this._mouseStarted =
999
                                (this._mouseStart(this._mouseDownEvent, event) !== false);
1000
                        (this._mouseStarted ? this._mouseDrag(event) : this._mouseUp(event));
1001
                }
1002
1003
                return !this._mouseStarted;
1004
        },
1005
1006
        _mouseUp: function(event) {
1007
                this.document
1008
                        .unbind( "mousemove." + this.widgetName, this._mouseMoveDelegate )
1009
                        .unbind( "mouseup." + this.widgetName, this._mouseUpDelegate );
1010
1011
                if (this._mouseStarted) {
1012
                        this._mouseStarted = false;
1013
1014
                        if (event.target === this._mouseDownEvent.target) {
1015
                                $.data(event.target, this.widgetName + ".preventClickEvent", true);
1016
                        }
1017
1018
                        this._mouseStop(event);
1019
                }
1020
1021
                mouseHandled = false;
1022
                return false;
1023
        },
1024
1025
        _mouseDistanceMet: function(event) {
1026
                return (Math.max(
1027
                                Math.abs(this._mouseDownEvent.pageX - event.pageX),
1028
                                Math.abs(this._mouseDownEvent.pageY - event.pageY)
1029
                        ) >= this.options.distance
1030
                );
1031
        },
1032
1033
        _mouseDelayMet: function(/* event */) {
1034
                return this.mouseDelayMet;
1035
        },
1036
1037
        // These are placeholder methods, to be overriden by extending plugin
1038
        _mouseStart: function(/* event */) {},
1039
        _mouseDrag: function(/* event */) {},
1040
        _mouseStop: function(/* event */) {},
1041
        _mouseCapture: function(/* event */) { return true; }
1042
});
1043
1044
1045
/*!
1046
 * jQuery UI Position 1.11.4
1047
 * http://jqueryui.com
1048
 *
1049
 * Copyright jQuery Foundation and other contributors
1050
 * Released under the MIT license.
1051
 * http://jquery.org/license
1052
 *
1053
 * http://api.jqueryui.com/position/
1054
 */
1055
1056
(function() {
1057
1058
$.ui = $.ui || {};
1059
1060
var cachedScrollbarWidth, supportsOffsetFractions,
1061
        max = Math.max,
1062
        abs = Math.abs,
1063
        round = Math.round,
1064
        rhorizontal = /left|center|right/,
1065
        rvertical = /top|center|bottom/,
1066
        roffset = /[\+\-]\d+(\.[\d]+)?%?/,
1067
        rposition = /^\w+/,
1068
        rpercent = /%$/,
1069
        _position = $.fn.position;
1070
1071
function getOffsets( offsets, width, height ) {
1072
        return [
1073
                parseFloat( offsets[ 0 ] ) * ( rpercent.test( offsets[ 0 ] ) ? width / 100 : 1 ),
1074
                parseFloat( offsets[ 1 ] ) * ( rpercent.test( offsets[ 1 ] ) ? height / 100 : 1 )
1075
        ];
1076
}
1077
1078
function parseCss( element, property ) {
1079
        return parseInt( $.css( element, property ), 10 ) || 0;
1080
}
1081
1082
function getDimensions( elem ) {
1083
        var raw = elem[0];
1084
        if ( raw.nodeType === 9 ) {
1085
                return {
1086
                        width: elem.width(),
1087
                        height: elem.height(),
1088
                        offset: { top: 0, left: 0 }
1089
                };
1090
        }
1091
        if ( $.isWindow( raw ) ) {
1092
                return {
1093
                        width: elem.width(),
1094
                        height: elem.height(),
1095
                        offset: { top: elem.scrollTop(), left: elem.scrollLeft() }
1096
                };
1097
        }
1098
        if ( raw.preventDefault ) {
1099
                return {
1100
                        width: 0,
1101
                        height: 0,
1102
                        offset: { top: raw.pageY, left: raw.pageX }
1103
                };
1104
        }
1105
        return {
1106
                width: elem.outerWidth(),
1107
                height: elem.outerHeight(),
1108
                offset: elem.offset()
1109
        };
1110
}
1111
1112
$.position = {
1113
        scrollbarWidth: function() {
1114
                if ( cachedScrollbarWidth !== undefined ) {
1115
                        return cachedScrollbarWidth;
1116
                }
1117
                var w1, w2,
1118
                        div = $( "<div style='display:block;position:absolute;width:50px;height:50px;overflow:hidden;'><div style='height:100px;width:auto;'></div></div>" ),
1119
                        innerDiv = div.children()[0];
1120
1121
                $( "body" ).append( div );
1122
                w1 = innerDiv.offsetWidth;
1123
                div.css( "overflow", "scroll" );
1124
1125
                w2 = innerDiv.offsetWidth;
1126
1127
                if ( w1 === w2 ) {
1128
                        w2 = div[0].clientWidth;
1129
                }
1130
1131
                div.remove();
1132
1133
                return (cachedScrollbarWidth = w1 - w2);
1134
        },
1135
        getScrollInfo: function( within ) {
1136
                var overflowX = within.isWindow || within.isDocument ? "" :
1137
                                within.element.css( "overflow-x" ),
1138
                        overflowY = within.isWindow || within.isDocument ? "" :
1139
                                within.element.css( "overflow-y" ),
1140
                        hasOverflowX = overflowX === "scroll" ||
1141
                                ( overflowX === "auto" && within.width < within.element[0].scrollWidth ),
1142
                        hasOverflowY = overflowY === "scroll" ||
1143
                                ( overflowY === "auto" && within.height < within.element[0].scrollHeight );
1144
                return {
1145
                        width: hasOverflowY ? $.position.scrollbarWidth() : 0,
1146
                        height: hasOverflowX ? $.position.scrollbarWidth() : 0
1147
                };
1148
        },
1149
        getWithinInfo: function( element ) {
1150
                var withinElement = $( element || window ),
1151
                        isWindow = $.isWindow( withinElement[0] ),
1152
                        isDocument = !!withinElement[ 0 ] && withinElement[ 0 ].nodeType === 9;
1153
                return {
1154
                        element: withinElement,
1155
                        isWindow: isWindow,
1156
                        isDocument: isDocument,
1157
                        offset: withinElement.offset() || { left: 0, top: 0 },
1158
                        scrollLeft: withinElement.scrollLeft(),
1159
                        scrollTop: withinElement.scrollTop(),
1160
1161
                        // support: jQuery 1.6.x
1162
                        // jQuery 1.6 doesn't support .outerWidth/Height() on documents or windows
1163
                        width: isWindow || isDocument ? withinElement.width() : withinElement.outerWidth(),
1164
                        height: isWindow || isDocument ? withinElement.height() : withinElement.outerHeight()
1165
                };
1166
        }
1167
};
1168
1169
$.fn.position = function( options ) {
1170
        if ( !options || !options.of ) {
1171
                return _position.apply( this, arguments );
1172
        }
1173
1174
        // make a copy, we don't want to modify arguments
1175
        options = $.extend( {}, options );
1176
1177
        var atOffset, targetWidth, targetHeight, targetOffset, basePosition, dimensions,
1178
                target = $( options.of ),
1179
                within = $.position.getWithinInfo( options.within ),
1180
                scrollInfo = $.position.getScrollInfo( within ),
1181
                collision = ( options.collision || "flip" ).split( " " ),
1182
                offsets = {};
1183
1184
        dimensions = getDimensions( target );
1185
        if ( target[0].preventDefault ) {
1186
                // force left top to allow flipping
1187
                options.at = "left top";
1188
        }
1189
        targetWidth = dimensions.width;
1190
        targetHeight = dimensions.height;
1191
        targetOffset = dimensions.offset;
1192
        // clone to reuse original targetOffset later
1193
        basePosition = $.extend( {}, targetOffset );
1194
1195
        // force my and at to have valid horizontal and vertical positions
1196
        // if a value is missing or invalid, it will be converted to center
1197
        $.each( [ "my", "at" ], function() {
1198
                var pos = ( options[ this ] || "" ).split( " " ),
1199
                        horizontalOffset,
1200
                        verticalOffset;
1201
1202
                if ( pos.length === 1) {
1203
                        pos = rhorizontal.test( pos[ 0 ] ) ?
1204
                                pos.concat( [ "center" ] ) :
1205
                                rvertical.test( pos[ 0 ] ) ?
1206
                                        [ "center" ].concat( pos ) :
1207
                                        [ "center", "center" ];
1208
                }
1209
                pos[ 0 ] = rhorizontal.test( pos[ 0 ] ) ? pos[ 0 ] : "center";
1210
                pos[ 1 ] = rvertical.test( pos[ 1 ] ) ? pos[ 1 ] : "center";
1211
1212
                // calculate offsets
1213
                horizontalOffset = roffset.exec( pos[ 0 ] );
1214
                verticalOffset = roffset.exec( pos[ 1 ] );
1215
                offsets[ this ] = [
1216
                        horizontalOffset ? horizontalOffset[ 0 ] : 0,
1217
                        verticalOffset ? verticalOffset[ 0 ] : 0
1218
                ];
1219
1220
                // reduce to just the positions without the offsets
1221
                options[ this ] = [
1222
                        rposition.exec( pos[ 0 ] )[ 0 ],
1223
                        rposition.exec( pos[ 1 ] )[ 0 ]
1224
                ];
1225
        });
1226
1227
        // normalize collision option
1228
        if ( collision.length === 1 ) {
1229
                collision[ 1 ] = collision[ 0 ];
1230
        }
1231
1232
        if ( options.at[ 0 ] === "right" ) {
1233
                basePosition.left += targetWidth;
1234
        } else if ( options.at[ 0 ] === "center" ) {
1235
                basePosition.left += targetWidth / 2;
1236
        }
1237
1238
        if ( options.at[ 1 ] === "bottom" ) {
1239
                basePosition.top += targetHeight;
1240
        } else if ( options.at[ 1 ] === "center" ) {
1241
                basePosition.top += targetHeight / 2;
1242
        }
1243
1244
        atOffset = getOffsets( offsets.at, targetWidth, targetHeight );
1245
        basePosition.left += atOffset[ 0 ];
1246
        basePosition.top += atOffset[ 1 ];
1247
1248
        return this.each(function() {
1249
                var collisionPosition, using,
1250
                        elem = $( this ),
1251
                        elemWidth = elem.outerWidth(),
1252
                        elemHeight = elem.outerHeight(),
1253
                        marginLeft = parseCss( this, "marginLeft" ),
1254
                        marginTop = parseCss( this, "marginTop" ),
1255
                        collisionWidth = elemWidth + marginLeft + parseCss( this, "marginRight" ) + scrollInfo.width,
1256
                        collisionHeight = elemHeight + marginTop + parseCss( this, "marginBottom" ) + scrollInfo.height,
1257
                        position = $.extend( {}, basePosition ),
1258
                        myOffset = getOffsets( offsets.my, elem.outerWidth(), elem.outerHeight() );
1259
1260
                if ( options.my[ 0 ] === "right" ) {
1261
                        position.left -= elemWidth;
1262
                } else if ( options.my[ 0 ] === "center" ) {
1263
                        position.left -= elemWidth / 2;
1264
                }
1265
1266
                if ( options.my[ 1 ] === "bottom" ) {
1267
                        position.top -= elemHeight;
1268
                } else if ( options.my[ 1 ] === "center" ) {
1269
                        position.top -= elemHeight / 2;
1270
                }
1271
1272
                position.left += myOffset[ 0 ];
1273
                position.top += myOffset[ 1 ];
1274
1275
                // if the browser doesn't support fractions, then round for consistent results
1276
                if ( !supportsOffsetFractions ) {
1277
                        position.left = round( position.left );
1278
                        position.top = round( position.top );
1279
                }
1280
1281
                collisionPosition = {
1282
                        marginLeft: marginLeft,
1283
                        marginTop: marginTop
1284
                };
1285
1286
                $.each( [ "left", "top" ], function( i, dir ) {
1287
                        if ( $.ui.position[ collision[ i ] ] ) {
1288
                                $.ui.position[ collision[ i ] ][ dir ]( position, {
1289
                                        targetWidth: targetWidth,
1290
                                        targetHeight: targetHeight,
1291
                                        elemWidth: elemWidth,
1292
                                        elemHeight: elemHeight,
1293
                                        collisionPosition: collisionPosition,
1294
                                        collisionWidth: collisionWidth,
1295
                                        collisionHeight: collisionHeight,
1296
                                        offset: [ atOffset[ 0 ] + myOffset[ 0 ], atOffset [ 1 ] + myOffset[ 1 ] ],
1297
                                        my: options.my,
1298
                                        at: options.at,
1299
                                        within: within,
1300
                                        elem: elem
1301
                                });
1302
                        }
1303
                });
1304
1305
                if ( options.using ) {
1306
                        // adds feedback as second argument to using callback, if present
1307
                        using = function( props ) {
1308
                                var left = targetOffset.left - position.left,
1309
                                        right = left + targetWidth - elemWidth,
1310
                                        top = targetOffset.top - position.top,
1311
                                        bottom = top + targetHeight - elemHeight,
1312
                                        feedback = {
1313
                                                target: {
1314
                                                        element: target,
1315
                                                        left: targetOffset.left,
1316
                                                        top: targetOffset.top,
1317
                                                        width: targetWidth,
1318
                                                        height: targetHeight
1319
                                                },
1320
                                                element: {
1321
                                                        element: elem,
1322
                                                        left: position.left,
1323
                                                        top: position.top,
1324
                                                        width: elemWidth,
1325
                                                        height: elemHeight
1326
                                                },
1327
                                                horizontal: right < 0 ? "left" : left > 0 ? "right" : "center",
1328
                                                vertical: bottom < 0 ? "top" : top > 0 ? "bottom" : "middle"
1329
                                        };
1330
                                if ( targetWidth < elemWidth && abs( left + right ) < targetWidth ) {
1331
                                        feedback.horizontal = "center";
1332
                                }
1333
                                if ( targetHeight < elemHeight && abs( top + bottom ) < targetHeight ) {
1334
                                        feedback.vertical = "middle";
1335
                                }
1336
                                if ( max( abs( left ), abs( right ) ) > max( abs( top ), abs( bottom ) ) ) {
1337
                                        feedback.important = "horizontal";
1338
                                } else {
1339
                                        feedback.important = "vertical";
1340
                                }
1341
                                options.using.call( this, props, feedback );
1342
                        };
1343
                }
1344
1345
                elem.offset( $.extend( position, { using: using } ) );
1346
        });
1347
};
1348
1349
$.ui.position = {
1350
        fit: {
1351
                left: function( position, data ) {
1352
                        var within = data.within,
1353
                                withinOffset = within.isWindow ? within.scrollLeft : within.offset.left,
1354
                                outerWidth = within.width,
1355
                                collisionPosLeft = position.left - data.collisionPosition.marginLeft,
1356
                                overLeft = withinOffset - collisionPosLeft,
1357
                                overRight = collisionPosLeft + data.collisionWidth - outerWidth - withinOffset,
1358
                                newOverRight;
1359
1360
                        // element is wider than within
1361
                        if ( data.collisionWidth > outerWidth ) {
1362
                                // element is initially over the left side of within
1363
                                if ( overLeft > 0 && overRight <= 0 ) {
1364
                                        newOverRight = position.left + overLeft + data.collisionWidth - outerWidth - withinOffset;
1365
                                        position.left += overLeft - newOverRight;
1366
                                // element is initially over right side of within
1367
                                } else if ( overRight > 0 && overLeft <= 0 ) {
1368
                                        position.left = withinOffset;
1369
                                // element is initially over both left and right sides of within
1370
                                } else {
1371
                                        if ( overLeft > overRight ) {
1372
                                                position.left = withinOffset + outerWidth - data.collisionWidth;
1373
                                        } else {
1374
                                                position.left = withinOffset;
1375
                                        }
1376
                                }
1377
                        // too far left -> align with left edge
1378
                        } else if ( overLeft > 0 ) {
1379
                                position.left += overLeft;
1380
                        // too far right -> align with right edge
1381
                        } else if ( overRight > 0 ) {
1382
                                position.left -= overRight;
1383
                        // adjust based on position and margin
1384
                        } else {
1385
                                position.left = max( position.left - collisionPosLeft, position.left );
1386
                        }
1387
                },
1388
                top: function( position, data ) {
1389
                        var within = data.within,
1390
                                withinOffset = within.isWindow ? within.scrollTop : within.offset.top,
1391
                                outerHeight = data.within.height,
1392
                                collisionPosTop = position.top - data.collisionPosition.marginTop,
1393
                                overTop = withinOffset - collisionPosTop,
1394
                                overBottom = collisionPosTop + data.collisionHeight - outerHeight - withinOffset,
1395
                                newOverBottom;
1396
1397
                        // element is taller than within
1398
                        if ( data.collisionHeight > outerHeight ) {
1399
                                // element is initially over the top of within
1400
                                if ( overTop > 0 && overBottom <= 0 ) {
1401
                                        newOverBottom = position.top + overTop + data.collisionHeight - outerHeight - withinOffset;
1402
                                        position.top += overTop - newOverBottom;
1403
                                // element is initially over bottom of within
1404
                                } else if ( overBottom > 0 && overTop <= 0 ) {
1405
                                        position.top = withinOffset;
1406
                                // element is initially over both top and bottom of within
1407
                                } else {
1408
                                        if ( overTop > overBottom ) {
1409
                                                position.top = withinOffset + outerHeight - data.collisionHeight;
1410
                                        } else {
1411
                                                position.top = withinOffset;
1412
                                        }
1413
                                }
1414
                        // too far up -> align with top
1415
                        } else if ( overTop > 0 ) {
1416
                                position.top += overTop;
1417
                        // too far down -> align with bottom edge
1418
                        } else if ( overBottom > 0 ) {
1419
                                position.top -= overBottom;
1420
                        // adjust based on position and margin
1421
                        } else {
1422
                                position.top = max( position.top - collisionPosTop, position.top );
1423
                        }
1424
                }
1425
        },
1426
        flip: {
1427
                left: function( position, data ) {
1428
                        var within = data.within,
1429
                                withinOffset = within.offset.left + within.scrollLeft,
1430
                                outerWidth = within.width,
1431
                                offsetLeft = within.isWindow ? within.scrollLeft : within.offset.left,
1432
                                collisionPosLeft = position.left - data.collisionPosition.marginLeft,
1433
                                overLeft = collisionPosLeft - offsetLeft,
1434
                                overRight = collisionPosLeft + data.collisionWidth - outerWidth - offsetLeft,
1435
                                myOffset = data.my[ 0 ] === "left" ?
1436
                                        -data.elemWidth :
1437
                                        data.my[ 0 ] === "right" ?
1438
                                                data.elemWidth :
1439
                                                0,
1440
                                atOffset = data.at[ 0 ] === "left" ?
1441
                                        data.targetWidth :
1442
                                        data.at[ 0 ] === "right" ?
1443
                                                -data.targetWidth :
1444
                                                0,
1445
                                offset = -2 * data.offset[ 0 ],
1446
                                newOverRight,
1447
                                newOverLeft;
1448
1449
                        if ( overLeft < 0 ) {
1450
                                newOverRight = position.left + myOffset + atOffset + offset + data.collisionWidth - outerWidth - withinOffset;
1451
                                if ( newOverRight < 0 || newOverRight < abs( overLeft ) ) {
1452
                                        position.left += myOffset + atOffset + offset;
1453
                                }
1454
                        } else if ( overRight > 0 ) {
1455
                                newOverLeft = position.left - data.collisionPosition.marginLeft + myOffset + atOffset + offset - offsetLeft;
1456
                                if ( newOverLeft > 0 || abs( newOverLeft ) < overRight ) {
1457
                                        position.left += myOffset + atOffset + offset;
1458
                                }
1459
                        }
1460
                },
1461
                top: function( position, data ) {
1462
                        var within = data.within,
1463
                                withinOffset = within.offset.top + within.scrollTop,
1464
                                outerHeight = within.height,
1465
                                offsetTop = within.isWindow ? within.scrollTop : within.offset.top,
1466
                                collisionPosTop = position.top - data.collisionPosition.marginTop,
1467
                                overTop = collisionPosTop - offsetTop,
1468
                                overBottom = collisionPosTop + data.collisionHeight - outerHeight - offsetTop,
1469
                                top = data.my[ 1 ] === "top",
1470
                                myOffset = top ?
1471
                                        -data.elemHeight :
1472
                                        data.my[ 1 ] === "bottom" ?
1473
                                                data.elemHeight :
1474
                                                0,
1475
                                atOffset = data.at[ 1 ] === "top" ?
1476
                                        data.targetHeight :
1477
                                        data.at[ 1 ] === "bottom" ?
1478
                                                -data.targetHeight :
1479
                                                0,
1480
                                offset = -2 * data.offset[ 1 ],
1481
                                newOverTop,
1482
                                newOverBottom;
1483
                        if ( overTop < 0 ) {
1484
                                newOverBottom = position.top + myOffset + atOffset + offset + data.collisionHeight - outerHeight - withinOffset;
1485
                                if ( newOverBottom < 0 || newOverBottom < abs( overTop ) ) {
1486
                                        position.top += myOffset + atOffset + offset;
1487
                                }
1488
                        } else if ( overBottom > 0 ) {
1489
                                newOverTop = position.top - data.collisionPosition.marginTop + myOffset + atOffset + offset - offsetTop;
1490
                                if ( newOverTop > 0 || abs( newOverTop ) < overBottom ) {
1491
                                        position.top += myOffset + atOffset + offset;
1492
                                }
1493
                        }
1494
                }
1495
        },
1496
        flipfit: {
1497
                left: function() {
1498
                        $.ui.position.flip.left.apply( this, arguments );
1499
                        $.ui.position.fit.left.apply( this, arguments );
1500
                },
1501
                top: function() {
1502
                        $.ui.position.flip.top.apply( this, arguments );
1503
                        $.ui.position.fit.top.apply( this, arguments );
1504
                }
1505
        }
1506
};
1507
1508
// fraction support test
1509
(function() {
1510
        var testElement, testElementParent, testElementStyle, offsetLeft, i,
1511
                body = document.getElementsByTagName( "body" )[ 0 ],
1512
                div = document.createElement( "div" );
1513
1514
        //Create a "fake body" for testing based on method used in jQuery.support
1515
        testElement = document.createElement( body ? "div" : "body" );
1516
        testElementStyle = {
1517
                visibility: "hidden",
1518
                width: 0,
1519
                height: 0,
1520
                border: 0,
1521
                margin: 0,
1522
                background: "none"
1523
        };
1524
        if ( body ) {
1525
                $.extend( testElementStyle, {
1526
                        position: "absolute",
1527
                        left: "-1000px",
1528
                        top: "-1000px"
1529
                });
1530
        }
1531
        for ( i in testElementStyle ) {
1532
                testElement.style[ i ] = testElementStyle[ i ];
1533
        }
1534
        testElement.appendChild( div );
1535
        testElementParent = body || document.documentElement;
1536
        testElementParent.insertBefore( testElement, testElementParent.firstChild );
1537
1538
        div.style.cssText = "position: absolute; left: 10.7432222px;";
1539
1540
        offsetLeft = $( div ).offset().left;
1541
        supportsOffsetFractions = offsetLeft > 10 && offsetLeft < 11;
1542
1543
        testElement.innerHTML = "";
1544
        testElementParent.removeChild( testElement );
1545
})();
1546
1547
})();
1548
1549
var position = $.ui.position;
1550
1551
1552
/*!
1553
 * jQuery UI Accordion 1.11.4
1554
 * http://jqueryui.com
1555
 *
1556
 * Copyright jQuery Foundation and other contributors
1557
 * Released under the MIT license.
1558
 * http://jquery.org/license
1559
 *
1560
 * http://api.jqueryui.com/accordion/
1561
 */
1562
1563
1564
var accordion = $.widget( "ui.accordion", {
1565
        version: "1.11.4",
1566
        options: {
1567
                active: 0,
1568
                animate: {},
1569
                collapsible: false,
1570
                event: "click",
1571
                header: "> li > :first-child,> :not(li):even",
1572
                heightStyle: "auto",
1573
                icons: {
1574
                        activeHeader: "ui-icon-triangle-1-s",
1575
                        header: "ui-icon-triangle-1-e"
1576
                },
1577
1578
                // callbacks
1579
                activate: null,
1580
                beforeActivate: null
1581
        },
1582
1583
        hideProps: {
1584
                borderTopWidth: "hide",
1585
                borderBottomWidth: "hide",
1586
                paddingTop: "hide",
1587
                paddingBottom: "hide",
1588
                height: "hide"
1589
        },
1590
1591
        showProps: {
1592
                borderTopWidth: "show",
1593
                borderBottomWidth: "show",
1594
                paddingTop: "show",
1595
                paddingBottom: "show",
1596
                height: "show"
1597
        },
1598
1599
        _create: function() {
1600
                var options = this.options;
1601
                this.prevShow = this.prevHide = $();
1602
                this.element.addClass( "ui-accordion ui-widget ui-helper-reset" )
1603
                        // ARIA
1604
                        .attr( "role", "tablist" );
1605
1606
                // don't allow collapsible: false and active: false / null
1607
                if ( !options.collapsible && (options.active === false || options.active == null) ) {
1608
                        options.active = 0;
1609
                }
1610
1611
                this._processPanels();
1612
                // handle negative values
1613
                if ( options.active < 0 ) {
1614
                        options.active += this.headers.length;
1615
                }
1616
                this._refresh();
1617
        },
1618
1619
        _getCreateEventData: function() {
1620
                return {
1621
                        header: this.active,
1622
                        panel: !this.active.length ? $() : this.active.next()
1623
                };
1624
        },
1625
1626
        _createIcons: function() {
1627
                var icons = this.options.icons;
1628
                if ( icons ) {
1629
                        $( "<span>" )
1630
                                .addClass( "ui-accordion-header-icon ui-icon " + icons.header )
1631
                                .prependTo( this.headers );
1632
                        this.active.children( ".ui-accordion-header-icon" )
1633
                                .removeClass( icons.header )
1634
                                .addClass( icons.activeHeader );
1635
                        this.headers.addClass( "ui-accordion-icons" );
1636
                }
1637
        },
1638
1639
        _destroyIcons: function() {
1640
                this.headers
1641
                        .removeClass( "ui-accordion-icons" )
1642
                        .children( ".ui-accordion-header-icon" )
1643
                                .remove();
1644
        },
1645
1646
        _destroy: function() {
1647
                var contents;
1648
1649
                // clean up main element
1650
                this.element
1651
                        .removeClass( "ui-accordion ui-widget ui-helper-reset" )
1652
                        .removeAttr( "role" );
1653
1654
                // clean up headers
1655
                this.headers
1656
                        .removeClass( "ui-accordion-header ui-accordion-header-active ui-state-default " +
1657
                                "ui-corner-all ui-state-active ui-state-disabled ui-corner-top" )
1658
                        .removeAttr( "role" )
1659
                        .removeAttr( "aria-expanded" )
1660
                        .removeAttr( "aria-selected" )
1661
                        .removeAttr( "aria-controls" )
1662
                        .removeAttr( "tabIndex" )
1663
                        .removeUniqueId();
1664
1665
                this._destroyIcons();
1666
1667
                // clean up content panels
1668
                contents = this.headers.next()
1669
                        .removeClass( "ui-helper-reset ui-widget-content ui-corner-bottom " +
1670
                                "ui-accordion-content ui-accordion-content-active ui-state-disabled" )
1671
                        .css( "display", "" )
1672
                        .removeAttr( "role" )
1673
                        .removeAttr( "aria-hidden" )
1674
                        .removeAttr( "aria-labelledby" )
1675
                        .removeUniqueId();
1676
1677
                if ( this.options.heightStyle !== "content" ) {
1678
                        contents.css( "height", "" );
1679
                }
1680
        },
1681
1682
        _setOption: function( key, value ) {
1683
                if ( key === "active" ) {
1684
                        // _activate() will handle invalid values and update this.options
1685
                        this._activate( value );
1686
                        return;
1687
                }
1688
1689
                if ( key === "event" ) {
1690
                        if ( this.options.event ) {
1691
                                this._off( this.headers, this.options.event );
1692
                        }
1693
                        this._setupEvents( value );
1694
                }
1695
1696
                this._super( key, value );
1697
1698
                // setting collapsible: false while collapsed; open first panel
1699
                if ( key === "collapsible" && !value && this.options.active === false ) {
1700
                        this._activate( 0 );
1701
                }
1702
1703
                if ( key === "icons" ) {
1704
                        this._destroyIcons();
1705
                        if ( value ) {
1706
                                this._createIcons();
1707
                        }
1708
                }
1709
1710
                // #5332 - opacity doesn't cascade to positioned elements in IE
1711
                // so we need to add the disabled class to the headers and panels
1712
                if ( key === "disabled" ) {
1713
                        this.element
1714
                                .toggleClass( "ui-state-disabled", !!value )
1715
                                .attr( "aria-disabled", value );
1716
                        this.headers.add( this.headers.next() )
1717
                                .toggleClass( "ui-state-disabled", !!value );
1718
                }
1719
        },
1720
1721
        _keydown: function( event ) {
1722
                if ( event.altKey || event.ctrlKey ) {
1723
                        return;
1724
                }
1725
1726
                var keyCode = $.ui.keyCode,
1727
                        length = this.headers.length,
1728
                        currentIndex = this.headers.index( event.target ),
1729
                        toFocus = false;
1730
1731
                switch ( event.keyCode ) {
1732
                        case keyCode.RIGHT:
1733
                        case keyCode.DOWN:
1734
                                toFocus = this.headers[ ( currentIndex + 1 ) % length ];
1735
                                break;
1736
                        case keyCode.LEFT:
1737
                        case keyCode.UP:
1738
                                toFocus = this.headers[ ( currentIndex - 1 + length ) % length ];
1739
                                break;
1740
                        case keyCode.SPACE:
1741
                        case keyCode.ENTER:
1742
                                this._eventHandler( event );
1743
                                break;
1744
                        case keyCode.HOME:
1745
                                toFocus = this.headers[ 0 ];
1746
                                break;
1747
                        case keyCode.END:
1748
                                toFocus = this.headers[ length - 1 ];
1749
                                break;
1750
                }
1751
1752
                if ( toFocus ) {
1753
                        $( event.target ).attr( "tabIndex", -1 );
1754
                        $( toFocus ).attr( "tabIndex", 0 );
1755
                        toFocus.focus();
1756
                        event.preventDefault();
1757
                }
1758
        },
1759
1760
        _panelKeyDown: function( event ) {
1761
                if ( event.keyCode === $.ui.keyCode.UP && event.ctrlKey ) {
1762
                        $( event.currentTarget ).prev().focus();
1763
                }
1764
        },
1765
1766
        refresh: function() {
1767
                var options = this.options;
1768
                this._processPanels();
1769
1770
                // was collapsed or no panel
1771
                if ( ( options.active === false && options.collapsible === true ) || !this.headers.length ) {
1772
                        options.active = false;
1773
                        this.active = $();
1774
                // active false only when collapsible is true
1775
                } else if ( options.active === false ) {
1776
                        this._activate( 0 );
1777
                // was active, but active panel is gone
1778
                } else if ( this.active.length && !$.contains( this.element[ 0 ], this.active[ 0 ] ) ) {
1779
                        // all remaining panel are disabled
1780
                        if ( this.headers.length === this.headers.find(".ui-state-disabled").length ) {
1781
                                options.active = false;
1782
                                this.active = $();
1783
                        // activate previous panel
1784
                        } else {
1785
                                this._activate( Math.max( 0, options.active - 1 ) );
1786
                        }
1787
                // was active, active panel still exists
1788
                } else {
1789
                        // make sure active index is correct
1790
                        options.active = this.headers.index( this.active );
1791
                }
1792
1793
                this._destroyIcons();
1794
1795
                this._refresh();
1796
        },
1797
1798
        _processPanels: function() {
1799
                var prevHeaders = this.headers,
1800
                        prevPanels = this.panels;
1801
1802
                this.headers = this.element.find( this.options.header )
1803
                        .addClass( "ui-accordion-header ui-state-default ui-corner-all" );
1804
1805
                this.panels = this.headers.next()
1806
                        .addClass( "ui-accordion-content ui-helper-reset ui-widget-content ui-corner-bottom" )
1807
                        .filter( ":not(.ui-accordion-content-active)" )
1808
                        .hide();
1809
1810
                // Avoid memory leaks (#10056)
1811
                if ( prevPanels ) {
1812
                        this._off( prevHeaders.not( this.headers ) );
1813
                        this._off( prevPanels.not( this.panels ) );
1814
                }
1815
        },
1816
1817
        _refresh: function() {
1818
                var maxHeight,
1819
                        options = this.options,
1820
                        heightStyle = options.heightStyle,
1821
                        parent = this.element.parent();
1822
1823
                this.active = this._findActive( options.active )
1824
                        .addClass( "ui-accordion-header-active ui-state-active ui-corner-top" )
1825
                        .removeClass( "ui-corner-all" );
1826
                this.active.next()
1827
                        .addClass( "ui-accordion-content-active" )
1828
                        .show();
1829
1830
                this.headers
1831
                        .attr( "role", "tab" )
1832
                        .each(function() {
1833
                                var header = $( this ),
1834
                                        headerId = header.uniqueId().attr( "id" ),
1835
                                        panel = header.next(),
1836
                                        panelId = panel.uniqueId().attr( "id" );
1837
                                header.attr( "aria-controls", panelId );
1838
                                panel.attr( "aria-labelledby", headerId );
1839
                        })
1840
                        .next()
1841
                                .attr( "role", "tabpanel" );
1842
1843
                this.headers
1844
                        .not( this.active )
1845
                        .attr({
1846
                                "aria-selected": "false",
1847
                                "aria-expanded": "false",
1848
                                tabIndex: -1
1849
                        })
1850
                        .next()
1851
                                .attr({
1852
                                        "aria-hidden": "true"
1853
                                })
1854
                                .hide();
1855
1856
                // make sure at least one header is in the tab order
1857
                if ( !this.active.length ) {
1858
                        this.headers.eq( 0 ).attr( "tabIndex", 0 );
1859
                } else {
1860
                        this.active.attr({
1861
                                "aria-selected": "true",
1862
                                "aria-expanded": "true",
1863
                                tabIndex: 0
1864
                        })
1865
                        .next()
1866
                                .attr({
1867
                                        "aria-hidden": "false"
1868
                                });
1869
                }
1870
1871
                this._createIcons();
1872
1873
                this._setupEvents( options.event );
1874
1875
                if ( heightStyle === "fill" ) {
1876
                        maxHeight = parent.height();
1877
                        this.element.siblings( ":visible" ).each(function() {
1878
                                var elem = $( this ),
1879
                                        position = elem.css( "position" );
1880
1881
                                if ( position === "absolute" || position === "fixed" ) {
1882
                                        return;
1883
                                }
1884
                                maxHeight -= elem.outerHeight( true );
1885
                        });
1886
1887
                        this.headers.each(function() {
1888
                                maxHeight -= $( this ).outerHeight( true );
1889
                        });
1890
1891
                        this.headers.next()
1892
                                .each(function() {
1893
                                        $( this ).height( Math.max( 0, maxHeight -
1894
                                                $( this ).innerHeight() + $( this ).height() ) );
1895
                                })
1896
                                .css( "overflow", "auto" );
1897
                } else if ( heightStyle === "auto" ) {
1898
                        maxHeight = 0;
1899
                        this.headers.next()
1900
                                .each(function() {
1901
                                        maxHeight = Math.max( maxHeight, $( this ).css( "height", "" ).height() );
1902
                                })
1903
                                .height( maxHeight );
1904
                }
1905
        },
1906
1907
        _activate: function( index ) {
1908
                var active = this._findActive( index )[ 0 ];
1909
1910
                // trying to activate the already active panel
1911
                if ( active === this.active[ 0 ] ) {
1912
                        return;
1913
                }
1914
1915
                // trying to collapse, simulate a click on the currently active header
1916
                active = active || this.active[ 0 ];
1917
1918
                this._eventHandler({
1919
                        target: active,
1920
                        currentTarget: active,
1921
                        preventDefault: $.noop
1922
                });
1923
        },
1924
1925
        _findActive: function( selector ) {
1926
                return typeof selector === "number" ? this.headers.eq( selector ) : $();
1927
        },
1928
1929
        _setupEvents: function( event ) {
1930
                var events = {
1931
                        keydown: "_keydown"
1932
                };
1933
                if ( event ) {
1934
                        $.each( event.split( " " ), function( index, eventName ) {
1935
                                events[ eventName ] = "_eventHandler";
1936
                        });
1937
                }
1938
1939
                this._off( this.headers.add( this.headers.next() ) );
1940
                this._on( this.headers, events );
1941
                this._on( this.headers.next(), { keydown: "_panelKeyDown" });
1942
                this._hoverable( this.headers );
1943
                this._focusable( this.headers );
1944
        },
1945
1946
        _eventHandler: function( event ) {
1947
                var options = this.options,
1948
                        active = this.active,
1949
                        clicked = $( event.currentTarget ),
1950
                        clickedIsActive = clicked[ 0 ] === active[ 0 ],
1951
                        collapsing = clickedIsActive && options.collapsible,
1952
                        toShow = collapsing ? $() : clicked.next(),
1953
                        toHide = active.next(),
1954
                        eventData = {
1955
                                oldHeader: active,
1956
                                oldPanel: toHide,
1957
                                newHeader: collapsing ? $() : clicked,
1958
                                newPanel: toShow
1959
                        };
1960
1961
                event.preventDefault();
1962
1963
                if (
1964
                                // click on active header, but not collapsible
1965
                                ( clickedIsActive && !options.collapsible ) ||
1966
                                // allow canceling activation
1967
                                ( this._trigger( "beforeActivate", event, eventData ) === false ) ) {
1968
                        return;
1969
                }
1970
1971
                options.active = collapsing ? false : this.headers.index( clicked );
1972
1973
                // when the call to ._toggle() comes after the class changes
1974
                // it causes a very odd bug in IE 8 (see #6720)
1975
                this.active = clickedIsActive ? $() : clicked;
1976
                this._toggle( eventData );
1977
1978
                // switch classes
1979
                // corner classes on the previously active header stay after the animation
1980
                active.removeClass( "ui-accordion-header-active ui-state-active" );
1981
                if ( options.icons ) {
1982
                        active.children( ".ui-accordion-header-icon" )
1983
                                .removeClass( options.icons.activeHeader )
1984
                                .addClass( options.icons.header );
1985
                }
1986
1987
                if ( !clickedIsActive ) {
1988
                        clicked
1989
                                .removeClass( "ui-corner-all" )
1990
                                .addClass( "ui-accordion-header-active ui-state-active ui-corner-top" );
1991
                        if ( options.icons ) {
1992
                                clicked.children( ".ui-accordion-header-icon" )
1993
                                        .removeClass( options.icons.header )
1994
                                        .addClass( options.icons.activeHeader );
1995
                        }
1996
1997
                        clicked
1998
                                .next()
1999
                                .addClass( "ui-accordion-content-active" );
2000
                }
2001
        },
2002
2003
        _toggle: function( data ) {
2004
                var toShow = data.newPanel,
2005
                        toHide = this.prevShow.length ? this.prevShow : data.oldPanel;
2006
2007
                // handle activating a panel during the animation for another activation
2008
                this.prevShow.add( this.prevHide ).stop( true, true );
2009
                this.prevShow = toShow;
2010
                this.prevHide = toHide;
2011
2012
                if ( this.options.animate ) {
2013
                        this._animate( toShow, toHide, data );
2014
                } else {
2015
                        toHide.hide();
2016
                        toShow.show();
2017
                        this._toggleComplete( data );
2018
                }
2019
2020
                toHide.attr({
2021
                        "aria-hidden": "true"
2022
                });
2023
                toHide.prev().attr({
2024
                        "aria-selected": "false",
2025
                        "aria-expanded": "false"
2026
                });
2027
                // if we're switching panels, remove the old header from the tab order
2028
                // if we're opening from collapsed state, remove the previous header from the tab order
2029
                // if we're collapsing, then keep the collapsing header in the tab order
2030
                if ( toShow.length && toHide.length ) {
2031
                        toHide.prev().attr({
2032
                                "tabIndex": -1,
2033
                                "aria-expanded": "false"
2034
                        });
2035
                } else if ( toShow.length ) {
2036
                        this.headers.filter(function() {
2037
                                return parseInt( $( this ).attr( "tabIndex" ), 10 ) === 0;
2038
                        })
2039
                        .attr( "tabIndex", -1 );
2040
                }
2041
2042
                toShow
2043
                        .attr( "aria-hidden", "false" )
2044
                        .prev()
2045
                                .attr({
2046
                                        "aria-selected": "true",
2047
                                        "aria-expanded": "true",
2048
                                        tabIndex: 0
2049
                                });
2050
        },
2051
2052
        _animate: function( toShow, toHide, data ) {
2053
                var total, easing, duration,
2054
                        that = this,
2055
                        adjust = 0,
2056
                        boxSizing = toShow.css( "box-sizing" ),
2057
                        down = toShow.length &&
2058
                                ( !toHide.length || ( toShow.index() < toHide.index() ) ),
2059
                        animate = this.options.animate || {},
2060
                        options = down && animate.down || animate,
2061
                        complete = function() {
2062
                                that._toggleComplete( data );
2063
                        };
2064
2065
                if ( typeof options === "number" ) {
2066
                        duration = options;
2067
                }
2068
                if ( typeof options === "string" ) {
2069
                        easing = options;
2070
                }
2071
                // fall back from options to animation in case of partial down settings
2072
                easing = easing || options.easing || animate.easing;
2073
                duration = duration || options.duration || animate.duration;
2074
2075
                if ( !toHide.length ) {
2076
                        return toShow.animate( this.showProps, duration, easing, complete );
2077
                }
2078
                if ( !toShow.length ) {
2079
                        return toHide.animate( this.hideProps, duration, easing, complete );
2080
                }
2081
2082
                total = toShow.show().outerHeight();
2083
                toHide.animate( this.hideProps, {
2084
                        duration: duration,
2085
                        easing: easing,
2086
                        step: function( now, fx ) {
2087
                                fx.now = Math.round( now );
2088
                        }
2089
                });
2090
                toShow
2091
                        .hide()
2092
                        .animate( this.showProps, {
2093
                                duration: duration,
2094
                                easing: easing,
2095
                                complete: complete,
2096
                                step: function( now, fx ) {
2097
                                        fx.now = Math.round( now );
2098
                                        if ( fx.prop !== "height" ) {
2099
                                                if ( boxSizing === "content-box" ) {
2100
                                                        adjust += fx.now;
2101
                                                }
2102
                                        } else if ( that.options.heightStyle !== "content" ) {
2103
                                                fx.now = Math.round( total - toHide.outerHeight() - adjust );
2104
                                                adjust = 0;
2105
                                        }
2106
                                }
2107
                        });
2108
        },
2109
2110
        _toggleComplete: function( data ) {
2111
                var toHide = data.oldPanel;
2112
2113
                toHide
2114
                        .removeClass( "ui-accordion-content-active" )
2115
                        .prev()
2116
                                .removeClass( "ui-corner-top" )
2117
                                .addClass( "ui-corner-all" );
2118
2119
                // Work around for rendering bug in IE (#5421)
2120
                if ( toHide.length ) {
2121
                        toHide.parent()[ 0 ].className = toHide.parent()[ 0 ].className;
2122
                }
2123
                this._trigger( "activate", null, data );
2124
        }
2125
});
2126
2127
2128
/*!
2129
 * jQuery UI Menu 1.11.4
2130
 * http://jqueryui.com
2131
 *
2132
 * Copyright jQuery Foundation and other contributors
2133
 * Released under the MIT license.
2134
 * http://jquery.org/license
2135
 *
2136
 * http://api.jqueryui.com/menu/
2137
 */
2138
2139
2140
var menu = $.widget( "ui.menu", {
2141
        version: "1.11.4",
2142
        defaultElement: "<ul>",
2143
        delay: 300,
2144
        options: {
2145
                icons: {
2146
                        submenu: "ui-icon-carat-1-e"
2147
                },
2148
                items: "> *",
2149
                menus: "ul",
2150
                position: {
2151
                        my: "left-1 top",
2152
                        at: "right top"
2153
                },
2154
                role: "menu",
2155
2156
                // callbacks
2157
                blur: null,
2158
                focus: null,
2159
                select: null
2160
        },
2161
2162
        _create: function() {
2163
                this.activeMenu = this.element;
2164
2165
                // Flag used to prevent firing of the click handler
2166
                // as the event bubbles up through nested menus
2167
                this.mouseHandled = false;
2168
                this.element
2169
                        .uniqueId()
2170
                        .addClass( "ui-menu ui-widget ui-widget-content" )
2171
                        .toggleClass( "ui-menu-icons", !!this.element.find( ".ui-icon" ).length )
2172
                        .attr({
2173
                                role: this.options.role,
2174
                                tabIndex: 0
2175
                        });
2176
2177
                if ( this.options.disabled ) {
2178
                        this.element
2179
                                .addClass( "ui-state-disabled" )
2180
                                .attr( "aria-disabled", "true" );
2181
                }
2182
2183
                this._on({
2184
                        // Prevent focus from sticking to links inside menu after clicking
2185
                        // them (focus should always stay on UL during navigation).
2186
                        "mousedown .ui-menu-item": function( event ) {
2187
                                event.preventDefault();
2188
                        },
2189
                        "click .ui-menu-item": function( event ) {
2190
                                var target = $( event.target );
2191
                                if ( !this.mouseHandled && target.not( ".ui-state-disabled" ).length ) {
2192
                                        this.select( event );
2193
2194
                                        // Only set the mouseHandled flag if the event will bubble, see #9469.
2195
                                        if ( !event.isPropagationStopped() ) {
2196
                                                this.mouseHandled = true;
2197
                                        }
2198
2199
                                        // Open submenu on click
2200
                                        if ( target.has( ".ui-menu" ).length ) {
2201
                                                this.expand( event );
2202
                                        } else if ( !this.element.is( ":focus" ) && $( this.document[ 0 ].activeElement ).closest( ".ui-menu" ).length ) {
2203
2204
                                                // Redirect focus to the menu
2205
                                                this.element.trigger( "focus", [ true ] );
2206
2207
                                                // If the active item is on the top level, let it stay active.
2208
                                                // Otherwise, blur the active item since it is no longer visible.
2209
                                                if ( this.active && this.active.parents( ".ui-menu" ).length === 1 ) {
2210
                                                        clearTimeout( this.timer );
2211
                                                }
2212
                                        }
2213
                                }
2214
                        },
2215
                        "mouseenter .ui-menu-item": function( event ) {
2216
                                // Ignore mouse events while typeahead is active, see #10458.
2217
                                // Prevents focusing the wrong item when typeahead causes a scroll while the mouse
2218
                                // is over an item in the menu
2219
                                if ( this.previousFilter ) {
2220
                                        return;
2221
                                }
2222
                                var target = $( event.currentTarget );
2223
                                // Remove ui-state-active class from siblings of the newly focused menu item
2224
                                // to avoid a jump caused by adjacent elements both having a class with a border
2225
                                target.siblings( ".ui-state-active" ).removeClass( "ui-state-active" );
2226
                                this.focus( event, target );
2227
                        },
2228
                        mouseleave: "collapseAll",
2229
                        "mouseleave .ui-menu": "collapseAll",
2230
                        focus: function( event, keepActiveItem ) {
2231
                                // If there's already an active item, keep it active
2232
                                // If not, activate the first item
2233
                                var item = this.active || this.element.find( this.options.items ).eq( 0 );
2234
2235
                                if ( !keepActiveItem ) {
2236
                                        this.focus( event, item );
2237
                                }
2238
                        },
2239
                        blur: function( event ) {
2240
                                this._delay(function() {
2241
                                        if ( !$.contains( this.element[0], this.document[0].activeElement ) ) {
2242
                                                this.collapseAll( event );
2243
                                        }
2244
                                });
2245
                        },
2246
                        keydown: "_keydown"
2247
                });
2248
2249
                this.refresh();
2250
2251
                // Clicks outside of a menu collapse any open menus
2252
                this._on( this.document, {
2253
                        click: function( event ) {
2254
                                if ( this._closeOnDocumentClick( event ) ) {
2255
                                        this.collapseAll( event );
2256
                                }
2257
2258
                                // Reset the mouseHandled flag
2259
                                this.mouseHandled = false;
2260
                        }
2261
                });
2262
        },
2263
2264
        _destroy: function() {
2265
                // Destroy (sub)menus
2266
                this.element
2267
                        .removeAttr( "aria-activedescendant" )
2268
                        .find( ".ui-menu" ).addBack()
2269
                                .removeClass( "ui-menu ui-widget ui-widget-content ui-menu-icons ui-front" )
2270
                                .removeAttr( "role" )
2271
                                .removeAttr( "tabIndex" )
2272
                                .removeAttr( "aria-labelledby" )
2273
                                .removeAttr( "aria-expanded" )
2274
                                .removeAttr( "aria-hidden" )
2275
                                .removeAttr( "aria-disabled" )
2276
                                .removeUniqueId()
2277
                                .show();
2278
2279
                // Destroy menu items
2280
                this.element.find( ".ui-menu-item" )
2281
                        .removeClass( "ui-menu-item" )
2282
                        .removeAttr( "role" )
2283
                        .removeAttr( "aria-disabled" )
2284
                        .removeUniqueId()
2285
                        .removeClass( "ui-state-hover" )
2286
                        .removeAttr( "tabIndex" )
2287
                        .removeAttr( "role" )
2288
                        .removeAttr( "aria-haspopup" )
2289
                        .children().each( function() {
2290
                                var elem = $( this );
2291
                                if ( elem.data( "ui-menu-submenu-carat" ) ) {
2292
                                        elem.remove();
2293
                                }
2294
                        });
2295
2296
                // Destroy menu dividers
2297
                this.element.find( ".ui-menu-divider" ).removeClass( "ui-menu-divider ui-widget-content" );
2298
        },
2299
2300
        _keydown: function( event ) {
2301
                var match, prev, character, skip,
2302
                        preventDefault = true;
2303
2304
                switch ( event.keyCode ) {
2305
                case $.ui.keyCode.PAGE_UP:
2306
                        this.previousPage( event );
2307
                        break;
2308
                case $.ui.keyCode.PAGE_DOWN:
2309
                        this.nextPage( event );
2310
                        break;
2311
                case $.ui.keyCode.HOME:
2312
                        this._move( "first", "first", event );
2313
                        break;
2314
                case $.ui.keyCode.END:
2315
                        this._move( "last", "last", event );
2316
                        break;
2317
                case $.ui.keyCode.UP:
2318
                        this.previous( event );
2319
                        break;
2320
                case $.ui.keyCode.DOWN:
2321
                        this.next( event );
2322
                        break;
2323
                case $.ui.keyCode.LEFT:
2324
                        this.collapse( event );
2325
                        break;
2326
                case $.ui.keyCode.RIGHT:
2327
                        if ( this.active && !this.active.is( ".ui-state-disabled" ) ) {
2328
                                this.expand( event );
2329
                        }
2330
                        break;
2331
                case $.ui.keyCode.ENTER:
2332
                case $.ui.keyCode.SPACE:
2333
                        this._activate( event );
2334
                        break;
2335
                case $.ui.keyCode.ESCAPE:
2336
                        this.collapse( event );
2337
                        break;
2338
                default:
2339
                        preventDefault = false;
2340
                        prev = this.previousFilter || "";
2341
                        character = String.fromCharCode( event.keyCode );
2342
                        skip = false;
2343
2344
                        clearTimeout( this.filterTimer );
2345
2346
                        if ( character === prev ) {
2347
                                skip = true;
2348
                        } else {
2349
                                character = prev + character;
2350
                        }
2351
2352
                        match = this._filterMenuItems( character );
2353
                        match = skip && match.index( this.active.next() ) !== -1 ?
2354
                                this.active.nextAll( ".ui-menu-item" ) :
2355
                                match;
2356
2357
                        // If no matches on the current filter, reset to the last character pressed
2358
                        // to move down the menu to the first item that starts with that character
2359
                        if ( !match.length ) {
2360
                                character = String.fromCharCode( event.keyCode );
2361
                                match = this._filterMenuItems( character );
2362
                        }
2363
2364
                        if ( match.length ) {
2365
                                this.focus( event, match );
2366
                                this.previousFilter = character;
2367
                                this.filterTimer = this._delay(function() {
2368
                                        delete this.previousFilter;
2369
                                }, 1000 );
2370
                        } else {
2371
                                delete this.previousFilter;
2372
                        }
2373
                }
2374
2375
                if ( preventDefault ) {
2376
                        event.preventDefault();
2377
                }
2378
        },
2379
2380
        _activate: function( event ) {
2381
                if ( !this.active.is( ".ui-state-disabled" ) ) {
2382
                        if ( this.active.is( "[aria-haspopup='true']" ) ) {
2383
                                this.expand( event );
2384
                        } else {
2385
                                this.select( event );
2386
                        }
2387
                }
2388
        },
2389
2390
        refresh: function() {
2391
                var menus, items,
2392
                        that = this,
2393
                        icon = this.options.icons.submenu,
2394
                        submenus = this.element.find( this.options.menus );
2395
2396
                this.element.toggleClass( "ui-menu-icons", !!this.element.find( ".ui-icon" ).length );
2397
2398
                // Initialize nested menus
2399
                submenus.filter( ":not(.ui-menu)" )
2400
                        .addClass( "ui-menu ui-widget ui-widget-content ui-front" )
2401
                        .hide()
2402
                        .attr({
2403
                                role: this.options.role,
2404
                                "aria-hidden": "true",
2405
                                "aria-expanded": "false"
2406
                        })
2407
                        .each(function() {
2408
                                var menu = $( this ),
2409
                                        item = menu.parent(),
2410
                                        submenuCarat = $( "<span>" )
2411
                                                .addClass( "ui-menu-icon ui-icon " + icon )
2412
                                                .data( "ui-menu-submenu-carat", true );
2413
2414
                                item
2415
                                        .attr( "aria-haspopup", "true" )
2416
                                        .prepend( submenuCarat );
2417
                                menu.attr( "aria-labelledby", item.attr( "id" ) );
2418
                        });
2419
2420
                menus = submenus.add( this.element );
2421
                items = menus.find( this.options.items );
2422
2423
                // Initialize menu-items containing spaces and/or dashes only as dividers
2424
                items.not( ".ui-menu-item" ).each(function() {
2425
                        var item = $( this );
2426
                        if ( that._isDivider( item ) ) {
2427
                                item.addClass( "ui-widget-content ui-menu-divider" );
2428
                        }
2429
                });
2430
2431
                // Don't refresh list items that are already adapted
2432
                items.not( ".ui-menu-item, .ui-menu-divider" )
2433
                        .addClass( "ui-menu-item" )
2434
                        .uniqueId()
2435
                        .attr({
2436
                                tabIndex: -1,
2437
                                role: this._itemRole()
2438
                        });
2439
2440
                // Add aria-disabled attribute to any disabled menu item
2441
                items.filter( ".ui-state-disabled" ).attr( "aria-disabled", "true" );
2442
2443
                // If the active item has been removed, blur the menu
2444
                if ( this.active && !$.contains( this.element[ 0 ], this.active[ 0 ] ) ) {
2445
                        this.blur();
2446
                }
2447
        },
2448
2449
        _itemRole: function() {
2450
                return {
2451
                        menu: "menuitem",
2452
                        listbox: "option"
2453
                }[ this.options.role ];
2454
        },
2455
2456
        _setOption: function( key, value ) {
2457
                if ( key === "icons" ) {
2458
                        this.element.find( ".ui-menu-icon" )
2459
                                .removeClass( this.options.icons.submenu )
2460
                                .addClass( value.submenu );
2461
                }
2462
                if ( key === "disabled" ) {
2463
                        this.element
2464
                                .toggleClass( "ui-state-disabled", !!value )
2465
                                .attr( "aria-disabled", value );
2466
                }
2467
                this._super( key, value );
2468
        },
2469
2470
        focus: function( event, item ) {
2471
                var nested, focused;
2472
                this.blur( event, event && event.type === "focus" );
2473
2474
                this._scrollIntoView( item );
2475
2476
                this.active = item.first();
2477
                focused = this.active.addClass( "ui-state-focus" ).removeClass( "ui-state-active" );
2478
                // Only update aria-activedescendant if there's a role
2479
                // otherwise we assume focus is managed elsewhere
2480
                if ( this.options.role ) {
2481
                        this.element.attr( "aria-activedescendant", focused.attr( "id" ) );
2482
                }
2483
2484
                // Highlight active parent menu item, if any
2485
                this.active
2486
                        .parent()
2487
                        .closest( ".ui-menu-item" )
2488
                        .addClass( "ui-state-active" );
2489
2490
                if ( event && event.type === "keydown" ) {
2491
                        this._close();
2492
                } else {
2493
                        this.timer = this._delay(function() {
2494
                                this._close();
2495
                        }, this.delay );
2496
                }
2497
2498
                nested = item.children( ".ui-menu" );
2499
                if ( nested.length && event && ( /^mouse/.test( event.type ) ) ) {
2500
                        this._startOpening(nested);
2501
                }
2502
                this.activeMenu = item.parent();
2503
2504
                this._trigger( "focus", event, { item: item } );
2505
        },
2506
2507
        _scrollIntoView: function( item ) {
2508
                var borderTop, paddingTop, offset, scroll, elementHeight, itemHeight;
2509
                if ( this._hasScroll() ) {
2510
                        borderTop = parseFloat( $.css( this.activeMenu[0], "borderTopWidth" ) ) || 0;
2511
                        paddingTop = parseFloat( $.css( this.activeMenu[0], "paddingTop" ) ) || 0;
2512
                        offset = item.offset().top - this.activeMenu.offset().top - borderTop - paddingTop;
2513
                        scroll = this.activeMenu.scrollTop();
2514
                        elementHeight = this.activeMenu.height();
2515
                        itemHeight = item.outerHeight();
2516
2517
                        if ( offset < 0 ) {
2518
                                this.activeMenu.scrollTop( scroll + offset );
2519
                        } else if ( offset + itemHeight > elementHeight ) {
2520
                                this.activeMenu.scrollTop( scroll + offset - elementHeight + itemHeight );
2521
                        }
2522
                }
2523
        },
2524
2525
        blur: function( event, fromFocus ) {
2526
                if ( !fromFocus ) {
2527
                        clearTimeout( this.timer );
2528
                }
2529
2530
                if ( !this.active ) {
2531
                        return;
2532
                }
2533
2534
                this.active.removeClass( "ui-state-focus" );
2535
                this.active = null;
2536
2537
                this._trigger( "blur", event, { item: this.active } );
2538
        },
2539
2540
        _startOpening: function( submenu ) {
2541
                clearTimeout( this.timer );
2542
2543
                // Don't open if already open fixes a Firefox bug that caused a .5 pixel
2544
                // shift in the submenu position when mousing over the carat icon
2545
                if ( submenu.attr( "aria-hidden" ) !== "true" ) {
2546
                        return;
2547
                }
2548
2549
                this.timer = this._delay(function() {
2550
                        this._close();
2551
                        this._open( submenu );
2552
                }, this.delay );
2553
        },
2554
2555
        _open: function( submenu ) {
2556
                var position = $.extend({
2557
                        of: this.active
2558
                }, this.options.position );
2559
2560
                clearTimeout( this.timer );
2561
                this.element.find( ".ui-menu" ).not( submenu.parents( ".ui-menu" ) )
2562
                        .hide()
2563
                        .attr( "aria-hidden", "true" );
2564
2565
                submenu
2566
                        .show()
2567
                        .removeAttr( "aria-hidden" )
2568
                        .attr( "aria-expanded", "true" )
2569
                        .position( position );
2570
        },
2571
2572
        collapseAll: function( event, all ) {
2573
                clearTimeout( this.timer );
2574
                this.timer = this._delay(function() {
2575
                        // If we were passed an event, look for the submenu that contains the event
2576
                        var currentMenu = all ? this.element :
2577
                                $( event && event.target ).closest( this.element.find( ".ui-menu" ) );
2578
2579
                        // If we found no valid submenu ancestor, use the main menu to close all sub menus anyway
2580
                        if ( !currentMenu.length ) {
2581
                                currentMenu = this.element;
2582
                        }
2583
2584
                        this._close( currentMenu );
2585
2586
                        this.blur( event );
2587
                        this.activeMenu = currentMenu;
2588
                }, this.delay );
2589
        },
2590
2591
        // With no arguments, closes the currently active menu - if nothing is active
2592
        // it closes all menus.  If passed an argument, it will search for menus BELOW
2593
        _close: function( startMenu ) {
2594
                if ( !startMenu ) {
2595
                        startMenu = this.active ? this.active.parent() : this.element;
2596
                }
2597
2598
                startMenu
2599
                        .find( ".ui-menu" )
2600
                                .hide()
2601
                                .attr( "aria-hidden", "true" )
2602
                                .attr( "aria-expanded", "false" )
2603
                        .end()
2604
                        .find( ".ui-state-active" ).not( ".ui-state-focus" )
2605
                                .removeClass( "ui-state-active" );
2606
        },
2607
2608
        _closeOnDocumentClick: function( event ) {
2609
                return !$( event.target ).closest( ".ui-menu" ).length;
2610
        },
2611
2612
        _isDivider: function( item ) {
2613
2614
                // Match hyphen, em dash, en dash
2615
                return !/[^\-\u2014\u2013\s]/.test( item.text() );
2616
        },
2617
2618
        collapse: function( event ) {
2619
                var newItem = this.active &&
2620
                        this.active.parent().closest( ".ui-menu-item", this.element );
2621
                if ( newItem && newItem.length ) {
2622
                        this._close();
2623
                        this.focus( event, newItem );
2624
                }
2625
        },
2626
2627
        expand: function( event ) {
2628
                var newItem = this.active &&
2629
                        this.active
2630
                                .children( ".ui-menu " )
2631
                                .find( this.options.items )
2632
                                .first();
2633
2634
                if ( newItem && newItem.length ) {
2635
                        this._open( newItem.parent() );
2636
2637
                        // Delay so Firefox will not hide activedescendant change in expanding submenu from AT
2638
                        this._delay(function() {
2639
                                this.focus( event, newItem );
2640
                        });
2641
                }
2642
        },
2643
2644
        next: function( event ) {
2645
                this._move( "next", "first", event );
2646
        },
2647
2648
        previous: function( event ) {
2649
                this._move( "prev", "last", event );
2650
        },
2651
2652
        isFirstItem: function() {
2653
                return this.active && !this.active.prevAll( ".ui-menu-item" ).length;
2654
        },
2655
2656
        isLastItem: function() {
2657
                return this.active && !this.active.nextAll( ".ui-menu-item" ).length;
2658
        },
2659
2660
        _move: function( direction, filter, event ) {
2661
                var next;
2662
                if ( this.active ) {
2663
                        if ( direction === "first" || direction === "last" ) {
2664
                                next = this.active
2665
                                        [ direction === "first" ? "prevAll" : "nextAll" ]( ".ui-menu-item" )
2666
                                        .eq( -1 );
2667
                        } else {
2668
                                next = this.active
2669
                                        [ direction + "All" ]( ".ui-menu-item" )
2670
                                        .eq( 0 );
2671
                        }
2672
                }
2673
                if ( !next || !next.length || !this.active ) {
2674
                        next = this.activeMenu.find( this.options.items )[ filter ]();
2675
                }
2676
2677
                this.focus( event, next );
2678
        },
2679
2680
        nextPage: function( event ) {
2681
                var item, base, height;
2682
2683
                if ( !this.active ) {
2684
                        this.next( event );
2685
                        return;
2686
                }
2687
                if ( this.isLastItem() ) {
2688
                        return;
2689
                }
2690
                if ( this._hasScroll() ) {
2691
                        base = this.active.offset().top;
2692
                        height = this.element.height();
2693
                        this.active.nextAll( ".ui-menu-item" ).each(function() {
2694
                                item = $( this );
2695
                                return item.offset().top - base - height < 0;
2696
                        });
2697
2698
                        this.focus( event, item );
2699
                } else {
2700
                        this.focus( event, this.activeMenu.find( this.options.items )
2701
                                [ !this.active ? "first" : "last" ]() );
2702
                }
2703
        },
2704
2705
        previousPage: function( event ) {
2706
                var item, base, height;
2707
                if ( !this.active ) {
2708
                        this.next( event );
2709
                        return;
2710
                }
2711
                if ( this.isFirstItem() ) {
2712
                        return;
2713
                }
2714
                if ( this._hasScroll() ) {
2715
                        base = this.active.offset().top;
2716
                        height = this.element.height();
2717
                        this.active.prevAll( ".ui-menu-item" ).each(function() {
2718
                                item = $( this );
2719
                                return item.offset().top - base + height > 0;
2720
                        });
2721
2722
                        this.focus( event, item );
2723
                } else {
2724
                        this.focus( event, this.activeMenu.find( this.options.items ).first() );
2725
                }
2726
        },
2727
2728
        _hasScroll: function() {
2729
                return this.element.outerHeight() < this.element.prop( "scrollHeight" );
2730
        },
2731
2732
        select: function( event ) {
2733
                // TODO: It should never be possible to not have an active item at this
2734
                // point, but the tests don't trigger mouseenter before click.
2735
                this.active = this.active || $( event.target ).closest( ".ui-menu-item" );
2736
                var ui = { item: this.active };
2737
                if ( !this.active.has( ".ui-menu" ).length ) {
2738
                        this.collapseAll( event, true );
2739
                }
2740
                this._trigger( "select", event, ui );
2741
        },
2742
2743
        _filterMenuItems: function(character) {
2744
                var escapedCharacter = character.replace( /[\-\[\]{}()*+?.,\\\^$|#\s]/g, "\\$&" ),
2745
                        regex = new RegExp( "^" + escapedCharacter, "i" );
2746
2747
                return this.activeMenu
2748
                        .find( this.options.items )
2749
2750
                        // Only match on items, not dividers or other content (#10571)
2751
                        .filter( ".ui-menu-item" )
2752
                        .filter(function() {
2753
                                return regex.test( $.trim( $( this ).text() ) );
2754
                        });
2755
        }
2756
});
2757
2758
2759
/*!
2760
 * jQuery UI Autocomplete 1.11.4
2761
 * http://jqueryui.com
2762
 *
2763
 * Copyright jQuery Foundation and other contributors
2764
 * Released under the MIT license.
2765
 * http://jquery.org/license
2766
 *
2767
 * http://api.jqueryui.com/autocomplete/
2768
 */
2769
2770
2771
$.widget( "ui.autocomplete", {
2772
        version: "1.11.4",
2773
        defaultElement: "<input>",
2774
        options: {
2775
                appendTo: null,
2776
                autoFocus: false,
2777
                delay: 300,
2778
                minLength: 1,
2779
                position: {
2780
                        my: "left top",
2781
                        at: "left bottom",
2782
                        collision: "none"
2783
                },
2784
                source: null,
2785
2786
                // callbacks
2787
                change: null,
2788
                close: null,
2789
                focus: null,
2790
                open: null,
2791
                response: null,
2792
                search: null,
2793
                select: null
2794
        },
2795
2796
        requestIndex: 0,
2797
        pending: 0,
2798
2799
        _create: function() {
2800
                // Some browsers only repeat keydown events, not keypress events,
2801
                // so we use the suppressKeyPress flag to determine if we've already
2802
                // handled the keydown event. #7269
2803
                // Unfortunately the code for & in keypress is the same as the up arrow,
2804
                // so we use the suppressKeyPressRepeat flag to avoid handling keypress
2805
                // events when we know the keydown event was used to modify the
2806
                // search term. #7799
2807
                var suppressKeyPress, suppressKeyPressRepeat, suppressInput,
2808
                        nodeName = this.element[ 0 ].nodeName.toLowerCase(),
2809
                        isTextarea = nodeName === "textarea",
2810
                        isInput = nodeName === "input";
2811
2812
                this.isMultiLine =
2813
                        // Textareas are always multi-line
2814
                        isTextarea ? true :
2815
                        // Inputs are always single-line, even if inside a contentEditable element
2816
                        // IE also treats inputs as contentEditable
2817
                        isInput ? false :
2818
                        // All other element types are determined by whether or not they're contentEditable
2819
                        this.element.prop( "isContentEditable" );
2820
2821
                this.valueMethod = this.element[ isTextarea || isInput ? "val" : "text" ];
2822
                this.isNewMenu = true;
2823
2824
                this.element
2825
                        .addClass( "ui-autocomplete-input" )
2826
                        .attr( "autocomplete", "off" );
2827
2828
                this._on( this.element, {
2829
                        keydown: function( event ) {
2830
                                if ( this.element.prop( "readOnly" ) ) {
2831
                                        suppressKeyPress = true;
2832
                                        suppressInput = true;
2833
                                        suppressKeyPressRepeat = true;
2834
                                        return;
2835
                                }
2836
2837
                                suppressKeyPress = false;
2838
                                suppressInput = false;
2839
                                suppressKeyPressRepeat = false;
2840
                                var keyCode = $.ui.keyCode;
2841
                                switch ( event.keyCode ) {
2842
                                case keyCode.PAGE_UP:
2843
                                        suppressKeyPress = true;
2844
                                        this._move( "previousPage", event );
2845
                                        break;
2846
                                case keyCode.PAGE_DOWN:
2847
                                        suppressKeyPress = true;
2848
                                        this._move( "nextPage", event );
2849
                                        break;
2850
                                case keyCode.UP:
2851
                                        suppressKeyPress = true;
2852
                                        this._keyEvent( "previous", event );
2853
                                        break;
2854
                                case keyCode.DOWN:
2855
                                        suppressKeyPress = true;
2856
                                        this._keyEvent( "next", event );
2857
                                        break;
2858
                                case keyCode.ENTER:
2859
                                        // when menu is open and has focus
2860
                                        if ( this.menu.active ) {
2861
                                                // #6055 - Opera still allows the keypress to occur
2862
                                                // which causes forms to submit
2863
                                                suppressKeyPress = true;
2864
                                                event.preventDefault();
2865
                                                this.menu.select( event );
2866
                                        }
2867
                                        break;
2868
                                case keyCode.TAB:
2869
                                        if ( this.menu.active ) {
2870
                                                this.menu.select( event );
2871
                                        }
2872
                                        break;
2873
                                case keyCode.ESCAPE:
2874
                                        if ( this.menu.element.is( ":visible" ) ) {
2875
                                                if ( !this.isMultiLine ) {
2876
                                                        this._value( this.term );
2877
                                                }
2878
                                                this.close( event );
2879
                                                // Different browsers have different default behavior for escape
2880
                                                // Single press can mean undo or clear
2881
                                                // Double press in IE means clear the whole form
2882
                                                event.preventDefault();
2883
                                        }
2884
                                        break;
2885
                                default:
2886
                                        suppressKeyPressRepeat = true;
2887
                                        // search timeout should be triggered before the input value is changed
2888
                                        this._searchTimeout( event );
2889
                                        break;
2890
                                }
2891
                        },
2892
                        keypress: function( event ) {
2893
                                if ( suppressKeyPress ) {
2894
                                        suppressKeyPress = false;
2895
                                        if ( !this.isMultiLine || this.menu.element.is( ":visible" ) ) {
2896
                                                event.preventDefault();
2897
                                        }
2898
                                        return;
2899
                                }
2900
                                if ( suppressKeyPressRepeat ) {
2901
                                        return;
2902
                                }
2903
2904
                                // replicate some key handlers to allow them to repeat in Firefox and Opera
2905
                                var keyCode = $.ui.keyCode;
2906
                                switch ( event.keyCode ) {
2907
                                case keyCode.PAGE_UP:
2908
                                        this._move( "previousPage", event );
2909
                                        break;
2910
                                case keyCode.PAGE_DOWN:
2911
                                        this._move( "nextPage", event );
2912
                                        break;
2913
                                case keyCode.UP:
2914
                                        this._keyEvent( "previous", event );
2915
                                        break;
2916
                                case keyCode.DOWN:
2917
                                        this._keyEvent( "next", event );
2918
                                        break;
2919
                                }
2920
                        },
2921
                        input: function( event ) {
2922
                                if ( suppressInput ) {
2923
                                        suppressInput = false;
2924
                                        event.preventDefault();
2925
                                        return;
2926
                                }
2927
                                this._searchTimeout( event );
2928
                        },
2929
                        focus: function() {
2930
                                this.selectedItem = null;
2931
                                this.previous = this._value();
2932
                        },
2933
                        blur: function( event ) {
2934
                                if ( this.cancelBlur ) {
2935
                                        delete this.cancelBlur;
2936
                                        return;
2937
                                }
2938
2939
                                clearTimeout( this.searching );
2940
                                this.close( event );
2941
                                this._change( event );
2942
                        }
2943
                });
2944
2945
                this._initSource();
2946
                this.menu = $( "<ul>" )
2947
                        .addClass( "ui-autocomplete ui-front" )
2948
                        .appendTo( this._appendTo() )
2949
                        .menu({
2950
                                // disable ARIA support, the live region takes care of that
2951
                                role: null
2952
                        })
2953
                        .hide()
2954
                        .menu( "instance" );
2955
2956
                this._on( this.menu.element, {
2957
                        mousedown: function( event ) {
2958
                                // prevent moving focus out of the text field
2959
                                event.preventDefault();
2960
2961
                                // IE doesn't prevent moving focus even with event.preventDefault()
2962
                                // so we set a flag to know when we should ignore the blur event
2963
                                this.cancelBlur = true;
2964
                                this._delay(function() {
2965
                                        delete this.cancelBlur;
2966
                                });
2967
2968
                                // clicking on the scrollbar causes focus to shift to the body
2969
                                // but we can't detect a mouseup or a click immediately afterward
2970
                                // so we have to track the next mousedown and close the menu if
2971
                                // the user clicks somewhere outside of the autocomplete
2972
                                var menuElement = this.menu.element[ 0 ];
2973
                                if ( !$( event.target ).closest( ".ui-menu-item" ).length ) {
2974
                                        this._delay(function() {
2975
                                                var that = this;
2976
                                                this.document.one( "mousedown", function( event ) {
2977
                                                        if ( event.target !== that.element[ 0 ] &&
2978
                                                                        event.target !== menuElement &&
2979
                                                                        !$.contains( menuElement, event.target ) ) {
2980
                                                                that.close();
2981
                                                        }
2982
                                                });
2983
                                        });
2984
                                }
2985
                        },
2986
                        menufocus: function( event, ui ) {
2987
                                var label, item;
2988
                                // support: Firefox
2989
                                // Prevent accidental activation of menu items in Firefox (#7024 #9118)
2990
                                if ( this.isNewMenu ) {
2991
                                        this.isNewMenu = false;
2992
                                        if ( event.originalEvent && /^mouse/.test( event.originalEvent.type ) ) {
2993
                                                this.menu.blur();
2994
2995
                                                this.document.one( "mousemove", function() {
2996
                                                        $( event.target ).trigger( event.originalEvent );
2997
                                                });
2998
2999
                                                return;
3000
                                        }
3001
                                }
3002
3003
                                item = ui.item.data( "ui-autocomplete-item" );
3004
                                if ( false !== this._trigger( "focus", event, { item: item } ) ) {
3005
                                        // use value to match what will end up in the input, if it was a key event
3006
                                        if ( event.originalEvent && /^key/.test( event.originalEvent.type ) ) {
3007
                                                this._value( item.value );
3008
                                        }
3009
                                }
3010
3011
                                // Announce the value in the liveRegion
3012
                                label = ui.item.attr( "aria-label" ) || item.value;
3013
                                if ( label && $.trim( label ).length ) {
3014
                                        this.liveRegion.children().hide();
3015
                                        $( "<div>" ).text( label ).appendTo( this.liveRegion );
3016
                                }
3017
                        },
3018
                        menuselect: function( event, ui ) {
3019
                                var item = ui.item.data( "ui-autocomplete-item" ),
3020
                                        previous = this.previous;
3021
3022
                                // only trigger when focus was lost (click on menu)
3023
                                if ( this.element[ 0 ] !== this.document[ 0 ].activeElement ) {
3024
                                        this.element.focus();
3025
                                        this.previous = previous;
3026
                                        // #6109 - IE triggers two focus events and the second
3027
                                        // is asynchronous, so we need to reset the previous
3028
                                        // term synchronously and asynchronously :-(
3029
                                        this._delay(function() {
3030
                                                this.previous = previous;
3031
                                                this.selectedItem = item;
3032
                                        });
3033
                                }
3034
3035
                                if ( false !== this._trigger( "select", event, { item: item } ) ) {
3036
                                        this._value( item.value );
3037
                                }
3038
                                // reset the term after the select event
3039
                                // this allows custom select handling to work properly
3040
                                this.term = this._value();
3041
3042
                                this.close( event );
3043
                                this.selectedItem = item;
3044
                        }
3045
                });
3046
3047
                this.liveRegion = $( "<span>", {
3048
                                role: "status",
3049
                                "aria-live": "assertive",
3050
                                "aria-relevant": "additions"
3051
                        })
3052
                        .addClass( "ui-helper-hidden-accessible" )
3053
                        .appendTo( this.document[ 0 ].body );
3054
3055
                // turning off autocomplete prevents the browser from remembering the
3056
                // value when navigating through history, so we re-enable autocomplete
3057
                // if the page is unloaded before the widget is destroyed. #7790
3058
                this._on( this.window, {
3059
                        beforeunload: function() {
3060
                                this.element.removeAttr( "autocomplete" );
3061
                        }
3062
                });
3063
        },
3064
3065
        _destroy: function() {
3066
                clearTimeout( this.searching );
3067
                this.element
3068
                        .removeClass( "ui-autocomplete-input" )
3069
                        .removeAttr( "autocomplete" );
3070
                this.menu.element.remove();
3071
                this.liveRegion.remove();
3072
        },
3073
3074
        _setOption: function( key, value ) {
3075
                this._super( key, value );
3076
                if ( key === "source" ) {
3077
                        this._initSource();
3078
                }
3079
                if ( key === "appendTo" ) {
3080
                        this.menu.element.appendTo( this._appendTo() );
3081
                }
3082
                if ( key === "disabled" && value && this.xhr ) {
3083
                        this.xhr.abort();
3084
                }
3085
        },
3086
3087
        _appendTo: function() {
3088
                var element = this.options.appendTo;
3089
3090
                if ( element ) {
3091
                        element = element.jquery || element.nodeType ?
3092
                                $( element ) :
3093
                                this.document.find( element ).eq( 0 );
3094
                }
3095
3096
                if ( !element || !element[ 0 ] ) {
3097
                        element = this.element.closest( ".ui-front" );
3098
                }
3099
3100
                if ( !element.length ) {
3101
                        element = this.document[ 0 ].body;
3102
                }
3103
3104
                return element;
3105
        },
3106
3107
        _initSource: function() {
3108
                var array, url,
3109
                        that = this;
3110
                if ( $.isArray( this.options.source ) ) {
3111
                        array = this.options.source;
3112
                        this.source = function( request, response ) {
3113
                                response( $.ui.autocomplete.filter( array, request.term ) );
3114
                        };
3115
                } else if ( typeof this.options.source === "string" ) {
3116
                        url = this.options.source;
3117
                        this.source = function( request, response ) {
3118
                                if ( that.xhr ) {
3119
                                        that.xhr.abort();
3120
                                }
3121
                                that.xhr = $.ajax({
3122
                                        url: url,
3123
                                        data: request,
3124
                                        dataType: "json",
3125
                                        success: function( data ) {
3126
                                                response( data );
3127
                                        },
3128
                                        error: function() {
3129
                                                response([]);
3130
                                        }
3131
                                });
3132
                        };
3133
                } else {
3134
                        this.source = this.options.source;
3135
                }
3136
        },
3137
3138
        _searchTimeout: function( event ) {
3139
                clearTimeout( this.searching );
3140
                this.searching = this._delay(function() {
3141
3142
                        // Search if the value has changed, or if the user retypes the same value (see #7434)
3143
                        var equalValues = this.term === this._value(),
3144
                                menuVisible = this.menu.element.is( ":visible" ),
3145
                                modifierKey = event.altKey || event.ctrlKey || event.metaKey || event.shiftKey;
3146
3147
                        if ( !equalValues || ( equalValues && !menuVisible && !modifierKey ) ) {
3148
                                this.selectedItem = null;
3149
                                this.search( null, event );
3150
                        }
3151
                }, this.options.delay );
3152
        },
3153
3154
        search: function( value, event ) {
3155
                value = value != null ? value : this._value();
3156
3157
                // always save the actual value, not the one passed as an argument
3158
                this.term = this._value();
3159
3160
                if ( value.length < this.options.minLength ) {
3161
                        return this.close( event );
3162
                }
3163
3164
                if ( this._trigger( "search", event ) === false ) {
3165
                        return;
3166
                }
3167
3168
                return this._search( value );
3169
        },
3170
3171
        _search: function( value ) {
3172
                this.pending++;
3173
                this.element.addClass( "ui-autocomplete-loading" );
3174
                this.cancelSearch = false;
3175
3176
                this.source( { term: value }, this._response() );
3177
        },
3178
3179
        _response: function() {
3180
                var index = ++this.requestIndex;
3181
3182
                return $.proxy(function( content ) {
3183
                        if ( index === this.requestIndex ) {
3184
                                this.__response( content );
3185
                        }
3186
3187
                        this.pending--;
3188
                        if ( !this.pending ) {
3189
                                this.element.removeClass( "ui-autocomplete-loading" );
3190
                        }
3191
                }, this );
3192
        },
3193
3194
        __response: function( content ) {
3195
                if ( content ) {
3196
                        content = this._normalize( content );
3197
                }
3198
                this._trigger( "response", null, { content: content } );
3199
                if ( !this.options.disabled && content && content.length && !this.cancelSearch ) {
3200
                        this._suggest( content );
3201
                        this._trigger( "open" );
3202
                } else {
3203
                        // use ._close() instead of .close() so we don't cancel future searches
3204
                        this._close();
3205
                }
3206
        },
3207
3208
        close: function( event ) {
3209
                this.cancelSearch = true;
3210
                this._close( event );
3211
        },
3212
3213
        _close: function( event ) {
3214
                if ( this.menu.element.is( ":visible" ) ) {
3215
                        this.menu.element.hide();
3216
                        this.menu.blur();
3217
                        this.isNewMenu = true;
3218
                        this._trigger( "close", event );
3219
                }
3220
        },
3221
3222
        _change: function( event ) {
3223
                if ( this.previous !== this._value() ) {
3224
                        this._trigger( "change", event, { item: this.selectedItem } );
3225
                }
3226
        },
3227
3228
        _normalize: function( items ) {
3229
                // assume all items have the right format when the first item is complete
3230
                if ( items.length && items[ 0 ].label && items[ 0 ].value ) {
3231
                        return items;
3232
                }
3233
                return $.map( items, function( item ) {
3234
                        if ( typeof item === "string" ) {
3235
                                return {
3236
                                        label: item,
3237
                                        value: item
3238
                                };
3239
                        }
3240
                        return $.extend( {}, item, {
3241
                                label: item.label || item.value,
3242
                                value: item.value || item.label
3243
                        });
3244
                });
3245
        },
3246
3247
        _suggest: function( items ) {
3248
                var ul = this.menu.element.empty();
3249
                this._renderMenu( ul, items );
3250
                this.isNewMenu = true;
3251
                this.menu.refresh();
3252
3253
                // size and position menu
3254
                ul.show();
3255
                this._resizeMenu();
3256
                ul.position( $.extend({
3257
                        of: this.element
3258
                }, this.options.position ) );
3259
3260
                if ( this.options.autoFocus ) {
3261
                        this.menu.next();
3262
                }
3263
        },
3264
3265
        _resizeMenu: function() {
3266
                var ul = this.menu.element;
3267
                ul.outerWidth( Math.max(
3268
                        // Firefox wraps long text (possibly a rounding bug)
3269
                        // so we add 1px to avoid the wrapping (#7513)
3270
                        ul.width( "" ).outerWidth() + 1,
3271
                        this.element.outerWidth()
3272
                ) );
3273
        },
3274
3275
        _renderMenu: function( ul, items ) {
3276
                var that = this;
3277
                $.each( items, function( index, item ) {
3278
                        that._renderItemData( ul, item );
3279
                });
3280
        },
3281
3282
        _renderItemData: function( ul, item ) {
3283
                return this._renderItem( ul, item ).data( "ui-autocomplete-item", item );
3284
        },
3285
3286
        _renderItem: function( ul, item ) {
3287
                return $( "<li>" ).text( item.label ).appendTo( ul );
3288
        },
3289
3290
        _move: function( direction, event ) {
3291
                if ( !this.menu.element.is( ":visible" ) ) {
3292
                        this.search( null, event );
3293
                        return;
3294
                }
3295
                if ( this.menu.isFirstItem() && /^previous/.test( direction ) ||
3296
                                this.menu.isLastItem() && /^next/.test( direction ) ) {
3297
3298
                        if ( !this.isMultiLine ) {
3299
                                this._value( this.term );
3300
                        }
3301
3302
                        this.menu.blur();
3303
                        return;
3304
                }
3305
                this.menu[ direction ]( event );
3306
        },
3307
3308
        widget: function() {
3309
                return this.menu.element;
3310
        },
3311
3312
        _value: function() {
3313
                return this.valueMethod.apply( this.element, arguments );
3314
        },
3315
3316
        _keyEvent: function( keyEvent, event ) {
3317
                if ( !this.isMultiLine || this.menu.element.is( ":visible" ) ) {
3318
                        this._move( keyEvent, event );
3319
3320
                        // prevents moving cursor to beginning/end of the text field in some browsers
3321
                        event.preventDefault();
3322
                }
3323
        }
3324
});
3325
3326
$.extend( $.ui.autocomplete, {
3327
        escapeRegex: function( value ) {
3328
                return value.replace( /[\-\[\]{}()*+?.,\\\^$|#\s]/g, "\\$&" );
3329
        },
3330
        filter: function( array, term ) {
3331
                var matcher = new RegExp( $.ui.autocomplete.escapeRegex( term ), "i" );
3332
                return $.grep( array, function( value ) {
3333
                        return matcher.test( value.label || value.value || value );
3334
                });
3335
        }
3336
});
3337
3338
// live region extension, adding a `messages` option
3339
// NOTE: This is an experimental API. We are still investigating
3340
// a full solution for string manipulation and internationalization.
3341
$.widget( "ui.autocomplete", $.ui.autocomplete, {
3342
        options: {
3343
                messages: {
3344
                        noResults: "No search results.",
3345
                        results: function( amount ) {
3346
                                return amount + ( amount > 1 ? " results are" : " result is" ) +
3347
                                        " available, use up and down arrow keys to navigate.";
3348
                        }
3349
                }
3350
        },
3351
3352
        __response: function( content ) {
3353
                var message;
3354
                this._superApply( arguments );
3355
                if ( this.options.disabled || this.cancelSearch ) {
3356
                        return;
3357
                }
3358
                if ( content && content.length ) {
3359
                        message = this.options.messages.results( content.length );
3360
                } else {
3361
                        message = this.options.messages.noResults;
3362
                }
3363
                this.liveRegion.children().hide();
3364
                $( "<div>" ).text( message ).appendTo( this.liveRegion );
3365
        }
3366
});
3367
3368
var autocomplete = $.ui.autocomplete;
3369
3370
3371
/*!
3372
 * jQuery UI Button 1.11.4
3373
 * http://jqueryui.com
3374
 *
3375
 * Copyright jQuery Foundation and other contributors
3376
 * Released under the MIT license.
3377
 * http://jquery.org/license
3378
 *
3379
 * http://api.jqueryui.com/button/
3380
 */
3381
3382
3383
var lastActive,
3384
        baseClasses = "ui-button ui-widget ui-state-default ui-corner-all",
3385
        typeClasses = "ui-button-icons-only ui-button-icon-only ui-button-text-icons ui-button-text-icon-primary ui-button-text-icon-secondary ui-button-text-only",
3386
        formResetHandler = function() {
3387
                var form = $( this );
3388
                setTimeout(function() {
3389
                        form.find( ":ui-button" ).button( "refresh" );
3390
                }, 1 );
3391
        },
3392
        radioGroup = function( radio ) {
3393
                var name = radio.name,
3394
                        form = radio.form,
3395
                        radios = $( [] );
3396
                if ( name ) {
3397
                        name = name.replace( /'/g, "\\'" );
3398
                        if ( form ) {
3399
                                radios = $( form ).find( "[name='" + name + "'][type=radio]" );
3400
                        } else {
3401
                                radios = $( "[name='" + name + "'][type=radio]", radio.ownerDocument )
3402
                                        .filter(function() {
3403
                                                return !this.form;
3404
                                        });
3405
                        }
3406
                }
3407
                return radios;
3408
        };
3409
3410
$.widget( "ui.button", {
3411
        version: "1.11.4",
3412
        defaultElement: "<button>",
3413
        options: {
3414
                disabled: null,
3415
                text: true,
3416
                label: null,
3417
                icons: {
3418
                        primary: null,
3419
                        secondary: null
3420
                }
3421
        },
3422
        _create: function() {
3423
                this.element.closest( "form" )
3424
                        .unbind( "reset" + this.eventNamespace )
3425
                        .bind( "reset" + this.eventNamespace, formResetHandler );
3426
3427
                if ( typeof this.options.disabled !== "boolean" ) {
3428
                        this.options.disabled = !!this.element.prop( "disabled" );
3429
                } else {
3430
                        this.element.prop( "disabled", this.options.disabled );
3431
                }
3432
3433
                this._determineButtonType();
3434
                this.hasTitle = !!this.buttonElement.attr( "title" );
3435
3436
                var that = this,
3437
                        options = this.options,
3438
                        toggleButton = this.type === "checkbox" || this.type === "radio",
3439
                        activeClass = !toggleButton ? "ui-state-active" : "";
3440
3441
                if ( options.label === null ) {
3442
                        options.label = (this.type === "input" ? this.buttonElement.val() : this.buttonElement.html());
3443
                }
3444
3445
                this._hoverable( this.buttonElement );
3446
3447
                this.buttonElement
3448
                        .addClass( baseClasses )
3449
                        .attr( "role", "button" )
3450
                        .bind( "mouseenter" + this.eventNamespace, function() {
3451
                                if ( options.disabled ) {
3452
                                        return;
3453
                                }
3454
                                if ( this === lastActive ) {
3455
                                        $( this ).addClass( "ui-state-active" );
3456
                                }
3457
                        })
3458
                        .bind( "mouseleave" + this.eventNamespace, function() {
3459
                                if ( options.disabled ) {
3460
                                        return;
3461
                                }
3462
                                $( this ).removeClass( activeClass );
3463
                        })
3464
                        .bind( "click" + this.eventNamespace, function( event ) {
3465
                                if ( options.disabled ) {
3466
                                        event.preventDefault();
3467
                                        event.stopImmediatePropagation();
3468
                                }
3469
                        });
3470
3471
                // Can't use _focusable() because the element that receives focus
3472
                // and the element that gets the ui-state-focus class are different
3473
                this._on({
3474
                        focus: function() {
3475
                                this.buttonElement.addClass( "ui-state-focus" );
3476
                        },
3477
                        blur: function() {
3478
                                this.buttonElement.removeClass( "ui-state-focus" );
3479
                        }
3480
                });
3481
3482
                if ( toggleButton ) {
3483
                        this.element.bind( "change" + this.eventNamespace, function() {
3484
                                that.refresh();
3485
                        });
3486
                }
3487
3488
                if ( this.type === "checkbox" ) {
3489
                        this.buttonElement.bind( "click" + this.eventNamespace, function() {
3490
                                if ( options.disabled ) {
3491
                                        return false;
3492
                                }
3493
                        });
3494
                } else if ( this.type === "radio" ) {
3495
                        this.buttonElement.bind( "click" + this.eventNamespace, function() {
3496
                                if ( options.disabled ) {
3497
                                        return false;
3498
                                }
3499
                                $( this ).addClass( "ui-state-active" );
3500
                                that.buttonElement.attr( "aria-pressed", "true" );
3501
3502
                                var radio = that.element[ 0 ];
3503
                                radioGroup( radio )
3504
                                        .not( radio )
3505
                                        .map(function() {
3506
                                                return $( this ).button( "widget" )[ 0 ];
3507
                                        })
3508
                                        .removeClass( "ui-state-active" )
3509
                                        .attr( "aria-pressed", "false" );
3510
                        });
3511
                } else {
3512
                        this.buttonElement
3513
                                .bind( "mousedown" + this.eventNamespace, function() {
3514
                                        if ( options.disabled ) {
3515
                                                return false;
3516
                                        }
3517
                                        $( this ).addClass( "ui-state-active" );
3518
                                        lastActive = this;
3519
                                        that.document.one( "mouseup", function() {
3520
                                                lastActive = null;
3521
                                        });
3522
                                })
3523
                                .bind( "mouseup" + this.eventNamespace, function() {
3524
                                        if ( options.disabled ) {
3525
                                                return false;
3526
                                        }
3527
                                        $( this ).removeClass( "ui-state-active" );
3528
                                })
3529
                                .bind( "keydown" + this.eventNamespace, function(event) {
3530
                                        if ( options.disabled ) {
3531
                                                return false;
3532
                                        }
3533
                                        if ( event.keyCode === $.ui.keyCode.SPACE || event.keyCode === $.ui.keyCode.ENTER ) {
3534
                                                $( this ).addClass( "ui-state-active" );
3535
                                        }
3536
                                })
3537
                                // see #8559, we bind to blur here in case the button element loses
3538
                                // focus between keydown and keyup, it would be left in an "active" state
3539
                                .bind( "keyup" + this.eventNamespace + " blur" + this.eventNamespace, function() {
3540
                                        $( this ).removeClass( "ui-state-active" );
3541
                                });
3542
3543
                        if ( this.buttonElement.is("a") ) {
3544
                                this.buttonElement.keyup(function(event) {
3545
                                        if ( event.keyCode === $.ui.keyCode.SPACE ) {
3546
                                                // TODO pass through original event correctly (just as 2nd argument doesn't work)
3547
                                                $( this ).click();
3548
                                        }
3549
                                });
3550
                        }
3551
                }
3552
3553
                this._setOption( "disabled", options.disabled );
3554
                this._resetButton();
3555
        },
3556
3557
        _determineButtonType: function() {
3558
                var ancestor, labelSelector, checked;
3559
3560
                if ( this.element.is("[type=checkbox]") ) {
3561
                        this.type = "checkbox";
3562
                } else if ( this.element.is("[type=radio]") ) {
3563
                        this.type = "radio";
3564
                } else if ( this.element.is("input") ) {
3565
                        this.type = "input";
3566
                } else {
3567
                        this.type = "button";
3568
                }
3569
3570
                if ( this.type === "checkbox" || this.type === "radio" ) {
3571
                        // we don't search against the document in case the element
3572
                        // is disconnected from the DOM
3573
                        ancestor = this.element.parents().last();
3574
                        labelSelector = "label[for='" + this.element.attr("id") + "']";
3575
                        this.buttonElement = ancestor.find( labelSelector );
3576
                        if ( !this.buttonElement.length ) {
3577
                                ancestor = ancestor.length ? ancestor.siblings() : this.element.siblings();
3578
                                this.buttonElement = ancestor.filter( labelSelector );
3579
                                if ( !this.buttonElement.length ) {
3580
                                        this.buttonElement = ancestor.find( labelSelector );
3581
                                }
3582
                        }
3583
                        this.element.addClass( "ui-helper-hidden-accessible" );
3584
3585
                        checked = this.element.is( ":checked" );
3586
                        if ( checked ) {
3587
                                this.buttonElement.addClass( "ui-state-active" );
3588
                        }
3589
                        this.buttonElement.prop( "aria-pressed", checked );
3590
                } else {
3591
                        this.buttonElement = this.element;
3592
                }
3593
        },
3594
3595
        widget: function() {
3596
                return this.buttonElement;
3597
        },
3598
3599
        _destroy: function() {
3600
                this.element
3601
                        .removeClass( "ui-helper-hidden-accessible" );
3602
                this.buttonElement
3603
                        .removeClass( baseClasses + " ui-state-active " + typeClasses )
3604
                        .removeAttr( "role" )
3605
                        .removeAttr( "aria-pressed" )
3606
                        .html( this.buttonElement.find(".ui-button-text").html() );
3607
3608
                if ( !this.hasTitle ) {
3609
                        this.buttonElement.removeAttr( "title" );
3610
                }
3611
        },
3612
3613
        _setOption: function( key, value ) {
3614
                this._super( key, value );
3615
                if ( key === "disabled" ) {
3616
                        this.widget().toggleClass( "ui-state-disabled", !!value );
3617
                        this.element.prop( "disabled", !!value );
3618
                        if ( value ) {
3619
                                if ( this.type === "checkbox" || this.type === "radio" ) {
3620
                                        this.buttonElement.removeClass( "ui-state-focus" );
3621
                                } else {
3622
                                        this.buttonElement.removeClass( "ui-state-focus ui-state-active" );
3623
                                }
3624
                        }
3625
                        return;
3626
                }
3627
                this._resetButton();
3628
        },
3629
3630
        refresh: function() {
3631
                //See #8237 & #8828
3632
                var isDisabled = this.element.is( "input, button" ) ? this.element.is( ":disabled" ) : this.element.hasClass( "ui-button-disabled" );
3633
3634
                if ( isDisabled !== this.options.disabled ) {
3635
                        this._setOption( "disabled", isDisabled );
3636
                }
3637
                if ( this.type === "radio" ) {
3638
                        radioGroup( this.element[0] ).each(function() {
3639
                                if ( $( this ).is( ":checked" ) ) {
3640
                                        $( this ).button( "widget" )
3641
                                                .addClass( "ui-state-active" )
3642
                                                .attr( "aria-pressed", "true" );
3643
                                } else {
3644
                                        $( this ).button( "widget" )
3645
                                                .removeClass( "ui-state-active" )
3646
                                                .attr( "aria-pressed", "false" );
3647
                                }
3648
                        });
3649
                } else if ( this.type === "checkbox" ) {
3650
                        if ( this.element.is( ":checked" ) ) {
3651
                                this.buttonElement
3652
                                        .addClass( "ui-state-active" )
3653
                                        .attr( "aria-pressed", "true" );
3654
                        } else {
3655
                                this.buttonElement
3656
                                        .removeClass( "ui-state-active" )
3657
                                        .attr( "aria-pressed", "false" );
3658
                        }
3659
                }
3660
        },
3661
3662
        _resetButton: function() {
3663
                if ( this.type === "input" ) {
3664
                        if ( this.options.label ) {
3665
                                this.element.val( this.options.label );
3666
                        }
3667
                        return;
3668
                }
3669
                var buttonElement = this.buttonElement.removeClass( typeClasses ),
3670
                        buttonText = $( "<span></span>", this.document[0] )
3671
                                .addClass( "ui-button-text" )
3672
                                .html( this.options.label )
3673
                                .appendTo( buttonElement.empty() )
3674
                                .text(),
3675
                        icons = this.options.icons,
3676
                        multipleIcons = icons.primary && icons.secondary,
3677
                        buttonClasses = [];
3678
3679
                if ( icons.primary || icons.secondary ) {
3680
                        if ( this.options.text ) {
3681
                                buttonClasses.push( "ui-button-text-icon" + ( multipleIcons ? "s" : ( icons.primary ? "-primary" : "-secondary" ) ) );
3682
                        }
3683
3684
                        if ( icons.primary ) {
3685
                                buttonElement.prepend( "<span class='ui-button-icon-primary ui-icon " + icons.primary + "'></span>" );
3686
                        }
3687
3688
                        if ( icons.secondary ) {
3689
                                buttonElement.append( "<span class='ui-button-icon-secondary ui-icon " + icons.secondary + "'></span>" );
3690
                        }
3691
3692
                        if ( !this.options.text ) {
3693
                                buttonClasses.push( multipleIcons ? "ui-button-icons-only" : "ui-button-icon-only" );
3694
3695
                                if ( !this.hasTitle ) {
3696
                                        buttonElement.attr( "title", $.trim( buttonText ) );
3697
                                }
3698
                        }
3699
                } else {
3700
                        buttonClasses.push( "ui-button-text-only" );
3701
                }
3702
                buttonElement.addClass( buttonClasses.join( " " ) );
3703
        }
3704
});
3705
3706
$.widget( "ui.buttonset", {
3707
        version: "1.11.4",
3708
        options: {
3709
                items: "button, input[type=button], input[type=submit], input[type=reset], input[type=checkbox], input[type=radio], a, :data(ui-button)"
3710
        },
3711
3712
        _create: function() {
3713
                this.element.addClass( "ui-buttonset" );
3714
        },
3715
3716
        _init: function() {
3717
                this.refresh();
3718
        },
3719
3720
        _setOption: function( key, value ) {
3721
                if ( key === "disabled" ) {
3722
                        this.buttons.button( "option", key, value );
3723
                }
3724
3725
                this._super( key, value );
3726
        },
3727
3728
        refresh: function() {
3729
                var rtl = this.element.css( "direction" ) === "rtl",
3730
                        allButtons = this.element.find( this.options.items ),
3731
                        existingButtons = allButtons.filter( ":ui-button" );
3732
3733
                // Initialize new buttons
3734
                allButtons.not( ":ui-button" ).button();
3735
3736
                // Refresh existing buttons
3737
                existingButtons.button( "refresh" );
3738
3739
                this.buttons = allButtons
3740
                        .map(function() {
3741
                                return $( this ).button( "widget" )[ 0 ];
3742
                        })
3743
                                .removeClass( "ui-corner-all ui-corner-left ui-corner-right" )
3744
                                .filter( ":first" )
3745
                                        .addClass( rtl ? "ui-corner-right" : "ui-corner-left" )
3746
                                .end()
3747
                                .filter( ":last" )
3748
                                        .addClass( rtl ? "ui-corner-left" : "ui-corner-right" )
3749
                                .end()
3750
                        .end();
3751
        },
3752
3753
        _destroy: function() {
3754
                this.element.removeClass( "ui-buttonset" );
3755
                this.buttons
3756
                        .map(function() {
3757
                                return $( this ).button( "widget" )[ 0 ];
3758
                        })
3759
                                .removeClass( "ui-corner-left ui-corner-right" )
3760
                        .end()
3761
                        .button( "destroy" );
3762
        }
3763
});
3764
3765
var button = $.ui.button;
3766
3767
3768
/*!
3769
 * jQuery UI Datepicker 1.11.4
3770
 * http://jqueryui.com
3771
 *
3772
 * Copyright jQuery Foundation and other contributors
3773
 * Released under the MIT license.
3774
 * http://jquery.org/license
3775
 *
3776
 * http://api.jqueryui.com/datepicker/
3777
 */
3778
3779
3780
$.extend($.ui, { datepicker: { version: "1.11.4" } });
3781
3782
var datepicker_instActive;
3783
3784
function datepicker_getZindex( elem ) {
3785
        var position, value;
3786
        while ( elem.length && elem[ 0 ] !== document ) {
3787
                // Ignore z-index if position is set to a value where z-index is ignored by the browser
3788
                // This makes behavior of this function consistent across browsers
3789
                // WebKit always returns auto if the element is positioned
3790
                position = elem.css( "position" );
3791
                if ( position === "absolute" || position === "relative" || position === "fixed" ) {
3792
                        // IE returns 0 when zIndex is not specified
3793
                        // other browsers return a string
3794
                        // we ignore the case of nested elements with an explicit value of 0
3795
                        // <div style="z-index: -10;"><div style="z-index: 0;"></div></div>
3796
                        value = parseInt( elem.css( "zIndex" ), 10 );
3797
                        if ( !isNaN( value ) && value !== 0 ) {
3798
                                return value;
3799
                        }
3800
                }
3801
                elem = elem.parent();
3802
        }
3803
3804
        return 0;
3805
}
3806
/* Date picker manager.
3807
   Use the singleton instance of this class, $.datepicker, to interact with the date picker.
3808
   Settings for (groups of) date pickers are maintained in an instance object,
3809
   allowing multiple different settings on the same page. */
3810
3811
function Datepicker() {
3812
        this._curInst = null; // The current instance in use
3813
        this._keyEvent = false; // If the last event was a key event
3814
        this._disabledInputs = []; // List of date picker inputs that have been disabled
3815
        this._datepickerShowing = false; // True if the popup picker is showing , false if not
3816
        this._inDialog = false; // True if showing within a "dialog", false if not
3817
        this._mainDivId = "ui-datepicker-div"; // The ID of the main datepicker division
3818
        this._inlineClass = "ui-datepicker-inline"; // The name of the inline marker class
3819
        this._appendClass = "ui-datepicker-append"; // The name of the append marker class
3820
        this._triggerClass = "ui-datepicker-trigger"; // The name of the trigger marker class
3821
        this._dialogClass = "ui-datepicker-dialog"; // The name of the dialog marker class
3822
        this._disableClass = "ui-datepicker-disabled"; // The name of the disabled covering marker class
3823
        this._unselectableClass = "ui-datepicker-unselectable"; // The name of the unselectable cell marker class
3824
        this._currentClass = "ui-datepicker-current-day"; // The name of the current day marker class
3825
        this._dayOverClass = "ui-datepicker-days-cell-over"; // The name of the day hover marker class
3826
        this.regional = []; // Available regional settings, indexed by language code
3827
        this.regional[""] = { // Default regional settings
3828
                closeText: "Done", // Display text for close link
3829
                prevText: "Prev", // Display text for previous month link
3830
                nextText: "Next", // Display text for next month link
3831
                currentText: "Today", // Display text for current month link
3832
                monthNames: ["January","February","March","April","May","June",
3833
                        "July","August","September","October","November","December"], // Names of months for drop-down and formatting
3834
                monthNamesShort: ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"], // For formatting
3835
                dayNames: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"], // For formatting
3836
                dayNamesShort: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"], // For formatting
3837
                dayNamesMin: ["Su","Mo","Tu","We","Th","Fr","Sa"], // Column headings for days starting at Sunday
3838
                weekHeader: "Wk", // Column header for week of the year
3839
                dateFormat: "mm/dd/yy", // See format options on parseDate
3840
                firstDay: 0, // The first day of the week, Sun = 0, Mon = 1, ...
3841
                isRTL: false, // True if right-to-left language, false if left-to-right
3842
                showMonthAfterYear: false, // True if the year select precedes month, false for month then year
3843
                yearSuffix: "" // Additional text to append to the year in the month headers
3844
        };
3845
        this._defaults = { // Global defaults for all the date picker instances
3846
                showOn: "focus", // "focus" for popup on focus,
3847
                        // "button" for trigger button, or "both" for either
3848
                showAnim: "fadeIn", // Name of jQuery animation for popup
3849
                showOptions: {}, // Options for enhanced animations
3850
                defaultDate: null, // Used when field is blank: actual date,
3851
                        // +/-number for offset from today, null for today
3852
                appendText: "", // Display text following the input box, e.g. showing the format
3853
                buttonText: "...", // Text for trigger button
3854
                buttonImage: "", // URL for trigger button image
3855
                buttonImageOnly: false, // True if the image appears alone, false if it appears on a button
3856
                hideIfNoPrevNext: false, // True to hide next/previous month links
3857
                        // if not applicable, false to just disable them
3858
                navigationAsDateFormat: false, // True if date formatting applied to prev/today/next links
3859
                gotoCurrent: false, // True if today link goes back to current selection instead
3860
                changeMonth: false, // True if month can be selected directly, false if only prev/next
3861
                changeYear: false, // True if year can be selected directly, false if only prev/next
3862
                yearRange: "c-10:c+10", // Range of years to display in drop-down,
3863
                        // either relative to today's year (-nn:+nn), relative to currently displayed year
3864
                        // (c-nn:c+nn), absolute (nnnn:nnnn), or a combination of the above (nnnn:-n)
3865
                showOtherMonths: false, // True to show dates in other months, false to leave blank
3866
                selectOtherMonths: false, // True to allow selection of dates in other months, false for unselectable
3867
                showWeek: false, // True to show week of the year, false to not show it
3868
                calculateWeek: this.iso8601Week, // How to calculate the week of the year,
3869
                        // takes a Date and returns the number of the week for it
3870
                shortYearCutoff: "+10", // Short year values < this are in the current century,
3871
                        // > this are in the previous century,
3872
                        // string value starting with "+" for current year + value
3873
                minDate: null, // The earliest selectable date, or null for no limit
3874
                maxDate: null, // The latest selectable date, or null for no limit
3875
                duration: "fast", // Duration of display/closure
3876
                beforeShowDay: null, // Function that takes a date and returns an array with
3877
                        // [0] = true if selectable, false if not, [1] = custom CSS class name(s) or "",
3878
                        // [2] = cell title (optional), e.g. $.datepicker.noWeekends
3879
                beforeShow: null, // Function that takes an input field and
3880
                        // returns a set of custom settings for the date picker
3881
                onSelect: null, // Define a callback function when a date is selected
3882
                onChangeMonthYear: null, // Define a callback function when the month or year is changed
3883
                onClose: null, // Define a callback function when the datepicker is closed
3884
                numberOfMonths: 1, // Number of months to show at a time
3885
                showCurrentAtPos: 0, // The position in multipe months at which to show the current month (starting at 0)
3886
                stepMonths: 1, // Number of months to step back/forward
3887
                stepBigMonths: 12, // Number of months to step back/forward for the big links
3888
                altField: "", // Selector for an alternate field to store selected dates into
3889
                altFormat: "", // The date format to use for the alternate field
3890
                constrainInput: true, // The input is constrained by the current date format
3891
                showButtonPanel: false, // True to show button panel, false to not show it
3892
                autoSize: false, // True to size the input for the date format, false to leave as is
3893
                disabled: false // The initial disabled state
3894
        };
3895
        $.extend(this._defaults, this.regional[""]);
3896
        this.regional.en = $.extend( true, {}, this.regional[ "" ]);
3897
        this.regional[ "en-US" ] = $.extend( true, {}, this.regional.en );
3898
        this.dpDiv = datepicker_bindHover($("<div id='" + this._mainDivId + "' class='ui-datepicker ui-widget ui-widget-content ui-helper-clearfix ui-corner-all'></div>"));
3899
}
3900
3901
$.extend(Datepicker.prototype, {
3902
        /* Class name added to elements to indicate already configured with a date picker. */
3903
        markerClassName: "hasDatepicker",
3904
3905
        //Keep track of the maximum number of rows displayed (see #7043)
3906
        maxRows: 4,
3907
3908
        // TODO rename to "widget" when switching to widget factory
3909
        _widgetDatepicker: function() {
3910
                return this.dpDiv;
3911
        },
3912
3913
        /* Override the default settings for all instances of the date picker.
3914
         * @param  settings  object - the new settings to use as defaults (anonymous object)
3915
         * @return the manager object
3916
         */
3917
        setDefaults: function(settings) {
3918
                datepicker_extendRemove(this._defaults, settings || {});
3919
                return this;
3920
        },
3921
3922
        /* Attach the date picker to a jQuery selection.
3923
         * @param  target        element - the target input field or division or span
3924
         * @param  settings  object - the new settings to use for this date picker instance (anonymous)
3925
         */
3926
        _attachDatepicker: function(target, settings) {
3927
                var nodeName, inline, inst;
3928
                nodeName = target.nodeName.toLowerCase();
3929
                inline = (nodeName === "div" || nodeName === "span");
3930
                if (!target.id) {
3931
                        this.uuid += 1;
3932
                        target.id = "dp" + this.uuid;
3933
                }
3934
                inst = this._newInst($(target), inline);
3935
                inst.settings = $.extend({}, settings || {});
3936
                if (nodeName === "input") {
3937
                        this._connectDatepicker(target, inst);
3938
                } else if (inline) {
3939
                        this._inlineDatepicker(target, inst);
3940
                }
3941
        },
3942
3943
        /* Create a new instance object. */
3944
        _newInst: function(target, inline) {
3945
                var id = target[0].id.replace(/([^A-Za-z0-9_\-])/g, "\\\\$1"); // escape jQuery meta chars
3946
                return {id: id, input: target, // associated target
3947
                        selectedDay: 0, selectedMonth: 0, selectedYear: 0, // current selection
3948
                        drawMonth: 0, drawYear: 0, // month being drawn
3949
                        inline: inline, // is datepicker inline or not
3950
                        dpDiv: (!inline ? this.dpDiv : // presentation div
3951
                        datepicker_bindHover($("<div class='" + this._inlineClass + " ui-datepicker ui-widget ui-widget-content ui-helper-clearfix ui-corner-all'></div>")))};
3952
        },
3953
3954
        /* Attach the date picker to an input field. */
3955
        _connectDatepicker: function(target, inst) {
3956
                var input = $(target);
3957
                inst.append = $([]);
3958
                inst.trigger = $([]);
3959
                if (input.hasClass(this.markerClassName)) {
3960
                        return;
3961
                }
3962
                this._attachments(input, inst);
3963
                input.addClass(this.markerClassName).keydown(this._doKeyDown).
3964
                        keypress(this._doKeyPress).keyup(this._doKeyUp);
3965
                this._autoSize(inst);
3966
                $.data(target, "datepicker", inst);
3967
                //If disabled option is true, disable the datepicker once it has been attached to the input (see ticket #5665)
3968
                if( inst.settings.disabled ) {
3969
                        this._disableDatepicker( target );
3970
                }
3971
        },
3972
3973
        /* Make attachments based on settings. */
3974
        _attachments: function(input, inst) {
3975
                var showOn, buttonText, buttonImage,
3976
                        appendText = this._get(inst, "appendText"),
3977
                        isRTL = this._get(inst, "isRTL");
3978
3979
                if (inst.append) {
3980
                        inst.append.remove();
3981
                }
3982
                if (appendText) {
3983
                        inst.append = $("<span class='" + this._appendClass + "'>" + appendText + "</span>");
3984
                        input[isRTL ? "before" : "after"](inst.append);
3985
                }
3986
3987
                input.unbind("focus", this._showDatepicker);
3988
3989
                if (inst.trigger) {
3990
                        inst.trigger.remove();
3991
                }
3992
3993
                showOn = this._get(inst, "showOn");
3994
                if (showOn === "focus" || showOn === "both") { // pop-up date picker when in the marked field
3995
                        input.focus(this._showDatepicker);
3996
                }
3997
                if (showOn === "button" || showOn === "both") { // pop-up date picker when button clicked
3998
                        buttonText = this._get(inst, "buttonText");
3999
                        buttonImage = this._get(inst, "buttonImage");
4000
                        inst.trigger = $(this._get(inst, "buttonImageOnly") ?
4001
                                $("<img/>").addClass(this._triggerClass).
4002
                                        attr({ src: buttonImage, alt: buttonText, title: buttonText }) :
4003
                                $("<button type='button'></button>").addClass(this._triggerClass).
4004
                                        html(!buttonImage ? buttonText : $("<img/>").attr(
4005
                                        { src:buttonImage, alt:buttonText, title:buttonText })));
4006
                        input[isRTL ? "before" : "after"](inst.trigger);
4007
                        inst.trigger.click(function() {
4008
                                if ($.datepicker._datepickerShowing && $.datepicker._lastInput === input[0]) {
4009
                                        $.datepicker._hideDatepicker();
4010
                                } else if ($.datepicker._datepickerShowing && $.datepicker._lastInput !== input[0]) {
4011
                                        $.datepicker._hideDatepicker();
4012
                                        $.datepicker._showDatepicker(input[0]);
4013
                                } else {
4014
                                        $.datepicker._showDatepicker(input[0]);
4015
                                }
4016
                                return false;
4017
                        });
4018
                }
4019
        },
4020
4021
        /* Apply the maximum length for the date format. */
4022
        _autoSize: function(inst) {
4023
                if (this._get(inst, "autoSize") && !inst.inline) {
4024
                        var findMax, max, maxI, i,
4025
                                date = new Date(2009, 12 - 1, 20), // Ensure double digits
4026
                                dateFormat = this._get(inst, "dateFormat");
4027
4028
                        if (dateFormat.match(/[DM]/)) {
4029
                                findMax = function(names) {
4030
                                        max = 0;
4031
                                        maxI = 0;
4032
                                        for (i = 0; i < names.length; i++) {
4033
                                                if (names[i].length > max) {
4034
                                                        max = names[i].length;
4035
                                                        maxI = i;
4036
                                                }
4037
                                        }
4038
                                        return maxI;
4039
                                };
4040
                                date.setMonth(findMax(this._get(inst, (dateFormat.match(/MM/) ?
4041
                                        "monthNames" : "monthNamesShort"))));
4042
                                date.setDate(findMax(this._get(inst, (dateFormat.match(/DD/) ?
4043
                                        "dayNames" : "dayNamesShort"))) + 20 - date.getDay());
4044
                        }
4045
                        inst.input.attr("size", this._formatDate(inst, date).length);
4046
                }
4047
        },
4048
4049
        /* Attach an inline date picker to a div. */
4050
        _inlineDatepicker: function(target, inst) {
4051
                var divSpan = $(target);
4052
                if (divSpan.hasClass(this.markerClassName)) {
4053
                        return;
4054
                }
4055
                divSpan.addClass(this.markerClassName).append(inst.dpDiv);
4056
                $.data(target, "datepicker", inst);
4057
                this._setDate(inst, this._getDefaultDate(inst), true);
4058
                this._updateDatepicker(inst);
4059
                this._updateAlternate(inst);
4060
                //If disabled option is true, disable the datepicker before showing it (see ticket #5665)
4061
                if( inst.settings.disabled ) {
4062
                        this._disableDatepicker( target );
4063
                }
4064
                // Set display:block in place of inst.dpDiv.show() which won't work on disconnected elements
4065
                // http://bugs.jqueryui.com/ticket/7552 - A Datepicker created on a detached div has zero height
4066
                inst.dpDiv.css( "display", "block" );
4067
        },
4068
4069
        /* Pop-up the date picker in a "dialog" box.
4070
         * @param  input element - ignored
4071
         * @param  date        string or Date - the initial date to display
4072
         * @param  onSelect  function - the function to call when a date is selected
4073
         * @param  settings  object - update the dialog date picker instance's settings (anonymous object)
4074
         * @param  pos int[2] - coordinates for the dialog's position within the screen or
4075
         *                                        event - with x/y coordinates or
4076
         *                                        leave empty for default (screen centre)
4077
         * @return the manager object
4078
         */
4079
        _dialogDatepicker: function(input, date, onSelect, settings, pos) {
4080
                var id, browserWidth, browserHeight, scrollX, scrollY,
4081
                        inst = this._dialogInst; // internal instance
4082
4083
                if (!inst) {
4084
                        this.uuid += 1;
4085
                        id = "dp" + this.uuid;
4086
                        this._dialogInput = $("<input type='text' id='" + id +
4087
                                "' style='position: absolute; top: -100px; width: 0px;'/>");
4088
                        this._dialogInput.keydown(this._doKeyDown);
4089
                        $("body").append(this._dialogInput);
4090
                        inst = this._dialogInst = this._newInst(this._dialogInput, false);
4091
                        inst.settings = {};
4092
                        $.data(this._dialogInput[0], "datepicker", inst);
4093
                }
4094
                datepicker_extendRemove(inst.settings, settings || {});
4095
                date = (date && date.constructor === Date ? this._formatDate(inst, date) : date);
4096
                this._dialogInput.val(date);
4097
4098
                this._pos = (pos ? (pos.length ? pos : [pos.pageX, pos.pageY]) : null);
4099
                if (!this._pos) {
4100
                        browserWidth = document.documentElement.clientWidth;
4101
                        browserHeight = document.documentElement.clientHeight;
4102
                        scrollX = document.documentElement.scrollLeft || document.body.scrollLeft;
4103
                        scrollY = document.documentElement.scrollTop || document.body.scrollTop;
4104
                        this._pos = // should use actual width/height below
4105
                                [(browserWidth / 2) - 100 + scrollX, (browserHeight / 2) - 150 + scrollY];
4106
                }
4107
4108
                // move input on screen for focus, but hidden behind dialog
4109
                this._dialogInput.css("left", (this._pos[0] + 20) + "px").css("top", this._pos[1] + "px");
4110
                inst.settings.onSelect = onSelect;
4111
                this._inDialog = true;
4112
                this.dpDiv.addClass(this._dialogClass);
4113
                this._showDatepicker(this._dialogInput[0]);
4114
                if ($.blockUI) {
4115
                        $.blockUI(this.dpDiv);
4116
                }
4117
                $.data(this._dialogInput[0], "datepicker", inst);
4118
                return this;
4119
        },
4120
4121
        /* Detach a datepicker from its control.
4122
         * @param  target        element - the target input field or division or span
4123
         */
4124
        _destroyDatepicker: function(target) {
4125
                var nodeName,
4126
                        $target = $(target),
4127
                        inst = $.data(target, "datepicker");
4128
4129
                if (!$target.hasClass(this.markerClassName)) {
4130
                        return;
4131
                }
4132
4133
                nodeName = target.nodeName.toLowerCase();
4134
                $.removeData(target, "datepicker");
4135
                if (nodeName === "input") {
4136
                        inst.append.remove();
4137
                        inst.trigger.remove();
4138
                        $target.removeClass(this.markerClassName).
4139
                                unbind("focus", this._showDatepicker).
4140
                                unbind("keydown", this._doKeyDown).
4141
                                unbind("keypress", this._doKeyPress).
4142
                                unbind("keyup", this._doKeyUp);
4143
                } else if (nodeName === "div" || nodeName === "span") {
4144
                        $target.removeClass(this.markerClassName).empty();
4145
                }
4146
4147
                if ( datepicker_instActive === inst ) {
4148
                        datepicker_instActive = null;
4149
                }
4150
        },
4151
4152
        /* Enable the date picker to a jQuery selection.
4153
         * @param  target        element - the target input field or division or span
4154
         */
4155
        _enableDatepicker: function(target) {
4156
                var nodeName, inline,
4157
                        $target = $(target),
4158
                        inst = $.data(target, "datepicker");
4159
4160
                if (!$target.hasClass(this.markerClassName)) {
4161
                        return;
4162
                }
4163
4164
                nodeName = target.nodeName.toLowerCase();
4165
                if (nodeName === "input") {
4166
                        target.disabled = false;
4167
                        inst.trigger.filter("button").
4168
                                each(function() { this.disabled = false; }).end().
4169
                                filter("img").css({opacity: "1.0", cursor: ""});
4170
                } else if (nodeName === "div" || nodeName === "span") {
4171
                        inline = $target.children("." + this._inlineClass);
4172
                        inline.children().removeClass("ui-state-disabled");
4173
                        inline.find("select.ui-datepicker-month, select.ui-datepicker-year").
4174
                                prop("disabled", false);
4175
                }
4176
                this._disabledInputs = $.map(this._disabledInputs,
4177
                        function(value) { return (value === target ? null : value); }); // delete entry
4178
        },
4179
4180
        /* Disable the date picker to a jQuery selection.
4181
         * @param  target        element - the target input field or division or span
4182
         */
4183
        _disableDatepicker: function(target) {
4184
                var nodeName, inline,
4185
                        $target = $(target),
4186
                        inst = $.data(target, "datepicker");
4187
4188
                if (!$target.hasClass(this.markerClassName)) {
4189
                        return;
4190
                }
4191
4192
                nodeName = target.nodeName.toLowerCase();
4193
                if (nodeName === "input") {
4194
                        target.disabled = true;
4195
                        inst.trigger.filter("button").
4196
                                each(function() { this.disabled = true; }).end().
4197
                                filter("img").css({opacity: "0.5", cursor: "default"});
4198
                } else if (nodeName === "div" || nodeName === "span") {
4199
                        inline = $target.children("." + this._inlineClass);
4200
                        inline.children().addClass("ui-state-disabled");
4201
                        inline.find("select.ui-datepicker-month, select.ui-datepicker-year").
4202
                                prop("disabled", true);
4203
                }
4204
                this._disabledInputs = $.map(this._disabledInputs,
4205
                        function(value) { return (value === target ? null : value); }); // delete entry
4206
                this._disabledInputs[this._disabledInputs.length] = target;
4207
        },
4208
4209
        /* Is the first field in a jQuery collection disabled as a datepicker?
4210
         * @param  target        element - the target input field or division or span
4211
         * @return boolean - true if disabled, false if enabled
4212
         */
4213
        _isDisabledDatepicker: function(target) {
4214
                if (!target) {
4215
                        return false;
4216
                }
4217
                for (var i = 0; i < this._disabledInputs.length; i++) {
4218
                        if (this._disabledInputs[i] === target) {
4219
                                return true;
4220
                        }
4221
                }
4222
                return false;
4223
        },
4224
4225
        /* Retrieve the instance data for the target control.
4226
         * @param  target  element - the target input field or division or span
4227
         * @return  object - the associated instance data
4228
         * @throws  error if a jQuery problem getting data
4229
         */
4230
        _getInst: function(target) {
4231
                try {
4232
                        return $.data(target, "datepicker");
4233
                }
4234
                catch (err) {
4235
                        throw "Missing instance data for this datepicker";
4236
                }
4237
        },
4238
4239
        /* Update or retrieve the settings for a date picker attached to an input field or division.
4240
         * @param  target  element - the target input field or division or span
4241
         * @param  name        object - the new settings to update or
4242
         *                                string - the name of the setting to change or retrieve,
4243
         *                                when retrieving also "all" for all instance settings or
4244
         *                                "defaults" for all global defaults
4245
         * @param  value   any - the new value for the setting
4246
         *                                (omit if above is an object or to retrieve a value)
4247
         */
4248
        _optionDatepicker: function(target, name, value) {
4249
                var settings, date, minDate, maxDate,
4250
                        inst = this._getInst(target);
4251
4252
                if (arguments.length === 2 && typeof name === "string") {
4253
                        return (name === "defaults" ? $.extend({}, $.datepicker._defaults) :
4254
                                (inst ? (name === "all" ? $.extend({}, inst.settings) :
4255
                                this._get(inst, name)) : null));
4256
                }
4257
4258
                settings = name || {};
4259
                if (typeof name === "string") {
4260
                        settings = {};
4261
                        settings[name] = value;
4262
                }
4263
4264
                if (inst) {
4265
                        if (this._curInst === inst) {
4266
                                this._hideDatepicker();
4267
                        }
4268
4269
                        date = this._getDateDatepicker(target, true);
4270
                        minDate = this._getMinMaxDate(inst, "min");
4271
                        maxDate = this._getMinMaxDate(inst, "max");
4272
                        datepicker_extendRemove(inst.settings, settings);
4273
                        // reformat the old minDate/maxDate values if dateFormat changes and a new minDate/maxDate isn't provided
4274
                        if (minDate !== null && settings.dateFormat !== undefined && settings.minDate === undefined) {
4275
                                inst.settings.minDate = this._formatDate(inst, minDate);
4276
                        }
4277
                        if (maxDate !== null && settings.dateFormat !== undefined && settings.maxDate === undefined) {
4278
                                inst.settings.maxDate = this._formatDate(inst, maxDate);
4279
                        }
4280
                        if ( "disabled" in settings ) {
4281
                                if ( settings.disabled ) {
4282
                                        this._disableDatepicker(target);
4283
                                } else {
4284
                                        this._enableDatepicker(target);
4285
                                }
4286
                        }
4287
                        this._attachments($(target), inst);
4288
                        this._autoSize(inst);
4289
                        this._setDate(inst, date);
4290
                        this._updateAlternate(inst);
4291
                        this._updateDatepicker(inst);
4292
                }
4293
        },
4294
4295
        // change method deprecated
4296
        _changeDatepicker: function(target, name, value) {
4297
                this._optionDatepicker(target, name, value);
4298
        },
4299
4300
        /* Redraw the date picker attached to an input field or division.
4301
         * @param  target  element - the target input field or division or span
4302
         */
4303
        _refreshDatepicker: function(target) {
4304
                var inst = this._getInst(target);
4305
                if (inst) {
4306
                        this._updateDatepicker(inst);
4307
                }
4308
        },
4309
4310
        /* Set the dates for a jQuery selection.
4311
         * @param  target element - the target input field or division or span
4312
         * @param  date        Date - the new date
4313
         */
4314
        _setDateDatepicker: function(target, date) {
4315
                var inst = this._getInst(target);
4316
                if (inst) {
4317
                        this._setDate(inst, date);
4318
                        this._updateDatepicker(inst);
4319
                        this._updateAlternate(inst);
4320
                }
4321
        },
4322
4323
        /* Get the date(s) for the first entry in a jQuery selection.
4324
         * @param  target element - the target input field or division or span
4325
         * @param  noDefault boolean - true if no default date is to be used
4326
         * @return Date - the current date
4327
         */
4328
        _getDateDatepicker: function(target, noDefault) {
4329
                var inst = this._getInst(target);
4330
                if (inst && !inst.inline) {
4331
                        this._setDateFromField(inst, noDefault);
4332
                }
4333
                return (inst ? this._getDate(inst) : null);
4334
        },
4335
4336
        /* Handle keystrokes. */
4337
        _doKeyDown: function(event) {
4338
                var onSelect, dateStr, sel,
4339
                        inst = $.datepicker._getInst(event.target),
4340
                        handled = true,
4341
                        isRTL = inst.dpDiv.is(".ui-datepicker-rtl");
4342
4343
                inst._keyEvent = true;
4344
                if ($.datepicker._datepickerShowing) {
4345
                        switch (event.keyCode) {
4346
                                case 9: $.datepicker._hideDatepicker();
4347
                                                handled = false;
4348
                                                break; // hide on tab out
4349
                                case 13: sel = $("td." + $.datepicker._dayOverClass + ":not(." +
4350
                                                                        $.datepicker._currentClass + ")", inst.dpDiv);
4351
                                                if (sel[0]) {
4352
                                                        $.datepicker._selectDay(event.target, inst.selectedMonth, inst.selectedYear, sel[0]);
4353
                                                }
4354
4355
                                                onSelect = $.datepicker._get(inst, "onSelect");
4356
                                                if (onSelect) {
4357
                                                        dateStr = $.datepicker._formatDate(inst);
4358
4359
                                                        // trigger custom callback
4360
                                                        onSelect.apply((inst.input ? inst.input[0] : null), [dateStr, inst]);
4361
                                                } else {
4362
                                                        $.datepicker._hideDatepicker();
4363
                                                }
4364
4365
                                                return false; // don't submit the form
4366
                                case 27: $.datepicker._hideDatepicker();
4367
                                                break; // hide on escape
4368
                                case 33: $.datepicker._adjustDate(event.target, (event.ctrlKey ?
4369
                                                        -$.datepicker._get(inst, "stepBigMonths") :
4370
                                                        -$.datepicker._get(inst, "stepMonths")), "M");
4371
                                                break; // previous month/year on page up/+ ctrl
4372
                                case 34: $.datepicker._adjustDate(event.target, (event.ctrlKey ?
4373
                                                        +$.datepicker._get(inst, "stepBigMonths") :
4374
                                                        +$.datepicker._get(inst, "stepMonths")), "M");
4375
                                                break; // next month/year on page down/+ ctrl
4376
                                case 35: if (event.ctrlKey || event.metaKey) {
4377
                                                        $.datepicker._clearDate(event.target);
4378
                                                }
4379
                                                handled = event.ctrlKey || event.metaKey;
4380
                                                break; // clear on ctrl or command +end
4381
                                case 36: if (event.ctrlKey || event.metaKey) {
4382
                                                        $.datepicker._gotoToday(event.target);
4383
                                                }
4384
                                                handled = event.ctrlKey || event.metaKey;
4385
                                                break; // current on ctrl or command +home
4386
                                case 37: if (event.ctrlKey || event.metaKey) {
4387
                                                        $.datepicker._adjustDate(event.target, (isRTL ? +1 : -1), "D");
4388
                                                }
4389
                                                handled = event.ctrlKey || event.metaKey;
4390
                                                // -1 day on ctrl or command +left
4391
                                                if (event.originalEvent.altKey) {
4392
                                                        $.datepicker._adjustDate(event.target, (event.ctrlKey ?
4393
                                                                -$.datepicker._get(inst, "stepBigMonths") :
4394
                                                                -$.datepicker._get(inst, "stepMonths")), "M");
4395
                                                }
4396
                                                // next month/year on alt +left on Mac
4397
                                                break;
4398
                                case 38: if (event.ctrlKey || event.metaKey) {
4399
                                                        $.datepicker._adjustDate(event.target, -7, "D");
4400
                                                }
4401
                                                handled = event.ctrlKey || event.metaKey;
4402
                                                break; // -1 week on ctrl or command +up
4403
                                case 39: if (event.ctrlKey || event.metaKey) {
4404
                                                        $.datepicker._adjustDate(event.target, (isRTL ? -1 : +1), "D");
4405
                                                }
4406
                                                handled = event.ctrlKey || event.metaKey;
4407
                                                // +1 day on ctrl or command +right
4408
                                                if (event.originalEvent.altKey) {
4409
                                                        $.datepicker._adjustDate(event.target, (event.ctrlKey ?
4410
                                                                +$.datepicker._get(inst, "stepBigMonths") :
4411
                                                                +$.datepicker._get(inst, "stepMonths")), "M");
4412
                                                }
4413
                                                // next month/year on alt +right
4414
                                                break;
4415
                                case 40: if (event.ctrlKey || event.metaKey) {
4416
                                                        $.datepicker._adjustDate(event.target, +7, "D");
4417
                                                }
4418
                                                handled = event.ctrlKey || event.metaKey;
4419
                                                break; // +1 week on ctrl or command +down
4420
                                default: handled = false;
4421
                        }
4422
                } else if (event.keyCode === 36 && event.ctrlKey) { // display the date picker on ctrl+home
4423
                        $.datepicker._showDatepicker(this);
4424
                } else {
4425
                        handled = false;
4426
                }
4427
4428
                if (handled) {
4429
                        event.preventDefault();
4430
                        event.stopPropagation();
4431
                }
4432
        },
4433
4434
        /* Filter entered characters - based on date format. */
4435
        _doKeyPress: function(event) {
4436
                var chars, chr,
4437
                        inst = $.datepicker._getInst(event.target);
4438
4439
                if ($.datepicker._get(inst, "constrainInput")) {
4440
                        chars = $.datepicker._possibleChars($.datepicker._get(inst, "dateFormat"));
4441
                        chr = String.fromCharCode(event.charCode == null ? event.keyCode : event.charCode);
4442
                        return event.ctrlKey || event.metaKey || (chr < " " || !chars || chars.indexOf(chr) > -1);
4443
                }
4444
        },
4445
4446
        /* Synchronise manual entry and field/alternate field. */
4447
        _doKeyUp: function(event) {
4448
                var date,
4449
                        inst = $.datepicker._getInst(event.target);
4450
4451
                if (inst.input.val() !== inst.lastVal) {
4452
                        try {
4453
                                date = $.datepicker.parseDate($.datepicker._get(inst, "dateFormat"),
4454
                                        (inst.input ? inst.input.val() : null),
4455
                                        $.datepicker._getFormatConfig(inst));
4456
4457
                                if (date) { // only if valid
4458
                                        $.datepicker._setDateFromField(inst);
4459
                                        $.datepicker._updateAlternate(inst);
4460
                                        $.datepicker._updateDatepicker(inst);
4461
                                }
4462
                        }
4463
                        catch (err) {
4464
                        }
4465
                }
4466
                return true;
4467
        },
4468
4469
        /* Pop-up the date picker for a given input field.
4470
         * If false returned from beforeShow event handler do not show.
4471
         * @param  input  element - the input field attached to the date picker or
4472
         *                                        event - if triggered by focus
4473
         */
4474
        _showDatepicker: function(input) {
4475
                input = input.target || input;
4476
                if (input.nodeName.toLowerCase() !== "input") { // find from button/image trigger
4477
                        input = $("input", input.parentNode)[0];
4478
                }
4479
4480
                if ($.datepicker._isDisabledDatepicker(input) || $.datepicker._lastInput === input) { // already here
4481
                        return;
4482
                }
4483
4484
                var inst, beforeShow, beforeShowSettings, isFixed,
4485
                        offset, showAnim, duration;
4486
4487
                inst = $.datepicker._getInst(input);
4488
                if ($.datepicker._curInst && $.datepicker._curInst !== inst) {
4489
                        $.datepicker._curInst.dpDiv.stop(true, true);
4490
                        if ( inst && $.datepicker._datepickerShowing ) {
4491
                                $.datepicker._hideDatepicker( $.datepicker._curInst.input[0] );
4492
                        }
4493
                }
4494
4495
                beforeShow = $.datepicker._get(inst, "beforeShow");
4496
                beforeShowSettings = beforeShow ? beforeShow.apply(input, [input, inst]) : {};
4497
                if(beforeShowSettings === false){
4498
                        return;
4499
                }
4500
                datepicker_extendRemove(inst.settings, beforeShowSettings);
4501
4502
                inst.lastVal = null;
4503
                $.datepicker._lastInput = input;
4504
                $.datepicker._setDateFromField(inst);
4505
4506
                if ($.datepicker._inDialog) { // hide cursor
4507
                        input.value = "";
4508
                }
4509
                if (!$.datepicker._pos) { // position below input
4510
                        $.datepicker._pos = $.datepicker._findPos(input);
4511
                        $.datepicker._pos[1] += input.offsetHeight; // add the height
4512
                }
4513
4514
                isFixed = false;
4515
                $(input).parents().each(function() {
4516
                        isFixed |= $(this).css("position") === "fixed";
4517
                        return !isFixed;
4518
                });
4519
4520
                offset = {left: $.datepicker._pos[0], top: $.datepicker._pos[1]};
4521
                $.datepicker._pos = null;
4522
                //to avoid flashes on Firefox
4523
                inst.dpDiv.empty();
4524
                // determine sizing offscreen
4525
                inst.dpDiv.css({position: "absolute", display: "block", top: "-1000px"});
4526
                $.datepicker._updateDatepicker(inst);
4527
                // fix width for dynamic number of date pickers
4528
                // and adjust position before showing
4529
                offset = $.datepicker._checkOffset(inst, offset, isFixed);
4530
                inst.dpDiv.css({position: ($.datepicker._inDialog && $.blockUI ?
4531
                        "static" : (isFixed ? "fixed" : "absolute")), display: "none",
4532
                        left: offset.left + "px", top: offset.top + "px"});
4533
4534
                if (!inst.inline) {
4535
                        showAnim = $.datepicker._get(inst, "showAnim");
4536
                        duration = $.datepicker._get(inst, "duration");
4537
                        inst.dpDiv.css( "z-index", datepicker_getZindex( $( input ) ) + 1 );
4538
                        $.datepicker._datepickerShowing = true;
4539
4540
                        if ( $.effects && $.effects.effect[ showAnim ] ) {
4541
                                inst.dpDiv.show(showAnim, $.datepicker._get(inst, "showOptions"), duration);
4542
                        } else {
4543
                                inst.dpDiv[showAnim || "show"](showAnim ? duration : null);
4544
                        }
4545
4546
                        if ( $.datepicker._shouldFocusInput( inst ) ) {
4547
                                inst.input.focus();
4548
                        }
4549
4550
                        $.datepicker._curInst = inst;
4551
                }
4552
        },
4553
4554
        /* Generate the date picker content. */
4555
        _updateDatepicker: function(inst) {
4556
                this.maxRows = 4; //Reset the max number of rows being displayed (see #7043)
4557
                datepicker_instActive = inst; // for delegate hover events
4558
                inst.dpDiv.empty().append(this._generateHTML(inst));
4559
                this._attachHandlers(inst);
4560
4561
                var origyearshtml,
4562
                        numMonths = this._getNumberOfMonths(inst),
4563
                        cols = numMonths[1],
4564
                        width = 17,
4565
                        activeCell = inst.dpDiv.find( "." + this._dayOverClass + " a" );
4566
4567
                if ( activeCell.length > 0 ) {
4568
                        datepicker_handleMouseover.apply( activeCell.get( 0 ) );
4569
                }
4570
4571
                inst.dpDiv.removeClass("ui-datepicker-multi-2 ui-datepicker-multi-3 ui-datepicker-multi-4").width("");
4572
                if (cols > 1) {
4573
                        inst.dpDiv.addClass("ui-datepicker-multi-" + cols).css("width", (width * cols) + "em");
4574
                }
4575
                inst.dpDiv[(numMonths[0] !== 1 || numMonths[1] !== 1 ? "add" : "remove") +
4576
                        "Class"]("ui-datepicker-multi");
4577
                inst.dpDiv[(this._get(inst, "isRTL") ? "add" : "remove") +
4578
                        "Class"]("ui-datepicker-rtl");
4579
4580
                if (inst === $.datepicker._curInst && $.datepicker._datepickerShowing && $.datepicker._shouldFocusInput( inst ) ) {
4581
                        inst.input.focus();
4582
                }
4583
4584
                // deffered render of the years select (to avoid flashes on Firefox)
4585
                if( inst.yearshtml ){
4586
                        origyearshtml = inst.yearshtml;
4587
                        setTimeout(function(){
4588
                                //assure that inst.yearshtml didn't change.
4589
                                if( origyearshtml === inst.yearshtml && inst.yearshtml ){
4590
                                        inst.dpDiv.find("select.ui-datepicker-year:first").replaceWith(inst.yearshtml);
4591
                                }
4592
                                origyearshtml = inst.yearshtml = null;
4593
                        }, 0);
4594
                }
4595
        },
4596
4597
        // #6694 - don't focus the input if it's already focused
4598
        // this breaks the change event in IE
4599
        // Support: IE and jQuery <1.9
4600
        _shouldFocusInput: function( inst ) {
4601
                return inst.input && inst.input.is( ":visible" ) && !inst.input.is( ":disabled" ) && !inst.input.is( ":focus" );
4602
        },
4603
4604
        /* Check positioning to remain on screen. */
4605
        _checkOffset: function(inst, offset, isFixed) {
4606
                var dpWidth = inst.dpDiv.outerWidth(),
4607
                        dpHeight = inst.dpDiv.outerHeight(),
4608
                        inputWidth = inst.input ? inst.input.outerWidth() : 0,
4609
                        inputHeight = inst.input ? inst.input.outerHeight() : 0,
4610
                        viewWidth = document.documentElement.clientWidth + (isFixed ? 0 : $(document).scrollLeft()),
4611
                        viewHeight = document.documentElement.clientHeight + (isFixed ? 0 : $(document).scrollTop());
4612
4613
                offset.left -= (this._get(inst, "isRTL") ? (dpWidth - inputWidth) : 0);
4614
                offset.left -= (isFixed && offset.left === inst.input.offset().left) ? $(document).scrollLeft() : 0;
4615
                offset.top -= (isFixed && offset.top === (inst.input.offset().top + inputHeight)) ? $(document).scrollTop() : 0;
4616
4617
                // now check if datepicker is showing outside window viewport - move to a better place if so.
4618
                offset.left -= Math.min(offset.left, (offset.left + dpWidth > viewWidth && viewWidth > dpWidth) ?
4619
                        Math.abs(offset.left + dpWidth - viewWidth) : 0);
4620
                offset.top -= Math.min(offset.top, (offset.top + dpHeight > viewHeight && viewHeight > dpHeight) ?
4621
                        Math.abs(dpHeight + inputHeight) : 0);
4622
4623
                return offset;
4624
        },
4625
4626
        /* Find an object's position on the screen. */
4627
        _findPos: function(obj) {
4628
                var position,
4629
                        inst = this._getInst(obj),
4630
                        isRTL = this._get(inst, "isRTL");
4631
4632
                while (obj && (obj.type === "hidden" || obj.nodeType !== 1 || $.expr.filters.hidden(obj))) {
4633
                        obj = obj[isRTL ? "previousSibling" : "nextSibling"];
4634
                }
4635
4636
                position = $(obj).offset();
4637
                return [position.left, position.top];
4638
        },
4639
4640
        /* Hide the date picker from view.
4641
         * @param  input  element - the input field attached to the date picker
4642
         */
4643
        _hideDatepicker: function(input) {
4644
                var showAnim, duration, postProcess, onClose,
4645
                        inst = this._curInst;
4646
4647
                if (!inst || (input && inst !== $.data(input, "datepicker"))) {
4648
                        return;
4649
                }
4650
4651
                if (this._datepickerShowing) {
4652
                        showAnim = this._get(inst, "showAnim");
4653
                        duration = this._get(inst, "duration");
4654
                        postProcess = function() {
4655
                                $.datepicker._tidyDialog(inst);
4656
                        };
4657
4658
                        // DEPRECATED: after BC for 1.8.x $.effects[ showAnim ] is not needed
4659
                        if ( $.effects && ( $.effects.effect[ showAnim ] || $.effects[ showAnim ] ) ) {
4660
                                inst.dpDiv.hide(showAnim, $.datepicker._get(inst, "showOptions"), duration, postProcess);
4661
                        } else {
4662
                                inst.dpDiv[(showAnim === "slideDown" ? "slideUp" :
4663
                                        (showAnim === "fadeIn" ? "fadeOut" : "hide"))]((showAnim ? duration : null), postProcess);
4664
                        }
4665
4666
                        if (!showAnim) {
4667
                                postProcess();
4668
                        }
4669
                        this._datepickerShowing = false;
4670
4671
                        onClose = this._get(inst, "onClose");
4672
                        if (onClose) {
4673
                                onClose.apply((inst.input ? inst.input[0] : null), [(inst.input ? inst.input.val() : ""), inst]);
4674
                        }
4675
4676
                        this._lastInput = null;
4677
                        if (this._inDialog) {
4678
                                this._dialogInput.css({ position: "absolute", left: "0", top: "-100px" });
4679
                                if ($.blockUI) {
4680
                                        $.unblockUI();
4681
                                        $("body").append(this.dpDiv);
4682
                                }
4683
                        }
4684
                        this._inDialog = false;
4685
                }
4686
        },
4687
4688
        /* Tidy up after a dialog display. */
4689
        _tidyDialog: function(inst) {
4690
                inst.dpDiv.removeClass(this._dialogClass).unbind(".ui-datepicker-calendar");
4691
        },
4692
4693
        /* Close date picker if clicked elsewhere. */
4694
        _checkExternalClick: function(event) {
4695
                if (!$.datepicker._curInst) {
4696
                        return;
4697
                }
4698
4699
                var $target = $(event.target),
4700
                        inst = $.datepicker._getInst($target[0]);
4701
4702
                if ( ( ( $target[0].id !== $.datepicker._mainDivId &&
4703
                                $target.parents("#" + $.datepicker._mainDivId).length === 0 &&
4704
                                !$target.hasClass($.datepicker.markerClassName) &&
4705
                                !$target.closest("." + $.datepicker._triggerClass).length &&
4706
                                $.datepicker._datepickerShowing && !($.datepicker._inDialog && $.blockUI) ) ) ||
4707
                        ( $target.hasClass($.datepicker.markerClassName) && $.datepicker._curInst !== inst ) ) {
4708
                                $.datepicker._hideDatepicker();
4709
                }
4710
        },
4711
4712
        /* Adjust one of the date sub-fields. */
4713
        _adjustDate: function(id, offset, period) {
4714
                var target = $(id),
4715
                        inst = this._getInst(target[0]);
4716
4717
                if (this._isDisabledDatepicker(target[0])) {
4718
                        return;
4719
                }
4720
                this._adjustInstDate(inst, offset +
4721
                        (period === "M" ? this._get(inst, "showCurrentAtPos") : 0), // undo positioning
4722
                        period);
4723
                this._updateDatepicker(inst);
4724
        },
4725
4726
        /* Action for current link. */
4727
        _gotoToday: function(id) {
4728
                var date,
4729
                        target = $(id),
4730
                        inst = this._getInst(target[0]);
4731
4732
                if (this._get(inst, "gotoCurrent") && inst.currentDay) {
4733
                        inst.selectedDay = inst.currentDay;
4734
                        inst.drawMonth = inst.selectedMonth = inst.currentMonth;
4735
                        inst.drawYear = inst.selectedYear = inst.currentYear;
4736
                } else {
4737
                        date = new Date();
4738
                        inst.selectedDay = date.getDate();
4739
                        inst.drawMonth = inst.selectedMonth = date.getMonth();
4740
                        inst.drawYear = inst.selectedYear = date.getFullYear();
4741
                }
4742
                this._notifyChange(inst);
4743
                this._adjustDate(target);
4744
        },
4745
4746
        /* Action for selecting a new month/year. */
4747
        _selectMonthYear: function(id, select, period) {
4748
                var target = $(id),
4749
                        inst = this._getInst(target[0]);
4750
4751
                inst["selected" + (period === "M" ? "Month" : "Year")] =
4752
                inst["draw" + (period === "M" ? "Month" : "Year")] =
4753
                        parseInt(select.options[select.selectedIndex].value,10);
4754
4755
                this._notifyChange(inst);
4756
                this._adjustDate(target);
4757
        },
4758
4759
        /* Action for selecting a day. */
4760
        _selectDay: function(id, month, year, td) {
4761
                var inst,
4762
                        target = $(id);
4763
4764
                if ($(td).hasClass(this._unselectableClass) || this._isDisabledDatepicker(target[0])) {
4765
                        return;
4766
                }
4767
4768
                inst = this._getInst(target[0]);
4769
                inst.selectedDay = inst.currentDay = $("a", td).html();
4770
                inst.selectedMonth = inst.currentMonth = month;
4771
                inst.selectedYear = inst.currentYear = year;
4772
                this._selectDate(id, this._formatDate(inst,
4773
                        inst.currentDay, inst.currentMonth, inst.currentYear));
4774
        },
4775
4776
        /* Erase the input field and hide the date picker. */
4777
        _clearDate: function(id) {
4778
                var target = $(id);
4779
                this._selectDate(target, "");
4780
        },
4781
4782
        /* Update the input field with the selected date. */
4783
        _selectDate: function(id, dateStr) {
4784
                var onSelect,
4785
                        target = $(id),
4786
                        inst = this._getInst(target[0]);
4787
4788
                dateStr = (dateStr != null ? dateStr : this._formatDate(inst));
4789
                if (inst.input) {
4790
                        inst.input.val(dateStr);
4791
                }
4792
                this._updateAlternate(inst);
4793
4794
                onSelect = this._get(inst, "onSelect");
4795
                if (onSelect) {
4796
                        onSelect.apply((inst.input ? inst.input[0] : null), [dateStr, inst]);  // trigger custom callback
4797
                } else if (inst.input) {
4798
                        inst.input.trigger("change"); // fire the change event
4799
                }
4800
4801
                if (inst.inline){
4802
                        this._updateDatepicker(inst);
4803
                } else {
4804
                        this._hideDatepicker();
4805
                        this._lastInput = inst.input[0];
4806
                        if (typeof(inst.input[0]) !== "object") {
4807
                                inst.input.focus(); // restore focus
4808
                        }
4809
                        this._lastInput = null;
4810
                }
4811
        },
4812
4813
        /* Update any alternate field to synchronise with the main field. */
4814
        _updateAlternate: function(inst) {
4815
                var altFormat, date, dateStr,
4816
                        altField = this._get(inst, "altField");
4817
4818
                if (altField) { // update alternate field too
4819
                        altFormat = this._get(inst, "altFormat") || this._get(inst, "dateFormat");
4820
                        date = this._getDate(inst);
4821
                        dateStr = this.formatDate(altFormat, date, this._getFormatConfig(inst));
4822
                        $(altField).each(function() { $(this).val(dateStr); });
4823
                }
4824
        },
4825
4826
        /* Set as beforeShowDay function to prevent selection of weekends.
4827
         * @param  date  Date - the date to customise
4828
         * @return [boolean, string] - is this date selectable?, what is its CSS class?
4829
         */
4830
        noWeekends: function(date) {
4831
                var day = date.getDay();
4832
                return [(day > 0 && day < 6), ""];
4833
        },
4834
4835
        /* Set as calculateWeek to determine the week of the year based on the ISO 8601 definition.
4836
         * @param  date  Date - the date to get the week for
4837
         * @return  number - the number of the week within the year that contains this date
4838
         */
4839
        iso8601Week: function(date) {
4840
                var time,
4841
                        checkDate = new Date(date.getTime());
4842
4843
                // Find Thursday of this week starting on Monday
4844
                checkDate.setDate(checkDate.getDate() + 4 - (checkDate.getDay() || 7));
4845
4846
                time = checkDate.getTime();
4847
                checkDate.setMonth(0); // Compare with Jan 1
4848
                checkDate.setDate(1);
4849
                return Math.floor(Math.round((time - checkDate) / 86400000) / 7) + 1;
4850
        },
4851
4852
        /* Parse a string value into a date object.
4853
         * See formatDate below for the possible formats.
4854
         *
4855
         * @param  format string - the expected format of the date
4856
         * @param  value string - the date in the above format
4857
         * @param  settings Object - attributes include:
4858
         *                                        shortYearCutoff  number - the cutoff year for determining the century (optional)
4859
         *                                        dayNamesShort        string[7] - abbreviated names of the days from Sunday (optional)
4860
         *                                        dayNames                string[7] - names of the days from Sunday (optional)
4861
         *                                        monthNamesShort string[12] - abbreviated names of the months (optional)
4862
         *                                        monthNames                string[12] - names of the months (optional)
4863
         * @return  Date - the extracted date value or null if value is blank
4864
         */
4865
        parseDate: function (format, value, settings) {
4866
                if (format == null || value == null) {
4867
                        throw "Invalid arguments";
4868
                }
4869
4870
                value = (typeof value === "object" ? value.toString() : value + "");
4871
                if (value === "") {
4872
                        return null;
4873
                }
4874
4875
                var iFormat, dim, extra,
4876
                        iValue = 0,
4877
                        shortYearCutoffTemp = (settings ? settings.shortYearCutoff : null) || this._defaults.shortYearCutoff,
4878
                        shortYearCutoff = (typeof shortYearCutoffTemp !== "string" ? shortYearCutoffTemp :
4879
                                new Date().getFullYear() % 100 + parseInt(shortYearCutoffTemp, 10)),
4880
                        dayNamesShort = (settings ? settings.dayNamesShort : null) || this._defaults.dayNamesShort,
4881
                        dayNames = (settings ? settings.dayNames : null) || this._defaults.dayNames,
4882
                        monthNamesShort = (settings ? settings.monthNamesShort : null) || this._defaults.monthNamesShort,
4883
                        monthNames = (settings ? settings.monthNames : null) || this._defaults.monthNames,
4884
                        year = -1,
4885
                        month = -1,
4886
                        day = -1,
4887
                        doy = -1,
4888
                        literal = false,
4889
                        date,
4890
                        // Check whether a format character is doubled
4891
                        lookAhead = function(match) {
4892
                                var matches = (iFormat + 1 < format.length && format.charAt(iFormat + 1) === match);
4893
                                if (matches) {
4894
                                        iFormat++;
4895
                                }
4896
                                return matches;
4897
                        },
4898
                        // Extract a number from the string value
4899
                        getNumber = function(match) {
4900
                                var isDoubled = lookAhead(match),
4901
                                        size = (match === "@" ? 14 : (match === "!" ? 20 :
4902
                                        (match === "y" && isDoubled ? 4 : (match === "o" ? 3 : 2)))),
4903
                                        minSize = (match === "y" ? size : 1),
4904
                                        digits = new RegExp("^\\d{" + minSize + "," + size + "}"),
4905
                                        num = value.substring(iValue).match(digits);
4906
                                if (!num) {
4907
                                        throw "Missing number at position " + iValue;
4908
                                }
4909
                                iValue += num[0].length;
4910
                                return parseInt(num[0], 10);
4911
                        },
4912
                        // Extract a name from the string value and convert to an index
4913
                        getName = function(match, shortNames, longNames) {
4914
                                var index = -1,
4915
                                        names = $.map(lookAhead(match) ? longNames : shortNames, function (v, k) {
4916
                                                return [ [k, v] ];
4917
                                        }).sort(function (a, b) {
4918
                                                return -(a[1].length - b[1].length);
4919
                                        });
4920
4921
                                $.each(names, function (i, pair) {
4922
                                        var name = pair[1];
4923
                                        if (value.substr(iValue, name.length).toLowerCase() === name.toLowerCase()) {
4924
                                                index = pair[0];
4925
                                                iValue += name.length;
4926
                                                return false;
4927
                                        }
4928
                                });
4929
                                if (index !== -1) {
4930
                                        return index + 1;
4931
                                } else {
4932
                                        throw "Unknown name at position " + iValue;
4933
                                }
4934
                        },
4935
                        // Confirm that a literal character matches the string value
4936
                        checkLiteral = function() {
4937
                                if (value.charAt(iValue) !== format.charAt(iFormat)) {
4938
                                        throw "Unexpected literal at position " + iValue;
4939
                                }
4940
                                iValue++;
4941
                        };
4942
4943
                for (iFormat = 0; iFormat < format.length; iFormat++) {
4944
                        if (literal) {
4945
                                if (format.charAt(iFormat) === "'" && !lookAhead("'")) {
4946
                                        literal = false;
4947
                                } else {
4948
                                        checkLiteral();
4949
                                }
4950
                        } else {
4951
                                switch (format.charAt(iFormat)) {
4952
                                        case "d":
4953
                                                day = getNumber("d");
4954
                                                break;
4955
                                        case "D":
4956
                                                getName("D", dayNamesShort, dayNames);
4957
                                                break;
4958
                                        case "o":
4959
                                                doy = getNumber("o");
4960
                                                break;
4961
                                        case "m":
4962
                                                month = getNumber("m");
4963
                                                break;
4964
                                        case "M":
4965
                                                month = getName("M", monthNamesShort, monthNames);
4966
                                                break;
4967
                                        case "y":
4968
                                                year = getNumber("y");
4969
                                                break;
4970
                                        case "@":
4971
                                                date = new Date(getNumber("@"));
4972
                                                year = date.getFullYear();
4973
                                                month = date.getMonth() + 1;
4974
                                                day = date.getDate();
4975
                                                break;
4976
                                        case "!":
4977
                                                date = new Date((getNumber("!") - this._ticksTo1970) / 10000);
4978
                                                year = date.getFullYear();
4979
                                                month = date.getMonth() + 1;
4980
                                                day = date.getDate();
4981
                                                break;
4982
                                        case "'":
4983
                                                if (lookAhead("'")){
4984
                                                        checkLiteral();
4985
                                                } else {
4986
                                                        literal = true;
4987
                                                }
4988
                                                break;
4989
                                        default:
4990
                                                checkLiteral();
4991
                                }
4992
                        }
4993
                }
4994
4995
                if (iValue < value.length){
4996
                        extra = value.substr(iValue);
4997
                        if (!/^\s+/.test(extra)) {
4998
                                throw "Extra/unparsed characters found in date: " + extra;
4999
                        }
5000
                }
5001
5002
                if (year === -1) {
5003
                        year = new Date().getFullYear();
5004
                } else if (year < 100) {
5005
                        year += new Date().getFullYear() - new Date().getFullYear() % 100 +
5006
                                (year <= shortYearCutoff ? 0 : -100);
5007
                }
5008
5009
                if (doy > -1) {
5010
                        month = 1;
5011
                        day = doy;
5012
                        do {
5013
                                dim = this._getDaysInMonth(year, month - 1);
5014
                                if (day <= dim) {
5015
                                        break;
5016
                                }
5017
                                month++;
5018
                                day -= dim;
5019
                        } while (true);
5020
                }
5021
5022
                date = this._daylightSavingAdjust(new Date(year, month - 1, day));
5023
                if (date.getFullYear() !== year || date.getMonth() + 1 !== month || date.getDate() !== day) {
5024
                        throw "Invalid date"; // E.g. 31/02/00
5025
                }
5026
                return date;
5027
        },
5028
5029
        /* Standard date formats. */
5030
        ATOM: "yy-mm-dd", // RFC 3339 (ISO 8601)
5031
        COOKIE: "D, dd M yy",
5032
        ISO_8601: "yy-mm-dd",
5033
        RFC_822: "D, d M y",
5034
        RFC_850: "DD, dd-M-y",
5035
        RFC_1036: "D, d M y",
5036
        RFC_1123: "D, d M yy",
5037
        RFC_2822: "D, d M yy",
5038
        RSS: "D, d M y", // RFC 822
5039
        TICKS: "!",
5040
        TIMESTAMP: "@",
5041
        W3C: "yy-mm-dd", // ISO 8601
5042
5043
        _ticksTo1970: (((1970 - 1) * 365 + Math.floor(1970 / 4) - Math.floor(1970 / 100) +
5044
                Math.floor(1970 / 400)) * 24 * 60 * 60 * 10000000),
5045
5046
        /* Format a date object into a string value.
5047
         * The format can be combinations of the following:
5048
         * d  - day of month (no leading zero)
5049
         * dd - day of month (two digit)
5050
         * o  - day of year (no leading zeros)
5051
         * oo - day of year (three digit)
5052
         * D  - day name short
5053
         * DD - day name long
5054
         * m  - month of year (no leading zero)
5055
         * mm - month of year (two digit)
5056
         * M  - month name short
5057
         * MM - month name long
5058
         * y  - year (two digit)
5059
         * yy - year (four digit)
5060
         * @ - Unix timestamp (ms since 01/01/1970)
5061
         * ! - Windows ticks (100ns since 01/01/0001)
5062
         * "..." - literal text
5063
         * '' - single quote
5064
         *
5065
         * @param  format string - the desired format of the date
5066
         * @param  date Date - the date value to format
5067
         * @param  settings Object - attributes include:
5068
         *                                        dayNamesShort        string[7] - abbreviated names of the days from Sunday (optional)
5069
         *                                        dayNames                string[7] - names of the days from Sunday (optional)
5070
         *                                        monthNamesShort string[12] - abbreviated names of the months (optional)
5071
         *                                        monthNames                string[12] - names of the months (optional)
5072
         * @return  string - the date in the above format
5073
         */
5074
        formatDate: function (format, date, settings) {
5075
                if (!date) {
5076
                        return "";
5077
                }
5078
5079
                var iFormat,
5080
                        dayNamesShort = (settings ? settings.dayNamesShort : null) || this._defaults.dayNamesShort,
5081
                        dayNames = (settings ? settings.dayNames : null) || this._defaults.dayNames,
5082
                        monthNamesShort = (settings ? settings.monthNamesShort : null) || this._defaults.monthNamesShort,
5083
                        monthNames = (settings ? settings.monthNames : null) || this._defaults.monthNames,
5084
                        // Check whether a format character is doubled
5085
                        lookAhead = function(match) {
5086
                                var matches = (iFormat + 1 < format.length && format.charAt(iFormat + 1) === match);
5087
                                if (matches) {
5088
                                        iFormat++;
5089
                                }
5090
                                return matches;
5091
                        },
5092
                        // Format a number, with leading zero if necessary
5093
                        formatNumber = function(match, value, len) {
5094
                                var num = "" + value;
5095
                                if (lookAhead(match)) {
5096
                                        while (num.length < len) {
5097
                                                num = "0" + num;
5098
                                        }
5099
                                }
5100
                                return num;
5101
                        },
5102
                        // Format a name, short or long as requested
5103
                        formatName = function(match, value, shortNames, longNames) {
5104
                                return (lookAhead(match) ? longNames[value] : shortNames[value]);
5105
                        },
5106
                        output = "",
5107
                        literal = false;
5108
5109
                if (date) {
5110
                        for (iFormat = 0; iFormat < format.length; iFormat++) {
5111
                                if (literal) {
5112
                                        if (format.charAt(iFormat) === "'" && !lookAhead("'")) {
5113
                                                literal = false;
5114
                                        } else {
5115
                                                output += format.charAt(iFormat);
5116
                                        }
5117
                                } else {
5118
                                        switch (format.charAt(iFormat)) {
5119
                                                case "d":
5120
                                                        output += formatNumber("d", date.getDate(), 2);
5121
                                                        break;
5122
                                                case "D":
5123
                                                        output += formatName("D", date.getDay(), dayNamesShort, dayNames);
5124
                                                        break;
5125
                                                case "o":
5126
                                                        output += formatNumber("o",
5127
                                                                Math.round((new Date(date.getFullYear(), date.getMonth(), date.getDate()).getTime() - new Date(date.getFullYear(), 0, 0).getTime()) / 86400000), 3);
5128
                                                        break;
5129
                                                case "m":
5130
                                                        output += formatNumber("m", date.getMonth() + 1, 2);
5131
                                                        break;
5132
                                                case "M":
5133
                                                        output += formatName("M", date.getMonth(), monthNamesShort, monthNames);
5134
                                                        break;
5135
                                                case "y":
5136
                                                        output += (lookAhead("y") ? date.getFullYear() :
5137
                                                                (date.getYear() % 100 < 10 ? "0" : "") + date.getYear() % 100);
5138
                                                        break;
5139
                                                case "@":
5140
                                                        output += date.getTime();
5141
                                                        break;
5142
                                                case "!":
5143
                                                        output += date.getTime() * 10000 + this._ticksTo1970;
5144
                                                        break;
5145
                                                case "'":
5146
                                                        if (lookAhead("'")) {
5147
                                                                output += "'";
5148
                                                        } else {
5149
                                                                literal = true;
5150
                                                        }
5151
                                                        break;
5152
                                                default:
5153
                                                        output += format.charAt(iFormat);
5154
                                        }
5155
                                }
5156
                        }
5157
                }
5158
                return output;
5159
        },
5160
5161
        /* Extract all possible characters from the date format. */
5162
        _possibleChars: function (format) {
5163
                var iFormat,
5164
                        chars = "",
5165
                        literal = false,
5166
                        // Check whether a format character is doubled
5167
                        lookAhead = function(match) {
5168
                                var matches = (iFormat + 1 < format.length && format.charAt(iFormat + 1) === match);
5169
                                if (matches) {
5170
                                        iFormat++;
5171
                                }
5172
                                return matches;
5173
                        };
5174
5175
                for (iFormat = 0; iFormat < format.length; iFormat++) {
5176
                        if (literal) {
5177
                                if (format.charAt(iFormat) === "'" && !lookAhead("'")) {
5178
                                        literal = false;
5179
                                } else {
5180
                                        chars += format.charAt(iFormat);
5181
                                }
5182
                        } else {
5183
                                switch (format.charAt(iFormat)) {
5184
                                        case "d": case "m": case "y": case "@":
5185
                                                chars += "0123456789";
5186
                                                break;
5187
                                        case "D": case "M":
5188
                                                return null; // Accept anything
5189
                                        case "'":
5190
                                                if (lookAhead("'")) {
5191
                                                        chars += "'";
5192
                                                } else {
5193
                                                        literal = true;
5194
                                                }
5195
                                                break;
5196
                                        default:
5197
                                                chars += format.charAt(iFormat);
5198
                                }
5199
                        }
5200
                }
5201
                return chars;
5202
        },
5203
5204
        /* Get a setting value, defaulting if necessary. */
5205
        _get: function(inst, name) {
5206
                return inst.settings[name] !== undefined ?
5207
                        inst.settings[name] : this._defaults[name];
5208
        },
5209
5210
        /* Parse existing date and initialise date picker. */
5211
        _setDateFromField: function(inst, noDefault) {
5212
                if (inst.input.val() === inst.lastVal) {
5213
                        return;
5214
                }
5215
5216
                var dateFormat = this._get(inst, "dateFormat"),
5217
                        dates = inst.lastVal = inst.input ? inst.input.val() : null,
5218
                        defaultDate = this._getDefaultDate(inst),
5219
                        date = defaultDate,
5220
                        settings = this._getFormatConfig(inst);
5221
5222
                try {
5223
                        date = this.parseDate(dateFormat, dates, settings) || defaultDate;
5224
                } catch (event) {
5225
                        dates = (noDefault ? "" : dates);
5226
                }
5227
                inst.selectedDay = date.getDate();
5228
                inst.drawMonth = inst.selectedMonth = date.getMonth();
5229
                inst.drawYear = inst.selectedYear = date.getFullYear();
5230
                inst.currentDay = (dates ? date.getDate() : 0);
5231
                inst.currentMonth = (dates ? date.getMonth() : 0);
5232
                inst.currentYear = (dates ? date.getFullYear() : 0);
5233
                this._adjustInstDate(inst);
5234
        },
5235
5236
        /* Retrieve the default date shown on opening. */
5237
        _getDefaultDate: function(inst) {
5238
                return this._restrictMinMax(inst,
5239
                        this._determineDate(inst, this._get(inst, "defaultDate"), new Date()));
5240
        },
5241
5242
        /* A date may be specified as an exact value or a relative one. */
5243
        _determineDate: function(inst, date, defaultDate) {
5244
                var offsetNumeric = function(offset) {
5245
                                var date = new Date();
5246
                                date.setDate(date.getDate() + offset);
5247
                                return date;
5248
                        },
5249
                        offsetString = function(offset) {
5250
                                try {
5251
                                        return $.datepicker.parseDate($.datepicker._get(inst, "dateFormat"),
5252
                                                offset, $.datepicker._getFormatConfig(inst));
5253
                                }
5254
                                catch (e) {
5255
                                        // Ignore
5256
                                }
5257
5258
                                var date = (offset.toLowerCase().match(/^c/) ?
5259
                                        $.datepicker._getDate(inst) : null) || new Date(),
5260
                                        year = date.getFullYear(),
5261
                                        month = date.getMonth(),
5262
                                        day = date.getDate(),
5263
                                        pattern = /([+\-]?[0-9]+)\s*(d|D|w|W|m|M|y|Y)?/g,
5264
                                        matches = pattern.exec(offset);
5265
5266
                                while (matches) {
5267
                                        switch (matches[2] || "d") {
5268
                                                case "d" : case "D" :
5269
                                                        day += parseInt(matches[1],10); break;
5270
                                                case "w" : case "W" :
5271
                                                        day += parseInt(matches[1],10) * 7; break;
5272
                                                case "m" : case "M" :
5273
                                                        month += parseInt(matches[1],10);
5274
                                                        day = Math.min(day, $.datepicker._getDaysInMonth(year, month));
5275
                                                        break;
5276
                                                case "y": case "Y" :
5277
                                                        year += parseInt(matches[1],10);
5278
                                                        day = Math.min(day, $.datepicker._getDaysInMonth(year, month));
5279
                                                        break;
5280
                                        }
5281
                                        matches = pattern.exec(offset);
5282
                                }
5283
                                return new Date(year, month, day);
5284
                        },
5285
                        newDate = (date == null || date === "" ? defaultDate : (typeof date === "string" ? offsetString(date) :
5286
                                (typeof date === "number" ? (isNaN(date) ? defaultDate : offsetNumeric(date)) : new Date(date.getTime()))));
5287
5288
                newDate = (newDate && newDate.toString() === "Invalid Date" ? defaultDate : newDate);
5289
                if (newDate) {
5290
                        newDate.setHours(0);
5291
                        newDate.setMinutes(0);
5292
                        newDate.setSeconds(0);
5293
                        newDate.setMilliseconds(0);
5294
                }
5295
                return this._daylightSavingAdjust(newDate);
5296
        },
5297
5298
        /* Handle switch to/from daylight saving.
5299
         * Hours may be non-zero on daylight saving cut-over:
5300
         * > 12 when midnight changeover, but then cannot generate
5301
         * midnight datetime, so jump to 1AM, otherwise reset.
5302
         * @param  date  (Date) the date to check
5303
         * @return  (Date) the corrected date
5304
         */
5305
        _daylightSavingAdjust: function(date) {
5306
                if (!date) {
5307
                        return null;
5308
                }
5309
                date.setHours(date.getHours() > 12 ? date.getHours() + 2 : 0);
5310
                return date;
5311
        },
5312
5313
        /* Set the date(s) directly. */
5314
        _setDate: function(inst, date, noChange) {
5315
                var clear = !date,
5316
                        origMonth = inst.selectedMonth,
5317
                        origYear = inst.selectedYear,
5318
                        newDate = this._restrictMinMax(inst, this._determineDate(inst, date, new Date()));
5319
5320
                inst.selectedDay = inst.currentDay = newDate.getDate();
5321
                inst.drawMonth = inst.selectedMonth = inst.currentMonth = newDate.getMonth();
5322
                inst.drawYear = inst.selectedYear = inst.currentYear = newDate.getFullYear();
5323
                if ((origMonth !== inst.selectedMonth || origYear !== inst.selectedYear) && !noChange) {
5324
                        this._notifyChange(inst);
5325
                }
5326
                this._adjustInstDate(inst);
5327
                if (inst.input) {
5328
                        inst.input.val(clear ? "" : this._formatDate(inst));
5329
                }
5330
        },
5331
5332
        /* Retrieve the date(s) directly. */
5333
        _getDate: function(inst) {
5334
                var startDate = (!inst.currentYear || (inst.input && inst.input.val() === "") ? null :
5335
                        this._daylightSavingAdjust(new Date(
5336
                        inst.currentYear, inst.currentMonth, inst.currentDay)));
5337
                        return startDate;
5338
        },
5339
5340
        /* Attach the onxxx handlers.  These are declared statically so
5341
         * they work with static code transformers like Caja.
5342
         */
5343
        _attachHandlers: function(inst) {
5344
                var stepMonths = this._get(inst, "stepMonths"),
5345
                        id = "#" + inst.id.replace( /\\\\/g, "\\" );
5346
                inst.dpDiv.find("[data-handler]").map(function () {
5347
                        var handler = {
5348
                                prev: function () {
5349
                                        $.datepicker._adjustDate(id, -stepMonths, "M");
5350
                                },
5351
                                next: function () {
5352
                                        $.datepicker._adjustDate(id, +stepMonths, "M");
5353
                                },
5354
                                hide: function () {
5355
                                        $.datepicker._hideDatepicker();
5356
                                },
5357
                                today: function () {
5358
                                        $.datepicker._gotoToday(id);
5359
                                },
5360
                                selectDay: function () {
5361
                                        $.datepicker._selectDay(id, +this.getAttribute("data-month"), +this.getAttribute("data-year"), this);
5362
                                        return false;
5363
                                },
5364
                                selectMonth: function () {
5365
                                        $.datepicker._selectMonthYear(id, this, "M");
5366
                                        return false;
5367
                                },
5368
                                selectYear: function () {
5369
                                        $.datepicker._selectMonthYear(id, this, "Y");
5370
                                        return false;
5371
                                }
5372
                        };
5373
                        $(this).bind(this.getAttribute("data-event"), handler[this.getAttribute("data-handler")]);
5374
                });
5375
        },
5376
5377
        /* Generate the HTML for the current state of the date picker. */
5378
        _generateHTML: function(inst) {
5379
                var maxDraw, prevText, prev, nextText, next, currentText, gotoDate,
5380
                        controls, buttonPanel, firstDay, showWeek, dayNames, dayNamesMin,
5381
                        monthNames, monthNamesShort, beforeShowDay, showOtherMonths,
5382
                        selectOtherMonths, defaultDate, html, dow, row, group, col, selectedDate,
5383
                        cornerClass, calender, thead, day, daysInMonth, leadDays, curRows, numRows,
5384
                        printDate, dRow, tbody, daySettings, otherMonth, unselectable,
5385
                        tempDate = new Date(),
5386
                        today = this._daylightSavingAdjust(
5387
                                new Date(tempDate.getFullYear(), tempDate.getMonth(), tempDate.getDate())), // clear time
5388
                        isRTL = this._get(inst, "isRTL"),
5389
                        showButtonPanel = this._get(inst, "showButtonPanel"),
5390
                        hideIfNoPrevNext = this._get(inst, "hideIfNoPrevNext"),
5391
                        navigationAsDateFormat = this._get(inst, "navigationAsDateFormat"),
5392
                        numMonths = this._getNumberOfMonths(inst),
5393
                        showCurrentAtPos = this._get(inst, "showCurrentAtPos"),
5394
                        stepMonths = this._get(inst, "stepMonths"),
5395
                        isMultiMonth = (numMonths[0] !== 1 || numMonths[1] !== 1),
5396
                        currentDate = this._daylightSavingAdjust((!inst.currentDay ? new Date(9999, 9, 9) :
5397
                                new Date(inst.currentYear, inst.currentMonth, inst.currentDay))),
5398
                        minDate = this._getMinMaxDate(inst, "min"),
5399
                        maxDate = this._getMinMaxDate(inst, "max"),
5400
                        drawMonth = inst.drawMonth - showCurrentAtPos,
5401
                        drawYear = inst.drawYear;
5402
5403
                if (drawMonth < 0) {
5404
                        drawMonth += 12;
5405
                        drawYear--;
5406
                }
5407
                if (maxDate) {
5408
                        maxDraw = this._daylightSavingAdjust(new Date(maxDate.getFullYear(),
5409
                                maxDate.getMonth() - (numMonths[0] * numMonths[1]) + 1, maxDate.getDate()));
5410
                        maxDraw = (minDate && maxDraw < minDate ? minDate : maxDraw);
5411
                        while (this._daylightSavingAdjust(new Date(drawYear, drawMonth, 1)) > maxDraw) {
5412
                                drawMonth--;
5413
                                if (drawMonth < 0) {
5414
                                        drawMonth = 11;
5415
                                        drawYear--;
5416
                                }
5417
                        }
5418
                }
5419
                inst.drawMonth = drawMonth;
5420
                inst.drawYear = drawYear;
5421
5422
                prevText = this._get(inst, "prevText");
5423
                prevText = (!navigationAsDateFormat ? prevText : this.formatDate(prevText,
5424
                        this._daylightSavingAdjust(new Date(drawYear, drawMonth - stepMonths, 1)),
5425
                        this._getFormatConfig(inst)));
5426
5427
                prev = (this._canAdjustMonth(inst, -1, drawYear, drawMonth) ?
5428
                        "<a class='ui-datepicker-prev ui-corner-all' data-handler='prev' data-event='click'" +
5429
                        " title='" + prevText + "'><span class='ui-icon ui-icon-circle-triangle-" + ( isRTL ? "e" : "w") + "'>" + prevText + "</span></a>" :
5430
                        (hideIfNoPrevNext ? "" : "<a class='ui-datepicker-prev ui-corner-all ui-state-disabled' title='"+ prevText +"'><span class='ui-icon ui-icon-circle-triangle-" + ( isRTL ? "e" : "w") + "'>" + prevText + "</span></a>"));
5431
5432
                nextText = this._get(inst, "nextText");
5433
                nextText = (!navigationAsDateFormat ? nextText : this.formatDate(nextText,
5434
                        this._daylightSavingAdjust(new Date(drawYear, drawMonth + stepMonths, 1)),
5435
                        this._getFormatConfig(inst)));
5436
5437
                next = (this._canAdjustMonth(inst, +1, drawYear, drawMonth) ?
5438
                        "<a class='ui-datepicker-next ui-corner-all' data-handler='next' data-event='click'" +
5439
                        " title='" + nextText + "'><span class='ui-icon ui-icon-circle-triangle-" + ( isRTL ? "w" : "e") + "'>" + nextText + "</span></a>" :
5440
                        (hideIfNoPrevNext ? "" : "<a class='ui-datepicker-next ui-corner-all ui-state-disabled' title='"+ nextText + "'><span class='ui-icon ui-icon-circle-triangle-" + ( isRTL ? "w" : "e") + "'>" + nextText + "</span></a>"));
5441
5442
                currentText = this._get(inst, "currentText");
5443
                gotoDate = (this._get(inst, "gotoCurrent") && inst.currentDay ? currentDate : today);
5444
                currentText = (!navigationAsDateFormat ? currentText :
5445
                        this.formatDate(currentText, gotoDate, this._getFormatConfig(inst)));
5446
5447
                controls = (!inst.inline ? "<button type='button' class='ui-datepicker-close ui-state-default ui-priority-primary ui-corner-all' data-handler='hide' data-event='click'>" +
5448
                        this._get(inst, "closeText") + "</button>" : "");
5449
5450
                buttonPanel = (showButtonPanel) ? "<div class='ui-datepicker-buttonpane ui-widget-content'>" + (isRTL ? controls : "") +
5451
                        (this._isInRange(inst, gotoDate) ? "<button type='button' class='ui-datepicker-current ui-state-default ui-priority-secondary ui-corner-all' data-handler='today' data-event='click'" +
5452
                        ">" + currentText + "</button>" : "") + (isRTL ? "" : controls) + "</div>" : "";
5453
5454
                firstDay = parseInt(this._get(inst, "firstDay"),10);
5455
                firstDay = (isNaN(firstDay) ? 0 : firstDay);
5456
5457
                showWeek = this._get(inst, "showWeek");
5458
                dayNames = this._get(inst, "dayNames");
5459
                dayNamesMin = this._get(inst, "dayNamesMin");
5460
                monthNames = this._get(inst, "monthNames");
5461
                monthNamesShort = this._get(inst, "monthNamesShort");
5462
                beforeShowDay = this._get(inst, "beforeShowDay");
5463
                showOtherMonths = this._get(inst, "showOtherMonths");
5464
                selectOtherMonths = this._get(inst, "selectOtherMonths");
5465
                defaultDate = this._getDefaultDate(inst);
5466
                html = "";
5467
                dow;
5468
                for (row = 0; row < numMonths[0]; row++) {
5469
                        group = "";
5470
                        this.maxRows = 4;
5471
                        for (col = 0; col < numMonths[1]; col++) {
5472
                                selectedDate = this._daylightSavingAdjust(new Date(drawYear, drawMonth, inst.selectedDay));
5473
                                cornerClass = " ui-corner-all";
5474
                                calender = "";
5475
                                if (isMultiMonth) {
5476
                                        calender += "<div class='ui-datepicker-group";
5477
                                        if (numMonths[1] > 1) {
5478
                                                switch (col) {
5479
                                                        case 0: calender += " ui-datepicker-group-first";
5480
                                                                cornerClass = " ui-corner-" + (isRTL ? "right" : "left"); break;
5481
                                                        case numMonths[1]-1: calender += " ui-datepicker-group-last";
5482
                                                                cornerClass = " ui-corner-" + (isRTL ? "left" : "right"); break;
5483
                                                        default: calender += " ui-datepicker-group-middle"; cornerClass = ""; break;
5484
                                                }
5485
                                        }
5486
                                        calender += "'>";
5487
                                }
5488
                                calender += "<div class='ui-datepicker-header ui-widget-header ui-helper-clearfix" + cornerClass + "'>" +
5489
                                        (/all|left/.test(cornerClass) && row === 0 ? (isRTL ? next : prev) : "") +
5490
                                        (/all|right/.test(cornerClass) && row === 0 ? (isRTL ? prev : next) : "") +
5491
                                        this._generateMonthYearHeader(inst, drawMonth, drawYear, minDate, maxDate,
5492
                                        row > 0 || col > 0, monthNames, monthNamesShort) + // draw month headers
5493
                                        "</div><table class='ui-datepicker-calendar'><thead>" +
5494
                                        "<tr>";
5495
                                thead = (showWeek ? "<th class='ui-datepicker-week-col'>" + this._get(inst, "weekHeader") + "</th>" : "");
5496
                                for (dow = 0; dow < 7; dow++) { // days of the week
5497
                                        day = (dow + firstDay) % 7;
5498
                                        thead += "<th scope='col'" + ((dow + firstDay + 6) % 7 >= 5 ? " class='ui-datepicker-week-end'" : "") + ">" +
5499
                                                "<span title='" + dayNames[day] + "'>" + dayNamesMin[day] + "</span></th>";
5500
                                }
5501
                                calender += thead + "</tr></thead><tbody>";
5502
                                daysInMonth = this._getDaysInMonth(drawYear, drawMonth);
5503
                                if (drawYear === inst.selectedYear && drawMonth === inst.selectedMonth) {
5504
                                        inst.selectedDay = Math.min(inst.selectedDay, daysInMonth);
5505
                                }
5506
                                leadDays = (this._getFirstDayOfMonth(drawYear, drawMonth) - firstDay + 7) % 7;
5507
                                curRows = Math.ceil((leadDays + daysInMonth) / 7); // calculate the number of rows to generate
5508
                                numRows = (isMultiMonth ? this.maxRows > curRows ? this.maxRows : curRows : curRows); //If multiple months, use the higher number of rows (see #7043)
5509
                                this.maxRows = numRows;
5510
                                printDate = this._daylightSavingAdjust(new Date(drawYear, drawMonth, 1 - leadDays));
5511
                                for (dRow = 0; dRow < numRows; dRow++) { // create date picker rows
5512
                                        calender += "<tr>";
5513
                                        tbody = (!showWeek ? "" : "<td class='ui-datepicker-week-col'>" +
5514
                                                this._get(inst, "calculateWeek")(printDate) + "</td>");
5515
                                        for (dow = 0; dow < 7; dow++) { // create date picker days
5516
                                                daySettings = (beforeShowDay ?
5517
                                                        beforeShowDay.apply((inst.input ? inst.input[0] : null), [printDate]) : [true, ""]);
5518
                                                otherMonth = (printDate.getMonth() !== drawMonth);
5519
                                                unselectable = (otherMonth && !selectOtherMonths) || !daySettings[0] ||
5520
                                                        (minDate && printDate < minDate) || (maxDate && printDate > maxDate);
5521
                                                tbody += "<td class='" +
5522
                                                        ((dow + firstDay + 6) % 7 >= 5 ? " ui-datepicker-week-end" : "") + // highlight weekends
5523
                                                        (otherMonth ? " ui-datepicker-other-month" : "") + // highlight days from other months
5524
                                                        ((printDate.getTime() === selectedDate.getTime() && drawMonth === inst.selectedMonth && inst._keyEvent) || // user pressed key
5525
                                                        (defaultDate.getTime() === printDate.getTime() && defaultDate.getTime() === selectedDate.getTime()) ?
5526
                                                        // or defaultDate is current printedDate and defaultDate is selectedDate
5527
                                                        " " + this._dayOverClass : "") + // highlight selected day
5528
                                                        (unselectable ? " " + this._unselectableClass + " ui-state-disabled": "") +  // highlight unselectable days
5529
                                                        (otherMonth && !showOtherMonths ? "" : " " + daySettings[1] + // highlight custom dates
5530
                                                        (printDate.getTime() === currentDate.getTime() ? " " + this._currentClass : "") + // highlight selected day
5531
                                                        (printDate.getTime() === today.getTime() ? " ui-datepicker-today" : "")) + "'" + // highlight today (if different)
5532
                                                        ((!otherMonth || showOtherMonths) && daySettings[2] ? " title='" + daySettings[2].replace(/'/g, "&#39;") + "'" : "") + // cell title
5533
                                                        (unselectable ? "" : " data-handler='selectDay' data-event='click' data-month='" + printDate.getMonth() + "' data-year='" + printDate.getFullYear() + "'") + ">" + // actions
5534
                                                        (otherMonth && !showOtherMonths ? "&#xa0;" : // display for other months
5535
                                                        (unselectable ? "<span class='ui-state-default'>" + printDate.getDate() + "</span>" : "<a class='ui-state-default" +
5536
                                                        (printDate.getTime() === today.getTime() ? " ui-state-highlight" : "") +
5537
                                                        (printDate.getTime() === currentDate.getTime() ? " ui-state-active" : "") + // highlight selected day
5538
                                                        (otherMonth ? " ui-priority-secondary" : "") + // distinguish dates from other months
5539
                                                        "' href='#'>" + printDate.getDate() + "</a>")) + "</td>"; // display selectable date
5540
                                                printDate.setDate(printDate.getDate() + 1);
5541
                                                printDate = this._daylightSavingAdjust(printDate);
5542
                                        }
5543
                                        calender += tbody + "</tr>";
5544
                                }
5545
                                drawMonth++;
5546
                                if (drawMonth > 11) {
5547
                                        drawMonth = 0;
5548
                                        drawYear++;
5549
                                }
5550
                                calender += "</tbody></table>" + (isMultiMonth ? "</div>" +
5551
                                                        ((numMonths[0] > 0 && col === numMonths[1]-1) ? "<div class='ui-datepicker-row-break'></div>" : "") : "");
5552
                                group += calender;
5553
                        }
5554
                        html += group;
5555
                }
5556
                html += buttonPanel;
5557
                inst._keyEvent = false;
5558
                return html;
5559
        },
5560
5561
        /* Generate the month and year header. */
5562
        _generateMonthYearHeader: function(inst, drawMonth, drawYear, minDate, maxDate,
5563
                        secondary, monthNames, monthNamesShort) {
5564
5565
                var inMinYear, inMaxYear, month, years, thisYear, determineYear, year, endYear,
5566
                        changeMonth = this._get(inst, "changeMonth"),
5567
                        changeYear = this._get(inst, "changeYear"),
5568
                        showMonthAfterYear = this._get(inst, "showMonthAfterYear"),
5569
                        html = "<div class='ui-datepicker-title'>",
5570
                        monthHtml = "";
5571
5572
                // month selection
5573
                if (secondary || !changeMonth) {
5574
                        monthHtml += "<span class='ui-datepicker-month'>" + monthNames[drawMonth] + "</span>";
5575
                } else {
5576
                        inMinYear = (minDate && minDate.getFullYear() === drawYear);
5577
                        inMaxYear = (maxDate && maxDate.getFullYear() === drawYear);
5578
                        monthHtml += "<select class='ui-datepicker-month' data-handler='selectMonth' data-event='change'>";
5579
                        for ( month = 0; month < 12; month++) {
5580
                                if ((!inMinYear || month >= minDate.getMonth()) && (!inMaxYear || month <= maxDate.getMonth())) {
5581
                                        monthHtml += "<option value='" + month + "'" +
5582
                                                (month === drawMonth ? " selected='selected'" : "") +
5583
                                                ">" + monthNamesShort[month] + "</option>";
5584
                                }
5585
                        }
5586
                        monthHtml += "</select>";
5587
                }
5588
5589
                if (!showMonthAfterYear) {
5590
                        html += monthHtml + (secondary || !(changeMonth && changeYear) ? "&#xa0;" : "");
5591
                }
5592
5593
                // year selection
5594
                if ( !inst.yearshtml ) {
5595
                        inst.yearshtml = "";
5596
                        if (secondary || !changeYear) {
5597
                                html += "<span class='ui-datepicker-year'>" + drawYear + "</span>";
5598
                        } else {
5599
                                // determine range of years to display
5600
                                years = this._get(inst, "yearRange").split(":");
5601
                                thisYear = new Date().getFullYear();
5602
                                determineYear = function(value) {
5603
                                        var year = (value.match(/c[+\-].*/) ? drawYear + parseInt(value.substring(1), 10) :
5604
                                                (value.match(/[+\-].*/) ? thisYear + parseInt(value, 10) :
5605
                                                parseInt(value, 10)));
5606
                                        return (isNaN(year) ? thisYear : year);
5607
                                };
5608
                                year = determineYear(years[0]);
5609
                                endYear = Math.max(year, determineYear(years[1] || ""));
5610
                                year = (minDate ? Math.max(year, minDate.getFullYear()) : year);
5611
                                endYear = (maxDate ? Math.min(endYear, maxDate.getFullYear()) : endYear);
5612
                                inst.yearshtml += "<select class='ui-datepicker-year' data-handler='selectYear' data-event='change'>";
5613
                                for (; year <= endYear; year++) {
5614
                                        inst.yearshtml += "<option value='" + year + "'" +
5615
                                                (year === drawYear ? " selected='selected'" : "") +
5616
                                                ">" + year + "</option>";
5617
                                }
5618
                                inst.yearshtml += "</select>";
5619
5620
                                html += inst.yearshtml;
5621
                                inst.yearshtml = null;
5622
                        }
5623
                }
5624
5625
                html += this._get(inst, "yearSuffix");
5626
                if (showMonthAfterYear) {
5627
                        html += (secondary || !(changeMonth && changeYear) ? "&#xa0;" : "") + monthHtml;
5628
                }
5629
                html += "</div>"; // Close datepicker_header
5630
                return html;
5631
        },
5632
5633
        /* Adjust one of the date sub-fields. */
5634
        _adjustInstDate: function(inst, offset, period) {
5635
                var year = inst.drawYear + (period === "Y" ? offset : 0),
5636
                        month = inst.drawMonth + (period === "M" ? offset : 0),
5637
                        day = Math.min(inst.selectedDay, this._getDaysInMonth(year, month)) + (period === "D" ? offset : 0),
5638
                        date = this._restrictMinMax(inst, this._daylightSavingAdjust(new Date(year, month, day)));
5639
5640
                inst.selectedDay = date.getDate();
5641
                inst.drawMonth = inst.selectedMonth = date.getMonth();
5642
                inst.drawYear = inst.selectedYear = date.getFullYear();
5643
                if (period === "M" || period === "Y") {
5644
                        this._notifyChange(inst);
5645
                }
5646
        },
5647
5648
        /* Ensure a date is within any min/max bounds. */
5649
        _restrictMinMax: function(inst, date) {
5650
                var minDate = this._getMinMaxDate(inst, "min"),
5651
                        maxDate = this._getMinMaxDate(inst, "max"),
5652
                        newDate = (minDate && date < minDate ? minDate : date);
5653
                return (maxDate && newDate > maxDate ? maxDate : newDate);
5654
        },
5655
5656
        /* Notify change of month/year. */
5657
        _notifyChange: function(inst) {
5658
                var onChange = this._get(inst, "onChangeMonthYear");
5659
                if (onChange) {
5660
                        onChange.apply((inst.input ? inst.input[0] : null),
5661
                                [inst.selectedYear, inst.selectedMonth + 1, inst]);
5662
                }
5663
        },
5664
5665
        /* Determine the number of months to show. */
5666
        _getNumberOfMonths: function(inst) {
5667
                var numMonths = this._get(inst, "numberOfMonths");
5668
                return (numMonths == null ? [1, 1] : (typeof numMonths === "number" ? [1, numMonths] : numMonths));
5669
        },
5670
5671
        /* Determine the current maximum date - ensure no time components are set. */
5672
        _getMinMaxDate: function(inst, minMax) {
5673
                return this._determineDate(inst, this._get(inst, minMax + "Date"), null);
5674
        },
5675
5676
        /* Find the number of days in a given month. */
5677
        _getDaysInMonth: function(year, month) {
5678
                return 32 - this._daylightSavingAdjust(new Date(year, month, 32)).getDate();
5679
        },
5680
5681
        /* Find the day of the week of the first of a month. */
5682
        _getFirstDayOfMonth: function(year, month) {
5683
                return new Date(year, month, 1).getDay();
5684
        },
5685
5686
        /* Determines if we should allow a "next/prev" month display change. */
5687
        _canAdjustMonth: function(inst, offset, curYear, curMonth) {
5688
                var numMonths = this._getNumberOfMonths(inst),
5689
                        date = this._daylightSavingAdjust(new Date(curYear,
5690
                        curMonth + (offset < 0 ? offset : numMonths[0] * numMonths[1]), 1));
5691
5692
                if (offset < 0) {
5693
                        date.setDate(this._getDaysInMonth(date.getFullYear(), date.getMonth()));
5694
                }
5695
                return this._isInRange(inst, date);
5696
        },
5697
5698
        /* Is the given date in the accepted range? */
5699
        _isInRange: function(inst, date) {
5700
                var yearSplit, currentYear,
5701
                        minDate = this._getMinMaxDate(inst, "min"),
5702
                        maxDate = this._getMinMaxDate(inst, "max"),
5703
                        minYear = null,
5704
                        maxYear = null,
5705
                        years = this._get(inst, "yearRange");
5706
                        if (years){
5707
                                yearSplit = years.split(":");
5708
                                currentYear = new Date().getFullYear();
5709
                                minYear = parseInt(yearSplit[0], 10);
5710
                                maxYear = parseInt(yearSplit[1], 10);
5711
                                if ( yearSplit[0].match(/[+\-].*/) ) {
5712
                                        minYear += currentYear;
5713
                                }
5714
                                if ( yearSplit[1].match(/[+\-].*/) ) {
5715
                                        maxYear += currentYear;
5716
                                }
5717
                        }
5718
5719
                return ((!minDate || date.getTime() >= minDate.getTime()) &&
5720
                        (!maxDate || date.getTime() <= maxDate.getTime()) &&
5721
                        (!minYear || date.getFullYear() >= minYear) &&
5722
                        (!maxYear || date.getFullYear() <= maxYear));
5723
        },
5724
5725
        /* Provide the configuration settings for formatting/parsing. */
5726
        _getFormatConfig: function(inst) {
5727
                var shortYearCutoff = this._get(inst, "shortYearCutoff");
5728
                shortYearCutoff = (typeof shortYearCutoff !== "string" ? shortYearCutoff :
5729
                        new Date().getFullYear() % 100 + parseInt(shortYearCutoff, 10));
5730
                return {shortYearCutoff: shortYearCutoff,
5731
                        dayNamesShort: this._get(inst, "dayNamesShort"), dayNames: this._get(inst, "dayNames"),
5732
                        monthNamesShort: this._get(inst, "monthNamesShort"), monthNames: this._get(inst, "monthNames")};
5733
        },
5734
5735
        /* Format the given date for display. */
5736
        _formatDate: function(inst, day, month, year) {
5737
                if (!day) {
5738
                        inst.currentDay = inst.selectedDay;
5739
                        inst.currentMonth = inst.selectedMonth;
5740
                        inst.currentYear = inst.selectedYear;
5741
                }
5742
                var date = (day ? (typeof day === "object" ? day :
5743
                        this._daylightSavingAdjust(new Date(year, month, day))) :
5744
                        this._daylightSavingAdjust(new Date(inst.currentYear, inst.currentMonth, inst.currentDay)));
5745
                return this.formatDate(this._get(inst, "dateFormat"), date, this._getFormatConfig(inst));
5746
        }
5747
});
5748
5749
/*
5750
 * Bind hover events for datepicker elements.
5751
 * Done via delegate so the binding only occurs once in the lifetime of the parent div.
5752
 * Global datepicker_instActive, set by _updateDatepicker allows the handlers to find their way back to the active picker.
5753
 */
5754
function datepicker_bindHover(dpDiv) {
5755
        var selector = "button, .ui-datepicker-prev, .ui-datepicker-next, .ui-datepicker-calendar td a";
5756
        return dpDiv.delegate(selector, "mouseout", function() {
5757
                        $(this).removeClass("ui-state-hover");
5758
                        if (this.className.indexOf("ui-datepicker-prev") !== -1) {
5759
                                $(this).removeClass("ui-datepicker-prev-hover");
5760
                        }
5761
                        if (this.className.indexOf("ui-datepicker-next") !== -1) {
5762
                                $(this).removeClass("ui-datepicker-next-hover");
5763
                        }
5764
                })
5765
                .delegate( selector, "mouseover", datepicker_handleMouseover );
5766
}
5767
5768
function datepicker_handleMouseover() {
5769
        if (!$.datepicker._isDisabledDatepicker( datepicker_instActive.inline? datepicker_instActive.dpDiv.parent()[0] : datepicker_instActive.input[0])) {
5770
                $(this).parents(".ui-datepicker-calendar").find("a").removeClass("ui-state-hover");
5771
                $(this).addClass("ui-state-hover");
5772
                if (this.className.indexOf("ui-datepicker-prev") !== -1) {
5773
                        $(this).addClass("ui-datepicker-prev-hover");
5774
                }
5775
                if (this.className.indexOf("ui-datepicker-next") !== -1) {
5776
                        $(this).addClass("ui-datepicker-next-hover");
5777
                }
5778
        }
5779
}
5780
5781
/* jQuery extend now ignores nulls! */
5782
function datepicker_extendRemove(target, props) {
5783
        $.extend(target, props);
5784
        for (var name in props) {
5785
                if (props[name] == null) {
5786
                        target[name] = props[name];
5787
                }
5788
        }
5789
        return target;
5790
}
5791
5792
/* Invoke the datepicker functionality.
5793
   @param  options  string - a command, optionally followed by additional parameters or
5794
                                        Object - settings for attaching new datepicker functionality
5795
   @return  jQuery object */
5796
$.fn.datepicker = function(options){
5797
5798
        /* Verify an empty collection wasn't passed - Fixes #6976 */
5799
        if ( !this.length ) {
5800
                return this;
5801
        }
5802
5803
        /* Initialise the date picker. */
5804
        if (!$.datepicker.initialized) {
5805
                $(document).mousedown($.datepicker._checkExternalClick);
5806
                $.datepicker.initialized = true;
5807
        }
5808
5809
        /* Append datepicker main container to body if not exist. */
5810
        if ($("#"+$.datepicker._mainDivId).length === 0) {
5811
                $("body").append($.datepicker.dpDiv);
5812
        }
5813
5814
        var otherArgs = Array.prototype.slice.call(arguments, 1);
5815
        if (typeof options === "string" && (options === "isDisabled" || options === "getDate" || options === "widget")) {
5816
                return $.datepicker["_" + options + "Datepicker"].
5817
                        apply($.datepicker, [this[0]].concat(otherArgs));
5818
        }
5819
        if (options === "option" && arguments.length === 2 && typeof arguments[1] === "string") {
5820
                return $.datepicker["_" + options + "Datepicker"].
5821
                        apply($.datepicker, [this[0]].concat(otherArgs));
5822
        }
5823
        return this.each(function() {
5824
                typeof options === "string" ?
5825
                        $.datepicker["_" + options + "Datepicker"].
5826
                                apply($.datepicker, [this].concat(otherArgs)) :
5827
                        $.datepicker._attachDatepicker(this, options);
5828
        });
5829
};
5830
5831
$.datepicker = new Datepicker(); // singleton instance
5832
$.datepicker.initialized = false;
5833
$.datepicker.uuid = new Date().getTime();
5834
$.datepicker.version = "1.11.4";
5835
5836
var datepicker = $.datepicker;
5837
5838
5839
/*!
5840
 * jQuery UI Draggable 1.11.4
5841
 * http://jqueryui.com
5842
 *
5843
 * Copyright jQuery Foundation and other contributors
5844
 * Released under the MIT license.
5845
 * http://jquery.org/license
5846
 *
5847
 * http://api.jqueryui.com/draggable/
5848
 */
5849
5850
5851
$.widget("ui.draggable", $.ui.mouse, {
5852
        version: "1.11.4",
5853
        widgetEventPrefix: "drag",
5854
        options: {
5855
                addClasses: true,
5856
                appendTo: "parent",
5857
                axis: false,
5858
                connectToSortable: false,
5859
                containment: false,
5860
                cursor: "auto",
5861
                cursorAt: false,
5862
                grid: false,
5863
                handle: false,
5864
                helper: "original",
5865
                iframeFix: false,
5866
                opacity: false,
5867
                refreshPositions: false,
5868
                revert: false,
5869
                revertDuration: 500,
5870
                scope: "default",
5871
                scroll: true,
5872
                scrollSensitivity: 20,
5873
                scrollSpeed: 20,
5874
                snap: false,
5875
                snapMode: "both",
5876
                snapTolerance: 20,
5877
                stack: false,
5878
                zIndex: false,
5879
5880
                // callbacks
5881
                drag: null,
5882
                start: null,
5883
                stop: null
5884
        },
5885
        _create: function() {
5886
5887
                if ( this.options.helper === "original" ) {
5888
                        this._setPositionRelative();
5889
                }
5890
                if (this.options.addClasses){
5891
                        this.element.addClass("ui-draggable");
5892
                }
5893
                if (this.options.disabled){
5894
                        this.element.addClass("ui-draggable-disabled");
5895
                }
5896
                this._setHandleClassName();
5897
5898
                this._mouseInit();
5899
        },
5900
5901
        _setOption: function( key, value ) {
5902
                this._super( key, value );
5903
                if ( key === "handle" ) {
5904
                        this._removeHandleClassName();
5905
                        this._setHandleClassName();
5906
                }
5907
        },
5908
5909
        _destroy: function() {
5910
                if ( ( this.helper || this.element ).is( ".ui-draggable-dragging" ) ) {
5911
                        this.destroyOnClear = true;
5912
                        return;
5913
                }
5914
                this.element.removeClass( "ui-draggable ui-draggable-dragging ui-draggable-disabled" );
5915
                this._removeHandleClassName();
5916
                this._mouseDestroy();
5917
        },
5918
5919
        _mouseCapture: function(event) {
5920
                var o = this.options;
5921
5922
                this._blurActiveElement( event );
5923
5924
                // among others, prevent a drag on a resizable-handle
5925
                if (this.helper || o.disabled || $(event.target).closest(".ui-resizable-handle").length > 0) {
5926
                        return false;
5927
                }
5928
5929
                //Quit if we're not on a valid handle
5930
                this.handle = this._getHandle(event);
5931
                if (!this.handle) {
5932
                        return false;
5933
                }
5934
5935
                this._blockFrames( o.iframeFix === true ? "iframe" : o.iframeFix );
5936
5937
                return true;
5938
5939
        },
5940
5941
        _blockFrames: function( selector ) {
5942
                this.iframeBlocks = this.document.find( selector ).map(function() {
5943
                        var iframe = $( this );
5944
5945
                        return $( "<div>" )
5946
                                .css( "position", "absolute" )
5947
                                .appendTo( iframe.parent() )
5948
                                .outerWidth( iframe.outerWidth() )
5949
                                .outerHeight( iframe.outerHeight() )
5950
                                .offset( iframe.offset() )[ 0 ];
5951
                });
5952
        },
5953
5954
        _unblockFrames: function() {
5955
                if ( this.iframeBlocks ) {
5956
                        this.iframeBlocks.remove();
5957
                        delete this.iframeBlocks;
5958
                }
5959
        },
5960
5961
        _blurActiveElement: function( event ) {
5962
                var document = this.document[ 0 ];
5963
5964
                // Only need to blur if the event occurred on the draggable itself, see #10527
5965
                if ( !this.handleElement.is( event.target ) ) {
5966
                        return;
5967
                }
5968
5969
                // support: IE9
5970
                // IE9 throws an "Unspecified error" accessing document.activeElement from an <iframe>
5971
                try {
5972
5973
                        // Support: IE9, IE10
5974
                        // If the <body> is blurred, IE will switch windows, see #9520
5975
                        if ( document.activeElement && document.activeElement.nodeName.toLowerCase() !== "body" ) {
5976
5977
                                // Blur any element that currently has focus, see #4261
5978
                                $( document.activeElement ).blur();
5979
                        }
5980
                } catch ( error ) {}
5981
        },
5982
5983
        _mouseStart: function(event) {
5984
5985
                var o = this.options;
5986
5987
                //Create and append the visible helper
5988
                this.helper = this._createHelper(event);
5989
5990
                this.helper.addClass("ui-draggable-dragging");
5991
5992
                //Cache the helper size
5993
                this._cacheHelperProportions();
5994
5995
                //If ddmanager is used for droppables, set the global draggable
5996
                if ($.ui.ddmanager) {
5997
                        $.ui.ddmanager.current = this;
5998
                }
5999
6000
                /*
6001
                 * - Position generation -
6002
                 * This block generates everything position related - it's the core of draggables.
6003
                 */
6004
6005
                //Cache the margins of the original element
6006
                this._cacheMargins();
6007
6008
                //Store the helper's css position
6009
                this.cssPosition = this.helper.css( "position" );
6010
                this.scrollParent = this.helper.scrollParent( true );
6011
                this.offsetParent = this.helper.offsetParent();
6012
                this.hasFixedAncestor = this.helper.parents().filter(function() {
6013
                                return $( this ).css( "position" ) === "fixed";
6014
                        }).length > 0;
6015
6016
                //The element's absolute position on the page minus margins
6017
                this.positionAbs = this.element.offset();
6018
                this._refreshOffsets( event );
6019
6020
                //Generate the original position
6021
                this.originalPosition = this.position = this._generatePosition( event, false );
6022
                this.originalPageX = event.pageX;
6023
                this.originalPageY = event.pageY;
6024
6025
                //Adjust the mouse offset relative to the helper if "cursorAt" is supplied
6026
                (o.cursorAt && this._adjustOffsetFromHelper(o.cursorAt));
6027
6028
                //Set a containment if given in the options
6029
                this._setContainment();
6030
6031
                //Trigger event + callbacks
6032
                if (this._trigger("start", event) === false) {
6033
                        this._clear();
6034
                        return false;
6035
                }
6036
6037
                //Recache the helper size
6038
                this._cacheHelperProportions();
6039
6040
                //Prepare the droppable offsets
6041
                if ($.ui.ddmanager && !o.dropBehaviour) {
6042
                        $.ui.ddmanager.prepareOffsets(this, event);
6043
                }
6044
6045
                // Reset helper's right/bottom css if they're set and set explicit width/height instead
6046
                // as this prevents resizing of elements with right/bottom set (see #7772)
6047
                this._normalizeRightBottom();
6048
6049
                this._mouseDrag(event, true); //Execute the drag once - this causes the helper not to be visible before getting its correct position
6050
6051
                //If the ddmanager is used for droppables, inform the manager that dragging has started (see #5003)
6052
                if ( $.ui.ddmanager ) {
6053
                        $.ui.ddmanager.dragStart(this, event);
6054
                }
6055
6056
                return true;
6057
        },
6058
6059
        _refreshOffsets: function( event ) {
6060
                this.offset = {
6061
                        top: this.positionAbs.top - this.margins.top,
6062
                        left: this.positionAbs.left - this.margins.left,
6063
                        scroll: false,
6064
                        parent: this._getParentOffset(),
6065
                        relative: this._getRelativeOffset()
6066
                };
6067
6068
                this.offset.click = {
6069
                        left: event.pageX - this.offset.left,
6070
                        top: event.pageY - this.offset.top
6071
                };
6072
        },
6073
6074
        _mouseDrag: function(event, noPropagation) {
6075
                // reset any necessary cached properties (see #5009)
6076
                if ( this.hasFixedAncestor ) {
6077
                        this.offset.parent = this._getParentOffset();
6078
                }
6079
6080
                //Compute the helpers position
6081
                this.position = this._generatePosition( event, true );
6082
                this.positionAbs = this._convertPositionTo("absolute");
6083
6084
                //Call plugins and callbacks and use the resulting position if something is returned
6085
                if (!noPropagation) {
6086
                        var ui = this._uiHash();
6087
                        if (this._trigger("drag", event, ui) === false) {
6088
                                this._mouseUp({});
6089
                                return false;
6090
                        }
6091
                        this.position = ui.position;
6092
                }
6093
6094
                this.helper[ 0 ].style.left = this.position.left + "px";
6095
                this.helper[ 0 ].style.top = this.position.top + "px";
6096
6097
                if ($.ui.ddmanager) {
6098
                        $.ui.ddmanager.drag(this, event);
6099
                }
6100
6101
                return false;
6102
        },
6103
6104
        _mouseStop: function(event) {
6105
6106
                //If we are using droppables, inform the manager about the drop
6107
                var that = this,
6108
                        dropped = false;
6109
                if ($.ui.ddmanager && !this.options.dropBehaviour) {
6110
                        dropped = $.ui.ddmanager.drop(this, event);
6111
                }
6112
6113
                //if a drop comes from outside (a sortable)
6114
                if (this.dropped) {
6115
                        dropped = this.dropped;
6116
                        this.dropped = false;
6117
                }
6118
6119
                if ((this.options.revert === "invalid" && !dropped) || (this.options.revert === "valid" && dropped) || this.options.revert === true || ($.isFunction(this.options.revert) && this.options.revert.call(this.element, dropped))) {
6120
                        $(this.helper).animate(this.originalPosition, parseInt(this.options.revertDuration, 10), function() {
6121
                                if (that._trigger("stop", event) !== false) {
6122
                                        that._clear();
6123
                                }
6124
                        });
6125
                } else {
6126
                        if (this._trigger("stop", event) !== false) {
6127
                                this._clear();
6128
                        }
6129
                }
6130
6131
                return false;
6132
        },
6133
6134
        _mouseUp: function( event ) {
6135
                this._unblockFrames();
6136
6137
                //If the ddmanager is used for droppables, inform the manager that dragging has stopped (see #5003)
6138
                if ( $.ui.ddmanager ) {
6139
                        $.ui.ddmanager.dragStop(this, event);
6140
                }
6141
6142
                // Only need to focus if the event occurred on the draggable itself, see #10527
6143
                if ( this.handleElement.is( event.target ) ) {
6144
                        // The interaction is over; whether or not the click resulted in a drag, focus the element
6145
                        this.element.focus();
6146
                }
6147
6148
                return $.ui.mouse.prototype._mouseUp.call(this, event);
6149
        },
6150
6151
        cancel: function() {
6152
6153
                if (this.helper.is(".ui-draggable-dragging")) {
6154
                        this._mouseUp({});
6155
                } else {
6156
                        this._clear();
6157
                }
6158
6159
                return this;
6160
6161
        },
6162
6163
        _getHandle: function(event) {
6164
                return this.options.handle ?
6165
                        !!$( event.target ).closest( this.element.find( this.options.handle ) ).length :
6166
                        true;
6167
        },
6168
6169
        _setHandleClassName: function() {
6170
                this.handleElement = this.options.handle ?
6171
                        this.element.find( this.options.handle ) : this.element;
6172
                this.handleElement.addClass( "ui-draggable-handle" );
6173
        },
6174
6175
        _removeHandleClassName: function() {
6176
                this.handleElement.removeClass( "ui-draggable-handle" );
6177
        },
6178
6179
        _createHelper: function(event) {
6180
6181
                var o = this.options,
6182
                        helperIsFunction = $.isFunction( o.helper ),
6183
                        helper = helperIsFunction ?
6184
                                $( o.helper.apply( this.element[ 0 ], [ event ] ) ) :
6185
                                ( o.helper === "clone" ?
6186
                                        this.element.clone().removeAttr( "id" ) :
6187
                                        this.element );
6188
6189
                if (!helper.parents("body").length) {
6190
                        helper.appendTo((o.appendTo === "parent" ? this.element[0].parentNode : o.appendTo));
6191
                }
6192
6193
                // http://bugs.jqueryui.com/ticket/9446
6194
                // a helper function can return the original element
6195
                // which wouldn't have been set to relative in _create
6196
                if ( helperIsFunction && helper[ 0 ] === this.element[ 0 ] ) {
6197
                        this._setPositionRelative();
6198
                }
6199
6200
                if (helper[0] !== this.element[0] && !(/(fixed|absolute)/).test(helper.css("position"))) {
6201
                        helper.css("position", "absolute");
6202
                }
6203
6204
                return helper;
6205
6206
        },
6207
6208
        _setPositionRelative: function() {
6209
                if ( !( /^(?:r|a|f)/ ).test( this.element.css( "position" ) ) ) {
6210
                        this.element[ 0 ].style.position = "relative";
6211
                }
6212
        },
6213
6214
        _adjustOffsetFromHelper: function(obj) {
6215
                if (typeof obj === "string") {
6216
                        obj = obj.split(" ");
6217
                }
6218
                if ($.isArray(obj)) {
6219
                        obj = { left: +obj[0], top: +obj[1] || 0 };
6220
                }
6221
                if ("left" in obj) {
6222
                        this.offset.click.left = obj.left + this.margins.left;
6223
                }
6224
                if ("right" in obj) {
6225
                        this.offset.click.left = this.helperProportions.width - obj.right + this.margins.left;
6226
                }
6227
                if ("top" in obj) {
6228
                        this.offset.click.top = obj.top + this.margins.top;
6229
                }
6230
                if ("bottom" in obj) {
6231
                        this.offset.click.top = this.helperProportions.height - obj.bottom + this.margins.top;
6232
                }
6233
        },
6234
6235
        _isRootNode: function( element ) {
6236
                return ( /(html|body)/i ).test( element.tagName ) || element === this.document[ 0 ];
6237
        },
6238
6239
        _getParentOffset: function() {
6240
6241
                //Get the offsetParent and cache its position
6242
                var po = this.offsetParent.offset(),
6243
                        document = this.document[ 0 ];
6244
6245
                // This is a special case where we need to modify a offset calculated on start, since the following happened:
6246
                // 1. The position of the helper is absolute, so it's position is calculated based on the next positioned parent
6247
                // 2. The actual offset parent is a child of the scroll parent, and the scroll parent isn't the document, which means that
6248
                //    the scroll is included in the initial calculation of the offset of the parent, and never recalculated upon drag
6249
                if (this.cssPosition === "absolute" && this.scrollParent[0] !== document && $.contains(this.scrollParent[0], this.offsetParent[0])) {
6250
                        po.left += this.scrollParent.scrollLeft();
6251
                        po.top += this.scrollParent.scrollTop();
6252
                }
6253
6254
                if ( this._isRootNode( this.offsetParent[ 0 ] ) ) {
6255
                        po = { top: 0, left: 0 };
6256
                }
6257
6258
                return {
6259
                        top: po.top + (parseInt(this.offsetParent.css("borderTopWidth"), 10) || 0),
6260
                        left: po.left + (parseInt(this.offsetParent.css("borderLeftWidth"), 10) || 0)
6261
                };
6262
6263
        },
6264
6265
        _getRelativeOffset: function() {
6266
                if ( this.cssPosition !== "relative" ) {
6267
                        return { top: 0, left: 0 };
6268
                }
6269
6270
                var p = this.element.position(),
6271
                        scrollIsRootNode = this._isRootNode( this.scrollParent[ 0 ] );
6272
6273
                return {
6274
                        top: p.top - ( parseInt(this.helper.css( "top" ), 10) || 0 ) + ( !scrollIsRootNode ? this.scrollParent.scrollTop() : 0 ),
6275
                        left: p.left - ( parseInt(this.helper.css( "left" ), 10) || 0 ) + ( !scrollIsRootNode ? this.scrollParent.scrollLeft() : 0 )
6276
                };
6277
6278
        },
6279
6280
        _cacheMargins: function() {
6281
                this.margins = {
6282
                        left: (parseInt(this.element.css("marginLeft"), 10) || 0),
6283
                        top: (parseInt(this.element.css("marginTop"), 10) || 0),
6284
                        right: (parseInt(this.element.css("marginRight"), 10) || 0),
6285
                        bottom: (parseInt(this.element.css("marginBottom"), 10) || 0)
6286
                };
6287
        },
6288
6289
        _cacheHelperProportions: function() {
6290
                this.helperProportions = {
6291
                        width: this.helper.outerWidth(),
6292
                        height: this.helper.outerHeight()
6293
                };
6294
        },
6295
6296
        _setContainment: function() {
6297
6298
                var isUserScrollable, c, ce,
6299
                        o = this.options,
6300
                        document = this.document[ 0 ];
6301
6302
                this.relativeContainer = null;
6303
6304
                if ( !o.containment ) {
6305
                        this.containment = null;
6306
                        return;
6307
                }
6308
6309
                if ( o.containment === "window" ) {
6310
                        this.containment = [
6311
                                $( window ).scrollLeft() - this.offset.relative.left - this.offset.parent.left,
6312
                                $( window ).scrollTop() - this.offset.relative.top - this.offset.parent.top,
6313
                                $( window ).scrollLeft() + $( window ).width() - this.helperProportions.width - this.margins.left,
6314
                                $( window ).scrollTop() + ( $( window ).height() || document.body.parentNode.scrollHeight ) - this.helperProportions.height - this.margins.top
6315
                        ];
6316
                        return;
6317
                }
6318
6319
                if ( o.containment === "document") {
6320
                        this.containment = [
6321
                                0,
6322
                                0,
6323
                                $( document ).width() - this.helperProportions.width - this.margins.left,
6324
                                ( $( document ).height() || document.body.parentNode.scrollHeight ) - this.helperProportions.height - this.margins.top
6325
                        ];
6326
                        return;
6327
                }
6328
6329
                if ( o.containment.constructor === Array ) {
6330
                        this.containment = o.containment;
6331
                        return;
6332
                }
6333
6334
                if ( o.containment === "parent" ) {
6335
                        o.containment = this.helper[ 0 ].parentNode;
6336
                }
6337
6338
                c = $( o.containment );
6339
                ce = c[ 0 ];
6340
6341
                if ( !ce ) {
6342
                        return;
6343
                }
6344
6345
                isUserScrollable = /(scroll|auto)/.test( c.css( "overflow" ) );
6346
6347
                this.containment = [
6348
                        ( parseInt( c.css( "borderLeftWidth" ), 10 ) || 0 ) + ( parseInt( c.css( "paddingLeft" ), 10 ) || 0 ),
6349
                        ( parseInt( c.css( "borderTopWidth" ), 10 ) || 0 ) + ( parseInt( c.css( "paddingTop" ), 10 ) || 0 ),
6350
                        ( isUserScrollable ? Math.max( ce.scrollWidth, ce.offsetWidth ) : ce.offsetWidth ) -
6351
                                ( parseInt( c.css( "borderRightWidth" ), 10 ) || 0 ) -
6352
                                ( parseInt( c.css( "paddingRight" ), 10 ) || 0 ) -
6353
                                this.helperProportions.width -
6354
                                this.margins.left -
6355
                                this.margins.right,
6356
                        ( isUserScrollable ? Math.max( ce.scrollHeight, ce.offsetHeight ) : ce.offsetHeight ) -
6357
                                ( parseInt( c.css( "borderBottomWidth" ), 10 ) || 0 ) -
6358
                                ( parseInt( c.css( "paddingBottom" ), 10 ) || 0 ) -
6359
                                this.helperProportions.height -
6360
                                this.margins.top -
6361
                                this.margins.bottom
6362
                ];
6363
                this.relativeContainer = c;
6364
        },
6365
6366
        _convertPositionTo: function(d, pos) {
6367
6368
                if (!pos) {
6369
                        pos = this.position;
6370
                }
6371
6372
                var mod = d === "absolute" ? 1 : -1,
6373
                        scrollIsRootNode = this._isRootNode( this.scrollParent[ 0 ] );
6374
6375
                return {
6376
                        top: (
6377
                                pos.top        +                                                                                                                                // The absolute mouse position
6378
                                this.offset.relative.top * mod +                                                                                // Only for relative positioned nodes: Relative offset from element to offset parent
6379
                                this.offset.parent.top * mod -                                                                                // The offsetParent's offset without borders (offset + border)
6380
                                ( ( this.cssPosition === "fixed" ? -this.offset.scroll.top : ( scrollIsRootNode ? 0 : this.offset.scroll.top ) ) * mod)
6381
                        ),
6382
                        left: (
6383
                                pos.left +                                                                                                                                // The absolute mouse position
6384
                                this.offset.relative.left * mod +                                                                                // Only for relative positioned nodes: Relative offset from element to offset parent
6385
                                this.offset.parent.left * mod        -                                                                                // The offsetParent's offset without borders (offset + border)
6386
                                ( ( this.cssPosition === "fixed" ? -this.offset.scroll.left : ( scrollIsRootNode ? 0 : this.offset.scroll.left ) ) * mod)
6387
                        )
6388
                };
6389
6390
        },
6391
6392
        _generatePosition: function( event, constrainPosition ) {
6393
6394
                var containment, co, top, left,
6395
                        o = this.options,
6396
                        scrollIsRootNode = this._isRootNode( this.scrollParent[ 0 ] ),
6397
                        pageX = event.pageX,
6398
                        pageY = event.pageY;
6399
6400
                // Cache the scroll
6401
                if ( !scrollIsRootNode || !this.offset.scroll ) {
6402
                        this.offset.scroll = {
6403
                                top: this.scrollParent.scrollTop(),
6404
                                left: this.scrollParent.scrollLeft()
6405
                        };
6406
                }
6407
6408
                /*
6409
                 * - Position constraining -
6410
                 * Constrain the position to a mix of grid, containment.
6411
                 */
6412
6413
                // If we are not dragging yet, we won't check for options
6414
                if ( constrainPosition ) {
6415
                        if ( this.containment ) {
6416
                                if ( this.relativeContainer ){
6417
                                        co = this.relativeContainer.offset();
6418
                                        containment = [
6419
                                                this.containment[ 0 ] + co.left,
6420
                                                this.containment[ 1 ] + co.top,
6421
                                                this.containment[ 2 ] + co.left,
6422
                                                this.containment[ 3 ] + co.top
6423
                                        ];
6424
                                } else {
6425
                                        containment = this.containment;
6426
                                }
6427
6428
                                if (event.pageX - this.offset.click.left < containment[0]) {
6429
                                        pageX = containment[0] + this.offset.click.left;
6430
                                }
6431
                                if (event.pageY - this.offset.click.top < containment[1]) {
6432
                                        pageY = containment[1] + this.offset.click.top;
6433
                                }
6434
                                if (event.pageX - this.offset.click.left > containment[2]) {
6435
                                        pageX = containment[2] + this.offset.click.left;
6436
                                }
6437
                                if (event.pageY - this.offset.click.top > containment[3]) {
6438
                                        pageY = containment[3] + this.offset.click.top;
6439
                                }
6440
                        }
6441
6442
                        if (o.grid) {
6443
                                //Check for grid elements set to 0 to prevent divide by 0 error causing invalid argument errors in IE (see ticket #6950)
6444
                                top = o.grid[1] ? this.originalPageY + Math.round((pageY - this.originalPageY) / o.grid[1]) * o.grid[1] : this.originalPageY;
6445
                                pageY = containment ? ((top - this.offset.click.top >= containment[1] || top - this.offset.click.top > containment[3]) ? top : ((top - this.offset.click.top >= containment[1]) ? top - o.grid[1] : top + o.grid[1])) : top;
6446
6447
                                left = o.grid[0] ? this.originalPageX + Math.round((pageX - this.originalPageX) / o.grid[0]) * o.grid[0] : this.originalPageX;
6448
                                pageX = containment ? ((left - this.offset.click.left >= containment[0] || left - this.offset.click.left > containment[2]) ? left : ((left - this.offset.click.left >= containment[0]) ? left - o.grid[0] : left + o.grid[0])) : left;
6449
                        }
6450
6451
                        if ( o.axis === "y" ) {
6452
                                pageX = this.originalPageX;
6453
                        }
6454
6455
                        if ( o.axis === "x" ) {
6456
                                pageY = this.originalPageY;
6457
                        }
6458
                }
6459
6460
                return {
6461
                        top: (
6462
                                pageY -                                                                                                                                        // The absolute mouse position
6463
                                this.offset.click.top        -                                                                                                // Click offset (relative to the element)
6464
                                this.offset.relative.top -                                                                                                // Only for relative positioned nodes: Relative offset from element to offset parent
6465
                                this.offset.parent.top +                                                                                                // The offsetParent's offset without borders (offset + border)
6466
                                ( this.cssPosition === "fixed" ? -this.offset.scroll.top : ( scrollIsRootNode ? 0 : this.offset.scroll.top ) )
6467
                        ),
6468
                        left: (
6469
                                pageX -                                                                                                                                        // The absolute mouse position
6470
                                this.offset.click.left -                                                                                                // Click offset (relative to the element)
6471
                                this.offset.relative.left -                                                                                                // Only for relative positioned nodes: Relative offset from element to offset parent
6472
                                this.offset.parent.left +                                                                                                // The offsetParent's offset without borders (offset + border)
6473
                                ( this.cssPosition === "fixed" ? -this.offset.scroll.left : ( scrollIsRootNode ? 0 : this.offset.scroll.left ) )
6474
                        )
6475
                };
6476
6477
        },
6478
6479
        _clear: function() {
6480
                this.helper.removeClass("ui-draggable-dragging");
6481
                if (this.helper[0] !== this.element[0] && !this.cancelHelperRemoval) {
6482
                        this.helper.remove();
6483
                }
6484
                this.helper = null;
6485
                this.cancelHelperRemoval = false;
6486
                if ( this.destroyOnClear ) {
6487
                        this.destroy();
6488
                }
6489
        },
6490
6491
        _normalizeRightBottom: function() {
6492
                if ( this.options.axis !== "y" && this.helper.css( "right" ) !== "auto" ) {
6493
                        this.helper.width( this.helper.width() );
6494
                        this.helper.css( "right", "auto" );
6495
                }
6496
                if ( this.options.axis !== "x" && this.helper.css( "bottom" ) !== "auto" ) {
6497
                        this.helper.height( this.helper.height() );
6498
                        this.helper.css( "bottom", "auto" );
6499
                }
6500
        },
6501
6502
        // From now on bulk stuff - mainly helpers
6503
6504
        _trigger: function( type, event, ui ) {
6505
                ui = ui || this._uiHash();
6506
                $.ui.plugin.call( this, type, [ event, ui, this ], true );
6507
6508
                // Absolute position and offset (see #6884 ) have to be recalculated after plugins
6509
                if ( /^(drag|start|stop)/.test( type ) ) {
6510
                        this.positionAbs = this._convertPositionTo( "absolute" );
6511
                        ui.offset = this.positionAbs;
6512
                }
6513
                return $.Widget.prototype._trigger.call( this, type, event, ui );
6514
        },
6515
6516
        plugins: {},
6517
6518
        _uiHash: function() {
6519
                return {
6520
                        helper: this.helper,
6521
                        position: this.position,
6522
                        originalPosition: this.originalPosition,
6523
                        offset: this.positionAbs
6524
                };
6525
        }
6526
6527
});
6528
6529
$.ui.plugin.add( "draggable", "connectToSortable", {
6530
        start: function( event, ui, draggable ) {
6531
                var uiSortable = $.extend( {}, ui, {
6532
                        item: draggable.element
6533
                });
6534
6535
                draggable.sortables = [];
6536
                $( draggable.options.connectToSortable ).each(function() {
6537
                        var sortable = $( this ).sortable( "instance" );
6538
6539
                        if ( sortable && !sortable.options.disabled ) {
6540
                                draggable.sortables.push( sortable );
6541
6542
                                // refreshPositions is called at drag start to refresh the containerCache
6543
                                // which is used in drag. This ensures it's initialized and synchronized
6544
                                // with any changes that might have happened on the page since initialization.
6545
                                sortable.refreshPositions();
6546
                                sortable._trigger("activate", event, uiSortable);
6547
                        }
6548
                });
6549
        },
6550
        stop: function( event, ui, draggable ) {
6551
                var uiSortable = $.extend( {}, ui, {
6552
                        item: draggable.element
6553
                });
6554
6555
                draggable.cancelHelperRemoval = false;
6556
6557
                $.each( draggable.sortables, function() {
6558
                        var sortable = this;
6559
6560
                        if ( sortable.isOver ) {
6561
                                sortable.isOver = 0;
6562
6563
                                // Allow this sortable to handle removing the helper
6564
                                draggable.cancelHelperRemoval = true;
6565
                                sortable.cancelHelperRemoval = false;
6566
6567
                                // Use _storedCSS To restore properties in the sortable,
6568
                                // as this also handles revert (#9675) since the draggable
6569
                                // may have modified them in unexpected ways (#8809)
6570
                                sortable._storedCSS = {
6571
                                        position: sortable.placeholder.css( "position" ),
6572
                                        top: sortable.placeholder.css( "top" ),
6573
                                        left: sortable.placeholder.css( "left" )
6574
                                };
6575
6576
                                sortable._mouseStop(event);
6577
6578
                                // Once drag has ended, the sortable should return to using
6579
                                // its original helper, not the shared helper from draggable
6580
                                sortable.options.helper = sortable.options._helper;
6581
                        } else {
6582
                                // Prevent this Sortable from removing the helper.
6583
                                // However, don't set the draggable to remove the helper
6584
                                // either as another connected Sortable may yet handle the removal.
6585
                                sortable.cancelHelperRemoval = true;
6586
6587
                                sortable._trigger( "deactivate", event, uiSortable );
6588
                        }
6589
                });
6590
        },
6591
        drag: function( event, ui, draggable ) {
6592
                $.each( draggable.sortables, function() {
6593
                        var innermostIntersecting = false,
6594
                                sortable = this;
6595
6596
                        // Copy over variables that sortable's _intersectsWith uses
6597
                        sortable.positionAbs = draggable.positionAbs;
6598
                        sortable.helperProportions = draggable.helperProportions;
6599
                        sortable.offset.click = draggable.offset.click;
6600
6601
                        if ( sortable._intersectsWith( sortable.containerCache ) ) {
6602
                                innermostIntersecting = true;
6603
6604
                                $.each( draggable.sortables, function() {
6605
                                        // Copy over variables that sortable's _intersectsWith uses
6606
                                        this.positionAbs = draggable.positionAbs;
6607
                                        this.helperProportions = draggable.helperProportions;
6608
                                        this.offset.click = draggable.offset.click;
6609
6610
                                        if ( this !== sortable &&
6611
                                                        this._intersectsWith( this.containerCache ) &&
6612
                                                        $.contains( sortable.element[ 0 ], this.element[ 0 ] ) ) {
6613
                                                innermostIntersecting = false;
6614
                                        }
6615
6616
                                        return innermostIntersecting;
6617
                                });
6618
                        }
6619
6620
                        if ( innermostIntersecting ) {
6621
                                // If it intersects, we use a little isOver variable and set it once,
6622
                                // so that the move-in stuff gets fired only once.
6623
                                if ( !sortable.isOver ) {
6624
                                        sortable.isOver = 1;
6625
6626
                                        // Store draggable's parent in case we need to reappend to it later.
6627
                                        draggable._parent = ui.helper.parent();
6628
6629
                                        sortable.currentItem = ui.helper
6630
                                                .appendTo( sortable.element )
6631
                                                .data( "ui-sortable-item", true );
6632
6633
                                        // Store helper option to later restore it
6634
                                        sortable.options._helper = sortable.options.helper;
6635
6636
                                        sortable.options.helper = function() {
6637
                                                return ui.helper[ 0 ];
6638
                                        };
6639
6640
                                        // Fire the start events of the sortable with our passed browser event,
6641
                                        // and our own helper (so it doesn't create a new one)
6642
                                        event.target = sortable.currentItem[ 0 ];
6643
                                        sortable._mouseCapture( event, true );
6644
                                        sortable._mouseStart( event, true, true );
6645
6646
                                        // Because the browser event is way off the new appended portlet,
6647
                                        // modify necessary variables to reflect the changes
6648
                                        sortable.offset.click.top = draggable.offset.click.top;
6649
                                        sortable.offset.click.left = draggable.offset.click.left;
6650
                                        sortable.offset.parent.left -= draggable.offset.parent.left -
6651
                                                sortable.offset.parent.left;
6652
                                        sortable.offset.parent.top -= draggable.offset.parent.top -
6653
                                                sortable.offset.parent.top;
6654
6655
                                        draggable._trigger( "toSortable", event );
6656
6657
                                        // Inform draggable that the helper is in a valid drop zone,
6658
                                        // used solely in the revert option to handle "valid/invalid".
6659
                                        draggable.dropped = sortable.element;
6660
6661
                                        // Need to refreshPositions of all sortables in the case that
6662
                                        // adding to one sortable changes the location of the other sortables (#9675)
6663
                                        $.each( draggable.sortables, function() {
6664
                                                this.refreshPositions();
6665
                                        });
6666
6667
                                        // hack so receive/update callbacks work (mostly)
6668
                                        draggable.currentItem = draggable.element;
6669
                                        sortable.fromOutside = draggable;
6670
                                }
6671
6672
                                if ( sortable.currentItem ) {
6673
                                        sortable._mouseDrag( event );
6674
                                        // Copy the sortable's position because the draggable's can potentially reflect
6675
                                        // a relative position, while sortable is always absolute, which the dragged
6676
                                        // element has now become. (#8809)
6677
                                        ui.position = sortable.position;
6678
                                }
6679
                        } else {
6680
                                // If it doesn't intersect with the sortable, and it intersected before,
6681
                                // we fake the drag stop of the sortable, but make sure it doesn't remove
6682
                                // the helper by using cancelHelperRemoval.
6683
                                if ( sortable.isOver ) {
6684
6685
                                        sortable.isOver = 0;
6686
                                        sortable.cancelHelperRemoval = true;
6687
6688
                                        // Calling sortable's mouseStop would trigger a revert,
6689
                                        // so revert must be temporarily false until after mouseStop is called.
6690
                                        sortable.options._revert = sortable.options.revert;
6691
                                        sortable.options.revert = false;
6692
6693
                                        sortable._trigger( "out", event, sortable._uiHash( sortable ) );
6694
                                        sortable._mouseStop( event, true );
6695
6696
                                        // restore sortable behaviors that were modfied
6697
                                        // when the draggable entered the sortable area (#9481)
6698
                                        sortable.options.revert = sortable.options._revert;
6699
                                        sortable.options.helper = sortable.options._helper;
6700
6701
                                        if ( sortable.placeholder ) {
6702
                                                sortable.placeholder.remove();
6703
                                        }
6704
6705
                                        // Restore and recalculate the draggable's offset considering the sortable
6706
                                        // may have modified them in unexpected ways. (#8809, #10669)
6707
                                        ui.helper.appendTo( draggable._parent );
6708
                                        draggable._refreshOffsets( event );
6709
                                        ui.position = draggable._generatePosition( event, true );
6710
6711
                                        draggable._trigger( "fromSortable", event );
6712
6713
                                        // Inform draggable that the helper is no longer in a valid drop zone
6714
                                        draggable.dropped = false;
6715
6716
                                        // Need to refreshPositions of all sortables just in case removing
6717
                                        // from one sortable changes the location of other sortables (#9675)
6718
                                        $.each( draggable.sortables, function() {
6719
                                                this.refreshPositions();
6720
                                        });
6721
                                }
6722
                        }
6723
                });
6724
        }
6725
});
6726
6727
$.ui.plugin.add("draggable", "cursor", {
6728
        start: function( event, ui, instance ) {
6729
                var t = $( "body" ),
6730
                        o = instance.options;
6731
6732
                if (t.css("cursor")) {
6733
                        o._cursor = t.css("cursor");
6734
                }
6735
                t.css("cursor", o.cursor);
6736
        },
6737
        stop: function( event, ui, instance ) {
6738
                var o = instance.options;
6739
                if (o._cursor) {
6740
                        $("body").css("cursor", o._cursor);
6741
                }
6742
        }
6743
});
6744
6745
$.ui.plugin.add("draggable", "opacity", {
6746
        start: function( event, ui, instance ) {
6747
                var t = $( ui.helper ),
6748
                        o = instance.options;
6749
                if (t.css("opacity")) {
6750
                        o._opacity = t.css("opacity");
6751
                }
6752
                t.css("opacity", o.opacity);
6753
        },
6754
        stop: function( event, ui, instance ) {
6755
                var o = instance.options;
6756
                if (o._opacity) {
6757
                        $(ui.helper).css("opacity", o._opacity);
6758
                }
6759
        }
6760
});
6761
6762
$.ui.plugin.add("draggable", "scroll", {
6763
        start: function( event, ui, i ) {
6764
                if ( !i.scrollParentNotHidden ) {
6765
                        i.scrollParentNotHidden = i.helper.scrollParent( false );
6766
                }
6767
6768
                if ( i.scrollParentNotHidden[ 0 ] !== i.document[ 0 ] && i.scrollParentNotHidden[ 0 ].tagName !== "HTML" ) {
6769
                        i.overflowOffset = i.scrollParentNotHidden.offset();
6770
                }
6771
        },
6772
        drag: function( event, ui, i  ) {
6773
6774
                var o = i.options,
6775
                        scrolled = false,
6776
                        scrollParent = i.scrollParentNotHidden[ 0 ],
6777
                        document = i.document[ 0 ];
6778
6779
                if ( scrollParent !== document && scrollParent.tagName !== "HTML" ) {
6780
                        if ( !o.axis || o.axis !== "x" ) {
6781
                                if ( ( i.overflowOffset.top + scrollParent.offsetHeight ) - event.pageY < o.scrollSensitivity ) {
6782
                                        scrollParent.scrollTop = scrolled = scrollParent.scrollTop + o.scrollSpeed;
6783
                                } else if ( event.pageY - i.overflowOffset.top < o.scrollSensitivity ) {
6784
                                        scrollParent.scrollTop = scrolled = scrollParent.scrollTop - o.scrollSpeed;
6785
                                }
6786
                        }
6787
6788
                        if ( !o.axis || o.axis !== "y" ) {
6789
                                if ( ( i.overflowOffset.left + scrollParent.offsetWidth ) - event.pageX < o.scrollSensitivity ) {
6790
                                        scrollParent.scrollLeft = scrolled = scrollParent.scrollLeft + o.scrollSpeed;
6791
                                } else if ( event.pageX - i.overflowOffset.left < o.scrollSensitivity ) {
6792
                                        scrollParent.scrollLeft = scrolled = scrollParent.scrollLeft - o.scrollSpeed;
6793
                                }
6794
                        }
6795
6796
                } else {
6797
6798
                        if (!o.axis || o.axis !== "x") {
6799
                                if (event.pageY - $(document).scrollTop() < o.scrollSensitivity) {
6800
                                        scrolled = $(document).scrollTop($(document).scrollTop() - o.scrollSpeed);
6801
                                } else if ($(window).height() - (event.pageY - $(document).scrollTop()) < o.scrollSensitivity) {
6802
                                        scrolled = $(document).scrollTop($(document).scrollTop() + o.scrollSpeed);
6803
                                }
6804
                        }
6805
6806
                        if (!o.axis || o.axis !== "y") {
6807
                                if (event.pageX - $(document).scrollLeft() < o.scrollSensitivity) {
6808
                                        scrolled = $(document).scrollLeft($(document).scrollLeft() - o.scrollSpeed);
6809
                                } else if ($(window).width() - (event.pageX - $(document).scrollLeft()) < o.scrollSensitivity) {
6810
                                        scrolled = $(document).scrollLeft($(document).scrollLeft() + o.scrollSpeed);
6811
                                }
6812
                        }
6813
6814
                }
6815
6816
                if (scrolled !== false && $.ui.ddmanager && !o.dropBehaviour) {
6817
                        $.ui.ddmanager.prepareOffsets(i, event);
6818
                }
6819
6820
        }
6821
});
6822
6823
$.ui.plugin.add("draggable", "snap", {
6824
        start: function( event, ui, i ) {
6825
6826
                var o = i.options;
6827
6828
                i.snapElements = [];
6829
6830
                $(o.snap.constructor !== String ? ( o.snap.items || ":data(ui-draggable)" ) : o.snap).each(function() {
6831
                        var $t = $(this),
6832
                                $o = $t.offset();
6833
                        if (this !== i.element[0]) {
6834
                                i.snapElements.push({
6835
                                        item: this,
6836
                                        width: $t.outerWidth(), height: $t.outerHeight(),
6837
                                        top: $o.top, left: $o.left
6838
                                });
6839
                        }
6840
                });
6841
6842
        },
6843
        drag: function( event, ui, inst ) {
6844
6845
                var ts, bs, ls, rs, l, r, t, b, i, first,
6846
                        o = inst.options,
6847
                        d = o.snapTolerance,
6848
                        x1 = ui.offset.left, x2 = x1 + inst.helperProportions.width,
6849
                        y1 = ui.offset.top, y2 = y1 + inst.helperProportions.height;
6850
6851
                for (i = inst.snapElements.length - 1; i >= 0; i--){
6852
6853
                        l = inst.snapElements[i].left - inst.margins.left;
6854
                        r = l + inst.snapElements[i].width;
6855
                        t = inst.snapElements[i].top - inst.margins.top;
6856
                        b = t + inst.snapElements[i].height;
6857
6858
                        if ( x2 < l - d || x1 > r + d || y2 < t - d || y1 > b + d || !$.contains( inst.snapElements[ i ].item.ownerDocument, inst.snapElements[ i ].item ) ) {
6859
                                if (inst.snapElements[i].snapping) {
6860
                                        (inst.options.snap.release && inst.options.snap.release.call(inst.element, event, $.extend(inst._uiHash(), { snapItem: inst.snapElements[i].item })));
6861
                                }
6862
                                inst.snapElements[i].snapping = false;
6863
                                continue;
6864
                        }
6865
6866
                        if (o.snapMode !== "inner") {
6867
                                ts = Math.abs(t - y2) <= d;
6868
                                bs = Math.abs(b - y1) <= d;
6869
                                ls = Math.abs(l - x2) <= d;
6870
                                rs = Math.abs(r - x1) <= d;
6871
                                if (ts) {
6872
                                        ui.position.top = inst._convertPositionTo("relative", { top: t - inst.helperProportions.height, left: 0 }).top;
6873
                                }
6874
                                if (bs) {
6875
                                        ui.position.top = inst._convertPositionTo("relative", { top: b, left: 0 }).top;
6876
                                }
6877
                                if (ls) {
6878
                                        ui.position.left = inst._convertPositionTo("relative", { top: 0, left: l - inst.helperProportions.width }).left;
6879
                                }
6880
                                if (rs) {
6881
                                        ui.position.left = inst._convertPositionTo("relative", { top: 0, left: r }).left;
6882
                                }
6883
                        }
6884
6885
                        first = (ts || bs || ls || rs);
6886
6887
                        if (o.snapMode !== "outer") {
6888
                                ts = Math.abs(t - y1) <= d;
6889
                                bs = Math.abs(b - y2) <= d;
6890
                                ls = Math.abs(l - x1) <= d;
6891
                                rs = Math.abs(r - x2) <= d;
6892
                                if (ts) {
6893
                                        ui.position.top = inst._convertPositionTo("relative", { top: t, left: 0 }).top;
6894
                                }
6895
                                if (bs) {
6896
                                        ui.position.top = inst._convertPositionTo("relative", { top: b - inst.helperProportions.height, left: 0 }).top;
6897
                                }
6898
                                if (ls) {
6899
                                        ui.position.left = inst._convertPositionTo("relative", { top: 0, left: l }).left;
6900
                                }
6901
                                if (rs) {
6902
                                        ui.position.left = inst._convertPositionTo("relative", { top: 0, left: r - inst.helperProportions.width }).left;
6903
                                }
6904
                        }
6905
6906
                        if (!inst.snapElements[i].snapping && (ts || bs || ls || rs || first)) {
6907
                                (inst.options.snap.snap && inst.options.snap.snap.call(inst.element, event, $.extend(inst._uiHash(), { snapItem: inst.snapElements[i].item })));
6908
                        }
6909
                        inst.snapElements[i].snapping = (ts || bs || ls || rs || first);
6910
6911
                }
6912
6913
        }
6914
});
6915
6916
$.ui.plugin.add("draggable", "stack", {
6917
        start: function( event, ui, instance ) {
6918
                var min,
6919
                        o = instance.options,
6920
                        group = $.makeArray($(o.stack)).sort(function(a, b) {
6921
                                return (parseInt($(a).css("zIndex"), 10) || 0) - (parseInt($(b).css("zIndex"), 10) || 0);
6922
                        });
6923
6924
                if (!group.length) { return; }
6925
6926
                min = parseInt($(group[0]).css("zIndex"), 10) || 0;
6927
                $(group).each(function(i) {
6928
                        $(this).css("zIndex", min + i);
6929
                });
6930
                this.css("zIndex", (min + group.length));
6931
        }
6932
});
6933
6934
$.ui.plugin.add("draggable", "zIndex", {
6935
        start: function( event, ui, instance ) {
6936
                var t = $( ui.helper ),
6937
                        o = instance.options;
6938
6939
                if (t.css("zIndex")) {
6940
                        o._zIndex = t.css("zIndex");
6941
                }
6942
                t.css("zIndex", o.zIndex);
6943
        },
6944
        stop: function( event, ui, instance ) {
6945
                var o = instance.options;
6946
6947
                if (o._zIndex) {
6948
                        $(ui.helper).css("zIndex", o._zIndex);
6949
                }
6950
        }
6951
});
6952
6953
var draggable = $.ui.draggable;
6954
6955
6956
/*!
6957
 * jQuery UI Resizable 1.11.4
6958
 * http://jqueryui.com
6959
 *
6960
 * Copyright jQuery Foundation and other contributors
6961
 * Released under the MIT license.
6962
 * http://jquery.org/license
6963
 *
6964
 * http://api.jqueryui.com/resizable/
6965
 */
6966
6967
6968
$.widget("ui.resizable", $.ui.mouse, {
6969
        version: "1.11.4",
6970
        widgetEventPrefix: "resize",
6971
        options: {
6972
                alsoResize: false,
6973
                animate: false,
6974
                animateDuration: "slow",
6975
                animateEasing: "swing",
6976
                aspectRatio: false,
6977
                autoHide: false,
6978
                containment: false,
6979
                ghost: false,
6980
                grid: false,
6981
                handles: "e,s,se",
6982
                helper: false,
6983
                maxHeight: null,
6984
                maxWidth: null,
6985
                minHeight: 10,
6986
                minWidth: 10,
6987
                // See #7960
6988
                zIndex: 90,
6989
6990
                // callbacks
6991
                resize: null,
6992
                start: null,
6993
                stop: null
6994
        },
6995
6996
        _num: function( value ) {
6997
                return parseInt( value, 10 ) || 0;
6998
        },
6999
7000
        _isNumber: function( value ) {
7001
                return !isNaN( parseInt( value, 10 ) );
7002
        },
7003
7004
        _hasScroll: function( el, a ) {
7005
7006
                if ( $( el ).css( "overflow" ) === "hidden") {
7007
                        return false;
7008
                }
7009
7010
                var scroll = ( a && a === "left" ) ? "scrollLeft" : "scrollTop",
7011
                        has = false;
7012
7013
                if ( el[ scroll ] > 0 ) {
7014
                        return true;
7015
                }
7016
7017
                // TODO: determine which cases actually cause this to happen
7018
                // if the element doesn't have the scroll set, see if it's possible to
7019
                // set the scroll
7020
                el[ scroll ] = 1;
7021
                has = ( el[ scroll ] > 0 );
7022
                el[ scroll ] = 0;
7023
                return has;
7024
        },
7025
7026
        _create: function() {
7027
7028
                var n, i, handle, axis, hname,
7029
                        that = this,
7030
                        o = this.options;
7031
                this.element.addClass("ui-resizable");
7032
7033
                $.extend(this, {
7034
                        _aspectRatio: !!(o.aspectRatio),
7035
                        aspectRatio: o.aspectRatio,
7036
                        originalElement: this.element,
7037
                        _proportionallyResizeElements: [],
7038
                        _helper: o.helper || o.ghost || o.animate ? o.helper || "ui-resizable-helper" : null
7039
                });
7040
7041
                // Wrap the element if it cannot hold child nodes
7042
                if (this.element[0].nodeName.match(/^(canvas|textarea|input|select|button|img)$/i)) {
7043
7044
                        this.element.wrap(
7045
                                $("<div class='ui-wrapper' style='overflow: hidden;'></div>").css({
7046
                                        position: this.element.css("position"),
7047
                                        width: this.element.outerWidth(),
7048
                                        height: this.element.outerHeight(),
7049
                                        top: this.element.css("top"),
7050
                                        left: this.element.css("left")
7051
                                })
7052
                        );
7053
7054
                        this.element = this.element.parent().data(
7055
                                "ui-resizable", this.element.resizable( "instance" )
7056
                        );
7057
7058
                        this.elementIsWrapper = true;
7059
7060
                        this.element.css({
7061
                                marginLeft: this.originalElement.css("marginLeft"),
7062
                                marginTop: this.originalElement.css("marginTop"),
7063
                                marginRight: this.originalElement.css("marginRight"),
7064
                                marginBottom: this.originalElement.css("marginBottom")
7065
                        });
7066
                        this.originalElement.css({
7067
                                marginLeft: 0,
7068
                                marginTop: 0,
7069
                                marginRight: 0,
7070
                                marginBottom: 0
7071
                        });
7072
                        // support: Safari
7073
                        // Prevent Safari textarea resize
7074
                        this.originalResizeStyle = this.originalElement.css("resize");
7075
                        this.originalElement.css("resize", "none");
7076
7077
                        this._proportionallyResizeElements.push( this.originalElement.css({
7078
                                position: "static",
7079
                                zoom: 1,
7080
                                display: "block"
7081
                        }) );
7082
7083
                        // support: IE9
7084
                        // avoid IE jump (hard set the margin)
7085
                        this.originalElement.css({ margin: this.originalElement.css("margin") });
7086
7087
                        this._proportionallyResize();
7088
                }
7089
7090
                this.handles = o.handles ||
7091
                        ( !$(".ui-resizable-handle", this.element).length ?
7092
                                "e,s,se" : {
7093
                                        n: ".ui-resizable-n",
7094
                                        e: ".ui-resizable-e",
7095
                                        s: ".ui-resizable-s",
7096
                                        w: ".ui-resizable-w",
7097
                                        se: ".ui-resizable-se",
7098
                                        sw: ".ui-resizable-sw",
7099
                                        ne: ".ui-resizable-ne",
7100
                                        nw: ".ui-resizable-nw"
7101
                                } );
7102
7103
                this._handles = $();
7104
                if ( this.handles.constructor === String ) {
7105
7106
                        if ( this.handles === "all") {
7107
                                this.handles = "n,e,s,w,se,sw,ne,nw";
7108
                        }
7109
7110
                        n = this.handles.split(",");
7111
                        this.handles = {};
7112
7113
                        for (i = 0; i < n.length; i++) {
7114
7115
                                handle = $.trim(n[i]);
7116
                                hname = "ui-resizable-" + handle;
7117
                                axis = $("<div class='ui-resizable-handle " + hname + "'></div>");
7118
7119
                                axis.css({ zIndex: o.zIndex });
7120
7121
                                // TODO : What's going on here?
7122
                                if ("se" === handle) {
7123
                                        axis.addClass("ui-icon ui-icon-gripsmall-diagonal-se");
7124
                                }
7125
7126
                                this.handles[handle] = ".ui-resizable-" + handle;
7127
                                this.element.append(axis);
7128
                        }
7129
7130
                }
7131
7132
                this._renderAxis = function(target) {
7133
7134
                        var i, axis, padPos, padWrapper;
7135
7136
                        target = target || this.element;
7137
7138
                        for (i in this.handles) {
7139
7140
                                if (this.handles[i].constructor === String) {
7141
                                        this.handles[i] = this.element.children( this.handles[ i ] ).first().show();
7142
                                } else if ( this.handles[ i ].jquery || this.handles[ i ].nodeType ) {
7143
                                        this.handles[ i ] = $( this.handles[ i ] );
7144
                                        this._on( this.handles[ i ], { "mousedown": that._mouseDown });
7145
                                }
7146
7147
                                if (this.elementIsWrapper && this.originalElement[0].nodeName.match(/^(textarea|input|select|button)$/i)) {
7148
7149
                                        axis = $(this.handles[i], this.element);
7150
7151
                                        padWrapper = /sw|ne|nw|se|n|s/.test(i) ? axis.outerHeight() : axis.outerWidth();
7152
7153
                                        padPos = [ "padding",
7154
                                                /ne|nw|n/.test(i) ? "Top" :
7155
                                                /se|sw|s/.test(i) ? "Bottom" :
7156
                                                /^e$/.test(i) ? "Right" : "Left" ].join("");
7157
7158
                                        target.css(padPos, padWrapper);
7159
7160
                                        this._proportionallyResize();
7161
                                }
7162
7163
                                this._handles = this._handles.add( this.handles[ i ] );
7164
                        }
7165
                };
7166
7167
                // TODO: make renderAxis a prototype function
7168
                this._renderAxis(this.element);
7169
7170
                this._handles = this._handles.add( this.element.find( ".ui-resizable-handle" ) );
7171
                this._handles.disableSelection();
7172
7173
                this._handles.mouseover(function() {
7174
                        if (!that.resizing) {
7175
                                if (this.className) {
7176
                                        axis = this.className.match(/ui-resizable-(se|sw|ne|nw|n|e|s|w)/i);
7177
                                }
7178
                                that.axis = axis && axis[1] ? axis[1] : "se";
7179
                        }
7180
                });
7181
7182
                if (o.autoHide) {
7183
                        this._handles.hide();
7184
                        $(this.element)
7185
                                .addClass("ui-resizable-autohide")
7186
                                .mouseenter(function() {
7187
                                        if (o.disabled) {
7188
                                                return;
7189
                                        }
7190
                                        $(this).removeClass("ui-resizable-autohide");
7191
                                        that._handles.show();
7192
                                })
7193
                                .mouseleave(function() {
7194
                                        if (o.disabled) {
7195
                                                return;
7196
                                        }
7197
                                        if (!that.resizing) {
7198
                                                $(this).addClass("ui-resizable-autohide");
7199
                                                that._handles.hide();
7200
                                        }
7201
                                });
7202
                }
7203
7204
                this._mouseInit();
7205
        },
7206
7207
        _destroy: function() {
7208
7209
                this._mouseDestroy();
7210
7211
                var wrapper,
7212
                        _destroy = function(exp) {
7213
                                $(exp)
7214
                                        .removeClass("ui-resizable ui-resizable-disabled ui-resizable-resizing")
7215
                                        .removeData("resizable")
7216
                                        .removeData("ui-resizable")
7217
                                        .unbind(".resizable")
7218
                                        .find(".ui-resizable-handle")
7219
                                                .remove();
7220
                        };
7221
7222
                // TODO: Unwrap at same DOM position
7223
                if (this.elementIsWrapper) {
7224
                        _destroy(this.element);
7225
                        wrapper = this.element;
7226
                        this.originalElement.css({
7227
                                position: wrapper.css("position"),
7228
                                width: wrapper.outerWidth(),
7229
                                height: wrapper.outerHeight(),
7230
                                top: wrapper.css("top"),
7231
                                left: wrapper.css("left")
7232
                        }).insertAfter( wrapper );
7233
                        wrapper.remove();
7234
                }
7235
7236
                this.originalElement.css("resize", this.originalResizeStyle);
7237
                _destroy(this.originalElement);
7238
7239
                return this;
7240
        },
7241
7242
        _mouseCapture: function(event) {
7243
                var i, handle,
7244
                        capture = false;
7245
7246
                for (i in this.handles) {
7247
                        handle = $(this.handles[i])[0];
7248
                        if (handle === event.target || $.contains(handle, event.target)) {
7249
                                capture = true;
7250
                        }
7251
                }
7252
7253
                return !this.options.disabled && capture;
7254
        },
7255
7256
        _mouseStart: function(event) {
7257
7258
                var curleft, curtop, cursor,
7259
                        o = this.options,
7260
                        el = this.element;
7261
7262
                this.resizing = true;
7263
7264
                this._renderProxy();
7265
7266
                curleft = this._num(this.helper.css("left"));
7267
                curtop = this._num(this.helper.css("top"));
7268
7269
                if (o.containment) {
7270
                        curleft += $(o.containment).scrollLeft() || 0;
7271
                        curtop += $(o.containment).scrollTop() || 0;
7272
                }
7273
7274
                this.offset = this.helper.offset();
7275
                this.position = { left: curleft, top: curtop };
7276
7277
                this.size = this._helper ? {
7278
                                width: this.helper.width(),
7279
                                height: this.helper.height()
7280
                        } : {
7281
                                width: el.width(),
7282
                                height: el.height()
7283
                        };
7284
7285
                this.originalSize = this._helper ? {
7286
                                width: el.outerWidth(),
7287
                                height: el.outerHeight()
7288
                        } : {
7289
                                width: el.width(),
7290
                                height: el.height()
7291
                        };
7292
7293
                this.sizeDiff = {
7294
                        width: el.outerWidth() - el.width(),
7295
                        height: el.outerHeight() - el.height()
7296
                };
7297
7298
                this.originalPosition = { left: curleft, top: curtop };
7299
                this.originalMousePosition = { left: event.pageX, top: event.pageY };
7300
7301
                this.aspectRatio = (typeof o.aspectRatio === "number") ?
7302
                        o.aspectRatio :
7303
                        ((this.originalSize.width / this.originalSize.height) || 1);
7304
7305
                cursor = $(".ui-resizable-" + this.axis).css("cursor");
7306
                $("body").css("cursor", cursor === "auto" ? this.axis + "-resize" : cursor);
7307
7308
                el.addClass("ui-resizable-resizing");
7309
                this._propagate("start", event);
7310
                return true;
7311
        },
7312
7313
        _mouseDrag: function(event) {
7314
7315
                var data, props,
7316
                        smp = this.originalMousePosition,
7317
                        a = this.axis,
7318
                        dx = (event.pageX - smp.left) || 0,
7319
                        dy = (event.pageY - smp.top) || 0,
7320
                        trigger = this._change[a];
7321
7322
                this._updatePrevProperties();
7323
7324
                if (!trigger) {
7325
                        return false;
7326
                }
7327
7328
                data = trigger.apply(this, [ event, dx, dy ]);
7329
7330
                this._updateVirtualBoundaries(event.shiftKey);
7331
                if (this._aspectRatio || event.shiftKey) {
7332
                        data = this._updateRatio(data, event);
7333
                }
7334
7335
                data = this._respectSize(data, event);
7336
7337
                this._updateCache(data);
7338
7339
                this._propagate("resize", event);
7340
7341
                props = this._applyChanges();
7342
7343
                if ( !this._helper && this._proportionallyResizeElements.length ) {
7344
                        this._proportionallyResize();
7345
                }
7346
7347
                if ( !$.isEmptyObject( props ) ) {
7348
                        this._updatePrevProperties();
7349
                        this._trigger( "resize", event, this.ui() );
7350
                        this._applyChanges();
7351
                }
7352
7353
                return false;
7354
        },
7355
7356
        _mouseStop: function(event) {
7357
7358
                this.resizing = false;
7359
                var pr, ista, soffseth, soffsetw, s, left, top,
7360
                        o = this.options, that = this;
7361
7362
                if (this._helper) {
7363
7364
                        pr = this._proportionallyResizeElements;
7365
                        ista = pr.length && (/textarea/i).test(pr[0].nodeName);
7366
                        soffseth = ista && this._hasScroll(pr[0], "left") ? 0 : that.sizeDiff.height;
7367
                        soffsetw = ista ? 0 : that.sizeDiff.width;
7368
7369
                        s = {
7370
                                width: (that.helper.width()  - soffsetw),
7371
                                height: (that.helper.height() - soffseth)
7372
                        };
7373
                        left = (parseInt(that.element.css("left"), 10) +
7374
                                (that.position.left - that.originalPosition.left)) || null;
7375
                        top = (parseInt(that.element.css("top"), 10) +
7376
                                (that.position.top - that.originalPosition.top)) || null;
7377
7378
                        if (!o.animate) {
7379
                                this.element.css($.extend(s, { top: top, left: left }));
7380
                        }
7381
7382
                        that.helper.height(that.size.height);
7383
                        that.helper.width(that.size.width);
7384
7385
                        if (this._helper && !o.animate) {
7386
                                this._proportionallyResize();
7387
                        }
7388
                }
7389
7390
                $("body").css("cursor", "auto");
7391
7392
                this.element.removeClass("ui-resizable-resizing");
7393
7394
                this._propagate("stop", event);
7395
7396
                if (this._helper) {
7397
                        this.helper.remove();
7398
                }
7399
7400
                return false;
7401
7402
        },
7403
7404
        _updatePrevProperties: function() {
7405
                this.prevPosition = {
7406
                        top: this.position.top,
7407
                        left: this.position.left
7408
                };
7409
                this.prevSize = {
7410
                        width: this.size.width,
7411
                        height: this.size.height
7412
                };
7413
        },
7414
7415
        _applyChanges: function() {
7416
                var props = {};
7417
7418
                if ( this.position.top !== this.prevPosition.top ) {
7419
                        props.top = this.position.top + "px";
7420
                }
7421
                if ( this.position.left !== this.prevPosition.left ) {
7422
                        props.left = this.position.left + "px";
7423
                }
7424
                if ( this.size.width !== this.prevSize.width ) {
7425
                        props.width = this.size.width + "px";
7426
                }
7427
                if ( this.size.height !== this.prevSize.height ) {
7428
                        props.height = this.size.height + "px";
7429
                }
7430
7431
                this.helper.css( props );
7432
7433
                return props;
7434
        },
7435
7436
        _updateVirtualBoundaries: function(forceAspectRatio) {
7437
                var pMinWidth, pMaxWidth, pMinHeight, pMaxHeight, b,
7438
                        o = this.options;
7439
7440
                b = {
7441
                        minWidth: this._isNumber(o.minWidth) ? o.minWidth : 0,
7442
                        maxWidth: this._isNumber(o.maxWidth) ? o.maxWidth : Infinity,
7443
                        minHeight: this._isNumber(o.minHeight) ? o.minHeight : 0,
7444
                        maxHeight: this._isNumber(o.maxHeight) ? o.maxHeight : Infinity
7445
                };
7446
7447
                if (this._aspectRatio || forceAspectRatio) {
7448
                        pMinWidth = b.minHeight * this.aspectRatio;
7449
                        pMinHeight = b.minWidth / this.aspectRatio;
7450
                        pMaxWidth = b.maxHeight * this.aspectRatio;
7451
                        pMaxHeight = b.maxWidth / this.aspectRatio;
7452
7453
                        if (pMinWidth > b.minWidth) {
7454
                                b.minWidth = pMinWidth;
7455
                        }
7456
                        if (pMinHeight > b.minHeight) {
7457
                                b.minHeight = pMinHeight;
7458
                        }
7459
                        if (pMaxWidth < b.maxWidth) {
7460
                                b.maxWidth = pMaxWidth;
7461
                        }
7462
                        if (pMaxHeight < b.maxHeight) {
7463
                                b.maxHeight = pMaxHeight;
7464
                        }
7465
                }
7466
                this._vBoundaries = b;
7467
        },
7468
7469
        _updateCache: function(data) {
7470
                this.offset = this.helper.offset();
7471
                if (this._isNumber(data.left)) {
7472
                        this.position.left = data.left;
7473
                }
7474
                if (this._isNumber(data.top)) {
7475
                        this.position.top = data.top;
7476
                }
7477
                if (this._isNumber(data.height)) {
7478
                        this.size.height = data.height;
7479
                }
7480
                if (this._isNumber(data.width)) {
7481
                        this.size.width = data.width;
7482
                }
7483
        },
7484
7485
        _updateRatio: function( data ) {
7486
7487
                var cpos = this.position,
7488
                        csize = this.size,
7489
                        a = this.axis;
7490
7491
                if (this._isNumber(data.height)) {
7492
                        data.width = (data.height * this.aspectRatio);
7493
                } else if (this._isNumber(data.width)) {
7494
                        data.height = (data.width / this.aspectRatio);
7495
                }
7496
7497
                if (a === "sw") {
7498
                        data.left = cpos.left + (csize.width - data.width);
7499
                        data.top = null;
7500
                }
7501
                if (a === "nw") {
7502
                        data.top = cpos.top + (csize.height - data.height);
7503
                        data.left = cpos.left + (csize.width - data.width);
7504
                }
7505
7506
                return data;
7507
        },
7508
7509
        _respectSize: function( data ) {
7510
7511
                var o = this._vBoundaries,
7512
                        a = this.axis,
7513
                        ismaxw = this._isNumber(data.width) && o.maxWidth && (o.maxWidth < data.width),
7514
                        ismaxh = this._isNumber(data.height) && o.maxHeight && (o.maxHeight < data.height),
7515
                        isminw = this._isNumber(data.width) && o.minWidth && (o.minWidth > data.width),
7516
                        isminh = this._isNumber(data.height) && o.minHeight && (o.minHeight > data.height),
7517
                        dw = this.originalPosition.left + this.originalSize.width,
7518
                        dh = this.position.top + this.size.height,
7519
                        cw = /sw|nw|w/.test(a), ch = /nw|ne|n/.test(a);
7520
                if (isminw) {
7521
                        data.width = o.minWidth;
7522
                }
7523
                if (isminh) {
7524
                        data.height = o.minHeight;
7525
                }
7526
                if (ismaxw) {
7527
                        data.width = o.maxWidth;
7528
                }
7529
                if (ismaxh) {
7530
                        data.height = o.maxHeight;
7531
                }
7532
7533
                if (isminw && cw) {
7534
                        data.left = dw - o.minWidth;
7535
                }
7536
                if (ismaxw && cw) {
7537
                        data.left = dw - o.maxWidth;
7538
                }
7539
                if (isminh && ch) {
7540
                        data.top = dh - o.minHeight;
7541
                }
7542
                if (ismaxh && ch) {
7543
                        data.top = dh - o.maxHeight;
7544
                }
7545
7546
                // Fixing jump error on top/left - bug #2330
7547
                if (!data.width && !data.height && !data.left && data.top) {
7548
                        data.top = null;
7549
                } else if (!data.width && !data.height && !data.top && data.left) {
7550
                        data.left = null;
7551
                }
7552
7553
                return data;
7554
        },
7555
7556
        _getPaddingPlusBorderDimensions: function( element ) {
7557
                var i = 0,
7558
                        widths = [],
7559
                        borders = [
7560
                                element.css( "borderTopWidth" ),
7561
                                element.css( "borderRightWidth" ),
7562
                                element.css( "borderBottomWidth" ),
7563
                                element.css( "borderLeftWidth" )
7564
                        ],
7565
                        paddings = [
7566
                                element.css( "paddingTop" ),
7567
                                element.css( "paddingRight" ),
7568
                                element.css( "paddingBottom" ),
7569
                                element.css( "paddingLeft" )
7570
                        ];
7571
7572
                for ( ; i < 4; i++ ) {
7573
                        widths[ i ] = ( parseInt( borders[ i ], 10 ) || 0 );
7574
                        widths[ i ] += ( parseInt( paddings[ i ], 10 ) || 0 );
7575
                }
7576
7577
                return {
7578
                        height: widths[ 0 ] + widths[ 2 ],
7579
                        width: widths[ 1 ] + widths[ 3 ]
7580
                };
7581
        },
7582
7583
        _proportionallyResize: function() {
7584
7585
                if (!this._proportionallyResizeElements.length) {
7586
                        return;
7587
                }
7588
7589
                var prel,
7590
                        i = 0,
7591
                        element = this.helper || this.element;
7592
7593
                for ( ; i < this._proportionallyResizeElements.length; i++) {
7594
7595
                        prel = this._proportionallyResizeElements[i];
7596
7597
                        // TODO: Seems like a bug to cache this.outerDimensions
7598
                        // considering that we are in a loop.
7599
                        if (!this.outerDimensions) {
7600
                                this.outerDimensions = this._getPaddingPlusBorderDimensions( prel );
7601
                        }
7602
7603
                        prel.css({
7604
                                height: (element.height() - this.outerDimensions.height) || 0,
7605
                                width: (element.width() - this.outerDimensions.width) || 0
7606
                        });
7607
7608
                }
7609
7610
        },
7611
7612
        _renderProxy: function() {
7613
7614
                var el = this.element, o = this.options;
7615
                this.elementOffset = el.offset();
7616
7617
                if (this._helper) {
7618
7619
                        this.helper = this.helper || $("<div style='overflow:hidden;'></div>");
7620
7621
                        this.helper.addClass(this._helper).css({
7622
                                width: this.element.outerWidth() - 1,
7623
                                height: this.element.outerHeight() - 1,
7624
                                position: "absolute",
7625
                                left: this.elementOffset.left + "px",
7626
                                top: this.elementOffset.top + "px",
7627
                                zIndex: ++o.zIndex //TODO: Don't modify option
7628
                        });
7629
7630
                        this.helper
7631
                                .appendTo("body")
7632
                                .disableSelection();
7633
7634
                } else {
7635
                        this.helper = this.element;
7636
                }
7637
7638
        },
7639
7640
        _change: {
7641
                e: function(event, dx) {
7642
                        return { width: this.originalSize.width + dx };
7643
                },
7644
                w: function(event, dx) {
7645
                        var cs = this.originalSize, sp = this.originalPosition;
7646
                        return { left: sp.left + dx, width: cs.width - dx };
7647
                },
7648
                n: function(event, dx, dy) {
7649
                        var cs = this.originalSize, sp = this.originalPosition;
7650
                        return { top: sp.top + dy, height: cs.height - dy };
7651
                },
7652
                s: function(event, dx, dy) {
7653
                        return { height: this.originalSize.height + dy };
7654
                },
7655
                se: function(event, dx, dy) {
7656
                        return $.extend(this._change.s.apply(this, arguments),
7657
                                this._change.e.apply(this, [ event, dx, dy ]));
7658
                },
7659
                sw: function(event, dx, dy) {
7660
                        return $.extend(this._change.s.apply(this, arguments),
7661
                                this._change.w.apply(this, [ event, dx, dy ]));
7662
                },
7663
                ne: function(event, dx, dy) {
7664
                        return $.extend(this._change.n.apply(this, arguments),
7665
                                this._change.e.apply(this, [ event, dx, dy ]));
7666
                },
7667
                nw: function(event, dx, dy) {
7668
                        return $.extend(this._change.n.apply(this, arguments),
7669
                                this._change.w.apply(this, [ event, dx, dy ]));
7670
                }
7671
        },
7672
7673
        _propagate: function(n, event) {
7674
                $.ui.plugin.call(this, n, [ event, this.ui() ]);
7675
                (n !== "resize" && this._trigger(n, event, this.ui()));
7676
        },
7677
7678
        plugins: {},
7679
7680
        ui: function() {
7681
                return {
7682
                        originalElement: this.originalElement,
7683
                        element: this.element,
7684
                        helper: this.helper,
7685
                        position: this.position,
7686
                        size: this.size,
7687
                        originalSize: this.originalSize,
7688
                        originalPosition: this.originalPosition
7689
                };
7690
        }
7691
7692
});
7693
7694
/*
7695
 * Resizable Extensions
7696
 */
7697
7698
$.ui.plugin.add("resizable", "animate", {
7699
7700
        stop: function( event ) {
7701
                var that = $(this).resizable( "instance" ),
7702
                        o = that.options,
7703
                        pr = that._proportionallyResizeElements,
7704
                        ista = pr.length && (/textarea/i).test(pr[0].nodeName),
7705
                        soffseth = ista && that._hasScroll(pr[0], "left") ? 0 : that.sizeDiff.height,
7706
                        soffsetw = ista ? 0 : that.sizeDiff.width,
7707
                        style = { width: (that.size.width - soffsetw), height: (that.size.height - soffseth) },
7708
                        left = (parseInt(that.element.css("left"), 10) +
7709
                                (that.position.left - that.originalPosition.left)) || null,
7710
                        top = (parseInt(that.element.css("top"), 10) +
7711
                                (that.position.top - that.originalPosition.top)) || null;
7712
7713
                that.element.animate(
7714
                        $.extend(style, top && left ? { top: top, left: left } : {}), {
7715
                                duration: o.animateDuration,
7716
                                easing: o.animateEasing,
7717
                                step: function() {
7718
7719
                                        var data = {
7720
                                                width: parseInt(that.element.css("width"), 10),
7721
                                                height: parseInt(that.element.css("height"), 10),
7722
                                                top: parseInt(that.element.css("top"), 10),
7723
                                                left: parseInt(that.element.css("left"), 10)
7724
                                        };
7725
7726
                                        if (pr && pr.length) {
7727
                                                $(pr[0]).css({ width: data.width, height: data.height });
7728
                                        }
7729
7730
                                        // propagating resize, and updating values for each animation step
7731
                                        that._updateCache(data);
7732
                                        that._propagate("resize", event);
7733
7734
                                }
7735
                        }
7736
                );
7737
        }
7738
7739
});
7740
7741
$.ui.plugin.add( "resizable", "containment", {
7742
7743
        start: function() {
7744
                var element, p, co, ch, cw, width, height,
7745
                        that = $( this ).resizable( "instance" ),
7746
                        o = that.options,
7747
                        el = that.element,
7748
                        oc = o.containment,
7749
                        ce = ( oc instanceof $ ) ? oc.get( 0 ) : ( /parent/.test( oc ) ) ? el.parent().get( 0 ) : oc;
7750
7751
                if ( !ce ) {
7752
                        return;
7753
                }
7754
7755
                that.containerElement = $( ce );
7756
7757
                if ( /document/.test( oc ) || oc === document ) {
7758
                        that.containerOffset = {
7759
                                left: 0,
7760
                                top: 0
7761
                        };
7762
                        that.containerPosition = {
7763
                                left: 0,
7764
                                top: 0
7765
                        };
7766
7767
                        that.parentData = {
7768
                                element: $( document ),
7769
                                left: 0,
7770
                                top: 0,
7771
                                width: $( document ).width(),
7772
                                height: $( document ).height() || document.body.parentNode.scrollHeight
7773
                        };
7774
                } else {
7775
                        element = $( ce );
7776
                        p = [];
7777
                        $([ "Top", "Right", "Left", "Bottom" ]).each(function( i, name ) {
7778
                                p[ i ] = that._num( element.css( "padding" + name ) );
7779
                        });
7780
7781
                        that.containerOffset = element.offset();
7782
                        that.containerPosition = element.position();
7783
                        that.containerSize = {
7784
                                height: ( element.innerHeight() - p[ 3 ] ),
7785
                                width: ( element.innerWidth() - p[ 1 ] )
7786
                        };
7787
7788
                        co = that.containerOffset;
7789
                        ch = that.containerSize.height;
7790
                        cw = that.containerSize.width;
7791
                        width = ( that._hasScroll ( ce, "left" ) ? ce.scrollWidth : cw );
7792
                        height = ( that._hasScroll ( ce ) ? ce.scrollHeight : ch ) ;
7793
7794
                        that.parentData = {
7795
                                element: ce,
7796
                                left: co.left,
7797
                                top: co.top,
7798
                                width: width,
7799
                                height: height
7800
                        };
7801
                }
7802
        },
7803
7804
        resize: function( event ) {
7805
                var woset, hoset, isParent, isOffsetRelative,
7806
                        that = $( this ).resizable( "instance" ),
7807
                        o = that.options,
7808
                        co = that.containerOffset,
7809
                        cp = that.position,
7810
                        pRatio = that._aspectRatio || event.shiftKey,
7811
                        cop = {
7812
                                top: 0,
7813
                                left: 0
7814
                        },
7815
                        ce = that.containerElement,
7816
                        continueResize = true;
7817
7818
                if ( ce[ 0 ] !== document && ( /static/ ).test( ce.css( "position" ) ) ) {
7819
                        cop = co;
7820
                }
7821
7822
                if ( cp.left < ( that._helper ? co.left : 0 ) ) {
7823
                        that.size.width = that.size.width +
7824
                                ( that._helper ?
7825
                                        ( that.position.left - co.left ) :
7826
                                        ( that.position.left - cop.left ) );
7827
7828
                        if ( pRatio ) {
7829
                                that.size.height = that.size.width / that.aspectRatio;
7830
                                continueResize = false;
7831
                        }
7832
                        that.position.left = o.helper ? co.left : 0;
7833
                }
7834
7835
                if ( cp.top < ( that._helper ? co.top : 0 ) ) {
7836
                        that.size.height = that.size.height +
7837
                                ( that._helper ?
7838
                                        ( that.position.top - co.top ) :
7839
                                        that.position.top );
7840
7841
                        if ( pRatio ) {
7842
                                that.size.width = that.size.height * that.aspectRatio;
7843
                                continueResize = false;
7844
                        }
7845
                        that.position.top = that._helper ? co.top : 0;
7846
                }
7847
7848
                isParent = that.containerElement.get( 0 ) === that.element.parent().get( 0 );
7849
                isOffsetRelative = /relative|absolute/.test( that.containerElement.css( "position" ) );
7850
7851
                if ( isParent && isOffsetRelative ) {
7852
                        that.offset.left = that.parentData.left + that.position.left;
7853
                        that.offset.top = that.parentData.top + that.position.top;
7854
                } else {
7855
                        that.offset.left = that.element.offset().left;
7856
                        that.offset.top = that.element.offset().top;
7857
                }
7858
7859
                woset = Math.abs( that.sizeDiff.width +
7860
                        (that._helper ?
7861
                                that.offset.left - cop.left :
7862
                                (that.offset.left - co.left)) );
7863
7864
                hoset = Math.abs( that.sizeDiff.height +
7865
                        (that._helper ?
7866
                                that.offset.top - cop.top :
7867
                                (that.offset.top - co.top)) );
7868
7869
                if ( woset + that.size.width >= that.parentData.width ) {
7870
                        that.size.width = that.parentData.width - woset;
7871
                        if ( pRatio ) {
7872
                                that.size.height = that.size.width / that.aspectRatio;
7873
                                continueResize = false;
7874
                        }
7875
                }
7876
7877
                if ( hoset + that.size.height >= that.parentData.height ) {
7878
                        that.size.height = that.parentData.height - hoset;
7879
                        if ( pRatio ) {
7880
                                that.size.width = that.size.height * that.aspectRatio;
7881
                                continueResize = false;
7882
                        }
7883
                }
7884
7885
                if ( !continueResize ) {
7886
                        that.position.left = that.prevPosition.left;
7887
                        that.position.top = that.prevPosition.top;
7888
                        that.size.width = that.prevSize.width;
7889
                        that.size.height = that.prevSize.height;
7890
                }
7891
        },
7892
7893
        stop: function() {
7894
                var that = $( this ).resizable( "instance" ),
7895
                        o = that.options,
7896
                        co = that.containerOffset,
7897
                        cop = that.containerPosition,
7898
                        ce = that.containerElement,
7899
                        helper = $( that.helper ),
7900
                        ho = helper.offset(),
7901
                        w = helper.outerWidth() - that.sizeDiff.width,
7902
                        h = helper.outerHeight() - that.sizeDiff.height;
7903
7904
                if ( that._helper && !o.animate && ( /relative/ ).test( ce.css( "position" ) ) ) {
7905
                        $( this ).css({
7906
                                left: ho.left - cop.left - co.left,
7907
                                width: w,
7908
                                height: h
7909
                        });
7910
                }
7911
7912
                if ( that._helper && !o.animate && ( /static/ ).test( ce.css( "position" ) ) ) {
7913
                        $( this ).css({
7914
                                left: ho.left - cop.left - co.left,
7915
                                width: w,
7916
                                height: h
7917
                        });
7918
                }
7919
        }
7920
});
7921
7922
$.ui.plugin.add("resizable", "alsoResize", {
7923
7924
        start: function() {
7925
                var that = $(this).resizable( "instance" ),
7926
                        o = that.options;
7927
7928
                $(o.alsoResize).each(function() {
7929
                        var el = $(this);
7930
                        el.data("ui-resizable-alsoresize", {
7931
                                width: parseInt(el.width(), 10), height: parseInt(el.height(), 10),
7932
                                left: parseInt(el.css("left"), 10), top: parseInt(el.css("top"), 10)
7933
                        });
7934
                });
7935
        },
7936
7937
        resize: function(event, ui) {
7938
                var that = $(this).resizable( "instance" ),
7939
                        o = that.options,
7940
                        os = that.originalSize,
7941
                        op = that.originalPosition,
7942
                        delta = {
7943
                                height: (that.size.height - os.height) || 0,
7944
                                width: (that.size.width - os.width) || 0,
7945
                                top: (that.position.top - op.top) || 0,
7946
                                left: (that.position.left - op.left) || 0
7947
                        };
7948
7949
                        $(o.alsoResize).each(function() {
7950
                                var el = $(this), start = $(this).data("ui-resizable-alsoresize"), style = {},
7951
                                        css = el.parents(ui.originalElement[0]).length ?
7952
                                                        [ "width", "height" ] :
7953
                                                        [ "width", "height", "top", "left" ];
7954
7955
                                $.each(css, function(i, prop) {
7956
                                        var sum = (start[prop] || 0) + (delta[prop] || 0);
7957
                                        if (sum && sum >= 0) {
7958
                                                style[prop] = sum || null;
7959
                                        }
7960
                                });
7961
7962
                                el.css(style);
7963
                        });
7964
        },
7965
7966
        stop: function() {
7967
                $(this).removeData("resizable-alsoresize");
7968
        }
7969
});
7970
7971
$.ui.plugin.add("resizable", "ghost", {
7972
7973
        start: function() {
7974
7975
                var that = $(this).resizable( "instance" ), o = that.options, cs = that.size;
7976
7977
                that.ghost = that.originalElement.clone();
7978
                that.ghost
7979
                        .css({
7980
                                opacity: 0.25,
7981
                                display: "block",
7982
                                position: "relative",
7983
                                height: cs.height,
7984
                                width: cs.width,
7985
                                margin: 0,
7986
                                left: 0,
7987
                                top: 0
7988
                        })
7989
                        .addClass("ui-resizable-ghost")
7990
                        .addClass(typeof o.ghost === "string" ? o.ghost : "");
7991
7992
                that.ghost.appendTo(that.helper);
7993
7994
        },
7995
7996
        resize: function() {
7997
                var that = $(this).resizable( "instance" );
7998
                if (that.ghost) {
7999
                        that.ghost.css({
8000
                                position: "relative",
8001
                                height: that.size.height,
8002
                                width: that.size.width
8003
                        });
8004
                }
8005
        },
8006
8007
        stop: function() {
8008
                var that = $(this).resizable( "instance" );
8009
                if (that.ghost && that.helper) {
8010
                        that.helper.get(0).removeChild(that.ghost.get(0));
8011
                }
8012
        }
8013
8014
});
8015
8016
$.ui.plugin.add("resizable", "grid", {
8017
8018
        resize: function() {
8019
                var outerDimensions,
8020
                        that = $(this).resizable( "instance" ),
8021
                        o = that.options,
8022
                        cs = that.size,
8023
                        os = that.originalSize,
8024
                        op = that.originalPosition,
8025
                        a = that.axis,
8026
                        grid = typeof o.grid === "number" ? [ o.grid, o.grid ] : o.grid,
8027
                        gridX = (grid[0] || 1),
8028
                        gridY = (grid[1] || 1),
8029
                        ox = Math.round((cs.width - os.width) / gridX) * gridX,
8030
                        oy = Math.round((cs.height - os.height) / gridY) * gridY,
8031
                        newWidth = os.width + ox,
8032
                        newHeight = os.height + oy,
8033
                        isMaxWidth = o.maxWidth && (o.maxWidth < newWidth),
8034
                        isMaxHeight = o.maxHeight && (o.maxHeight < newHeight),
8035
                        isMinWidth = o.minWidth && (o.minWidth > newWidth),
8036
                        isMinHeight = o.minHeight && (o.minHeight > newHeight);
8037
8038
                o.grid = grid;
8039
8040
                if (isMinWidth) {
8041
                        newWidth += gridX;
8042
                }
8043
                if (isMinHeight) {
8044
                        newHeight += gridY;
8045
                }
8046
                if (isMaxWidth) {
8047
                        newWidth -= gridX;
8048
                }
8049
                if (isMaxHeight) {
8050
                        newHeight -= gridY;
8051
                }
8052
8053
                if (/^(se|s|e)$/.test(a)) {
8054
                        that.size.width = newWidth;
8055
                        that.size.height = newHeight;
8056
                } else if (/^(ne)$/.test(a)) {
8057
                        that.size.width = newWidth;
8058
                        that.size.height = newHeight;
8059
                        that.position.top = op.top - oy;
8060
                } else if (/^(sw)$/.test(a)) {
8061
                        that.size.width = newWidth;
8062
                        that.size.height = newHeight;
8063
                        that.position.left = op.left - ox;
8064
                } else {
8065
                        if ( newHeight - gridY <= 0 || newWidth - gridX <= 0) {
8066
                                outerDimensions = that._getPaddingPlusBorderDimensions( this );
8067
                        }
8068
8069
                        if ( newHeight - gridY > 0 ) {
8070
                                that.size.height = newHeight;
8071
                                that.position.top = op.top - oy;
8072
                        } else {
8073
                                newHeight = gridY - outerDimensions.height;
8074
                                that.size.height = newHeight;
8075
                                that.position.top = op.top + os.height - newHeight;
8076
                        }
8077
                        if ( newWidth - gridX > 0 ) {
8078
                                that.size.width = newWidth;
8079
                                that.position.left = op.left - ox;
8080
                        } else {
8081
                                newWidth = gridX - outerDimensions.width;
8082
                                that.size.width = newWidth;
8083
                                that.position.left = op.left + os.width - newWidth;
8084
                        }
8085
                }
8086
        }
8087
8088
});
8089
8090
var resizable = $.ui.resizable;
8091
8092
8093
/*!
8094
 * jQuery UI Dialog 1.11.4
8095
 * http://jqueryui.com
8096
 *
8097
 * Copyright jQuery Foundation and other contributors
8098
 * Released under the MIT license.
8099
 * http://jquery.org/license
8100
 *
8101
 * http://api.jqueryui.com/dialog/
8102
 */
8103
8104
8105
var dialog = $.widget( "ui.dialog", {
8106
        version: "1.11.4",
8107
        options: {
8108
                appendTo: "body",
8109
                autoOpen: true,
8110
                buttons: [],
8111
                closeOnEscape: true,
8112
                closeText: "Close",
8113
                dialogClass: "",
8114
                draggable: true,
8115
                hide: null,
8116
                height: "auto",
8117
                maxHeight: null,
8118
                maxWidth: null,
8119
                minHeight: 150,
8120
                minWidth: 150,
8121
                modal: false,
8122
                position: {
8123
                        my: "center",
8124
                        at: "center",
8125
                        of: window,
8126
                        collision: "fit",
8127
                        // Ensure the titlebar is always visible
8128
                        using: function( pos ) {
8129
                                var topOffset = $( this ).css( pos ).offset().top;
8130
                                if ( topOffset < 0 ) {
8131
                                        $( this ).css( "top", pos.top - topOffset );
8132
                                }
8133
                        }
8134
                },
8135
                resizable: true,
8136
                show: null,
8137
                title: null,
8138
                width: 300,
8139
8140
                // callbacks
8141
                beforeClose: null,
8142
                close: null,
8143
                drag: null,
8144
                dragStart: null,
8145
                dragStop: null,
8146
                focus: null,
8147
                open: null,
8148
                resize: null,
8149
                resizeStart: null,
8150
                resizeStop: null
8151
        },
8152
8153
        sizeRelatedOptions: {
8154
                buttons: true,
8155
                height: true,
8156
                maxHeight: true,
8157
                maxWidth: true,
8158
                minHeight: true,
8159
                minWidth: true,
8160
                width: true
8161
        },
8162
8163
        resizableRelatedOptions: {
8164
                maxHeight: true,
8165
                maxWidth: true,
8166
                minHeight: true,
8167
                minWidth: true
8168
        },
8169
8170
        _create: function() {
8171
                this.originalCss = {
8172
                        display: this.element[ 0 ].style.display,
8173
                        width: this.element[ 0 ].style.width,
8174
                        minHeight: this.element[ 0 ].style.minHeight,
8175
                        maxHeight: this.element[ 0 ].style.maxHeight,
8176
                        height: this.element[ 0 ].style.height
8177
                };
8178
                this.originalPosition = {
8179
                        parent: this.element.parent(),
8180
                        index: this.element.parent().children().index( this.element )
8181
                };
8182
                this.originalTitle = this.element.attr( "title" );
8183
                this.options.title = this.options.title || this.originalTitle;
8184
8185
                this._createWrapper();
8186
8187
                this.element
8188
                        .show()
8189
                        .removeAttr( "title" )
8190
                        .addClass( "ui-dialog-content ui-widget-content" )
8191
                        .appendTo( this.uiDialog );
8192
8193
                this._createTitlebar();
8194
                this._createButtonPane();
8195
8196
                if ( this.options.draggable && $.fn.draggable ) {
8197
                        this._makeDraggable();
8198
                }
8199
                if ( this.options.resizable && $.fn.resizable ) {
8200
                        this._makeResizable();
8201
                }
8202
8203
                this._isOpen = false;
8204
8205
                this._trackFocus();
8206
        },
8207
8208
        _init: function() {
8209
                if ( this.options.autoOpen ) {
8210
                        this.open();
8211
                }
8212
        },
8213
8214
        _appendTo: function() {
8215
                var element = this.options.appendTo;
8216
                if ( element && (element.jquery || element.nodeType) ) {
8217
                        return $( element );
8218
                }
8219
                return this.document.find( element || "body" ).eq( 0 );
8220
        },
8221
8222
        _destroy: function() {
8223
                var next,
8224
                        originalPosition = this.originalPosition;
8225
8226
                this._untrackInstance();
8227
                this._destroyOverlay();
8228
8229
                this.element
8230
                        .removeUniqueId()
8231
                        .removeClass( "ui-dialog-content ui-widget-content" )
8232
                        .css( this.originalCss )
8233
                        // Without detaching first, the following becomes really slow
8234
                        .detach();
8235
8236
                this.uiDialog.stop( true, true ).remove();
8237
8238
                if ( this.originalTitle ) {
8239
                        this.element.attr( "title", this.originalTitle );
8240
                }
8241
8242
                next = originalPosition.parent.children().eq( originalPosition.index );
8243
                // Don't try to place the dialog next to itself (#8613)
8244
                if ( next.length && next[ 0 ] !== this.element[ 0 ] ) {
8245
                        next.before( this.element );
8246
                } else {
8247
                        originalPosition.parent.append( this.element );
8248
                }
8249
        },
8250
8251
        widget: function() {
8252
                return this.uiDialog;
8253
        },
8254
8255
        disable: $.noop,
8256
        enable: $.noop,
8257
8258
        close: function( event ) {
8259
                var activeElement,
8260
                        that = this;
8261
8262
                if ( !this._isOpen || this._trigger( "beforeClose", event ) === false ) {
8263
                        return;
8264
                }
8265
8266
                this._isOpen = false;
8267
                this._focusedElement = null;
8268
                this._destroyOverlay();
8269
                this._untrackInstance();
8270
8271
                if ( !this.opener.filter( ":focusable" ).focus().length ) {
8272
8273
                        // support: IE9
8274
                        // IE9 throws an "Unspecified error" accessing document.activeElement from an <iframe>
8275
                        try {
8276
                                activeElement = this.document[ 0 ].activeElement;
8277
8278
                                // Support: IE9, IE10
8279
                                // If the <body> is blurred, IE will switch windows, see #4520
8280
                                if ( activeElement && activeElement.nodeName.toLowerCase() !== "body" ) {
8281
8282
                                        // Hiding a focused element doesn't trigger blur in WebKit
8283
                                        // so in case we have nothing to focus on, explicitly blur the active element
8284
                                        // https://bugs.webkit.org/show_bug.cgi?id=47182
8285
                                        $( activeElement ).blur();
8286
                                }
8287
                        } catch ( error ) {}
8288
                }
8289
8290
                this._hide( this.uiDialog, this.options.hide, function() {
8291
                        that._trigger( "close", event );
8292
                });
8293
        },
8294
8295
        isOpen: function() {
8296
                return this._isOpen;
8297
        },
8298
8299
        moveToTop: function() {
8300
                this._moveToTop();
8301
        },
8302
8303
        _moveToTop: function( event, silent ) {
8304
                var moved = false,
8305
                        zIndices = this.uiDialog.siblings( ".ui-front:visible" ).map(function() {
8306
                                return +$( this ).css( "z-index" );
8307
                        }).get(),
8308
                        zIndexMax = Math.max.apply( null, zIndices );
8309
8310
                if ( zIndexMax >= +this.uiDialog.css( "z-index" ) ) {
8311
                        this.uiDialog.css( "z-index", zIndexMax + 1 );
8312
                        moved = true;
8313
                }
8314
8315
                if ( moved && !silent ) {
8316
                        this._trigger( "focus", event );
8317
                }
8318
                return moved;
8319
        },
8320
8321
        open: function() {
8322
                var that = this;
8323
                if ( this._isOpen ) {
8324
                        if ( this._moveToTop() ) {
8325
                                this._focusTabbable();
8326
                        }
8327
                        return;
8328
                }
8329
8330
                this._isOpen = true;
8331
                this.opener = $( this.document[ 0 ].activeElement );
8332
8333
                this._size();
8334
                this._position();
8335
                this._createOverlay();
8336
                this._moveToTop( null, true );
8337
8338
                // Ensure the overlay is moved to the top with the dialog, but only when
8339
                // opening. The overlay shouldn't move after the dialog is open so that
8340
                // modeless dialogs opened after the modal dialog stack properly.
8341
                if ( this.overlay ) {
8342
                        this.overlay.css( "z-index", this.uiDialog.css( "z-index" ) - 1 );
8343
                }
8344
8345
                this._show( this.uiDialog, this.options.show, function() {
8346
                        that._focusTabbable();
8347
                        that._trigger( "focus" );
8348
                });
8349
8350
                // Track the dialog immediately upon openening in case a focus event
8351
                // somehow occurs outside of the dialog before an element inside the
8352
                // dialog is focused (#10152)
8353
                this._makeFocusTarget();
8354
8355
                this._trigger( "open" );
8356
        },
8357
8358
        _focusTabbable: function() {
8359
                // Set focus to the first match:
8360
                // 1. An element that was focused previously
8361
                // 2. First element inside the dialog matching [autofocus]
8362
                // 3. Tabbable element inside the content element
8363
                // 4. Tabbable element inside the buttonpane
8364
                // 5. The close button
8365
                // 6. The dialog itself
8366
                var hasFocus = this._focusedElement;
8367
                if ( !hasFocus ) {
8368
                        hasFocus = this.element.find( "[autofocus]" );
8369
                }
8370
                if ( !hasFocus.length ) {
8371
                        hasFocus = this.element.find( ":tabbable" );
8372
                }
8373
                if ( !hasFocus.length ) {
8374
                        hasFocus = this.uiDialogButtonPane.find( ":tabbable" );
8375
                }
8376
                if ( !hasFocus.length ) {
8377
                        hasFocus = this.uiDialogTitlebarClose.filter( ":tabbable" );
8378
                }
8379
                if ( !hasFocus.length ) {
8380
                        hasFocus = this.uiDialog;
8381
                }
8382
                hasFocus.eq( 0 ).focus();
8383
        },
8384
8385
        _keepFocus: function( event ) {
8386
                function checkFocus() {
8387
                        var activeElement = this.document[0].activeElement,
8388
                                isActive = this.uiDialog[0] === activeElement ||
8389
                                        $.contains( this.uiDialog[0], activeElement );
8390
                        if ( !isActive ) {
8391
                                this._focusTabbable();
8392
                        }
8393
                }
8394
                event.preventDefault();
8395
                checkFocus.call( this );
8396
                // support: IE
8397
                // IE <= 8 doesn't prevent moving focus even with event.preventDefault()
8398
                // so we check again later
8399
                this._delay( checkFocus );
8400
        },
8401
8402
        _createWrapper: function() {
8403
                this.uiDialog = $("<div>")
8404
                        .addClass( "ui-dialog ui-widget ui-widget-content ui-corner-all ui-front " +
8405
                                this.options.dialogClass )
8406
                        .hide()
8407
                        .attr({
8408
                                // Setting tabIndex makes the div focusable
8409
                                tabIndex: -1,
8410
                                role: "dialog"
8411
                        })
8412
                        .appendTo( this._appendTo() );
8413
8414
                this._on( this.uiDialog, {
8415
                        keydown: function( event ) {
8416
                                if ( this.options.closeOnEscape && !event.isDefaultPrevented() && event.keyCode &&
8417
                                                event.keyCode === $.ui.keyCode.ESCAPE ) {
8418
                                        event.preventDefault();
8419
                                        this.close( event );
8420
                                        return;
8421
                                }
8422
8423
                                // prevent tabbing out of dialogs
8424
                                if ( event.keyCode !== $.ui.keyCode.TAB || event.isDefaultPrevented() ) {
8425
                                        return;
8426
                                }
8427
                                var tabbables = this.uiDialog.find( ":tabbable" ),
8428
                                        first = tabbables.filter( ":first" ),
8429
                                        last = tabbables.filter( ":last" );
8430
8431
                                if ( ( event.target === last[0] || event.target === this.uiDialog[0] ) && !event.shiftKey ) {
8432
                                        this._delay(function() {
8433
                                                first.focus();
8434
                                        });
8435
                                        event.preventDefault();
8436
                                } else if ( ( event.target === first[0] || event.target === this.uiDialog[0] ) && event.shiftKey ) {
8437
                                        this._delay(function() {
8438
                                                last.focus();
8439
                                        });
8440
                                        event.preventDefault();
8441
                                }
8442
                        },
8443
                        mousedown: function( event ) {
8444
                                if ( this._moveToTop( event ) ) {
8445
                                        this._focusTabbable();
8446
                                }
8447
                        }
8448
                });
8449
8450
                // We assume that any existing aria-describedby attribute means
8451
                // that the dialog content is marked up properly
8452
                // otherwise we brute force the content as the description
8453
                if ( !this.element.find( "[aria-describedby]" ).length ) {
8454
                        this.uiDialog.attr({
8455
                                "aria-describedby": this.element.uniqueId().attr( "id" )
8456
                        });
8457
                }
8458
        },
8459
8460
        _createTitlebar: function() {
8461
                var uiDialogTitle;
8462
8463
                this.uiDialogTitlebar = $( "<div>" )
8464
                        .addClass( "ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix" )
8465
                        .prependTo( this.uiDialog );
8466
                this._on( this.uiDialogTitlebar, {
8467
                        mousedown: function( event ) {
8468
                                // Don't prevent click on close button (#8838)
8469
                                // Focusing a dialog that is partially scrolled out of view
8470
                                // causes the browser to scroll it into view, preventing the click event
8471
                                if ( !$( event.target ).closest( ".ui-dialog-titlebar-close" ) ) {
8472
                                        // Dialog isn't getting focus when dragging (#8063)
8473
                                        this.uiDialog.focus();
8474
                                }
8475
                        }
8476
                });
8477
8478
                // support: IE
8479
                // Use type="button" to prevent enter keypresses in textboxes from closing the
8480
                // dialog in IE (#9312)
8481
                this.uiDialogTitlebarClose = $( "<button type='button'></button>" )
8482
                        .button({
8483
                                label: this.options.closeText,
8484
                                icons: {
8485
                                        primary: "ui-icon-closethick"
8486
                                },
8487
                                text: false
8488
                        })
8489
                        .addClass( "ui-dialog-titlebar-close" )
8490
                        .appendTo( this.uiDialogTitlebar );
8491
                this._on( this.uiDialogTitlebarClose, {
8492
                        click: function( event ) {
8493
                                event.preventDefault();
8494
                                this.close( event );
8495
                        }
8496
                });
8497
8498
                uiDialogTitle = $( "<span>" )
8499
                        .uniqueId()
8500
                        .addClass( "ui-dialog-title" )
8501
                        .prependTo( this.uiDialogTitlebar );
8502
                this._title( uiDialogTitle );
8503
8504
                this.uiDialog.attr({
8505
                        "aria-labelledby": uiDialogTitle.attr( "id" )
8506
                });
8507
        },
8508
8509
        _title: function( title ) {
8510
                if ( !this.options.title ) {
8511
                        title.html( "&#160;" );
8512
                }
8513
                title.text( this.options.title );
8514
        },
8515
8516
        _createButtonPane: function() {
8517
                this.uiDialogButtonPane = $( "<div>" )
8518
                        .addClass( "ui-dialog-buttonpane ui-widget-content ui-helper-clearfix" );
8519
8520
                this.uiButtonSet = $( "<div>" )
8521
                        .addClass( "ui-dialog-buttonset" )
8522
                        .appendTo( this.uiDialogButtonPane );
8523
8524
                this._createButtons();
8525
        },
8526
8527
        _createButtons: function() {
8528
                var that = this,
8529
                        buttons = this.options.buttons;
8530
8531
                // if we already have a button pane, remove it
8532
                this.uiDialogButtonPane.remove();
8533
                this.uiButtonSet.empty();
8534
8535
                if ( $.isEmptyObject( buttons ) || ($.isArray( buttons ) && !buttons.length) ) {
8536
                        this.uiDialog.removeClass( "ui-dialog-buttons" );
8537
                        return;
8538
                }
8539
8540
                $.each( buttons, function( name, props ) {
8541
                        var click, buttonOptions;
8542
                        props = $.isFunction( props ) ?
8543
                                { click: props, text: name } :
8544
                                props;
8545
                        // Default to a non-submitting button
8546
                        props = $.extend( { type: "button" }, props );
8547
                        // Change the context for the click callback to be the main element
8548
                        click = props.click;
8549
                        props.click = function() {
8550
                                click.apply( that.element[ 0 ], arguments );
8551
                        };
8552
                        buttonOptions = {
8553
                                icons: props.icons,
8554
                                text: props.showText
8555
                        };
8556
                        delete props.icons;
8557
                        delete props.showText;
8558
                        $( "<button></button>", props )
8559
                                .button( buttonOptions )
8560
                                .appendTo( that.uiButtonSet );
8561
                });
8562
                this.uiDialog.addClass( "ui-dialog-buttons" );
8563
                this.uiDialogButtonPane.appendTo( this.uiDialog );
8564
        },
8565
8566
        _makeDraggable: function() {
8567
                var that = this,
8568
                        options = this.options;
8569
8570
                function filteredUi( ui ) {
8571
                        return {
8572
                                position: ui.position,
8573
                                offset: ui.offset
8574
                        };
8575
                }
8576
8577
                this.uiDialog.draggable({
8578
                        cancel: ".ui-dialog-content, .ui-dialog-titlebar-close",
8579
                        handle: ".ui-dialog-titlebar",
8580
                        containment: "document",
8581
                        start: function( event, ui ) {
8582
                                $( this ).addClass( "ui-dialog-dragging" );
8583
                                that._blockFrames();
8584
                                that._trigger( "dragStart", event, filteredUi( ui ) );
8585
                        },
8586
                        drag: function( event, ui ) {
8587
                                that._trigger( "drag", event, filteredUi( ui ) );
8588
                        },
8589
                        stop: function( event, ui ) {
8590
                                var left = ui.offset.left - that.document.scrollLeft(),
8591
                                        top = ui.offset.top - that.document.scrollTop();
8592
8593
                                options.position = {
8594
                                        my: "left top",
8595
                                        at: "left" + (left >= 0 ? "+" : "") + left + " " +
8596
                                                "top" + (top >= 0 ? "+" : "") + top,
8597
                                        of: that.window
8598
                                };
8599
                                $( this ).removeClass( "ui-dialog-dragging" );
8600
                                that._unblockFrames();
8601
                                that._trigger( "dragStop", event, filteredUi( ui ) );
8602
                        }
8603
                });
8604
        },
8605
8606
        _makeResizable: function() {
8607
                var that = this,
8608
                        options = this.options,
8609
                        handles = options.resizable,
8610
                        // .ui-resizable has position: relative defined in the stylesheet
8611
                        // but dialogs have to use absolute or fixed positioning
8612
                        position = this.uiDialog.css("position"),
8613
                        resizeHandles = typeof handles === "string" ?
8614
                                handles        :
8615
                                "n,e,s,w,se,sw,ne,nw";
8616
8617
                function filteredUi( ui ) {
8618
                        return {
8619
                                originalPosition: ui.originalPosition,
8620
                                originalSize: ui.originalSize,
8621
                                position: ui.position,
8622
                                size: ui.size
8623
                        };
8624
                }
8625
8626
                this.uiDialog.resizable({
8627
                        cancel: ".ui-dialog-content",
8628
                        containment: "document",
8629
                        alsoResize: this.element,
8630
                        maxWidth: options.maxWidth,
8631
                        maxHeight: options.maxHeight,
8632
                        minWidth: options.minWidth,
8633
                        minHeight: this._minHeight(),
8634
                        handles: resizeHandles,
8635
                        start: function( event, ui ) {
8636
                                $( this ).addClass( "ui-dialog-resizing" );
8637
                                that._blockFrames();
8638
                                that._trigger( "resizeStart", event, filteredUi( ui ) );
8639
                        },
8640
                        resize: function( event, ui ) {
8641
                                that._trigger( "resize", event, filteredUi( ui ) );
8642
                        },
8643
                        stop: function( event, ui ) {
8644
                                var offset = that.uiDialog.offset(),
8645
                                        left = offset.left - that.document.scrollLeft(),
8646
                                        top = offset.top - that.document.scrollTop();
8647
8648
                                options.height = that.uiDialog.height();
8649
                                options.width = that.uiDialog.width();
8650
                                options.position = {
8651
                                        my: "left top",
8652
                                        at: "left" + (left >= 0 ? "+" : "") + left + " " +
8653
                                                "top" + (top >= 0 ? "+" : "") + top,
8654
                                        of: that.window
8655
                                };
8656
                                $( this ).removeClass( "ui-dialog-resizing" );
8657
                                that._unblockFrames();
8658
                                that._trigger( "resizeStop", event, filteredUi( ui ) );
8659
                        }
8660
                })
8661
                .css( "position", position );
8662
        },
8663
8664
        _trackFocus: function() {
8665
                this._on( this.widget(), {
8666
                        focusin: function( event ) {
8667
                                this._makeFocusTarget();
8668
                                this._focusedElement = $( event.target );
8669
                        }
8670
                });
8671
        },
8672
8673
        _makeFocusTarget: function() {
8674
                this._untrackInstance();
8675
                this._trackingInstances().unshift( this );
8676
        },
8677
8678
        _untrackInstance: function() {
8679
                var instances = this._trackingInstances(),
8680
                        exists = $.inArray( this, instances );
8681
                if ( exists !== -1 ) {
8682
                        instances.splice( exists, 1 );
8683
                }
8684
        },
8685
8686
        _trackingInstances: function() {
8687
                var instances = this.document.data( "ui-dialog-instances" );
8688
                if ( !instances ) {
8689
                        instances = [];
8690
                        this.document.data( "ui-dialog-instances", instances );
8691
                }
8692
                return instances;
8693
        },
8694
8695
        _minHeight: function() {
8696
                var options = this.options;
8697
8698
                return options.height === "auto" ?
8699
                        options.minHeight :
8700
                        Math.min( options.minHeight, options.height );
8701
        },
8702
8703
        _position: function() {
8704
                // Need to show the dialog to get the actual offset in the position plugin
8705
                var isVisible = this.uiDialog.is( ":visible" );
8706
                if ( !isVisible ) {
8707
                        this.uiDialog.show();
8708
                }
8709
                this.uiDialog.position( this.options.position );
8710
                if ( !isVisible ) {
8711
                        this.uiDialog.hide();
8712
                }
8713
        },
8714
8715
        _setOptions: function( options ) {
8716
                var that = this,
8717
                        resize = false,
8718
                        resizableOptions = {};
8719
8720
                $.each( options, function( key, value ) {
8721
                        that._setOption( key, value );
8722
8723
                        if ( key in that.sizeRelatedOptions ) {
8724
                                resize = true;
8725
                        }
8726
                        if ( key in that.resizableRelatedOptions ) {
8727
                                resizableOptions[ key ] = value;
8728
                        }
8729
                });
8730
8731
                if ( resize ) {
8732
                        this._size();
8733
                        this._position();
8734
                }
8735
                if ( this.uiDialog.is( ":data(ui-resizable)" ) ) {
8736
                        this.uiDialog.resizable( "option", resizableOptions );
8737
                }
8738
        },
8739
8740
        _setOption: function( key, value ) {
8741
                var isDraggable, isResizable,
8742
                        uiDialog = this.uiDialog;
8743
8744
                if ( key === "dialogClass" ) {
8745
                        uiDialog
8746
                                .removeClass( this.options.dialogClass )
8747
                                .addClass( value );
8748
                }
8749
8750
                if ( key === "disabled" ) {
8751
                        return;
8752
                }
8753
8754
                this._super( key, value );
8755
8756
                if ( key === "appendTo" ) {
8757
                        this.uiDialog.appendTo( this._appendTo() );
8758
                }
8759
8760
                if ( key === "buttons" ) {
8761
                        this._createButtons();
8762
                }
8763
8764
                if ( key === "closeText" ) {
8765
                        this.uiDialogTitlebarClose.button({
8766
                                // Ensure that we always pass a string
8767
                                label: "" + value
8768
                        });
8769
                }
8770
8771
                if ( key === "draggable" ) {
8772
                        isDraggable = uiDialog.is( ":data(ui-draggable)" );
8773
                        if ( isDraggable && !value ) {
8774
                                uiDialog.draggable( "destroy" );
8775
                        }
8776
8777
                        if ( !isDraggable && value ) {
8778
                                this._makeDraggable();
8779
                        }
8780
                }
8781
8782
                if ( key === "position" ) {
8783
                        this._position();
8784
                }
8785
8786
                if ( key === "resizable" ) {
8787
                        // currently resizable, becoming non-resizable
8788
                        isResizable = uiDialog.is( ":data(ui-resizable)" );
8789
                        if ( isResizable && !value ) {
8790
                                uiDialog.resizable( "destroy" );
8791
                        }
8792
8793
                        // currently resizable, changing handles
8794
                        if ( isResizable && typeof value === "string" ) {
8795
                                uiDialog.resizable( "option", "handles", value );
8796
                        }
8797
8798
                        // currently non-resizable, becoming resizable
8799
                        if ( !isResizable && value !== false ) {
8800
                                this._makeResizable();
8801
                        }
8802
                }
8803
8804
                if ( key === "title" ) {
8805
                        this._title( this.uiDialogTitlebar.find( ".ui-dialog-title" ) );
8806
                }
8807
        },
8808
8809
        _size: function() {
8810
                // If the user has resized the dialog, the .ui-dialog and .ui-dialog-content
8811
                // divs will both have width and height set, so we need to reset them
8812
                var nonContentHeight, minContentHeight, maxContentHeight,
8813
                        options = this.options;
8814
8815
                // Reset content sizing
8816
                this.element.show().css({
8817
                        width: "auto",
8818
                        minHeight: 0,
8819
                        maxHeight: "none",
8820
                        height: 0
8821
                });
8822
8823
                if ( options.minWidth > options.width ) {
8824
                        options.width = options.minWidth;
8825
                }
8826
8827
                // reset wrapper sizing
8828
                // determine the height of all the non-content elements
8829
                nonContentHeight = this.uiDialog.css({
8830
                                height: "auto",
8831
                                width: options.width
8832
                        })
8833
                        .outerHeight();
8834
                minContentHeight = Math.max( 0, options.minHeight - nonContentHeight );
8835
                maxContentHeight = typeof options.maxHeight === "number" ?
8836
                        Math.max( 0, options.maxHeight - nonContentHeight ) :
8837
                        "none";
8838
8839
                if ( options.height === "auto" ) {
8840
                        this.element.css({
8841
                                minHeight: minContentHeight,
8842
                                maxHeight: maxContentHeight,
8843
                                height: "auto"
8844
                        });
8845
                } else {
8846
                        this.element.height( Math.max( 0, options.height - nonContentHeight ) );
8847
                }
8848
8849
                if ( this.uiDialog.is( ":data(ui-resizable)" ) ) {
8850
                        this.uiDialog.resizable( "option", "minHeight", this._minHeight() );
8851
                }
8852
        },
8853
8854
        _blockFrames: function() {
8855
                this.iframeBlocks = this.document.find( "iframe" ).map(function() {
8856
                        var iframe = $( this );
8857
8858
                        return $( "<div>" )
8859
                                .css({
8860
                                        position: "absolute",
8861
                                        width: iframe.outerWidth(),
8862
                                        height: iframe.outerHeight()
8863
                                })
8864
                                .appendTo( iframe.parent() )
8865
                                .offset( iframe.offset() )[0];
8866
                });
8867
        },
8868
8869
        _unblockFrames: function() {
8870
                if ( this.iframeBlocks ) {
8871
                        this.iframeBlocks.remove();
8872
                        delete this.iframeBlocks;
8873
                }
8874
        },
8875
8876
        _allowInteraction: function( event ) {
8877
                if ( $( event.target ).closest( ".ui-dialog" ).length ) {
8878
                        return true;
8879
                }
8880
8881
                // TODO: Remove hack when datepicker implements
8882
                // the .ui-front logic (#8989)
8883
                return !!$( event.target ).closest( ".ui-datepicker" ).length;
8884
        },
8885
8886
        _createOverlay: function() {
8887
                if ( !this.options.modal ) {
8888
                        return;
8889
                }
8890
8891
                // We use a delay in case the overlay is created from an
8892
                // event that we're going to be cancelling (#2804)
8893
                var isOpening = true;
8894
                this._delay(function() {
8895
                        isOpening = false;
8896
                });
8897
8898
                if ( !this.document.data( "ui-dialog-overlays" ) ) {
8899
8900
                        // Prevent use of anchors and inputs
8901
                        // Using _on() for an event handler shared across many instances is
8902
                        // safe because the dialogs stack and must be closed in reverse order
8903
                        this._on( this.document, {
8904
                                focusin: function( event ) {
8905
                                        if ( isOpening ) {
8906
                                                return;
8907
                                        }
8908
8909
                                        if ( !this._allowInteraction( event ) ) {
8910
                                                event.preventDefault();
8911
                                                this._trackingInstances()[ 0 ]._focusTabbable();
8912
                                        }
8913
                                }
8914
                        });
8915
                }
8916
8917
                this.overlay = $( "<div>" )
8918
                        .addClass( "ui-widget-overlay ui-front" )
8919
                        .appendTo( this._appendTo() );
8920
                this._on( this.overlay, {
8921
                        mousedown: "_keepFocus"
8922
                });
8923
                this.document.data( "ui-dialog-overlays",
8924
                        (this.document.data( "ui-dialog-overlays" ) || 0) + 1 );
8925
        },
8926
8927
        _destroyOverlay: function() {
8928
                if ( !this.options.modal ) {
8929
                        return;
8930
                }
8931
8932
                if ( this.overlay ) {
8933
                        var overlays = this.document.data( "ui-dialog-overlays" ) - 1;
8934
8935
                        if ( !overlays ) {
8936
                                this.document
8937
                                        .unbind( "focusin" )
8938
                                        .removeData( "ui-dialog-overlays" );
8939
                        } else {
8940
                                this.document.data( "ui-dialog-overlays", overlays );
8941
                        }
8942
8943
                        this.overlay.remove();
8944
                        this.overlay = null;
8945
                }
8946
        }
8947
});
8948
8949
8950
/*!
8951
 * jQuery UI Droppable 1.11.4
8952
 * http://jqueryui.com
8953
 *
8954
 * Copyright jQuery Foundation and other contributors
8955
 * Released under the MIT license.
8956
 * http://jquery.org/license
8957
 *
8958
 * http://api.jqueryui.com/droppable/
8959
 */
8960
8961
8962
$.widget( "ui.droppable", {
8963
        version: "1.11.4",
8964
        widgetEventPrefix: "drop",
8965
        options: {
8966
                accept: "*",
8967
                activeClass: false,
8968
                addClasses: true,
8969
                greedy: false,
8970
                hoverClass: false,
8971
                scope: "default",
8972
                tolerance: "intersect",
8973
8974
                // callbacks
8975
                activate: null,
8976
                deactivate: null,
8977
                drop: null,
8978
                out: null,
8979
                over: null
8980
        },
8981
        _create: function() {
8982
8983
                var proportions,
8984
                        o = this.options,
8985
                        accept = o.accept;
8986
8987
                this.isover = false;
8988
                this.isout = true;
8989
8990
                this.accept = $.isFunction( accept ) ? accept : function( d ) {
8991
                        return d.is( accept );
8992
                };
8993
8994
                this.proportions = function( /* valueToWrite */ ) {
8995
                        if ( arguments.length ) {
8996
                                // Store the droppable's proportions
8997
                                proportions = arguments[ 0 ];
8998
                        } else {
8999
                                // Retrieve or derive the droppable's proportions
9000
                                return proportions ?
9001
                                        proportions :
9002
                                        proportions = {
9003
                                                width: this.element[ 0 ].offsetWidth,
9004
                                                height: this.element[ 0 ].offsetHeight
9005
                                        };
9006
                        }
9007
                };
9008
9009
                this._addToManager( o.scope );
9010
9011
                o.addClasses && this.element.addClass( "ui-droppable" );
9012
9013
        },
9014
9015
        _addToManager: function( scope ) {
9016
                // Add the reference and positions to the manager
9017
                $.ui.ddmanager.droppables[ scope ] = $.ui.ddmanager.droppables[ scope ] || [];
9018
                $.ui.ddmanager.droppables[ scope ].push( this );
9019
        },
9020
9021
        _splice: function( drop ) {
9022
                var i = 0;
9023
                for ( ; i < drop.length; i++ ) {
9024
                        if ( drop[ i ] === this ) {
9025
                                drop.splice( i, 1 );
9026
                        }
9027
                }
9028
        },
9029
9030
        _destroy: function() {
9031
                var drop = $.ui.ddmanager.droppables[ this.options.scope ];
9032
9033
                this._splice( drop );
9034
9035
                this.element.removeClass( "ui-droppable ui-droppable-disabled" );
9036
        },
9037
9038
        _setOption: function( key, value ) {
9039
9040
                if ( key === "accept" ) {
9041
                        this.accept = $.isFunction( value ) ? value : function( d ) {
9042
                                return d.is( value );
9043
                        };
9044
                } else if ( key === "scope" ) {
9045
                        var drop = $.ui.ddmanager.droppables[ this.options.scope ];
9046
9047
                        this._splice( drop );
9048
                        this._addToManager( value );
9049
                }
9050
9051
                this._super( key, value );
9052
        },
9053
9054
        _activate: function( event ) {
9055
                var draggable = $.ui.ddmanager.current;
9056
                if ( this.options.activeClass ) {
9057
                        this.element.addClass( this.options.activeClass );
9058
                }
9059
                if ( draggable ){
9060
                        this._trigger( "activate", event, this.ui( draggable ) );
9061
                }
9062
        },
9063
9064
        _deactivate: function( event ) {
9065
                var draggable = $.ui.ddmanager.current;
9066
                if ( this.options.activeClass ) {
9067
                        this.element.removeClass( this.options.activeClass );
9068
                }
9069
                if ( draggable ){
9070
                        this._trigger( "deactivate", event, this.ui( draggable ) );
9071
                }
9072
        },
9073
9074
        _over: function( event ) {
9075
9076
                var draggable = $.ui.ddmanager.current;
9077
9078
                // Bail if draggable and droppable are same element
9079
                if ( !draggable || ( draggable.currentItem || draggable.element )[ 0 ] === this.element[ 0 ] ) {
9080
                        return;
9081
                }
9082
9083
                if ( this.accept.call( this.element[ 0 ], ( draggable.currentItem || draggable.element ) ) ) {
9084
                        if ( this.options.hoverClass ) {
9085
                                this.element.addClass( this.options.hoverClass );
9086
                        }
9087
                        this._trigger( "over", event, this.ui( draggable ) );
9088
                }
9089
9090
        },
9091
9092
        _out: function( event ) {
9093
9094
                var draggable = $.ui.ddmanager.current;
9095
9096
                // Bail if draggable and droppable are same element
9097
                if ( !draggable || ( draggable.currentItem || draggable.element )[ 0 ] === this.element[ 0 ] ) {
9098
                        return;
9099
                }
9100
9101
                if ( this.accept.call( this.element[ 0 ], ( draggable.currentItem || draggable.element ) ) ) {
9102
                        if ( this.options.hoverClass ) {
9103
                                this.element.removeClass( this.options.hoverClass );
9104
                        }
9105
                        this._trigger( "out", event, this.ui( draggable ) );
9106
                }
9107
9108
        },
9109
9110
        _drop: function( event, custom ) {
9111
9112
                var draggable = custom || $.ui.ddmanager.current,
9113
                        childrenIntersection = false;
9114
9115
                // Bail if draggable and droppable are same element
9116
                if ( !draggable || ( draggable.currentItem || draggable.element )[ 0 ] === this.element[ 0 ] ) {
9117
                        return false;
9118
                }
9119
9120
                this.element.find( ":data(ui-droppable)" ).not( ".ui-draggable-dragging" ).each(function() {
9121
                        var inst = $( this ).droppable( "instance" );
9122
                        if (
9123
                                inst.options.greedy &&
9124
                                !inst.options.disabled &&
9125
                                inst.options.scope === draggable.options.scope &&
9126
                                inst.accept.call( inst.element[ 0 ], ( draggable.currentItem || draggable.element ) ) &&
9127
                                $.ui.intersect( draggable, $.extend( inst, { offset: inst.element.offset() } ), inst.options.tolerance, event )
9128
                        ) { childrenIntersection = true; return false; }
9129
                });
9130
                if ( childrenIntersection ) {
9131
                        return false;
9132
                }
9133
9134
                if ( this.accept.call( this.element[ 0 ], ( draggable.currentItem || draggable.element ) ) ) {
9135
                        if ( this.options.activeClass ) {
9136
                                this.element.removeClass( this.options.activeClass );
9137
                        }
9138
                        if ( this.options.hoverClass ) {
9139
                                this.element.removeClass( this.options.hoverClass );
9140
                        }
9141
                        this._trigger( "drop", event, this.ui( draggable ) );
9142
                        return this.element;
9143
                }
9144
9145
                return false;
9146
9147
        },
9148
9149
        ui: function( c ) {
9150
                return {
9151
                        draggable: ( c.currentItem || c.element ),
9152
                        helper: c.helper,
9153
                        position: c.position,
9154
                        offset: c.positionAbs
9155
                };
9156
        }
9157
9158
});
9159
9160
$.ui.intersect = (function() {
9161
        function isOverAxis( x, reference, size ) {
9162
                return ( x >= reference ) && ( x < ( reference + size ) );
9163
        }
9164
9165
        return function( draggable, droppable, toleranceMode, event ) {
9166
9167
                if ( !droppable.offset ) {
9168
                        return false;
9169
                }
9170
9171
                var x1 = ( draggable.positionAbs || draggable.position.absolute ).left + draggable.margins.left,
9172
                        y1 = ( draggable.positionAbs || draggable.position.absolute ).top + draggable.margins.top,
9173
                        x2 = x1 + draggable.helperProportions.width,
9174
                        y2 = y1 + draggable.helperProportions.height,
9175
                        l = droppable.offset.left,
9176
                        t = droppable.offset.top,
9177
                        r = l + droppable.proportions().width,
9178
                        b = t + droppable.proportions().height;
9179
9180
                switch ( toleranceMode ) {
9181
                case "fit":
9182
                        return ( l <= x1 && x2 <= r && t <= y1 && y2 <= b );
9183
                case "intersect":
9184
                        return ( l < x1 + ( draggable.helperProportions.width / 2 ) && // Right Half
9185
                                x2 - ( draggable.helperProportions.width / 2 ) < r && // Left Half
9186
                                t < y1 + ( draggable.helperProportions.height / 2 ) && // Bottom Half
9187
                                y2 - ( draggable.helperProportions.height / 2 ) < b ); // Top Half
9188
                case "pointer":
9189
                        return isOverAxis( event.pageY, t, droppable.proportions().height ) && isOverAxis( event.pageX, l, droppable.proportions().width );
9190
                case "touch":
9191
                        return (
9192
                                ( y1 >= t && y1 <= b ) || // Top edge touching
9193
                                ( y2 >= t && y2 <= b ) || // Bottom edge touching
9194
                                ( y1 < t && y2 > b ) // Surrounded vertically
9195
                        ) && (
9196
                                ( x1 >= l && x1 <= r ) || // Left edge touching
9197
                                ( x2 >= l && x2 <= r ) || // Right edge touching
9198
                                ( x1 < l && x2 > r ) // Surrounded horizontally
9199
                        );
9200
                default:
9201
                        return false;
9202
                }
9203
        };
9204
})();
9205
9206
/*
9207
        This manager tracks offsets of draggables and droppables
9208
*/
9209
$.ui.ddmanager = {
9210
        current: null,
9211
        droppables: { "default": [] },
9212
        prepareOffsets: function( t, event ) {
9213
9214
                var i, j,
9215
                        m = $.ui.ddmanager.droppables[ t.options.scope ] || [],
9216
                        type = event ? event.type : null, // workaround for #2317
9217
                        list = ( t.currentItem || t.element ).find( ":data(ui-droppable)" ).addBack();
9218
9219
                droppablesLoop: for ( i = 0; i < m.length; i++ ) {
9220
9221
                        // No disabled and non-accepted
9222
                        if ( m[ i ].options.disabled || ( t && !m[ i ].accept.call( m[ i ].element[ 0 ], ( t.currentItem || t.element ) ) ) ) {
9223
                                continue;
9224
                        }
9225
9226
                        // Filter out elements in the current dragged item
9227
                        for ( j = 0; j < list.length; j++ ) {
9228
                                if ( list[ j ] === m[ i ].element[ 0 ] ) {
9229
                                        m[ i ].proportions().height = 0;
9230
                                        continue droppablesLoop;
9231
                                }
9232
                        }
9233
9234
                        m[ i ].visible = m[ i ].element.css( "display" ) !== "none";
9235
                        if ( !m[ i ].visible ) {
9236
                                continue;
9237
                        }
9238
9239
                        // Activate the droppable if used directly from draggables
9240
                        if ( type === "mousedown" ) {
9241
                                m[ i ]._activate.call( m[ i ], event );
9242
                        }
9243
9244
                        m[ i ].offset = m[ i ].element.offset();
9245
                        m[ i ].proportions({ width: m[ i ].element[ 0 ].offsetWidth, height: m[ i ].element[ 0 ].offsetHeight });
9246
9247
                }
9248
9249
        },
9250
        drop: function( draggable, event ) {
9251
9252
                var dropped = false;
9253
                // Create a copy of the droppables in case the list changes during the drop (#9116)
9254
                $.each( ( $.ui.ddmanager.droppables[ draggable.options.scope ] || [] ).slice(), function() {
9255
9256
                        if ( !this.options ) {
9257
                                return;
9258
                        }
9259
                        if ( !this.options.disabled && this.visible && $.ui.intersect( draggable, this, this.options.tolerance, event ) ) {
9260
                                dropped = this._drop.call( this, event ) || dropped;
9261
                        }
9262
9263
                        if ( !this.options.disabled && this.visible && this.accept.call( this.element[ 0 ], ( draggable.currentItem || draggable.element ) ) ) {
9264
                                this.isout = true;
9265
                                this.isover = false;
9266
                                this._deactivate.call( this, event );
9267
                        }
9268
9269
                });
9270
                return dropped;
9271
9272
        },
9273
        dragStart: function( draggable, event ) {
9274
                // Listen for scrolling so that if the dragging causes scrolling the position of the droppables can be recalculated (see #5003)
9275
                draggable.element.parentsUntil( "body" ).bind( "scroll.droppable", function() {
9276
                        if ( !draggable.options.refreshPositions ) {
9277
                                $.ui.ddmanager.prepareOffsets( draggable, event );
9278
                        }
9279
                });
9280
        },
9281
        drag: function( draggable, event ) {
9282
9283
                // If you have a highly dynamic page, you might try this option. It renders positions every time you move the mouse.
9284
                if ( draggable.options.refreshPositions ) {
9285
                        $.ui.ddmanager.prepareOffsets( draggable, event );
9286
                }
9287
9288
                // Run through all droppables and check their positions based on specific tolerance options
9289
                $.each( $.ui.ddmanager.droppables[ draggable.options.scope ] || [], function() {
9290
9291
                        if ( this.options.disabled || this.greedyChild || !this.visible ) {
9292
                                return;
9293
                        }
9294
9295
                        var parentInstance, scope, parent,
9296
                                intersects = $.ui.intersect( draggable, this, this.options.tolerance, event ),
9297
                                c = !intersects && this.isover ? "isout" : ( intersects && !this.isover ? "isover" : null );
9298
                        if ( !c ) {
9299
                                return;
9300
                        }
9301
9302
                        if ( this.options.greedy ) {
9303
                                // find droppable parents with same scope
9304
                                scope = this.options.scope;
9305
                                parent = this.element.parents( ":data(ui-droppable)" ).filter(function() {
9306
                                        return $( this ).droppable( "instance" ).options.scope === scope;
9307
                                });
9308
9309
                                if ( parent.length ) {
9310
                                        parentInstance = $( parent[ 0 ] ).droppable( "instance" );
9311
                                        parentInstance.greedyChild = ( c === "isover" );
9312
                                }
9313
                        }
9314
9315
                        // we just moved into a greedy child
9316
                        if ( parentInstance && c === "isover" ) {
9317
                                parentInstance.isover = false;
9318
                                parentInstance.isout = true;
9319
                                parentInstance._out.call( parentInstance, event );
9320
                        }
9321
9322
                        this[ c ] = true;
9323
                        this[c === "isout" ? "isover" : "isout"] = false;
9324
                        this[c === "isover" ? "_over" : "_out"].call( this, event );
9325
9326
                        // we just moved out of a greedy child
9327
                        if ( parentInstance && c === "isout" ) {
9328
                                parentInstance.isout = false;
9329
                                parentInstance.isover = true;
9330
                                parentInstance._over.call( parentInstance, event );
9331
                        }
9332
                });
9333
9334
        },
9335
        dragStop: function( draggable, event ) {
9336
                draggable.element.parentsUntil( "body" ).unbind( "scroll.droppable" );
9337
                // Call prepareOffsets one final time since IE does not fire return scroll events when overflow was caused by drag (see #5003)
9338
                if ( !draggable.options.refreshPositions ) {
9339
                        $.ui.ddmanager.prepareOffsets( draggable, event );
9340
                }
9341
        }
9342
};
9343
9344
var droppable = $.ui.droppable;
9345
9346
9347
/*!
9348
 * jQuery UI Effects 1.11.4
9349
 * http://jqueryui.com
9350
 *
9351
 * Copyright jQuery Foundation and other contributors
9352
 * Released under the MIT license.
9353
 * http://jquery.org/license
9354
 *
9355
 * http://api.jqueryui.com/category/effects-core/
9356
 */
9357
9358
9359
var dataSpace = "ui-effects-",
9360
9361
        // Create a local jQuery because jQuery Color relies on it and the
9362
        // global may not exist with AMD and a custom build (#10199)
9363
        jQuery = $;
9364
9365
$.effects = {
9366
        effect: {}
9367
};
9368
9369
/*!
9370
 * jQuery Color Animations v2.1.2
9371
 * https://github.com/jquery/jquery-color
9372
 *
9373
 * Copyright 2014 jQuery Foundation and other contributors
9374
 * Released under the MIT license.
9375
 * http://jquery.org/license
9376
 *
9377
 * Date: Wed Jan 16 08:47:09 2013 -0600
9378
 */
9379
(function( jQuery, undefined ) {
9380
9381
        var stepHooks = "backgroundColor borderBottomColor borderLeftColor borderRightColor borderTopColor color columnRuleColor outlineColor textDecorationColor textEmphasisColor",
9382
9383
        // plusequals test for += 100 -= 100
9384
        rplusequals = /^([\-+])=\s*(\d+\.?\d*)/,
9385
        // a set of RE's that can match strings and generate color tuples.
9386
        stringParsers = [ {
9387
                        re: /rgba?\(\s*(\d{1,3})\s*,\s*(\d{1,3})\s*,\s*(\d{1,3})\s*(?:,\s*(\d?(?:\.\d+)?)\s*)?\)/,
9388
                        parse: function( execResult ) {
9389
                                return [
9390
                                        execResult[ 1 ],
9391
                                        execResult[ 2 ],
9392
                                        execResult[ 3 ],
9393
                                        execResult[ 4 ]
9394
                                ];
9395
                        }
9396
                }, {
9397
                        re: /rgba?\(\s*(\d+(?:\.\d+)?)\%\s*,\s*(\d+(?:\.\d+)?)\%\s*,\s*(\d+(?:\.\d+)?)\%\s*(?:,\s*(\d?(?:\.\d+)?)\s*)?\)/,
9398
                        parse: function( execResult ) {
9399
                                return [
9400
                                        execResult[ 1 ] * 2.55,
9401
                                        execResult[ 2 ] * 2.55,
9402
                                        execResult[ 3 ] * 2.55,
9403
                                        execResult[ 4 ]
9404
                                ];
9405
                        }
9406
                }, {
9407
                        // this regex ignores A-F because it's compared against an already lowercased string
9408
                        re: /#([a-f0-9]{2})([a-f0-9]{2})([a-f0-9]{2})/,
9409
                        parse: function( execResult ) {
9410
                                return [
9411
                                        parseInt( execResult[ 1 ], 16 ),
9412
                                        parseInt( execResult[ 2 ], 16 ),
9413
                                        parseInt( execResult[ 3 ], 16 )
9414
                                ];
9415
                        }
9416
                }, {
9417
                        // this regex ignores A-F because it's compared against an already lowercased string
9418
                        re: /#([a-f0-9])([a-f0-9])([a-f0-9])/,
9419
                        parse: function( execResult ) {
9420
                                return [
9421
                                        parseInt( execResult[ 1 ] + execResult[ 1 ], 16 ),
9422
                                        parseInt( execResult[ 2 ] + execResult[ 2 ], 16 ),
9423
                                        parseInt( execResult[ 3 ] + execResult[ 3 ], 16 )
9424
                                ];
9425
                        }
9426
                }, {
9427
                        re: /hsla?\(\s*(\d+(?:\.\d+)?)\s*,\s*(\d+(?:\.\d+)?)\%\s*,\s*(\d+(?:\.\d+)?)\%\s*(?:,\s*(\d?(?:\.\d+)?)\s*)?\)/,
9428
                        space: "hsla",
9429
                        parse: function( execResult ) {
9430
                                return [
9431
                                        execResult[ 1 ],
9432
                                        execResult[ 2 ] / 100,
9433
                                        execResult[ 3 ] / 100,
9434
                                        execResult[ 4 ]
9435
                                ];
9436
                        }
9437
                } ],
9438
9439
        // jQuery.Color( )
9440
        color = jQuery.Color = function( color, green, blue, alpha ) {
9441
                return new jQuery.Color.fn.parse( color, green, blue, alpha );
9442
        },
9443
        spaces = {
9444
                rgba: {
9445
                        props: {
9446
                                red: {
9447
                                        idx: 0,
9448
                                        type: "byte"
9449
                                },
9450
                                green: {
9451
                                        idx: 1,
9452
                                        type: "byte"
9453
                                },
9454
                                blue: {
9455
                                        idx: 2,
9456
                                        type: "byte"
9457
                                }
9458
                        }
9459
                },
9460
9461
                hsla: {
9462
                        props: {
9463
                                hue: {
9464
                                        idx: 0,
9465
                                        type: "degrees"
9466
                                },
9467
                                saturation: {
9468
                                        idx: 1,
9469
                                        type: "percent"
9470
                                },
9471
                                lightness: {
9472
                                        idx: 2,
9473
                                        type: "percent"
9474
                                }
9475
                        }
9476
                }
9477
        },
9478
        propTypes = {
9479
                "byte": {
9480
                        floor: true,
9481
                        max: 255
9482
                },
9483
                "percent": {
9484
                        max: 1
9485
                },
9486
                "degrees": {
9487
                        mod: 360,
9488
                        floor: true
9489
                }
9490
        },
9491
        support = color.support = {},
9492
9493
        // element for support tests
9494
        supportElem = jQuery( "<p>" )[ 0 ],
9495
9496
        // colors = jQuery.Color.names
9497
        colors,
9498
9499
        // local aliases of functions called often
9500
        each = jQuery.each;
9501
9502
// determine rgba support immediately
9503
supportElem.style.cssText = "background-color:rgba(1,1,1,.5)";
9504
support.rgba = supportElem.style.backgroundColor.indexOf( "rgba" ) > -1;
9505
9506
// define cache name and alpha properties
9507
// for rgba and hsla spaces
9508
each( spaces, function( spaceName, space ) {
9509
        space.cache = "_" + spaceName;
9510
        space.props.alpha = {
9511
                idx: 3,
9512
                type: "percent",
9513
                def: 1
9514
        };
9515
});
9516
9517
function clamp( value, prop, allowEmpty ) {
9518
        var type = propTypes[ prop.type ] || {};
9519
9520
        if ( value == null ) {
9521
                return (allowEmpty || !prop.def) ? null : prop.def;
9522
        }
9523
9524
        // ~~ is an short way of doing floor for positive numbers
9525
        value = type.floor ? ~~value : parseFloat( value );
9526
9527
        // IE will pass in empty strings as value for alpha,
9528
        // which will hit this case
9529
        if ( isNaN( value ) ) {
9530
                return prop.def;
9531
        }
9532
9533
        if ( type.mod ) {
9534
                // we add mod before modding to make sure that negatives values
9535
                // get converted properly: -10 -> 350
9536
                return (value + type.mod) % type.mod;
9537
        }
9538
9539
        // for now all property types without mod have min and max
9540
        return 0 > value ? 0 : type.max < value ? type.max : value;
9541
}
9542
9543
function stringParse( string ) {
9544
        var inst = color(),
9545
                rgba = inst._rgba = [];
9546
9547
        string = string.toLowerCase();
9548
9549
        each( stringParsers, function( i, parser ) {
9550
                var parsed,
9551
                        match = parser.re.exec( string ),
9552
                        values = match && parser.parse( match ),
9553
                        spaceName = parser.space || "rgba";
9554
9555
                if ( values ) {
9556
                        parsed = inst[ spaceName ]( values );
9557
9558
                        // if this was an rgba parse the assignment might happen twice
9559
                        // oh well....
9560
                        inst[ spaces[ spaceName ].cache ] = parsed[ spaces[ spaceName ].cache ];
9561
                        rgba = inst._rgba = parsed._rgba;
9562
9563
                        // exit each( stringParsers ) here because we matched
9564
                        return false;
9565
                }
9566
        });
9567
9568
        // Found a stringParser that handled it
9569
        if ( rgba.length ) {
9570
9571
                // if this came from a parsed string, force "transparent" when alpha is 0
9572
                // chrome, (and maybe others) return "transparent" as rgba(0,0,0,0)
9573
                if ( rgba.join() === "0,0,0,0" ) {
9574
                        jQuery.extend( rgba, colors.transparent );
9575
                }
9576
                return inst;
9577
        }
9578
9579
        // named colors
9580
        return colors[ string ];
9581
}
9582
9583
color.fn = jQuery.extend( color.prototype, {
9584
        parse: function( red, green, blue, alpha ) {
9585
                if ( red === undefined ) {
9586
                        this._rgba = [ null, null, null, null ];
9587
                        return this;
9588
                }
9589
                if ( red.jquery || red.nodeType ) {
9590
                        red = jQuery( red ).css( green );
9591
                        green = undefined;
9592
                }
9593
9594
                var inst = this,
9595
                        type = jQuery.type( red ),
9596
                        rgba = this._rgba = [];
9597
9598
                // more than 1 argument specified - assume ( red, green, blue, alpha )
9599
                if ( green !== undefined ) {
9600
                        red = [ red, green, blue, alpha ];
9601
                        type = "array";
9602
                }
9603
9604
                if ( type === "string" ) {
9605
                        return this.parse( stringParse( red ) || colors._default );
9606
                }
9607
9608
                if ( type === "array" ) {
9609
                        each( spaces.rgba.props, function( key, prop ) {
9610
                                rgba[ prop.idx ] = clamp( red[ prop.idx ], prop );
9611
                        });
9612
                        return this;
9613
                }
9614
9615
                if ( type === "object" ) {
9616
                        if ( red instanceof color ) {
9617
                                each( spaces, function( spaceName, space ) {
9618
                                        if ( red[ space.cache ] ) {
9619
                                                inst[ space.cache ] = red[ space.cache ].slice();
9620
                                        }
9621
                                });
9622
                        } else {
9623
                                each( spaces, function( spaceName, space ) {
9624
                                        var cache = space.cache;
9625
                                        each( space.props, function( key, prop ) {
9626
9627
                                                // if the cache doesn't exist, and we know how to convert
9628
                                                if ( !inst[ cache ] && space.to ) {
9629
9630
                                                        // if the value was null, we don't need to copy it
9631
                                                        // if the key was alpha, we don't need to copy it either
9632
                                                        if ( key === "alpha" || red[ key ] == null ) {
9633
                                                                return;
9634
                                                        }
9635
                                                        inst[ cache ] = space.to( inst._rgba );
9636
                                                }
9637
9638
                                                // this is the only case where we allow nulls for ALL properties.
9639
                                                // call clamp with alwaysAllowEmpty
9640
                                                inst[ cache ][ prop.idx ] = clamp( red[ key ], prop, true );
9641
                                        });
9642
9643
                                        // everything defined but alpha?
9644
                                        if ( inst[ cache ] && jQuery.inArray( null, inst[ cache ].slice( 0, 3 ) ) < 0 ) {
9645
                                                // use the default of 1
9646
                                                inst[ cache ][ 3 ] = 1;
9647
                                                if ( space.from ) {
9648
                                                        inst._rgba = space.from( inst[ cache ] );
9649
                                                }
9650
                                        }
9651
                                });
9652
                        }
9653
                        return this;
9654
                }
9655
        },
9656
        is: function( compare ) {
9657
                var is = color( compare ),
9658
                        same = true,
9659
                        inst = this;
9660
9661
                each( spaces, function( _, space ) {
9662
                        var localCache,
9663
                                isCache = is[ space.cache ];
9664
                        if (isCache) {
9665
                                localCache = inst[ space.cache ] || space.to && space.to( inst._rgba ) || [];
9666
                                each( space.props, function( _, prop ) {
9667
                                        if ( isCache[ prop.idx ] != null ) {
9668
                                                same = ( isCache[ prop.idx ] === localCache[ prop.idx ] );
9669
                                                return same;
9670
                                        }
9671
                                });
9672
                        }
9673
                        return same;
9674
                });
9675
                return same;
9676
        },
9677
        _space: function() {
9678
                var used = [],
9679
                        inst = this;
9680
                each( spaces, function( spaceName, space ) {
9681
                        if ( inst[ space.cache ] ) {
9682
                                used.push( spaceName );
9683
                        }
9684
                });
9685
                return used.pop();
9686
        },
9687
        transition: function( other, distance ) {
9688
                var end = color( other ),
9689
                        spaceName = end._space(),
9690
                        space = spaces[ spaceName ],
9691
                        startColor = this.alpha() === 0 ? color( "transparent" ) : this,
9692
                        start = startColor[ space.cache ] || space.to( startColor._rgba ),
9693
                        result = start.slice();
9694
9695
                end = end[ space.cache ];
9696
                each( space.props, function( key, prop ) {
9697
                        var index = prop.idx,
9698
                                startValue = start[ index ],
9699
                                endValue = end[ index ],
9700
                                type = propTypes[ prop.type ] || {};
9701
9702
                        // if null, don't override start value
9703
                        if ( endValue === null ) {
9704
                                return;
9705
                        }
9706
                        // if null - use end
9707
                        if ( startValue === null ) {
9708
                                result[ index ] = endValue;
9709
                        } else {
9710
                                if ( type.mod ) {
9711
                                        if ( endValue - startValue > type.mod / 2 ) {
9712
                                                startValue += type.mod;
9713
                                        } else if ( startValue - endValue > type.mod / 2 ) {
9714
                                                startValue -= type.mod;
9715
                                        }
9716
                                }
9717
                                result[ index ] = clamp( ( endValue - startValue ) * distance + startValue, prop );
9718
                        }
9719
                });
9720
                return this[ spaceName ]( result );
9721
        },
9722
        blend: function( opaque ) {
9723
                // if we are already opaque - return ourself
9724
                if ( this._rgba[ 3 ] === 1 ) {
9725
                        return this;
9726
                }
9727
9728
                var rgb = this._rgba.slice(),
9729
                        a = rgb.pop(),
9730
                        blend = color( opaque )._rgba;
9731
9732
                return color( jQuery.map( rgb, function( v, i ) {
9733
                        return ( 1 - a ) * blend[ i ] + a * v;
9734
                }));
9735
        },
9736
        toRgbaString: function() {
9737
                var prefix = "rgba(",
9738
                        rgba = jQuery.map( this._rgba, function( v, i ) {
9739
                                return v == null ? ( i > 2 ? 1 : 0 ) : v;
9740
                        });
9741
9742
                if ( rgba[ 3 ] === 1 ) {
9743
                        rgba.pop();
9744
                        prefix = "rgb(";
9745
                }
9746
9747
                return prefix + rgba.join() + ")";
9748
        },
9749
        toHslaString: function() {
9750
                var prefix = "hsla(",
9751
                        hsla = jQuery.map( this.hsla(), function( v, i ) {
9752
                                if ( v == null ) {
9753
                                        v = i > 2 ? 1 : 0;
9754
                                }
9755
9756
                                // catch 1 and 2
9757
                                if ( i && i < 3 ) {
9758
                                        v = Math.round( v * 100 ) + "%";
9759
                                }
9760
                                return v;
9761
                        });
9762
9763
                if ( hsla[ 3 ] === 1 ) {
9764
                        hsla.pop();
9765
                        prefix = "hsl(";
9766
                }
9767
                return prefix + hsla.join() + ")";
9768
        },
9769
        toHexString: function( includeAlpha ) {
9770
                var rgba = this._rgba.slice(),
9771
                        alpha = rgba.pop();
9772
9773
                if ( includeAlpha ) {
9774
                        rgba.push( ~~( alpha * 255 ) );
9775
                }
9776
9777
                return "#" + jQuery.map( rgba, function( v ) {
9778
9779
                        // default to 0 when nulls exist
9780
                        v = ( v || 0 ).toString( 16 );
9781
                        return v.length === 1 ? "0" + v : v;
9782
                }).join("");
9783
        },
9784
        toString: function() {
9785
                return this._rgba[ 3 ] === 0 ? "transparent" : this.toRgbaString();
9786
        }
9787
});
9788
color.fn.parse.prototype = color.fn;
9789
9790
// hsla conversions adapted from:
9791
// https://code.google.com/p/maashaack/source/browse/packages/graphics/trunk/src/graphics/colors/HUE2RGB.as?r=5021
9792
9793
function hue2rgb( p, q, h ) {
9794
        h = ( h + 1 ) % 1;
9795
        if ( h * 6 < 1 ) {
9796
                return p + ( q - p ) * h * 6;
9797
        }
9798
        if ( h * 2 < 1) {
9799
                return q;
9800
        }
9801
        if ( h * 3 < 2 ) {
9802
                return p + ( q - p ) * ( ( 2 / 3 ) - h ) * 6;
9803
        }
9804
        return p;
9805
}
9806
9807
spaces.hsla.to = function( rgba ) {
9808
        if ( rgba[ 0 ] == null || rgba[ 1 ] == null || rgba[ 2 ] == null ) {
9809
                return [ null, null, null, rgba[ 3 ] ];
9810
        }
9811
        var r = rgba[ 0 ] / 255,
9812
                g = rgba[ 1 ] / 255,
9813
                b = rgba[ 2 ] / 255,
9814
                a = rgba[ 3 ],
9815
                max = Math.max( r, g, b ),
9816
                min = Math.min( r, g, b ),
9817
                diff = max - min,
9818
                add = max + min,
9819
                l = add * 0.5,
9820
                h, s;
9821
9822
        if ( min === max ) {
9823
                h = 0;
9824
        } else if ( r === max ) {
9825
                h = ( 60 * ( g - b ) / diff ) + 360;
9826
        } else if ( g === max ) {
9827
                h = ( 60 * ( b - r ) / diff ) + 120;
9828
        } else {
9829
                h = ( 60 * ( r - g ) / diff ) + 240;
9830
        }
9831
9832
        // chroma (diff) == 0 means greyscale which, by definition, saturation = 0%
9833
        // otherwise, saturation is based on the ratio of chroma (diff) to lightness (add)
9834
        if ( diff === 0 ) {
9835
                s = 0;
9836
        } else if ( l <= 0.5 ) {
9837
                s = diff / add;
9838
        } else {
9839
                s = diff / ( 2 - add );
9840
        }
9841
        return [ Math.round(h) % 360, s, l, a == null ? 1 : a ];
9842
};
9843
9844
spaces.hsla.from = function( hsla ) {
9845
        if ( hsla[ 0 ] == null || hsla[ 1 ] == null || hsla[ 2 ] == null ) {
9846
                return [ null, null, null, hsla[ 3 ] ];
9847
        }
9848
        var h = hsla[ 0 ] / 360,
9849
                s = hsla[ 1 ],
9850
                l = hsla[ 2 ],
9851
                a = hsla[ 3 ],
9852
                q = l <= 0.5 ? l * ( 1 + s ) : l + s - l * s,
9853
                p = 2 * l - q;
9854
9855
        return [
9856
                Math.round( hue2rgb( p, q, h + ( 1 / 3 ) ) * 255 ),
9857
                Math.round( hue2rgb( p, q, h ) * 255 ),
9858
                Math.round( hue2rgb( p, q, h - ( 1 / 3 ) ) * 255 ),
9859
                a
9860
        ];
9861
};
9862
9863
each( spaces, function( spaceName, space ) {
9864
        var props = space.props,
9865
                cache = space.cache,
9866
                to = space.to,
9867
                from = space.from;
9868
9869
        // makes rgba() and hsla()
9870
        color.fn[ spaceName ] = function( value ) {
9871
9872
                // generate a cache for this space if it doesn't exist
9873
                if ( to && !this[ cache ] ) {
9874
                        this[ cache ] = to( this._rgba );
9875
                }
9876
                if ( value === undefined ) {
9877
                        return this[ cache ].slice();
9878
                }
9879
9880
                var ret,
9881
                        type = jQuery.type( value ),
9882
                        arr = ( type === "array" || type === "object" ) ? value : arguments,
9883
                        local = this[ cache ].slice();
9884
9885
                each( props, function( key, prop ) {
9886
                        var val = arr[ type === "object" ? key : prop.idx ];
9887
                        if ( val == null ) {
9888
                                val = local[ prop.idx ];
9889
                        }
9890
                        local[ prop.idx ] = clamp( val, prop );
9891
                });
9892
9893
                if ( from ) {
9894
                        ret = color( from( local ) );
9895
                        ret[ cache ] = local;
9896
                        return ret;
9897
                } else {
9898
                        return color( local );
9899
                }
9900
        };
9901
9902
        // makes red() green() blue() alpha() hue() saturation() lightness()
9903
        each( props, function( key, prop ) {
9904
                // alpha is included in more than one space
9905
                if ( color.fn[ key ] ) {
9906
                        return;
9907
                }
9908
                color.fn[ key ] = function( value ) {
9909
                        var vtype = jQuery.type( value ),
9910
                                fn = ( key === "alpha" ? ( this._hsla ? "hsla" : "rgba" ) : spaceName ),
9911
                                local = this[ fn ](),
9912
                                cur = local[ prop.idx ],
9913
                                match;
9914
9915
                        if ( vtype === "undefined" ) {
9916
                                return cur;
9917
                        }
9918
9919
                        if ( vtype === "function" ) {
9920
                                value = value.call( this, cur );
9921
                                vtype = jQuery.type( value );
9922
                        }
9923
                        if ( value == null && prop.empty ) {
9924
                                return this;
9925
                        }
9926
                        if ( vtype === "string" ) {
9927
                                match = rplusequals.exec( value );
9928
                                if ( match ) {
9929
                                        value = cur + parseFloat( match[ 2 ] ) * ( match[ 1 ] === "+" ? 1 : -1 );
9930
                                }
9931
                        }
9932
                        local[ prop.idx ] = value;
9933
                        return this[ fn ]( local );
9934
                };
9935
        });
9936
});
9937
9938
// add cssHook and .fx.step function for each named hook.
9939
// accept a space separated string of properties
9940
color.hook = function( hook ) {
9941
        var hooks = hook.split( " " );
9942
        each( hooks, function( i, hook ) {
9943
                jQuery.cssHooks[ hook ] = {
9944
                        set: function( elem, value ) {
9945
                                var parsed, curElem,
9946
                                        backgroundColor = "";
9947
9948
                                if ( value !== "transparent" && ( jQuery.type( value ) !== "string" || ( parsed = stringParse( value ) ) ) ) {
9949
                                        value = color( parsed || value );
9950
                                        if ( !support.rgba && value._rgba[ 3 ] !== 1 ) {
9951
                                                curElem = hook === "backgroundColor" ? elem.parentNode : elem;
9952
                                                while (
9953
                                                        (backgroundColor === "" || backgroundColor === "transparent") &&
9954
                                                        curElem && curElem.style
9955
                                                ) {
9956
                                                        try {
9957
                                                                backgroundColor = jQuery.css( curElem, "backgroundColor" );
9958
                                                                curElem = curElem.parentNode;
9959
                                                        } catch ( e ) {
9960
                                                        }
9961
                                                }
9962
9963
                                                value = value.blend( backgroundColor && backgroundColor !== "transparent" ?
9964
                                                        backgroundColor :
9965
                                                        "_default" );
9966
                                        }
9967
9968
                                        value = value.toRgbaString();
9969
                                }
9970
                                try {
9971
                                        elem.style[ hook ] = value;
9972
                                } catch ( e ) {
9973
                                        // wrapped to prevent IE from throwing errors on "invalid" values like 'auto' or 'inherit'
9974
                                }
9975
                        }
9976
                };
9977
                jQuery.fx.step[ hook ] = function( fx ) {
9978
                        if ( !fx.colorInit ) {
9979
                                fx.start = color( fx.elem, hook );
9980
                                fx.end = color( fx.end );
9981
                                fx.colorInit = true;
9982
                        }
9983
                        jQuery.cssHooks[ hook ].set( fx.elem, fx.start.transition( fx.end, fx.pos ) );
9984
                };
9985
        });
9986
9987
};
9988
9989
color.hook( stepHooks );
9990
9991
jQuery.cssHooks.borderColor = {
9992
        expand: function( value ) {
9993
                var expanded = {};
9994
9995
                each( [ "Top", "Right", "Bottom", "Left" ], function( i, part ) {
9996
                        expanded[ "border" + part + "Color" ] = value;
9997
                });
9998
                return expanded;
9999
        }
10000
};
10001
10002
// Basic color names only.
10003
// Usage of any of the other color names requires adding yourself or including
10004
// jquery.color.svg-names.js.
10005
colors = jQuery.Color.names = {
10006
        // 4.1. Basic color keywords
10007
        aqua: "#00ffff",
10008
        black: "#000000",
10009
        blue: "#0000ff",
10010
        fuchsia: "#ff00ff",
10011
        gray: "#808080",
10012
        green: "#008000",
10013
        lime: "#00ff00",
10014
        maroon: "#800000",
10015
        navy: "#000080",
10016
        olive: "#808000",
10017
        purple: "#800080",
10018
        red: "#ff0000",
10019
        silver: "#c0c0c0",
10020
        teal: "#008080",
10021
        white: "#ffffff",
10022
        yellow: "#ffff00",
10023
10024
        // 4.2.3. "transparent" color keyword
10025
        transparent: [ null, null, null, 0 ],
10026
10027
        _default: "#ffffff"
10028
};
10029
10030
})( jQuery );
10031
10032
/******************************************************************************/
10033
/****************************** CLASS ANIMATIONS ******************************/
10034
/******************************************************************************/
10035
(function() {
10036
10037
var classAnimationActions = [ "add", "remove", "toggle" ],
10038
        shorthandStyles = {
10039
                border: 1,
10040
                borderBottom: 1,
10041
                borderColor: 1,
10042
                borderLeft: 1,
10043
                borderRight: 1,
10044
                borderTop: 1,
10045
                borderWidth: 1,
10046
                margin: 1,
10047
                padding: 1
10048
        };
10049
10050
$.each([ "borderLeftStyle", "borderRightStyle", "borderBottomStyle", "borderTopStyle" ], function( _, prop ) {
10051
        $.fx.step[ prop ] = function( fx ) {
10052
                if ( fx.end !== "none" && !fx.setAttr || fx.pos === 1 && !fx.setAttr ) {
10053
                        jQuery.style( fx.elem, prop, fx.end );
10054
                        fx.setAttr = true;
10055
                }
10056
        };
10057
});
10058
10059
function getElementStyles( elem ) {
10060
        var key, len,
10061
                style = elem.ownerDocument.defaultView ?
10062
                        elem.ownerDocument.defaultView.getComputedStyle( elem, null ) :
10063
                        elem.currentStyle,
10064
                styles = {};
10065
10066
        if ( style && style.length && style[ 0 ] && style[ style[ 0 ] ] ) {
10067
                len = style.length;
10068
                while ( len-- ) {
10069
                        key = style[ len ];
10070
                        if ( typeof style[ key ] === "string" ) {
10071
                                styles[ $.camelCase( key ) ] = style[ key ];
10072
                        }
10073
                }
10074
        // support: Opera, IE <9
10075
        } else {
10076
                for ( key in style ) {
10077
                        if ( typeof style[ key ] === "string" ) {
10078
                                styles[ key ] = style[ key ];
10079
                        }
10080
                }
10081
        }
10082
10083
        return styles;
10084
}
10085
10086
function styleDifference( oldStyle, newStyle ) {
10087
        var diff = {},
10088
                name, value;
10089
10090
        for ( name in newStyle ) {
10091
                value = newStyle[ name ];
10092
                if ( oldStyle[ name ] !== value ) {
10093
                        if ( !shorthandStyles[ name ] ) {
10094
                                if ( $.fx.step[ name ] || !isNaN( parseFloat( value ) ) ) {
10095
                                        diff[ name ] = value;
10096
                                }
10097
                        }
10098
                }
10099
        }
10100
10101
        return diff;
10102
}
10103
10104
// support: jQuery <1.8
10105
if ( !$.fn.addBack ) {
10106
        $.fn.addBack = function( selector ) {
10107
                return this.add( selector == null ?
10108
                        this.prevObject : this.prevObject.filter( selector )
10109
                );
10110
        };
10111
}
10112
10113
$.effects.animateClass = function( value, duration, easing, callback ) {
10114
        var o = $.speed( duration, easing, callback );
10115
10116
        return this.queue( function() {
10117
                var animated = $( this ),
10118
                        baseClass = animated.attr( "class" ) || "",
10119
                        applyClassChange,
10120
                        allAnimations = o.children ? animated.find( "*" ).addBack() : animated;
10121
10122
                // map the animated objects to store the original styles.
10123
                allAnimations = allAnimations.map(function() {
10124
                        var el = $( this );
10125
                        return {
10126
                                el: el,
10127
                                start: getElementStyles( this )
10128
                        };
10129
                });
10130
10131
                // apply class change
10132
                applyClassChange = function() {
10133
                        $.each( classAnimationActions, function(i, action) {
10134
                                if ( value[ action ] ) {
10135
                                        animated[ action + "Class" ]( value[ action ] );
10136
                                }
10137
                        });
10138
                };
10139
                applyClassChange();
10140
10141
                // map all animated objects again - calculate new styles and diff
10142
                allAnimations = allAnimations.map(function() {
10143
                        this.end = getElementStyles( this.el[ 0 ] );
10144
                        this.diff = styleDifference( this.start, this.end );
10145
                        return this;
10146
                });
10147
10148
                // apply original class
10149
                animated.attr( "class", baseClass );
10150
10151
                // map all animated objects again - this time collecting a promise
10152
                allAnimations = allAnimations.map(function() {
10153
                        var styleInfo = this,
10154
                                dfd = $.Deferred(),
10155
                                opts = $.extend({}, o, {
10156
                                        queue: false,
10157
                                        complete: function() {
10158
                                                dfd.resolve( styleInfo );
10159
                                        }
10160
                                });
10161
10162
                        this.el.animate( this.diff, opts );
10163
                        return dfd.promise();
10164
                });
10165
10166
                // once all animations have completed:
10167
                $.when.apply( $, allAnimations.get() ).done(function() {
10168
10169
                        // set the final class
10170
                        applyClassChange();
10171
10172
                        // for each animated element,
10173
                        // clear all css properties that were animated
10174
                        $.each( arguments, function() {
10175
                                var el = this.el;
10176
                                $.each( this.diff, function(key) {
10177
                                        el.css( key, "" );
10178
                                });
10179
                        });
10180
10181
                        // this is guarnteed to be there if you use jQuery.speed()
10182
                        // it also handles dequeuing the next anim...
10183
                        o.complete.call( animated[ 0 ] );
10184
                });
10185
        });
10186
};
10187
10188
$.fn.extend({
10189
        addClass: (function( orig ) {
10190
                return function( classNames, speed, easing, callback ) {
10191
                        return speed ?
10192
                                $.effects.animateClass.call( this,
10193
                                        { add: classNames }, speed, easing, callback ) :
10194
                                orig.apply( this, arguments );
10195
                };
10196
        })( $.fn.addClass ),
10197
10198
        removeClass: (function( orig ) {
10199
                return function( classNames, speed, easing, callback ) {
10200
                        return arguments.length > 1 ?
10201
                                $.effects.animateClass.call( this,
10202
                                        { remove: classNames }, speed, easing, callback ) :
10203
                                orig.apply( this, arguments );
10204
                };
10205
        })( $.fn.removeClass ),
10206
10207
        toggleClass: (function( orig ) {
10208
                return function( classNames, force, speed, easing, callback ) {
10209
                        if ( typeof force === "boolean" || force === undefined ) {
10210
                                if ( !speed ) {
10211
                                        // without speed parameter
10212
                                        return orig.apply( this, arguments );
10213
                                } else {
10214
                                        return $.effects.animateClass.call( this,
10215
                                                (force ? { add: classNames } : { remove: classNames }),
10216
                                                speed, easing, callback );
10217
                                }
10218
                        } else {
10219
                                // without force parameter
10220
                                return $.effects.animateClass.call( this,
10221
                                        { toggle: classNames }, force, speed, easing );
10222
                        }
10223
                };
10224
        })( $.fn.toggleClass ),
10225
10226
        switchClass: function( remove, add, speed, easing, callback) {
10227
                return $.effects.animateClass.call( this, {
10228
                        add: add,
10229
                        remove: remove
10230
                }, speed, easing, callback );
10231
        }
10232
});
10233
10234
})();
10235
10236
/******************************************************************************/
10237
/*********************************** EFFECTS **********************************/
10238
/******************************************************************************/
10239
10240
(function() {
10241
10242
$.extend( $.effects, {
10243
        version: "1.11.4",
10244
10245
        // Saves a set of properties in a data storage
10246
        save: function( element, set ) {
10247
                for ( var i = 0; i < set.length; i++ ) {
10248
                        if ( set[ i ] !== null ) {
10249
                                element.data( dataSpace + set[ i ], element[ 0 ].style[ set[ i ] ] );
10250
                        }
10251
                }
10252
        },
10253
10254
        // Restores a set of previously saved properties from a data storage
10255
        restore: function( element, set ) {
10256
                var val, i;
10257
                for ( i = 0; i < set.length; i++ ) {
10258
                        if ( set[ i ] !== null ) {
10259
                                val = element.data( dataSpace + set[ i ] );
10260
                                // support: jQuery 1.6.2
10261
                                // http://bugs.jquery.com/ticket/9917
10262
                                // jQuery 1.6.2 incorrectly returns undefined for any falsy value.
10263
                                // We can't differentiate between "" and 0 here, so we just assume
10264
                                // empty string since it's likely to be a more common value...
10265
                                if ( val === undefined ) {
10266
                                        val = "";
10267
                                }
10268
                                element.css( set[ i ], val );
10269
                        }
10270
                }
10271
        },
10272
10273
        setMode: function( el, mode ) {
10274
                if (mode === "toggle") {
10275
                        mode = el.is( ":hidden" ) ? "show" : "hide";
10276
                }
10277
                return mode;
10278
        },
10279
10280
        // Translates a [top,left] array into a baseline value
10281
        // this should be a little more flexible in the future to handle a string & hash
10282
        getBaseline: function( origin, original ) {
10283
                var y, x;
10284
                switch ( origin[ 0 ] ) {
10285
                        case "top": y = 0; break;
10286
                        case "middle": y = 0.5; break;
10287
                        case "bottom": y = 1; break;
10288
                        default: y = origin[ 0 ] / original.height;
10289
                }
10290
                switch ( origin[ 1 ] ) {
10291
                        case "left": x = 0; break;
10292
                        case "center": x = 0.5; break;
10293
                        case "right": x = 1; break;
10294
                        default: x = origin[ 1 ] / original.width;
10295
                }
10296
                return {
10297
                        x: x,
10298
                        y: y
10299
                };
10300
        },
10301
10302
        // Wraps the element around a wrapper that copies position properties
10303
        createWrapper: function( element ) {
10304
10305
                // if the element is already wrapped, return it
10306
                if ( element.parent().is( ".ui-effects-wrapper" )) {
10307
                        return element.parent();
10308
                }
10309
10310
                // wrap the element
10311
                var props = {
10312
                                width: element.outerWidth(true),
10313
                                height: element.outerHeight(true),
10314
                                "float": element.css( "float" )
10315
                        },
10316
                        wrapper = $( "<div></div>" )
10317
                                .addClass( "ui-effects-wrapper" )
10318
                                .css({
10319
                                        fontSize: "100%",
10320
                                        background: "transparent",
10321
                                        border: "none",
10322
                                        margin: 0,
10323
                                        padding: 0
10324
                                }),
10325
                        // Store the size in case width/height are defined in % - Fixes #5245
10326
                        size = {
10327
                                width: element.width(),
10328
                                height: element.height()
10329
                        },
10330
                        active = document.activeElement;
10331
10332
                // support: Firefox
10333
                // Firefox incorrectly exposes anonymous content
10334
                // https://bugzilla.mozilla.org/show_bug.cgi?id=561664
10335
                try {
10336
                        active.id;
10337
                } catch ( e ) {
10338
                        active = document.body;
10339
                }
10340
10341
                element.wrap( wrapper );
10342
10343
                // Fixes #7595 - Elements lose focus when wrapped.
10344
                if ( element[ 0 ] === active || $.contains( element[ 0 ], active ) ) {
10345
                        $( active ).focus();
10346
                }
10347
10348
                wrapper = element.parent(); //Hotfix for jQuery 1.4 since some change in wrap() seems to actually lose the reference to the wrapped element
10349
10350
                // transfer positioning properties to the wrapper
10351
                if ( element.css( "position" ) === "static" ) {
10352
                        wrapper.css({ position: "relative" });
10353
                        element.css({ position: "relative" });
10354
                } else {
10355
                        $.extend( props, {
10356
                                position: element.css( "position" ),
10357
                                zIndex: element.css( "z-index" )
10358
                        });
10359
                        $.each([ "top", "left", "bottom", "right" ], function(i, pos) {
10360
                                props[ pos ] = element.css( pos );
10361
                                if ( isNaN( parseInt( props[ pos ], 10 ) ) ) {
10362
                                        props[ pos ] = "auto";
10363
                                }
10364
                        });
10365
                        element.css({
10366
                                position: "relative",
10367
                                top: 0,
10368
                                left: 0,
10369
                                right: "auto",
10370
                                bottom: "auto"
10371
                        });
10372
                }
10373
                element.css(size);
10374
10375
                return wrapper.css( props ).show();
10376
        },
10377
10378
        removeWrapper: function( element ) {
10379
                var active = document.activeElement;
10380
10381
                if ( element.parent().is( ".ui-effects-wrapper" ) ) {
10382
                        element.parent().replaceWith( element );
10383
10384
                        // Fixes #7595 - Elements lose focus when wrapped.
10385
                        if ( element[ 0 ] === active || $.contains( element[ 0 ], active ) ) {
10386
                                $( active ).focus();
10387
                        }
10388
                }
10389
10390
                return element;
10391
        },
10392
10393
        setTransition: function( element, list, factor, value ) {
10394
                value = value || {};
10395
                $.each( list, function( i, x ) {
10396
                        var unit = element.cssUnit( x );
10397
                        if ( unit[ 0 ] > 0 ) {
10398
                                value[ x ] = unit[ 0 ] * factor + unit[ 1 ];
10399
                        }
10400
                });
10401
                return value;
10402
        }
10403
});
10404
10405
// return an effect options object for the given parameters:
10406
function _normalizeArguments( effect, options, speed, callback ) {
10407
10408
        // allow passing all options as the first parameter
10409
        if ( $.isPlainObject( effect ) ) {
10410
                options = effect;
10411
                effect = effect.effect;
10412
        }
10413
10414
        // convert to an object
10415
        effect = { effect: effect };
10416
10417
        // catch (effect, null, ...)
10418
        if ( options == null ) {
10419
                options = {};
10420
        }
10421
10422
        // catch (effect, callback)
10423
        if ( $.isFunction( options ) ) {
10424
                callback = options;
10425
                speed = null;
10426
                options = {};
10427
        }
10428
10429
        // catch (effect, speed, ?)
10430
        if ( typeof options === "number" || $.fx.speeds[ options ] ) {
10431
                callback = speed;
10432
                speed = options;
10433
                options = {};
10434
        }
10435
10436
        // catch (effect, options, callback)
10437
        if ( $.isFunction( speed ) ) {
10438
                callback = speed;
10439
                speed = null;
10440
        }
10441
10442
        // add options to effect
10443
        if ( options ) {
10444
                $.extend( effect, options );
10445
        }
10446
10447
        speed = speed || options.duration;
10448
        effect.duration = $.fx.off ? 0 :
10449
                typeof speed === "number" ? speed :
10450
                speed in $.fx.speeds ? $.fx.speeds[ speed ] :
10451
                $.fx.speeds._default;
10452
10453
        effect.complete = callback || options.complete;
10454
10455
        return effect;
10456
}
10457
10458
function standardAnimationOption( option ) {
10459
        // Valid standard speeds (nothing, number, named speed)
10460
        if ( !option || typeof option === "number" || $.fx.speeds[ option ] ) {
10461
                return true;
10462
        }
10463
10464
        // Invalid strings - treat as "normal" speed
10465
        if ( typeof option === "string" && !$.effects.effect[ option ] ) {
10466
                return true;
10467
        }
10468
10469
        // Complete callback
10470
        if ( $.isFunction( option ) ) {
10471
                return true;
10472
        }
10473
10474
        // Options hash (but not naming an effect)
10475
        if ( typeof option === "object" && !option.effect ) {
10476
                return true;
10477
        }
10478
10479
        // Didn't match any standard API
10480
        return false;
10481
}
10482
10483
$.fn.extend({
10484
        effect: function( /* effect, options, speed, callback */ ) {
10485
                var args = _normalizeArguments.apply( this, arguments ),
10486
                        mode = args.mode,
10487
                        queue = args.queue,
10488
                        effectMethod = $.effects.effect[ args.effect ];
10489
10490
                if ( $.fx.off || !effectMethod ) {
10491
                        // delegate to the original method (e.g., .show()) if possible
10492
                        if ( mode ) {
10493
                                return this[ mode ]( args.duration, args.complete );
10494
                        } else {
10495
                                return this.each( function() {
10496
                                        if ( args.complete ) {
10497
                                                args.complete.call( this );
10498
                                        }
10499
                                });
10500
                        }
10501
                }
10502
10503
                function run( next ) {
10504
                        var elem = $( this ),
10505
                                complete = args.complete,
10506
                                mode = args.mode;
10507
10508
                        function done() {
10509
                                if ( $.isFunction( complete ) ) {
10510
                                        complete.call( elem[0] );
10511
                                }
10512
                                if ( $.isFunction( next ) ) {
10513
                                        next();
10514
                                }
10515
                        }
10516
10517
                        // If the element already has the correct final state, delegate to
10518
                        // the core methods so the internal tracking of "olddisplay" works.
10519
                        if ( elem.is( ":hidden" ) ? mode === "hide" : mode === "show" ) {
10520
                                elem[ mode ]();
10521
                                done();
10522
                        } else {
10523
                                effectMethod.call( elem[0], args, done );
10524
                        }
10525
                }
10526
10527
                return queue === false ? this.each( run ) : this.queue( queue || "fx", run );
10528
        },
10529
10530
        show: (function( orig ) {
10531
                return function( option ) {
10532
                        if ( standardAnimationOption( option ) ) {
10533
                                return orig.apply( this, arguments );
10534
                        } else {
10535
                                var args = _normalizeArguments.apply( this, arguments );
10536
                                args.mode = "show";
10537
                                return this.effect.call( this, args );
10538
                        }
10539
                };
10540
        })( $.fn.show ),
10541
10542
        hide: (function( orig ) {
10543
                return function( option ) {
10544
                        if ( standardAnimationOption( option ) ) {
10545
                                return orig.apply( this, arguments );
10546
                        } else {
10547
                                var args = _normalizeArguments.apply( this, arguments );
10548
                                args.mode = "hide";
10549
                                return this.effect.call( this, args );
10550
                        }
10551
                };
10552
        })( $.fn.hide ),
10553
10554
        toggle: (function( orig ) {
10555
                return function( option ) {
10556
                        if ( standardAnimationOption( option ) || typeof option === "boolean" ) {
10557
                                return orig.apply( this, arguments );
10558
                        } else {
10559
                                var args = _normalizeArguments.apply( this, arguments );
10560
                                args.mode = "toggle";
10561
                                return this.effect.call( this, args );
10562
                        }
10563
                };
10564
        })( $.fn.toggle ),
10565
10566
        // helper functions
10567
        cssUnit: function(key) {
10568
                var style = this.css( key ),
10569
                        val = [];
10570
10571
                $.each( [ "em", "px", "%", "pt" ], function( i, unit ) {
10572
                        if ( style.indexOf( unit ) > 0 ) {
10573
                                val = [ parseFloat( style ), unit ];
10574
                        }
10575
                });
10576
                return val;
10577
        }
10578
});
10579
10580
})();
10581
10582
/******************************************************************************/
10583
/*********************************** EASING ***********************************/
10584
/******************************************************************************/
10585
10586
(function() {
10587
10588
// based on easing equations from Robert Penner (http://www.robertpenner.com/easing)
10589
10590
var baseEasings = {};
10591
10592
$.each( [ "Quad", "Cubic", "Quart", "Quint", "Expo" ], function( i, name ) {
10593
        baseEasings[ name ] = function( p ) {
10594
                return Math.pow( p, i + 2 );
10595
        };
10596
});
10597
10598
$.extend( baseEasings, {
10599
        Sine: function( p ) {
10600
                return 1 - Math.cos( p * Math.PI / 2 );
10601
        },
10602
        Circ: function( p ) {
10603
                return 1 - Math.sqrt( 1 - p * p );
10604
        },
10605
        Elastic: function( p ) {
10606
                return p === 0 || p === 1 ? p :
10607
                        -Math.pow( 2, 8 * (p - 1) ) * Math.sin( ( (p - 1) * 80 - 7.5 ) * Math.PI / 15 );
10608
        },
10609
        Back: function( p ) {
10610
                return p * p * ( 3 * p - 2 );
10611
        },
10612
        Bounce: function( p ) {
10613
                var pow2,
10614
                        bounce = 4;
10615
10616
                while ( p < ( ( pow2 = Math.pow( 2, --bounce ) ) - 1 ) / 11 ) {}
10617
                return 1 / Math.pow( 4, 3 - bounce ) - 7.5625 * Math.pow( ( pow2 * 3 - 2 ) / 22 - p, 2 );
10618
        }
10619
});
10620
10621
$.each( baseEasings, function( name, easeIn ) {
10622
        $.easing[ "easeIn" + name ] = easeIn;
10623
        $.easing[ "easeOut" + name ] = function( p ) {
10624
                return 1 - easeIn( 1 - p );
10625
        };
10626
        $.easing[ "easeInOut" + name ] = function( p ) {
10627
                return p < 0.5 ?
10628
                        easeIn( p * 2 ) / 2 :
10629
                        1 - easeIn( p * -2 + 2 ) / 2;
10630
        };
10631
});
10632
10633
})();
10634
10635
var effect = $.effects;
10636
10637
10638
/*!
10639
 * jQuery UI Effects Blind 1.11.4
10640
 * http://jqueryui.com
10641
 *
10642
 * Copyright jQuery Foundation and other contributors
10643
 * Released under the MIT license.
10644
 * http://jquery.org/license
10645
 *
10646
 * http://api.jqueryui.com/blind-effect/
10647
 */
10648
10649
10650
var effectBlind = $.effects.effect.blind = function( o, done ) {
10651
        // Create element
10652
        var el = $( this ),
10653
                rvertical = /up|down|vertical/,
10654
                rpositivemotion = /up|left|vertical|horizontal/,
10655
                props = [ "position", "top", "bottom", "left", "right", "height", "width" ],
10656
                mode = $.effects.setMode( el, o.mode || "hide" ),
10657
                direction = o.direction || "up",
10658
                vertical = rvertical.test( direction ),
10659
                ref = vertical ? "height" : "width",
10660
                ref2 = vertical ? "top" : "left",
10661
                motion = rpositivemotion.test( direction ),
10662
                animation = {},
10663
                show = mode === "show",
10664
                wrapper, distance, margin;
10665
10666
        // if already wrapped, the wrapper's properties are my property. #6245
10667
        if ( el.parent().is( ".ui-effects-wrapper" ) ) {
10668
                $.effects.save( el.parent(), props );
10669
        } else {
10670
                $.effects.save( el, props );
10671
        }
10672
        el.show();
10673
        wrapper = $.effects.createWrapper( el ).css({
10674
                overflow: "hidden"
10675
        });
10676
10677
        distance = wrapper[ ref ]();
10678
        margin = parseFloat( wrapper.css( ref2 ) ) || 0;
10679
10680
        animation[ ref ] = show ? distance : 0;
10681
        if ( !motion ) {
10682
                el
10683
                        .css( vertical ? "bottom" : "right", 0 )
10684
                        .css( vertical ? "top" : "left", "auto" )
10685
                        .css({ position: "absolute" });
10686
10687
                animation[ ref2 ] = show ? margin : distance + margin;
10688
        }
10689
10690
        // start at 0 if we are showing
10691
        if ( show ) {
10692
                wrapper.css( ref, 0 );
10693
                if ( !motion ) {
10694
                        wrapper.css( ref2, margin + distance );
10695
                }
10696
        }
10697
10698
        // Animate
10699
        wrapper.animate( animation, {
10700
                duration: o.duration,
10701
                easing: o.easing,
10702
                queue: false,
10703
                complete: function() {
10704
                        if ( mode === "hide" ) {
10705
                                el.hide();
10706
                        }
10707
                        $.effects.restore( el, props );
10708
                        $.effects.removeWrapper( el );
10709
                        done();
10710
                }
10711
        });
10712
};
10713
10714
10715
/*!
10716
 * jQuery UI Effects Bounce 1.11.4
10717
 * http://jqueryui.com
10718
 *
10719
 * Copyright jQuery Foundation and other contributors
10720
 * Released under the MIT license.
10721
 * http://jquery.org/license
10722
 *
10723
 * http://api.jqueryui.com/bounce-effect/
10724
 */
10725
10726
10727
var effectBounce = $.effects.effect.bounce = function( o, done ) {
10728
        var el = $( this ),
10729
                props = [ "position", "top", "bottom", "left", "right", "height", "width" ],
10730
10731
                // defaults:
10732
                mode = $.effects.setMode( el, o.mode || "effect" ),
10733
                hide = mode === "hide",
10734
                show = mode === "show",
10735
                direction = o.direction || "up",
10736
                distance = o.distance,
10737
                times = o.times || 5,
10738
10739
                // number of internal animations
10740
                anims = times * 2 + ( show || hide ? 1 : 0 ),
10741
                speed = o.duration / anims,
10742
                easing = o.easing,
10743
10744
                // utility:
10745
                ref = ( direction === "up" || direction === "down" ) ? "top" : "left",
10746
                motion = ( direction === "up" || direction === "left" ),
10747
                i,
10748
                upAnim,
10749
                downAnim,
10750
10751
                // we will need to re-assemble the queue to stack our animations in place
10752
                queue = el.queue(),
10753
                queuelen = queue.length;
10754
10755
        // Avoid touching opacity to prevent clearType and PNG issues in IE
10756
        if ( show || hide ) {
10757
                props.push( "opacity" );
10758
        }
10759
10760
        $.effects.save( el, props );
10761
        el.show();
10762
        $.effects.createWrapper( el ); // Create Wrapper
10763
10764
        // default distance for the BIGGEST bounce is the outer Distance / 3
10765
        if ( !distance ) {
10766
                distance = el[ ref === "top" ? "outerHeight" : "outerWidth" ]() / 3;
10767
        }
10768
10769
        if ( show ) {
10770
                downAnim = { opacity: 1 };
10771
                downAnim[ ref ] = 0;
10772
10773
                // if we are showing, force opacity 0 and set the initial position
10774
                // then do the "first" animation
10775
                el.css( "opacity", 0 )
10776
                        .css( ref, motion ? -distance * 2 : distance * 2 )
10777
                        .animate( downAnim, speed, easing );
10778
        }
10779
10780
        // start at the smallest distance if we are hiding
10781
        if ( hide ) {
10782
                distance = distance / Math.pow( 2, times - 1 );
10783
        }
10784
10785
        downAnim = {};
10786
        downAnim[ ref ] = 0;
10787
        // Bounces up/down/left/right then back to 0 -- times * 2 animations happen here
10788
        for ( i = 0; i < times; i++ ) {
10789
                upAnim = {};
10790
                upAnim[ ref ] = ( motion ? "-=" : "+=" ) + distance;
10791
10792
                el.animate( upAnim, speed, easing )
10793
                        .animate( downAnim, speed, easing );
10794
10795
                distance = hide ? distance * 2 : distance / 2;
10796
        }
10797
10798
        // Last Bounce when Hiding
10799
        if ( hide ) {
10800
                upAnim = { opacity: 0 };
10801
                upAnim[ ref ] = ( motion ? "-=" : "+=" ) + distance;
10802
10803
                el.animate( upAnim, speed, easing );
10804
        }
10805
10806
        el.queue(function() {
10807
                if ( hide ) {
10808
                        el.hide();
10809
                }
10810
                $.effects.restore( el, props );
10811
                $.effects.removeWrapper( el );
10812
                done();
10813
        });
10814
10815
        // inject all the animations we just queued to be first in line (after "inprogress")
10816
        if ( queuelen > 1) {
10817
                queue.splice.apply( queue,
10818
                        [ 1, 0 ].concat( queue.splice( queuelen, anims + 1 ) ) );
10819
        }
10820
        el.dequeue();
10821
10822
};
10823
10824
10825
/*!
10826
 * jQuery UI Effects Clip 1.11.4
10827
 * http://jqueryui.com
10828
 *
10829
 * Copyright jQuery Foundation and other contributors
10830
 * Released under the MIT license.
10831
 * http://jquery.org/license
10832
 *
10833
 * http://api.jqueryui.com/clip-effect/
10834
 */
10835
10836
10837
var effectClip = $.effects.effect.clip = function( o, done ) {
10838
        // Create element
10839
        var el = $( this ),
10840
                props = [ "position", "top", "bottom", "left", "right", "height", "width" ],
10841
                mode = $.effects.setMode( el, o.mode || "hide" ),
10842
                show = mode === "show",
10843
                direction = o.direction || "vertical",
10844
                vert = direction === "vertical",
10845
                size = vert ? "height" : "width",
10846
                position = vert ? "top" : "left",
10847
                animation = {},
10848
                wrapper, animate, distance;
10849
10850
        // Save & Show
10851
        $.effects.save( el, props );
10852
        el.show();
10853
10854
        // Create Wrapper
10855
        wrapper = $.effects.createWrapper( el ).css({
10856
                overflow: "hidden"
10857
        });
10858
        animate = ( el[0].tagName === "IMG" ) ? wrapper : el;
10859
        distance = animate[ size ]();
10860
10861
        // Shift
10862
        if ( show ) {
10863
                animate.css( size, 0 );
10864
                animate.css( position, distance / 2 );
10865
        }
10866
10867
        // Create Animation Object:
10868
        animation[ size ] = show ? distance : 0;
10869
        animation[ position ] = show ? 0 : distance / 2;
10870
10871
        // Animate
10872
        animate.animate( animation, {
10873
                queue: false,
10874
                duration: o.duration,
10875
                easing: o.easing,
10876
                complete: function() {
10877
                        if ( !show ) {
10878
                                el.hide();
10879
                        }
10880
                        $.effects.restore( el, props );
10881
                        $.effects.removeWrapper( el );
10882
                        done();
10883
                }
10884
        });
10885
10886
};
10887
10888
10889
/*!
10890
 * jQuery UI Effects Drop 1.11.4
10891
 * http://jqueryui.com
10892
 *
10893
 * Copyright jQuery Foundation and other contributors
10894
 * Released under the MIT license.
10895
 * http://jquery.org/license
10896
 *
10897
 * http://api.jqueryui.com/drop-effect/
10898
 */
10899
10900
10901
var effectDrop = $.effects.effect.drop = function( o, done ) {
10902
10903
        var el = $( this ),
10904
                props = [ "position", "top", "bottom", "left", "right", "opacity", "height", "width" ],
10905
                mode = $.effects.setMode( el, o.mode || "hide" ),
10906
                show = mode === "show",
10907
                direction = o.direction || "left",
10908
                ref = ( direction === "up" || direction === "down" ) ? "top" : "left",
10909
                motion = ( direction === "up" || direction === "left" ) ? "pos" : "neg",
10910
                animation = {
10911
                        opacity: show ? 1 : 0
10912
                },
10913
                distance;
10914
10915
        // Adjust
10916
        $.effects.save( el, props );
10917
        el.show();
10918
        $.effects.createWrapper( el );
10919
10920
        distance = o.distance || el[ ref === "top" ? "outerHeight" : "outerWidth" ]( true ) / 2;
10921
10922
        if ( show ) {
10923
                el
10924
                        .css( "opacity", 0 )
10925
                        .css( ref, motion === "pos" ? -distance : distance );
10926
        }
10927
10928
        // Animation
10929
        animation[ ref ] = ( show ?
10930
                ( motion === "pos" ? "+=" : "-=" ) :
10931
                ( motion === "pos" ? "-=" : "+=" ) ) +
10932
                distance;
10933
10934
        // Animate
10935
        el.animate( animation, {
10936
                queue: false,
10937
                duration: o.duration,
10938
                easing: o.easing,
10939
                complete: function() {
10940
                        if ( mode === "hide" ) {
10941
                                el.hide();
10942
                        }
10943
                        $.effects.restore( el, props );
10944
                        $.effects.removeWrapper( el );
10945
                        done();
10946
                }
10947
        });
10948
};
10949
10950
10951
/*!
10952
 * jQuery UI Effects Explode 1.11.4
10953
 * http://jqueryui.com
10954
 *
10955
 * Copyright jQuery Foundation and other contributors
10956
 * Released under the MIT license.
10957
 * http://jquery.org/license
10958
 *
10959
 * http://api.jqueryui.com/explode-effect/
10960
 */
10961
10962
10963
var effectExplode = $.effects.effect.explode = function( o, done ) {
10964
10965
        var rows = o.pieces ? Math.round( Math.sqrt( o.pieces ) ) : 3,
10966
                cells = rows,
10967
                el = $( this ),
10968
                mode = $.effects.setMode( el, o.mode || "hide" ),
10969
                show = mode === "show",
10970
10971
                // show and then visibility:hidden the element before calculating offset
10972
                offset = el.show().css( "visibility", "hidden" ).offset(),
10973
10974
                // width and height of a piece
10975
                width = Math.ceil( el.outerWidth() / cells ),
10976
                height = Math.ceil( el.outerHeight() / rows ),
10977
                pieces = [],
10978
10979
                // loop
10980
                i, j, left, top, mx, my;
10981
10982
        // children animate complete:
10983
        function childComplete() {
10984
                pieces.push( this );
10985
                if ( pieces.length === rows * cells ) {
10986
                        animComplete();
10987
                }
10988
        }
10989
10990
        // clone the element for each row and cell.
10991
        for ( i = 0; i < rows ; i++ ) { // ===>
10992
                top = offset.top + i * height;
10993
                my = i - ( rows - 1 ) / 2 ;
10994
10995
                for ( j = 0; j < cells ; j++ ) { // |||
10996
                        left = offset.left + j * width;
10997
                        mx = j - ( cells - 1 ) / 2 ;
10998
10999
                        // Create a clone of the now hidden main element that will be absolute positioned
11000
                        // within a wrapper div off the -left and -top equal to size of our pieces
11001
                        el
11002
                                .clone()
11003
                                .appendTo( "body" )
11004
                                .wrap( "<div></div>" )
11005
                                .css({
11006
                                        position: "absolute",
11007
                                        visibility: "visible",
11008
                                        left: -j * width,
11009
                                        top: -i * height
11010
                                })
11011
11012
                        // select the wrapper - make it overflow: hidden and absolute positioned based on
11013
                        // where the original was located +left and +top equal to the size of pieces
11014
                                .parent()
11015
                                .addClass( "ui-effects-explode" )
11016
                                .css({
11017
                                        position: "absolute",
11018
                                        overflow: "hidden",
11019
                                        width: width,
11020
                                        height: height,
11021
                                        left: left + ( show ? mx * width : 0 ),
11022
                                        top: top + ( show ? my * height : 0 ),
11023
                                        opacity: show ? 0 : 1
11024
                                }).animate({
11025
                                        left: left + ( show ? 0 : mx * width ),
11026
                                        top: top + ( show ? 0 : my * height ),
11027
                                        opacity: show ? 1 : 0
11028
                                }, o.duration || 500, o.easing, childComplete );
11029
                }
11030
        }
11031
11032
        function animComplete() {
11033
                el.css({
11034
                        visibility: "visible"
11035
                });
11036
                $( pieces ).remove();
11037
                if ( !show ) {
11038
                        el.hide();
11039
                }
11040
                done();
11041
        }
11042
};
11043
11044
11045
/*!
11046
 * jQuery UI Effects Fade 1.11.4
11047
 * http://jqueryui.com
11048
 *
11049
 * Copyright jQuery Foundation and other contributors
11050
 * Released under the MIT license.
11051
 * http://jquery.org/license
11052
 *
11053
 * http://api.jqueryui.com/fade-effect/
11054
 */
11055
11056
11057
var effectFade = $.effects.effect.fade = function( o, done ) {
11058
        var el = $( this ),
11059
                mode = $.effects.setMode( el, o.mode || "toggle" );
11060
11061
        el.animate({
11062
                opacity: mode
11063
        }, {
11064
                queue: false,
11065
                duration: o.duration,
11066
                easing: o.easing,
11067
                complete: done
11068
        });
11069
};
11070
11071
11072
/*!
11073
 * jQuery UI Effects Fold 1.11.4
11074
 * http://jqueryui.com
11075
 *
11076
 * Copyright jQuery Foundation and other contributors
11077
 * Released under the MIT license.
11078
 * http://jquery.org/license
11079
 *
11080
 * http://api.jqueryui.com/fold-effect/
11081
 */
11082
11083
11084
var effectFold = $.effects.effect.fold = function( o, done ) {
11085
11086
        // Create element
11087
        var el = $( this ),
11088
                props = [ "position", "top", "bottom", "left", "right", "height", "width" ],
11089
                mode = $.effects.setMode( el, o.mode || "hide" ),
11090
                show = mode === "show",
11091
                hide = mode === "hide",
11092
                size = o.size || 15,
11093
                percent = /([0-9]+)%/.exec( size ),
11094
                horizFirst = !!o.horizFirst,
11095
                widthFirst = show !== horizFirst,
11096
                ref = widthFirst ? [ "width", "height" ] : [ "height", "width" ],
11097
                duration = o.duration / 2,
11098
                wrapper, distance,
11099
                animation1 = {},
11100
                animation2 = {};
11101
11102
        $.effects.save( el, props );
11103
        el.show();
11104
11105
        // Create Wrapper
11106
        wrapper = $.effects.createWrapper( el ).css({
11107
                overflow: "hidden"
11108
        });
11109
        distance = widthFirst ?
11110
                [ wrapper.width(), wrapper.height() ] :
11111
                [ wrapper.height(), wrapper.width() ];
11112
11113
        if ( percent ) {
11114
                size = parseInt( percent[ 1 ], 10 ) / 100 * distance[ hide ? 0 : 1 ];
11115
        }
11116
        if ( show ) {
11117
                wrapper.css( horizFirst ? {
11118
                        height: 0,
11119
                        width: size
11120
                } : {
11121
                        height: size,
11122
                        width: 0
11123
                });
11124
        }
11125
11126
        // Animation
11127
        animation1[ ref[ 0 ] ] = show ? distance[ 0 ] : size;
11128
        animation2[ ref[ 1 ] ] = show ? distance[ 1 ] : 0;
11129
11130
        // Animate
11131
        wrapper
11132
                .animate( animation1, duration, o.easing )
11133
                .animate( animation2, duration, o.easing, function() {
11134
                        if ( hide ) {
11135
                                el.hide();
11136
                        }
11137
                        $.effects.restore( el, props );
11138
                        $.effects.removeWrapper( el );
11139
                        done();
11140
                });
11141
11142
};
11143
11144
11145
/*!
11146
 * jQuery UI Effects Highlight 1.11.4
11147
 * http://jqueryui.com
11148
 *
11149
 * Copyright jQuery Foundation and other contributors
11150
 * Released under the MIT license.
11151
 * http://jquery.org/license
11152
 *
11153
 * http://api.jqueryui.com/highlight-effect/
11154
 */
11155
11156
11157
var effectHighlight = $.effects.effect.highlight = function( o, done ) {
11158
        var elem = $( this ),
11159
                props = [ "backgroundImage", "backgroundColor", "opacity" ],
11160
                mode = $.effects.setMode( elem, o.mode || "show" ),
11161
                animation = {
11162
                        backgroundColor: elem.css( "backgroundColor" )
11163
                };
11164
11165
        if (mode === "hide") {
11166
                animation.opacity = 0;
11167
        }
11168
11169
        $.effects.save( elem, props );
11170
11171
        elem
11172
                .show()
11173
                .css({
11174
                        backgroundImage: "none",
11175
                        backgroundColor: o.color || "#ffff99"
11176
                })
11177
                .animate( animation, {
11178
                        queue: false,
11179
                        duration: o.duration,
11180
                        easing: o.easing,
11181
                        complete: function() {
11182
                                if ( mode === "hide" ) {
11183
                                        elem.hide();
11184
                                }
11185
                                $.effects.restore( elem, props );
11186
                                done();
11187
                        }
11188
                });
11189
};
11190
11191
11192
/*!
11193
 * jQuery UI Effects Size 1.11.4
11194
 * http://jqueryui.com
11195
 *
11196
 * Copyright jQuery Foundation and other contributors
11197
 * Released under the MIT license.
11198
 * http://jquery.org/license
11199
 *
11200
 * http://api.jqueryui.com/size-effect/
11201
 */
11202
11203
11204
var effectSize = $.effects.effect.size = function( o, done ) {
11205
11206
        // Create element
11207
        var original, baseline, factor,
11208
                el = $( this ),
11209
                props0 = [ "position", "top", "bottom", "left", "right", "width", "height", "overflow", "opacity" ],
11210
11211
                // Always restore
11212
                props1 = [ "position", "top", "bottom", "left", "right", "overflow", "opacity" ],
11213
11214
                // Copy for children
11215
                props2 = [ "width", "height", "overflow" ],
11216
                cProps = [ "fontSize" ],
11217
                vProps = [ "borderTopWidth", "borderBottomWidth", "paddingTop", "paddingBottom" ],
11218
                hProps = [ "borderLeftWidth", "borderRightWidth", "paddingLeft", "paddingRight" ],
11219
11220
                // Set options
11221
                mode = $.effects.setMode( el, o.mode || "effect" ),
11222
                restore = o.restore || mode !== "effect",
11223
                scale = o.scale || "both",
11224
                origin = o.origin || [ "middle", "center" ],
11225
                position = el.css( "position" ),
11226
                props = restore ? props0 : props1,
11227
                zero = {
11228
                        height: 0,
11229
                        width: 0,
11230
                        outerHeight: 0,
11231
                        outerWidth: 0
11232
                };
11233
11234
        if ( mode === "show" ) {
11235
                el.show();
11236
        }
11237
        original = {
11238
                height: el.height(),
11239
                width: el.width(),
11240
                outerHeight: el.outerHeight(),
11241
                outerWidth: el.outerWidth()
11242
        };
11243
11244
        if ( o.mode === "toggle" && mode === "show" ) {
11245
                el.from = o.to || zero;
11246
                el.to = o.from || original;
11247
        } else {
11248
                el.from = o.from || ( mode === "show" ? zero : original );
11249
                el.to = o.to || ( mode === "hide" ? zero : original );
11250
        }
11251
11252
        // Set scaling factor
11253
        factor = {
11254
                from: {
11255
                        y: el.from.height / original.height,
11256
                        x: el.from.width / original.width
11257
                },
11258
                to: {
11259
                        y: el.to.height / original.height,
11260
                        x: el.to.width / original.width
11261
                }
11262
        };
11263
11264
        // Scale the css box
11265
        if ( scale === "box" || scale === "both" ) {
11266
11267
                // Vertical props scaling
11268
                if ( factor.from.y !== factor.to.y ) {
11269
                        props = props.concat( vProps );
11270
                        el.from = $.effects.setTransition( el, vProps, factor.from.y, el.from );
11271
                        el.to = $.effects.setTransition( el, vProps, factor.to.y, el.to );
11272
                }
11273
11274
                // Horizontal props scaling
11275
                if ( factor.from.x !== factor.to.x ) {
11276
                        props = props.concat( hProps );
11277
                        el.from = $.effects.setTransition( el, hProps, factor.from.x, el.from );
11278
                        el.to = $.effects.setTransition( el, hProps, factor.to.x, el.to );
11279
                }
11280
        }
11281
11282
        // Scale the content
11283
        if ( scale === "content" || scale === "both" ) {
11284
11285
                // Vertical props scaling
11286
                if ( factor.from.y !== factor.to.y ) {
11287
                        props = props.concat( cProps ).concat( props2 );
11288
                        el.from = $.effects.setTransition( el, cProps, factor.from.y, el.from );
11289
                        el.to = $.effects.setTransition( el, cProps, factor.to.y, el.to );
11290
                }
11291
        }
11292
11293
        $.effects.save( el, props );
11294
        el.show();
11295
        $.effects.createWrapper( el );
11296
        el.css( "overflow", "hidden" ).css( el.from );
11297
11298
        // Adjust
11299
        if (origin) { // Calculate baseline shifts
11300
                baseline = $.effects.getBaseline( origin, original );
11301
                el.from.top = ( original.outerHeight - el.outerHeight() ) * baseline.y;
11302
                el.from.left = ( original.outerWidth - el.outerWidth() ) * baseline.x;
11303
                el.to.top = ( original.outerHeight - el.to.outerHeight ) * baseline.y;
11304
                el.to.left = ( original.outerWidth - el.to.outerWidth ) * baseline.x;
11305
        }
11306
        el.css( el.from ); // set top & left
11307
11308
        // Animate
11309
        if ( scale === "content" || scale === "both" ) { // Scale the children
11310
11311
                // Add margins/font-size
11312
                vProps = vProps.concat([ "marginTop", "marginBottom" ]).concat(cProps);
11313
                hProps = hProps.concat([ "marginLeft", "marginRight" ]);
11314
                props2 = props0.concat(vProps).concat(hProps);
11315
11316
                el.find( "*[width]" ).each( function() {
11317
                        var child = $( this ),
11318
                                c_original = {
11319
                                        height: child.height(),
11320
                                        width: child.width(),
11321
                                        outerHeight: child.outerHeight(),
11322
                                        outerWidth: child.outerWidth()
11323
                                };
11324
                        if (restore) {
11325
                                $.effects.save(child, props2);
11326
                        }
11327
11328
                        child.from = {
11329
                                height: c_original.height * factor.from.y,
11330
                                width: c_original.width * factor.from.x,
11331
                                outerHeight: c_original.outerHeight * factor.from.y,
11332
                                outerWidth: c_original.outerWidth * factor.from.x
11333
                        };
11334
                        child.to = {
11335
                                height: c_original.height * factor.to.y,
11336
                                width: c_original.width * factor.to.x,
11337
                                outerHeight: c_original.height * factor.to.y,
11338
                                outerWidth: c_original.width * factor.to.x
11339
                        };
11340
11341
                        // Vertical props scaling
11342
                        if ( factor.from.y !== factor.to.y ) {
11343
                                child.from = $.effects.setTransition( child, vProps, factor.from.y, child.from );
11344
                                child.to = $.effects.setTransition( child, vProps, factor.to.y, child.to );
11345
                        }
11346
11347
                        // Horizontal props scaling
11348
                        if ( factor.from.x !== factor.to.x ) {
11349
                                child.from = $.effects.setTransition( child, hProps, factor.from.x, child.from );
11350
                                child.to = $.effects.setTransition( child, hProps, factor.to.x, child.to );
11351
                        }
11352
11353
                        // Animate children
11354
                        child.css( child.from );
11355
                        child.animate( child.to, o.duration, o.easing, function() {
11356
11357
                                // Restore children
11358
                                if ( restore ) {
11359
                                        $.effects.restore( child, props2 );
11360
                                }
11361
                        });
11362
                });
11363
        }
11364
11365
        // Animate
11366
        el.animate( el.to, {
11367
                queue: false,
11368
                duration: o.duration,
11369
                easing: o.easing,
11370
                complete: function() {
11371
                        if ( el.to.opacity === 0 ) {
11372
                                el.css( "opacity", el.from.opacity );
11373
                        }
11374
                        if ( mode === "hide" ) {
11375
                                el.hide();
11376
                        }
11377
                        $.effects.restore( el, props );
11378
                        if ( !restore ) {
11379
11380
                                // we need to calculate our new positioning based on the scaling
11381
                                if ( position === "static" ) {
11382
                                        el.css({
11383
                                                position: "relative",
11384
                                                top: el.to.top,
11385
                                                left: el.to.left
11386
                                        });
11387
                                } else {
11388
                                        $.each([ "top", "left" ], function( idx, pos ) {
11389
                                                el.css( pos, function( _, str ) {
11390
                                                        var val = parseInt( str, 10 ),
11391
                                                                toRef = idx ? el.to.left : el.to.top;
11392
11393
                                                        // if original was "auto", recalculate the new value from wrapper
11394
                                                        if ( str === "auto" ) {
11395
                                                                return toRef + "px";
11396
                                                        }
11397
11398
                                                        return val + toRef + "px";
11399
                                                });
11400
                                        });
11401
                                }
11402
                        }
11403
11404
                        $.effects.removeWrapper( el );
11405
                        done();
11406
                }
11407
        });
11408
11409
};
11410
11411
11412
/*!
11413
 * jQuery UI Effects Scale 1.11.4
11414
 * http://jqueryui.com
11415
 *
11416
 * Copyright jQuery Foundation and other contributors
11417
 * Released under the MIT license.
11418
 * http://jquery.org/license
11419
 *
11420
 * http://api.jqueryui.com/scale-effect/
11421
 */
11422
11423
11424
var effectScale = $.effects.effect.scale = function( o, done ) {
11425
11426
        // Create element
11427
        var el = $( this ),
11428
                options = $.extend( true, {}, o ),
11429
                mode = $.effects.setMode( el, o.mode || "effect" ),
11430
                percent = parseInt( o.percent, 10 ) ||
11431
                        ( parseInt( o.percent, 10 ) === 0 ? 0 : ( mode === "hide" ? 0 : 100 ) ),
11432
                direction = o.direction || "both",
11433
                origin = o.origin,
11434
                original = {
11435
                        height: el.height(),
11436
                        width: el.width(),
11437
                        outerHeight: el.outerHeight(),
11438
                        outerWidth: el.outerWidth()
11439
                },
11440
                factor = {
11441
                        y: direction !== "horizontal" ? (percent / 100) : 1,
11442
                        x: direction !== "vertical" ? (percent / 100) : 1
11443
                };
11444
11445
        // We are going to pass this effect to the size effect:
11446
        options.effect = "size";
11447
        options.queue = false;
11448
        options.complete = done;
11449
11450
        // Set default origin and restore for show/hide
11451
        if ( mode !== "effect" ) {
11452
                options.origin = origin || [ "middle", "center" ];
11453
                options.restore = true;
11454
        }
11455
11456
        options.from = o.from || ( mode === "show" ? {
11457
                height: 0,
11458
                width: 0,
11459
                outerHeight: 0,
11460
                outerWidth: 0
11461
        } : original );
11462
        options.to = {
11463
                height: original.height * factor.y,
11464
                width: original.width * factor.x,
11465
                outerHeight: original.outerHeight * factor.y,
11466
                outerWidth: original.outerWidth * factor.x
11467
        };
11468
11469
        // Fade option to support puff
11470
        if ( options.fade ) {
11471
                if ( mode === "show" ) {
11472
                        options.from.opacity = 0;
11473
                        options.to.opacity = 1;
11474
                }
11475
                if ( mode === "hide" ) {
11476
                        options.from.opacity = 1;
11477
                        options.to.opacity = 0;
11478
                }
11479
        }
11480
11481
        // Animate
11482
        el.effect( options );
11483
11484
};
11485
11486
11487
/*!
11488
 * jQuery UI Effects Puff 1.11.4
11489
 * http://jqueryui.com
11490
 *
11491
 * Copyright jQuery Foundation and other contributors
11492
 * Released under the MIT license.
11493
 * http://jquery.org/license
11494
 *
11495
 * http://api.jqueryui.com/puff-effect/
11496
 */
11497
11498
11499
var effectPuff = $.effects.effect.puff = function( o, done ) {
11500
        var elem = $( this ),
11501
                mode = $.effects.setMode( elem, o.mode || "hide" ),
11502
                hide = mode === "hide",
11503
                percent = parseInt( o.percent, 10 ) || 150,
11504
                factor = percent / 100,
11505
                original = {
11506
                        height: elem.height(),
11507
                        width: elem.width(),
11508
                        outerHeight: elem.outerHeight(),
11509
                        outerWidth: elem.outerWidth()
11510
                };
11511
11512
        $.extend( o, {
11513
                effect: "scale",
11514
                queue: false,
11515
                fade: true,
11516
                mode: mode,
11517
                complete: done,
11518
                percent: hide ? percent : 100,
11519
                from: hide ?
11520
                        original :
11521
                        {
11522
                                height: original.height * factor,
11523
                                width: original.width * factor,
11524
                                outerHeight: original.outerHeight * factor,
11525
                                outerWidth: original.outerWidth * factor
11526
                        }
11527
        });
11528
11529
        elem.effect( o );
11530
};
11531
11532
11533
/*!
11534
 * jQuery UI Effects Pulsate 1.11.4
11535
 * http://jqueryui.com
11536
 *
11537
 * Copyright jQuery Foundation and other contributors
11538
 * Released under the MIT license.
11539
 * http://jquery.org/license
11540
 *
11541
 * http://api.jqueryui.com/pulsate-effect/
11542
 */
11543
11544
11545
var effectPulsate = $.effects.effect.pulsate = function( o, done ) {
11546
        var elem = $( this ),
11547
                mode = $.effects.setMode( elem, o.mode || "show" ),
11548
                show = mode === "show",
11549
                hide = mode === "hide",
11550
                showhide = ( show || mode === "hide" ),
11551
11552
                // showing or hiding leaves of the "last" animation
11553
                anims = ( ( o.times || 5 ) * 2 ) + ( showhide ? 1 : 0 ),
11554
                duration = o.duration / anims,
11555
                animateTo = 0,
11556
                queue = elem.queue(),
11557
                queuelen = queue.length,
11558
                i;
11559
11560
        if ( show || !elem.is(":visible")) {
11561
                elem.css( "opacity", 0 ).show();
11562
                animateTo = 1;
11563
        }
11564
11565
        // anims - 1 opacity "toggles"
11566
        for ( i = 1; i < anims; i++ ) {
11567
                elem.animate({
11568
                        opacity: animateTo
11569
                }, duration, o.easing );
11570
                animateTo = 1 - animateTo;
11571
        }
11572
11573
        elem.animate({
11574
                opacity: animateTo
11575
        }, duration, o.easing);
11576
11577
        elem.queue(function() {
11578
                if ( hide ) {
11579
                        elem.hide();
11580
                }
11581
                done();
11582
        });
11583
11584
        // We just queued up "anims" animations, we need to put them next in the queue
11585
        if ( queuelen > 1 ) {
11586
                queue.splice.apply( queue,
11587
                        [ 1, 0 ].concat( queue.splice( queuelen, anims + 1 ) ) );
11588
        }
11589
        elem.dequeue();
11590
};
11591
11592
11593
/*!
11594
 * jQuery UI Effects Shake 1.11.4
11595
 * http://jqueryui.com
11596
 *
11597
 * Copyright jQuery Foundation and other contributors
11598
 * Released under the MIT license.
11599
 * http://jquery.org/license
11600
 *
11601
 * http://api.jqueryui.com/shake-effect/
11602
 */
11603
11604
11605
var effectShake = $.effects.effect.shake = function( o, done ) {
11606
11607
        var el = $( this ),
11608
                props = [ "position", "top", "bottom", "left", "right", "height", "width" ],
11609
                mode = $.effects.setMode( el, o.mode || "effect" ),
11610
                direction = o.direction || "left",
11611
                distance = o.distance || 20,
11612
                times = o.times || 3,
11613
                anims = times * 2 + 1,
11614
                speed = Math.round( o.duration / anims ),
11615
                ref = (direction === "up" || direction === "down") ? "top" : "left",
11616
                positiveMotion = (direction === "up" || direction === "left"),
11617
                animation = {},
11618
                animation1 = {},
11619
                animation2 = {},
11620
                i,
11621
11622
                // we will need to re-assemble the queue to stack our animations in place
11623
                queue = el.queue(),
11624
                queuelen = queue.length;
11625
11626
        $.effects.save( el, props );
11627
        el.show();
11628
        $.effects.createWrapper( el );
11629
11630
        // Animation
11631
        animation[ ref ] = ( positiveMotion ? "-=" : "+=" ) + distance;
11632
        animation1[ ref ] = ( positiveMotion ? "+=" : "-=" ) + distance * 2;
11633
        animation2[ ref ] = ( positiveMotion ? "-=" : "+=" ) + distance * 2;
11634
11635
        // Animate
11636
        el.animate( animation, speed, o.easing );
11637
11638
        // Shakes
11639
        for ( i = 1; i < times; i++ ) {
11640
                el.animate( animation1, speed, o.easing ).animate( animation2, speed, o.easing );
11641
        }
11642
        el
11643
                .animate( animation1, speed, o.easing )
11644
                .animate( animation, speed / 2, o.easing )
11645
                .queue(function() {
11646
                        if ( mode === "hide" ) {
11647
                                el.hide();
11648
                        }
11649
                        $.effects.restore( el, props );
11650
                        $.effects.removeWrapper( el );
11651
                        done();
11652
                });
11653
11654
        // inject all the animations we just queued to be first in line (after "inprogress")
11655
        if ( queuelen > 1) {
11656
                queue.splice.apply( queue,
11657
                        [ 1, 0 ].concat( queue.splice( queuelen, anims + 1 ) ) );
11658
        }
11659
        el.dequeue();
11660
11661
};
11662
11663
11664
/*!
11665
 * jQuery UI Effects Slide 1.11.4
11666
 * http://jqueryui.com
11667
 *
11668
 * Copyright jQuery Foundation and other contributors
11669
 * Released under the MIT license.
11670
 * http://jquery.org/license
11671
 *
11672
 * http://api.jqueryui.com/slide-effect/
11673
 */
11674
11675
11676
var effectSlide = $.effects.effect.slide = function( o, done ) {
11677
11678
        // Create element
11679
        var el = $( this ),
11680
                props = [ "position", "top", "bottom", "left", "right", "width", "height" ],
11681
                mode = $.effects.setMode( el, o.mode || "show" ),
11682
                show = mode === "show",
11683
                direction = o.direction || "left",
11684
                ref = (direction === "up" || direction === "down") ? "top" : "left",
11685
                positiveMotion = (direction === "up" || direction === "left"),
11686
                distance,
11687
                animation = {};
11688
11689
        // Adjust
11690
        $.effects.save( el, props );
11691
        el.show();
11692
        distance = o.distance || el[ ref === "top" ? "outerHeight" : "outerWidth" ]( true );
11693
11694
        $.effects.createWrapper( el ).css({
11695
                overflow: "hidden"
11696
        });
11697
11698
        if ( show ) {
11699
                el.css( ref, positiveMotion ? (isNaN(distance) ? "-" + distance : -distance) : distance );
11700
        }
11701
11702
        // Animation
11703
        animation[ ref ] = ( show ?
11704
                ( positiveMotion ? "+=" : "-=") :
11705
                ( positiveMotion ? "-=" : "+=")) +
11706
                distance;
11707
11708
        // Animate
11709
        el.animate( animation, {
11710
                queue: false,
11711
                duration: o.duration,
11712
                easing: o.easing,
11713
                complete: function() {
11714
                        if ( mode === "hide" ) {
11715
                                el.hide();
11716
                        }
11717
                        $.effects.restore( el, props );
11718
                        $.effects.removeWrapper( el );
11719
                        done();
11720
                }
11721
        });
11722
};
11723
11724
11725
/*!
11726
 * jQuery UI Effects Transfer 1.11.4
11727
 * http://jqueryui.com
11728
 *
11729
 * Copyright jQuery Foundation and other contributors
11730
 * Released under the MIT license.
11731
 * http://jquery.org/license
11732
 *
11733
 * http://api.jqueryui.com/transfer-effect/
11734
 */
11735
11736
11737
var effectTransfer = $.effects.effect.transfer = function( o, done ) {
11738
        var elem = $( this ),
11739
                target = $( o.to ),
11740
                targetFixed = target.css( "position" ) === "fixed",
11741
                body = $("body"),
11742
                fixTop = targetFixed ? body.scrollTop() : 0,
11743
                fixLeft = targetFixed ? body.scrollLeft() : 0,
11744
                endPosition = target.offset(),
11745
                animation = {
11746
                        top: endPosition.top - fixTop,
11747
                        left: endPosition.left - fixLeft,
11748
                        height: target.innerHeight(),
11749
                        width: target.innerWidth()
11750
                },
11751
                startPosition = elem.offset(),
11752
                transfer = $( "<div class='ui-effects-transfer'></div>" )
11753
                        .appendTo( document.body )
11754
                        .addClass( o.className )
11755
                        .css({
11756
                                top: startPosition.top - fixTop,
11757
                                left: startPosition.left - fixLeft,
11758
                                height: elem.innerHeight(),
11759
                                width: elem.innerWidth(),
11760
                                position: targetFixed ? "fixed" : "absolute"
11761
                        })
11762
                        .animate( animation, o.duration, o.easing, function() {
11763
                                transfer.remove();
11764
                                done();
11765
                        });
11766
};
11767
11768
11769
/*!
11770
 * jQuery UI Progressbar 1.11.4
11771
 * http://jqueryui.com
11772
 *
11773
 * Copyright jQuery Foundation and other contributors
11774
 * Released under the MIT license.
11775
 * http://jquery.org/license
11776
 *
11777
 * http://api.jqueryui.com/progressbar/
11778
 */
11779
11780
11781
var progressbar = $.widget( "ui.progressbar", {
11782
        version: "1.11.4",
11783
        options: {
11784
                max: 100,
11785
                value: 0,
11786
11787
                change: null,
11788
                complete: null
11789
        },
11790
11791
        min: 0,
11792
11793
        _create: function() {
11794
                // Constrain initial value
11795
                this.oldValue = this.options.value = this._constrainedValue();
11796
11797
                this.element
11798
                        .addClass( "ui-progressbar ui-widget ui-widget-content ui-corner-all" )
11799
                        .attr({
11800
                                // Only set static values, aria-valuenow and aria-valuemax are
11801
                                // set inside _refreshValue()
11802
                                role: "progressbar",
11803
                                "aria-valuemin": this.min
11804
                        });
11805
11806
                this.valueDiv = $( "<div class='ui-progressbar-value ui-widget-header ui-corner-left'></div>" )
11807
                        .appendTo( this.element );
11808
11809
                this._refreshValue();
11810
        },
11811
11812
        _destroy: function() {
11813
                this.element
11814
                        .removeClass( "ui-progressbar ui-widget ui-widget-content ui-corner-all" )
11815
                        .removeAttr( "role" )
11816
                        .removeAttr( "aria-valuemin" )
11817
                        .removeAttr( "aria-valuemax" )
11818
                        .removeAttr( "aria-valuenow" );
11819
11820
                this.valueDiv.remove();
11821
        },
11822
11823
        value: function( newValue ) {
11824
                if ( newValue === undefined ) {
11825
                        return this.options.value;
11826
                }
11827
11828
                this.options.value = this._constrainedValue( newValue );
11829
                this._refreshValue();
11830
        },
11831
11832
        _constrainedValue: function( newValue ) {
11833
                if ( newValue === undefined ) {
11834
                        newValue = this.options.value;
11835
                }
11836
11837
                this.indeterminate = newValue === false;
11838
11839
                // sanitize value
11840
                if ( typeof newValue !== "number" ) {
11841
                        newValue = 0;
11842
                }
11843
11844
                return this.indeterminate ? false :
11845
                        Math.min( this.options.max, Math.max( this.min, newValue ) );
11846
        },
11847
11848
        _setOptions: function( options ) {
11849
                // Ensure "value" option is set after other values (like max)
11850
                var value = options.value;
11851
                delete options.value;
11852
11853
                this._super( options );
11854
11855
                this.options.value = this._constrainedValue( value );
11856
                this._refreshValue();
11857
        },
11858
11859
        _setOption: function( key, value ) {
11860
                if ( key === "max" ) {
11861
                        // Don't allow a max less than min
11862
                        value = Math.max( this.min, value );
11863
                }
11864
                if ( key === "disabled" ) {
11865
                        this.element
11866
                                .toggleClass( "ui-state-disabled", !!value )
11867
                                .attr( "aria-disabled", value );
11868
                }
11869
                this._super( key, value );
11870
        },
11871
11872
        _percentage: function() {
11873
                return this.indeterminate ? 100 : 100 * ( this.options.value - this.min ) / ( this.options.max - this.min );
11874
        },
11875
11876
        _refreshValue: function() {
11877
                var value = this.options.value,
11878
                        percentage = this._percentage();
11879
11880
                this.valueDiv
11881
                        .toggle( this.indeterminate || value > this.min )
11882
                        .toggleClass( "ui-corner-right", value === this.options.max )
11883
                        .width( percentage.toFixed(0) + "%" );
11884
11885
                this.element.toggleClass( "ui-progressbar-indeterminate", this.indeterminate );
11886
11887
                if ( this.indeterminate ) {
11888
                        this.element.removeAttr( "aria-valuenow" );
11889
                        if ( !this.overlayDiv ) {
11890
                                this.overlayDiv = $( "<div class='ui-progressbar-overlay'></div>" ).appendTo( this.valueDiv );
11891
                        }
11892
                } else {
11893
                        this.element.attr({
11894
                                "aria-valuemax": this.options.max,
11895
                                "aria-valuenow": value
11896
                        });
11897
                        if ( this.overlayDiv ) {
11898
                                this.overlayDiv.remove();
11899
                                this.overlayDiv = null;
11900
                        }
11901
                }
11902
11903
                if ( this.oldValue !== value ) {
11904
                        this.oldValue = value;
11905
                        this._trigger( "change" );
11906
                }
11907
                if ( value === this.options.max ) {
11908
                        this._trigger( "complete" );
11909
                }
11910
        }
11911
});
11912
11913
11914
/*!
11915
 * jQuery UI Selectable 1.11.4
11916
 * http://jqueryui.com
11917
 *
11918
 * Copyright jQuery Foundation and other contributors
11919
 * Released under the MIT license.
11920
 * http://jquery.org/license
11921
 *
11922
 * http://api.jqueryui.com/selectable/
11923
 */
11924
11925
11926
var selectable = $.widget("ui.selectable", $.ui.mouse, {
11927
        version: "1.11.4",
11928
        options: {
11929
                appendTo: "body",
11930
                autoRefresh: true,
11931
                distance: 0,
11932
                filter: "*",
11933
                tolerance: "touch",
11934
11935
                // callbacks
11936
                selected: null,
11937
                selecting: null,
11938
                start: null,
11939
                stop: null,
11940
                unselected: null,
11941
                unselecting: null
11942
        },
11943
        _create: function() {
11944
                var selectees,
11945
                        that = this;
11946
11947
                this.element.addClass("ui-selectable");
11948
11949
                this.dragged = false;
11950
11951
                // cache selectee children based on filter
11952
                this.refresh = function() {
11953
                        selectees = $(that.options.filter, that.element[0]);
11954
                        selectees.addClass("ui-selectee");
11955
                        selectees.each(function() {
11956
                                var $this = $(this),
11957
                                        pos = $this.offset();
11958
                                $.data(this, "selectable-item", {
11959
                                        element: this,
11960
                                        $element: $this,
11961
                                        left: pos.left,
11962
                                        top: pos.top,
11963
                                        right: pos.left + $this.outerWidth(),
11964
                                        bottom: pos.top + $this.outerHeight(),
11965
                                        startselected: false,
11966
                                        selected: $this.hasClass("ui-selected"),
11967
                                        selecting: $this.hasClass("ui-selecting"),
11968
                                        unselecting: $this.hasClass("ui-unselecting")
11969
                                });
11970
                        });
11971
                };
11972
                this.refresh();
11973
11974
                this.selectees = selectees.addClass("ui-selectee");
11975
11976
                this._mouseInit();
11977
11978
                this.helper = $("<div class='ui-selectable-helper'></div>");
11979
        },
11980
11981
        _destroy: function() {
11982
                this.selectees
11983
                        .removeClass("ui-selectee")
11984
                        .removeData("selectable-item");
11985
                this.element
11986
                        .removeClass("ui-selectable ui-selectable-disabled");
11987
                this._mouseDestroy();
11988
        },
11989
11990
        _mouseStart: function(event) {
11991
                var that = this,
11992
                        options = this.options;
11993
11994
                this.opos = [ event.pageX, event.pageY ];
11995
11996
                if (this.options.disabled) {
11997
                        return;
11998
                }
11999
12000
                this.selectees = $(options.filter, this.element[0]);
12001
12002
                this._trigger("start", event);
12003
12004
                $(options.appendTo).append(this.helper);
12005
                // position helper (lasso)
12006
                this.helper.css({
12007
                        "left": event.pageX,
12008
                        "top": event.pageY,
12009
                        "width": 0,
12010
                        "height": 0
12011
                });
12012
12013
                if (options.autoRefresh) {
12014
                        this.refresh();
12015
                }
12016
12017
                this.selectees.filter(".ui-selected").each(function() {
12018
                        var selectee = $.data(this, "selectable-item");
12019
                        selectee.startselected = true;
12020
                        if (!event.metaKey && !event.ctrlKey) {
12021
                                selectee.$element.removeClass("ui-selected");
12022
                                selectee.selected = false;
12023
                                selectee.$element.addClass("ui-unselecting");
12024
                                selectee.unselecting = true;
12025
                                // selectable UNSELECTING callback
12026
                                that._trigger("unselecting", event, {
12027
                                        unselecting: selectee.element
12028
                                });
12029
                        }
12030
                });
12031
12032
                $(event.target).parents().addBack().each(function() {
12033
                        var doSelect,
12034
                                selectee = $.data(this, "selectable-item");
12035
                        if (selectee) {
12036
                                doSelect = (!event.metaKey && !event.ctrlKey) || !selectee.$element.hasClass("ui-selected");
12037
                                selectee.$element
12038
                                        .removeClass(doSelect ? "ui-unselecting" : "ui-selected")
12039
                                        .addClass(doSelect ? "ui-selecting" : "ui-unselecting");
12040
                                selectee.unselecting = !doSelect;
12041
                                selectee.selecting = doSelect;
12042
                                selectee.selected = doSelect;
12043
                                // selectable (UN)SELECTING callback
12044
                                if (doSelect) {
12045
                                        that._trigger("selecting", event, {
12046
                                                selecting: selectee.element
12047
                                        });
12048
                                } else {
12049
                                        that._trigger("unselecting", event, {
12050
                                                unselecting: selectee.element
12051
                                        });
12052
                                }
12053
                                return false;
12054
                        }
12055
                });
12056
12057
        },
12058
12059
        _mouseDrag: function(event) {
12060
12061
                this.dragged = true;
12062
12063
                if (this.options.disabled) {
12064
                        return;
12065
                }
12066
12067
                var tmp,
12068
                        that = this,
12069
                        options = this.options,
12070
                        x1 = this.opos[0],
12071
                        y1 = this.opos[1],
12072
                        x2 = event.pageX,
12073
                        y2 = event.pageY;
12074
12075
                if (x1 > x2) { tmp = x2; x2 = x1; x1 = tmp; }
12076
                if (y1 > y2) { tmp = y2; y2 = y1; y1 = tmp; }
12077
                this.helper.css({ left: x1, top: y1, width: x2 - x1, height: y2 - y1 });
12078
12079
                this.selectees.each(function() {
12080
                        var selectee = $.data(this, "selectable-item"),
12081
                                hit = false;
12082
12083
                        //prevent helper from being selected if appendTo: selectable
12084
                        if (!selectee || selectee.element === that.element[0]) {
12085
                                return;
12086
                        }
12087
12088
                        if (options.tolerance === "touch") {
12089
                                hit = ( !(selectee.left > x2 || selectee.right < x1 || selectee.top > y2 || selectee.bottom < y1) );
12090
                        } else if (options.tolerance === "fit") {
12091
                                hit = (selectee.left > x1 && selectee.right < x2 && selectee.top > y1 && selectee.bottom < y2);
12092
                        }
12093
12094
                        if (hit) {
12095
                                // SELECT
12096
                                if (selectee.selected) {
12097
                                        selectee.$element.removeClass("ui-selected");
12098
                                        selectee.selected = false;
12099
                                }
12100
                                if (selectee.unselecting) {
12101
                                        selectee.$element.removeClass("ui-unselecting");
12102
                                        selectee.unselecting = false;
12103
                                }
12104
                                if (!selectee.selecting) {
12105
                                        selectee.$element.addClass("ui-selecting");
12106
                                        selectee.selecting = true;
12107
                                        // selectable SELECTING callback
12108
                                        that._trigger("selecting", event, {
12109
                                                selecting: selectee.element
12110
                                        });
12111
                                }
12112
                        } else {
12113
                                // UNSELECT
12114
                                if (selectee.selecting) {
12115
                                        if ((event.metaKey || event.ctrlKey) && selectee.startselected) {
12116
                                                selectee.$element.removeClass("ui-selecting");
12117
                                                selectee.selecting = false;
12118
                                                selectee.$element.addClass("ui-selected");
12119
                                                selectee.selected = true;
12120
                                        } else {
12121
                                                selectee.$element.removeClass("ui-selecting");
12122
                                                selectee.selecting = false;
12123
                                                if (selectee.startselected) {
12124
                                                        selectee.$element.addClass("ui-unselecting");
12125
                                                        selectee.unselecting = true;
12126
                                                }
12127
                                                // selectable UNSELECTING callback
12128
                                                that._trigger("unselecting", event, {
12129
                                                        unselecting: selectee.element
12130
                                                });
12131
                                        }
12132
                                }
12133
                                if (selectee.selected) {
12134
                                        if (!event.metaKey && !event.ctrlKey && !selectee.startselected) {
12135
                                                selectee.$element.removeClass("ui-selected");
12136
                                                selectee.selected = false;
12137
12138
                                                selectee.$element.addClass("ui-unselecting");
12139
                                                selectee.unselecting = true;
12140
                                                // selectable UNSELECTING callback
12141
                                                that._trigger("unselecting", event, {
12142
                                                        unselecting: selectee.element
12143
                                                });
12144
                                        }
12145
                                }
12146
                        }
12147
                });
12148
12149
                return false;
12150
        },
12151
12152
        _mouseStop: function(event) {
12153
                var that = this;
12154
12155
                this.dragged = false;
12156
12157
                $(".ui-unselecting", this.element[0]).each(function() {
12158
                        var selectee = $.data(this, "selectable-item");
12159
                        selectee.$element.removeClass("ui-unselecting");
12160
                        selectee.unselecting = false;
12161
                        selectee.startselected = false;
12162
                        that._trigger("unselected", event, {
12163
                                unselected: selectee.element
12164
                        });
12165
                });
12166
                $(".ui-selecting", this.element[0]).each(function() {
12167
                        var selectee = $.data(this, "selectable-item");
12168
                        selectee.$element.removeClass("ui-selecting").addClass("ui-selected");
12169
                        selectee.selecting = false;
12170
                        selectee.selected = true;
12171
                        selectee.startselected = true;
12172
                        that._trigger("selected", event, {
12173
                                selected: selectee.element
12174
                        });
12175
                });
12176
                this._trigger("stop", event);
12177
12178
                this.helper.remove();
12179
12180
                return false;
12181
        }
12182
12183
});
12184
12185
12186
/*!
12187
 * jQuery UI Selectmenu 1.11.4
12188
 * http://jqueryui.com
12189
 *
12190
 * Copyright jQuery Foundation and other contributors
12191
 * Released under the MIT license.
12192
 * http://jquery.org/license
12193
 *
12194
 * http://api.jqueryui.com/selectmenu
12195
 */
12196
12197
12198
var selectmenu = $.widget( "ui.selectmenu", {
12199
        version: "1.11.4",
12200
        defaultElement: "<select>",
12201
        options: {
12202
                appendTo: null,
12203
                disabled: null,
12204
                icons: {
12205
                        button: "ui-icon-triangle-1-s"
12206
                },
12207
                position: {
12208
                        my: "left top",
12209
                        at: "left bottom",
12210
                        collision: "none"
12211
                },
12212
                width: null,
12213
12214
                // callbacks
12215
                change: null,
12216
                close: null,
12217
                focus: null,
12218
                open: null,
12219
                select: null
12220
        },
12221
12222
        _create: function() {
12223
                var selectmenuId = this.element.uniqueId().attr( "id" );
12224
                this.ids = {
12225
                        element: selectmenuId,
12226
                        button: selectmenuId + "-button",
12227
                        menu: selectmenuId + "-menu"
12228
                };
12229
12230
                this._drawButton();
12231
                this._drawMenu();
12232
12233
                if ( this.options.disabled ) {
12234
                        this.disable();
12235
                }
12236
        },
12237
12238
        _drawButton: function() {
12239
                var that = this;
12240
12241
                // Associate existing label with the new button
12242
                this.label = $( "label[for='" + this.ids.element + "']" ).attr( "for", this.ids.button );
12243
                this._on( this.label, {
12244
                        click: function( event ) {
12245
                                this.button.focus();
12246
                                event.preventDefault();
12247
                        }
12248
                });
12249
12250
                // Hide original select element
12251
                this.element.hide();
12252
12253
                // Create button
12254
                this.button = $( "<span>", {
12255
                        "class": "ui-selectmenu-button ui-widget ui-state-default ui-corner-all",
12256
                        tabindex: this.options.disabled ? -1 : 0,
12257
                        id: this.ids.button,
12258
                        role: "combobox",
12259
                        "aria-expanded": "false",
12260
                        "aria-autocomplete": "list",
12261
                        "aria-owns": this.ids.menu,
12262
                        "aria-haspopup": "true"
12263
                })
12264
                        .insertAfter( this.element );
12265
12266
                $( "<span>", {
12267
                        "class": "ui-icon " + this.options.icons.button
12268
                })
12269
                        .prependTo( this.button );
12270
12271
                this.buttonText = $( "<span>", {
12272
                        "class": "ui-selectmenu-text"
12273
                })
12274
                        .appendTo( this.button );
12275
12276
                this._setText( this.buttonText, this.element.find( "option:selected" ).text() );
12277
                this._resizeButton();
12278
12279
                this._on( this.button, this._buttonEvents );
12280
                this.button.one( "focusin", function() {
12281
12282
                        // Delay rendering the menu items until the button receives focus.
12283
                        // The menu may have already been rendered via a programmatic open.
12284
                        if ( !that.menuItems ) {
12285
                                that._refreshMenu();
12286
                        }
12287
                });
12288
                this._hoverable( this.button );
12289
                this._focusable( this.button );
12290
        },
12291
12292
        _drawMenu: function() {
12293
                var that = this;
12294
12295
                // Create menu
12296
                this.menu = $( "<ul>", {
12297
                        "aria-hidden": "true",
12298
                        "aria-labelledby": this.ids.button,
12299
                        id: this.ids.menu
12300
                });
12301
12302
                // Wrap menu
12303
                this.menuWrap = $( "<div>", {
12304
                        "class": "ui-selectmenu-menu ui-front"
12305
                })
12306
                        .append( this.menu )
12307
                        .appendTo( this._appendTo() );
12308
12309
                // Initialize menu widget
12310
                this.menuInstance = this.menu
12311
                        .menu({
12312
                                role: "listbox",
12313
                                select: function( event, ui ) {
12314
                                        event.preventDefault();
12315
12316
                                        // support: IE8
12317
                                        // If the item was selected via a click, the text selection
12318
                                        // will be destroyed in IE
12319
                                        that._setSelection();
12320
12321
                                        that._select( ui.item.data( "ui-selectmenu-item" ), event );
12322
                                },
12323
                                focus: function( event, ui ) {
12324
                                        var item = ui.item.data( "ui-selectmenu-item" );
12325
12326
                                        // Prevent inital focus from firing and check if its a newly focused item
12327
                                        if ( that.focusIndex != null && item.index !== that.focusIndex ) {
12328
                                                that._trigger( "focus", event, { item: item } );
12329
                                                if ( !that.isOpen ) {
12330
                                                        that._select( item, event );
12331
                                                }
12332
                                        }
12333
                                        that.focusIndex = item.index;
12334
12335
                                        that.button.attr( "aria-activedescendant",
12336
                                                that.menuItems.eq( item.index ).attr( "id" ) );
12337
                                }
12338
                        })
12339
                        .menu( "instance" );
12340
12341
                // Adjust menu styles to dropdown
12342
                this.menu
12343
                        .addClass( "ui-corner-bottom" )
12344
                        .removeClass( "ui-corner-all" );
12345
12346
                // Don't close the menu on mouseleave
12347
                this.menuInstance._off( this.menu, "mouseleave" );
12348
12349
                // Cancel the menu's collapseAll on document click
12350
                this.menuInstance._closeOnDocumentClick = function() {
12351
                        return false;
12352
                };
12353
12354
                // Selects often contain empty items, but never contain dividers
12355
                this.menuInstance._isDivider = function() {
12356
                        return false;
12357
                };
12358
        },
12359
12360
        refresh: function() {
12361
                this._refreshMenu();
12362
                this._setText( this.buttonText, this._getSelectedItem().text() );
12363
                if ( !this.options.width ) {
12364
                        this._resizeButton();
12365
                }
12366
        },
12367
12368
        _refreshMenu: function() {
12369
                this.menu.empty();
12370
12371
                var item,
12372
                        options = this.element.find( "option" );
12373
12374
                if ( !options.length ) {
12375
                        return;
12376
                }
12377
12378
                this._parseOptions( options );
12379
                this._renderMenu( this.menu, this.items );
12380
12381
                this.menuInstance.refresh();
12382
                this.menuItems = this.menu.find( "li" ).not( ".ui-selectmenu-optgroup" );
12383
12384
                item = this._getSelectedItem();
12385
12386
                // Update the menu to have the correct item focused
12387
                this.menuInstance.focus( null, item );
12388
                this._setAria( item.data( "ui-selectmenu-item" ) );
12389
12390
                // Set disabled state
12391
                this._setOption( "disabled", this.element.prop( "disabled" ) );
12392
        },
12393
12394
        open: function( event ) {
12395
                if ( this.options.disabled ) {
12396
                        return;
12397
                }
12398
12399
                // If this is the first time the menu is being opened, render the items
12400
                if ( !this.menuItems ) {
12401
                        this._refreshMenu();
12402
                } else {
12403
12404
                        // Menu clears focus on close, reset focus to selected item
12405
                        this.menu.find( ".ui-state-focus" ).removeClass( "ui-state-focus" );
12406
                        this.menuInstance.focus( null, this._getSelectedItem() );
12407
                }
12408
12409
                this.isOpen = true;
12410
                this._toggleAttr();
12411
                this._resizeMenu();
12412
                this._position();
12413
12414
                this._on( this.document, this._documentClick );
12415
12416
                this._trigger( "open", event );
12417
        },
12418
12419
        _position: function() {
12420
                this.menuWrap.position( $.extend( { of: this.button }, this.options.position ) );
12421
        },
12422
12423
        close: function( event ) {
12424
                if ( !this.isOpen ) {
12425
                        return;
12426
                }
12427
12428
                this.isOpen = false;
12429
                this._toggleAttr();
12430
12431
                this.range = null;
12432
                this._off( this.document );
12433
12434
                this._trigger( "close", event );
12435
        },
12436
12437
        widget: function() {
12438
                return this.button;
12439
        },
12440
12441
        menuWidget: function() {
12442
                return this.menu;
12443
        },
12444
12445
        _renderMenu: function( ul, items ) {
12446
                var that = this,
12447
                        currentOptgroup = "";
12448
12449
                $.each( items, function( index, item ) {
12450
                        if ( item.optgroup !== currentOptgroup ) {
12451
                                $( "<li>", {
12452
                                        "class": "ui-selectmenu-optgroup ui-menu-divider" +
12453
                                                ( item.element.parent( "optgroup" ).prop( "disabled" ) ?
12454
                                                        " ui-state-disabled" :
12455
                                                        "" ),
12456
                                        text: item.optgroup
12457
                                })
12458
                                        .appendTo( ul );
12459
12460
                                currentOptgroup = item.optgroup;
12461
                        }
12462
12463
                        that._renderItemData( ul, item );
12464
                });
12465
        },
12466
12467
        _renderItemData: function( ul, item ) {
12468
                return this._renderItem( ul, item ).data( "ui-selectmenu-item", item );
12469
        },
12470
12471
        _renderItem: function( ul, item ) {
12472
                var li = $( "<li>" );
12473
12474
                if ( item.disabled ) {
12475
                        li.addClass( "ui-state-disabled" );
12476
                }
12477
                this._setText( li, item.label );
12478
12479
                return li.appendTo( ul );
12480
        },
12481
12482
        _setText: function( element, value ) {
12483
                if ( value ) {
12484
                        element.text( value );
12485
                } else {
12486
                        element.html( "&#160;" );
12487
                }
12488
        },
12489
12490
        _move: function( direction, event ) {
12491
                var item, next,
12492
                        filter = ".ui-menu-item";
12493
12494
                if ( this.isOpen ) {
12495
                        item = this.menuItems.eq( this.focusIndex );
12496
                } else {
12497
                        item = this.menuItems.eq( this.element[ 0 ].selectedIndex );
12498
                        filter += ":not(.ui-state-disabled)";
12499
                }
12500
12501
                if ( direction === "first" || direction === "last" ) {
12502
                        next = item[ direction === "first" ? "prevAll" : "nextAll" ]( filter ).eq( -1 );
12503
                } else {
12504
                        next = item[ direction + "All" ]( filter ).eq( 0 );
12505
                }
12506
12507
                if ( next.length ) {
12508
                        this.menuInstance.focus( event, next );
12509
                }
12510
        },
12511
12512
        _getSelectedItem: function() {
12513
                return this.menuItems.eq( this.element[ 0 ].selectedIndex );
12514
        },
12515
12516
        _toggle: function( event ) {
12517
                this[ this.isOpen ? "close" : "open" ]( event );
12518
        },
12519
12520
        _setSelection: function() {
12521
                var selection;
12522
12523
                if ( !this.range ) {
12524
                        return;
12525
                }
12526
12527
                if ( window.getSelection ) {
12528
                        selection = window.getSelection();
12529
                        selection.removeAllRanges();
12530
                        selection.addRange( this.range );
12531
12532
                // support: IE8
12533
                } else {
12534
                        this.range.select();
12535
                }
12536
12537
                // support: IE
12538
                // Setting the text selection kills the button focus in IE, but
12539
                // restoring the focus doesn't kill the selection.
12540
                this.button.focus();
12541
        },
12542
12543
        _documentClick: {
12544
                mousedown: function( event ) {
12545
                        if ( !this.isOpen ) {
12546
                                return;
12547
                        }
12548
12549
                        if ( !$( event.target ).closest( ".ui-selectmenu-menu, #" + this.ids.button ).length ) {
12550
                                this.close( event );
12551
                        }
12552
                }
12553
        },
12554
12555
        _buttonEvents: {
12556
12557
                // Prevent text selection from being reset when interacting with the selectmenu (#10144)
12558
                mousedown: function() {
12559
                        var selection;
12560
12561
                        if ( window.getSelection ) {
12562
                                selection = window.getSelection();
12563
                                if ( selection.rangeCount ) {
12564
                                        this.range = selection.getRangeAt( 0 );
12565
                                }
12566
12567
                        // support: IE8
12568
                        } else {
12569
                                this.range = document.selection.createRange();
12570
                        }
12571
                },
12572
12573
                click: function( event ) {
12574
                        this._setSelection();
12575
                        this._toggle( event );
12576
                },
12577
12578
                keydown: function( event ) {
12579
                        var preventDefault = true;
12580
                        switch ( event.keyCode ) {
12581
                                case $.ui.keyCode.TAB:
12582
                                case $.ui.keyCode.ESCAPE:
12583
                                        this.close( event );
12584
                                        preventDefault = false;
12585
                                        break;
12586
                                case $.ui.keyCode.ENTER:
12587
                                        if ( this.isOpen ) {
12588
                                                this._selectFocusedItem( event );
12589
                                        }
12590
                                        break;
12591
                                case $.ui.keyCode.UP:
12592
                                        if ( event.altKey ) {
12593
                                                this._toggle( event );
12594
                                        } else {
12595
                                                this._move( "prev", event );
12596
                                        }
12597
                                        break;
12598
                                case $.ui.keyCode.DOWN:
12599
                                        if ( event.altKey ) {
12600
                                                this._toggle( event );
12601
                                        } else {
12602
                                                this._move( "next", event );
12603
                                        }
12604
                                        break;
12605
                                case $.ui.keyCode.SPACE:
12606
                                        if ( this.isOpen ) {
12607
                                                this._selectFocusedItem( event );
12608
                                        } else {
12609
                                                this._toggle( event );
12610
                                        }
12611
                                        break;
12612
                                case $.ui.keyCode.LEFT:
12613
                                        this._move( "prev", event );
12614
                                        break;
12615
                                case $.ui.keyCode.RIGHT:
12616
                                        this._move( "next", event );
12617
                                        break;
12618
                                case $.ui.keyCode.HOME:
12619
                                case $.ui.keyCode.PAGE_UP:
12620
                                        this._move( "first", event );
12621
                                        break;
12622
                                case $.ui.keyCode.END:
12623
                                case $.ui.keyCode.PAGE_DOWN:
12624
                                        this._move( "last", event );
12625
                                        break;
12626
                                default:
12627
                                        this.menu.trigger( event );
12628
                                        preventDefault = false;
12629
                        }
12630
12631
                        if ( preventDefault ) {
12632
                                event.preventDefault();
12633
                        }
12634
                }
12635
        },
12636
12637
        _selectFocusedItem: function( event ) {
12638
                var item = this.menuItems.eq( this.focusIndex );
12639
                if ( !item.hasClass( "ui-state-disabled" ) ) {
12640
                        this._select( item.data( "ui-selectmenu-item" ), event );
12641
                }
12642
        },
12643
12644
        _select: function( item, event ) {
12645
                var oldIndex = this.element[ 0 ].selectedIndex;
12646
12647
                // Change native select element
12648
                this.element[ 0 ].selectedIndex = item.index;
12649
                this._setText( this.buttonText, item.label );
12650
                this._setAria( item );
12651
                this._trigger( "select", event, { item: item } );
12652
12653
                if ( item.index !== oldIndex ) {
12654
                        this._trigger( "change", event, { item: item } );
12655
                }
12656
12657
                this.close( event );
12658
        },
12659
12660
        _setAria: function( item ) {
12661
                var id = this.menuItems.eq( item.index ).attr( "id" );
12662
12663
                this.button.attr({
12664
                        "aria-labelledby": id,
12665
                        "aria-activedescendant": id
12666
                });
12667
                this.menu.attr( "aria-activedescendant", id );
12668
        },
12669
12670
        _setOption: function( key, value ) {
12671
                if ( key === "icons" ) {
12672
                        this.button.find( "span.ui-icon" )
12673
                                .removeClass( this.options.icons.button )
12674
                                .addClass( value.button );
12675
                }
12676
12677
                this._super( key, value );
12678
12679
                if ( key === "appendTo" ) {
12680
                        this.menuWrap.appendTo( this._appendTo() );
12681
                }
12682
12683
                if ( key === "disabled" ) {
12684
                        this.menuInstance.option( "disabled", value );
12685
                        this.button
12686
                                .toggleClass( "ui-state-disabled", value )
12687
                                .attr( "aria-disabled", value );
12688
12689
                        this.element.prop( "disabled", value );
12690
                        if ( value ) {
12691
                                this.button.attr( "tabindex", -1 );
12692
                                this.close();
12693
                        } else {
12694
                                this.button.attr( "tabindex", 0 );
12695
                        }
12696
                }
12697
12698
                if ( key === "width" ) {
12699
                        this._resizeButton();
12700
                }
12701
        },
12702
12703
        _appendTo: function() {
12704
                var element = this.options.appendTo;
12705
12706
                if ( element ) {
12707
                        element = element.jquery || element.nodeType ?
12708
                                $( element ) :
12709
                                this.document.find( element ).eq( 0 );
12710
                }
12711
12712
                if ( !element || !element[ 0 ] ) {
12713
                        element = this.element.closest( ".ui-front" );
12714
                }
12715
12716
                if ( !element.length ) {
12717
                        element = this.document[ 0 ].body;
12718
                }
12719
12720
                return element;
12721
        },
12722
12723
        _toggleAttr: function() {
12724
                this.button
12725
                        .toggleClass( "ui-corner-top", this.isOpen )
12726
                        .toggleClass( "ui-corner-all", !this.isOpen )
12727
                        .attr( "aria-expanded", this.isOpen );
12728
                this.menuWrap.toggleClass( "ui-selectmenu-open", this.isOpen );
12729
                this.menu.attr( "aria-hidden", !this.isOpen );
12730
        },
12731
12732
        _resizeButton: function() {
12733
                var width = this.options.width;
12734
12735
                if ( !width ) {
12736
                        width = this.element.show().outerWidth();
12737
                        this.element.hide();
12738
                }
12739
12740
                this.button.outerWidth( width );
12741
        },
12742
12743
        _resizeMenu: function() {
12744
                this.menu.outerWidth( Math.max(
12745
                        this.button.outerWidth(),
12746
12747
                        // support: IE10
12748
                        // IE10 wraps long text (possibly a rounding bug)
12749
                        // so we add 1px to avoid the wrapping
12750
                        this.menu.width( "" ).outerWidth() + 1
12751
                ) );
12752
        },
12753
12754
        _getCreateOptions: function() {
12755
                return { disabled: this.element.prop( "disabled" ) };
12756
        },
12757
12758
        _parseOptions: function( options ) {
12759
                var data = [];
12760
                options.each(function( index, item ) {
12761
                        var option = $( item ),
12762
                                optgroup = option.parent( "optgroup" );
12763
                        data.push({
12764
                                element: option,
12765
                                index: index,
12766
                                value: option.val(),
12767
                                label: option.text(),
12768
                                optgroup: optgroup.attr( "label" ) || "",
12769
                                disabled: optgroup.prop( "disabled" ) || option.prop( "disabled" )
12770
                        });
12771
                });
12772
                this.items = data;
12773
        },
12774
12775
        _destroy: function() {
12776
                this.menuWrap.remove();
12777
                this.button.remove();
12778
                this.element.show();
12779
                this.element.removeUniqueId();
12780
                this.label.attr( "for", this.ids.element );
12781
        }
12782
});
12783
12784
12785
/*!
12786
 * jQuery UI Slider 1.11.4
12787
 * http://jqueryui.com
12788
 *
12789
 * Copyright jQuery Foundation and other contributors
12790
 * Released under the MIT license.
12791
 * http://jquery.org/license
12792
 *
12793
 * http://api.jqueryui.com/slider/
12794
 */
12795
12796
12797
var slider = $.widget( "ui.slider", $.ui.mouse, {
12798
        version: "1.11.4",
12799
        widgetEventPrefix: "slide",
12800
12801
        options: {
12802
                animate: false,
12803
                distance: 0,
12804
                max: 100,
12805
                min: 0,
12806
                orientation: "horizontal",
12807
                range: false,
12808
                step: 1,
12809
                value: 0,
12810
                values: null,
12811
12812
                // callbacks
12813
                change: null,
12814
                slide: null,
12815
                start: null,
12816
                stop: null
12817
        },
12818
12819
        // number of pages in a slider
12820
        // (how many times can you page up/down to go through the whole range)
12821
        numPages: 5,
12822
12823
        _create: function() {
12824
                this._keySliding = false;
12825
                this._mouseSliding = false;
12826
                this._animateOff = true;
12827
                this._handleIndex = null;
12828
                this._detectOrientation();
12829
                this._mouseInit();
12830
                this._calculateNewMax();
12831
12832
                this.element
12833
                        .addClass( "ui-slider" +
12834
                                " ui-slider-" + this.orientation +
12835
                                " ui-widget" +
12836
                                " ui-widget-content" +
12837
                                " ui-corner-all");
12838
12839
                this._refresh();
12840
                this._setOption( "disabled", this.options.disabled );
12841
12842
                this._animateOff = false;
12843
        },
12844
12845
        _refresh: function() {
12846
                this._createRange();
12847
                this._createHandles();
12848
                this._setupEvents();
12849
                this._refreshValue();
12850
        },
12851
12852
        _createHandles: function() {
12853
                var i, handleCount,
12854
                        options = this.options,
12855
                        existingHandles = this.element.find( ".ui-slider-handle" ).addClass( "ui-state-default ui-corner-all" ),
12856
                        handle = "<span class='ui-slider-handle ui-state-default ui-corner-all' tabindex='0'></span>",
12857
                        handles = [];
12858
12859
                handleCount = ( options.values && options.values.length ) || 1;
12860
12861
                if ( existingHandles.length > handleCount ) {
12862
                        existingHandles.slice( handleCount ).remove();
12863
                        existingHandles = existingHandles.slice( 0, handleCount );
12864
                }
12865
12866
                for ( i = existingHandles.length; i < handleCount; i++ ) {
12867
                        handles.push( handle );
12868
                }
12869
12870
                this.handles = existingHandles.add( $( handles.join( "" ) ).appendTo( this.element ) );
12871
12872
                this.handle = this.handles.eq( 0 );
12873
12874
                this.handles.each(function( i ) {
12875
                        $( this ).data( "ui-slider-handle-index", i );
12876
                });
12877
        },
12878
12879
        _createRange: function() {
12880
                var options = this.options,
12881
                        classes = "";
12882
12883
                if ( options.range ) {
12884
                        if ( options.range === true ) {
12885
                                if ( !options.values ) {
12886
                                        options.values = [ this._valueMin(), this._valueMin() ];
12887
                                } else if ( options.values.length && options.values.length !== 2 ) {
12888
                                        options.values = [ options.values[0], options.values[0] ];
12889
                                } else if ( $.isArray( options.values ) ) {
12890
                                        options.values = options.values.slice(0);
12891
                                }
12892
                        }
12893
12894
                        if ( !this.range || !this.range.length ) {
12895
                                this.range = $( "<div></div>" )
12896
                                        .appendTo( this.element );
12897
12898
                                classes = "ui-slider-range" +
12899
                                // note: this isn't the most fittingly semantic framework class for this element,
12900
                                // but worked best visually with a variety of themes
12901
                                " ui-widget-header ui-corner-all";
12902
                        } else {
12903
                                this.range.removeClass( "ui-slider-range-min ui-slider-range-max" )
12904
                                        // Handle range switching from true to min/max
12905
                                        .css({
12906
                                                "left": "",
12907
                                                "bottom": ""
12908
                                        });
12909
                        }
12910
12911
                        this.range.addClass( classes +
12912
                                ( ( options.range === "min" || options.range === "max" ) ? " ui-slider-range-" + options.range : "" ) );
12913
                } else {
12914
                        if ( this.range ) {
12915
                                this.range.remove();
12916
                        }
12917
                        this.range = null;
12918
                }
12919
        },
12920
12921
        _setupEvents: function() {
12922
                this._off( this.handles );
12923
                this._on( this.handles, this._handleEvents );
12924
                this._hoverable( this.handles );
12925
                this._focusable( this.handles );
12926
        },
12927
12928
        _destroy: function() {
12929
                this.handles.remove();
12930
                if ( this.range ) {
12931
                        this.range.remove();
12932
                }
12933
12934
                this.element
12935
                        .removeClass( "ui-slider" +
12936
                                " ui-slider-horizontal" +
12937
                                " ui-slider-vertical" +
12938
                                " ui-widget" +
12939
                                " ui-widget-content" +
12940
                                " ui-corner-all" );
12941
12942
                this._mouseDestroy();
12943
        },
12944
12945
        _mouseCapture: function( event ) {
12946
                var position, normValue, distance, closestHandle, index, allowed, offset, mouseOverHandle,
12947
                        that = this,
12948
                        o = this.options;
12949
12950
                if ( o.disabled ) {
12951
                        return false;
12952
                }
12953
12954
                this.elementSize = {
12955
                        width: this.element.outerWidth(),
12956
                        height: this.element.outerHeight()
12957
                };
12958
                this.elementOffset = this.element.offset();
12959
12960
                position = { x: event.pageX, y: event.pageY };
12961
                normValue = this._normValueFromMouse( position );
12962
                distance = this._valueMax() - this._valueMin() + 1;
12963
                this.handles.each(function( i ) {
12964
                        var thisDistance = Math.abs( normValue - that.values(i) );
12965
                        if (( distance > thisDistance ) ||
12966
                                ( distance === thisDistance &&
12967
                                        (i === that._lastChangedValue || that.values(i) === o.min ))) {
12968
                                distance = thisDistance;
12969
                                closestHandle = $( this );
12970
                                index = i;
12971
                        }
12972
                });
12973
12974
                allowed = this._start( event, index );
12975
                if ( allowed === false ) {
12976
                        return false;
12977
                }
12978
                this._mouseSliding = true;
12979
12980
                this._handleIndex = index;
12981
12982
                closestHandle
12983
                        .addClass( "ui-state-active" )
12984
                        .focus();
12985
12986
                offset = closestHandle.offset();
12987
                mouseOverHandle = !$( event.target ).parents().addBack().is( ".ui-slider-handle" );
12988
                this._clickOffset = mouseOverHandle ? { left: 0, top: 0 } : {
12989
                        left: event.pageX - offset.left - ( closestHandle.width() / 2 ),
12990
                        top: event.pageY - offset.top -
12991
                                ( closestHandle.height() / 2 ) -
12992
                                ( parseInt( closestHandle.css("borderTopWidth"), 10 ) || 0 ) -
12993
                                ( parseInt( closestHandle.css("borderBottomWidth"), 10 ) || 0) +
12994
                                ( parseInt( closestHandle.css("marginTop"), 10 ) || 0)
12995
                };
12996
12997
                if ( !this.handles.hasClass( "ui-state-hover" ) ) {
12998
                        this._slide( event, index, normValue );
12999
                }
13000
                this._animateOff = true;
13001
                return true;
13002
        },
13003
13004
        _mouseStart: function() {
13005
                return true;
13006
        },
13007
13008
        _mouseDrag: function( event ) {
13009
                var position = { x: event.pageX, y: event.pageY },
13010
                        normValue = this._normValueFromMouse( position );
13011
13012
                this._slide( event, this._handleIndex, normValue );
13013
13014
                return false;
13015
        },
13016
13017
        _mouseStop: function( event ) {
13018
                this.handles.removeClass( "ui-state-active" );
13019
                this._mouseSliding = false;
13020
13021
                this._stop( event, this._handleIndex );
13022
                this._change( event, this._handleIndex );
13023
13024
                this._handleIndex = null;
13025
                this._clickOffset = null;
13026
                this._animateOff = false;
13027
13028
                return false;
13029
        },
13030
13031
        _detectOrientation: function() {
13032
                this.orientation = ( this.options.orientation === "vertical" ) ? "vertical" : "horizontal";
13033
        },
13034
13035
        _normValueFromMouse: function( position ) {
13036
                var pixelTotal,
13037
                        pixelMouse,
13038
                        percentMouse,
13039
                        valueTotal,
13040
                        valueMouse;
13041
13042
                if ( this.orientation === "horizontal" ) {
13043
                        pixelTotal = this.elementSize.width;
13044
                        pixelMouse = position.x - this.elementOffset.left - ( this._clickOffset ? this._clickOffset.left : 0 );
13045
                } else {
13046
                        pixelTotal = this.elementSize.height;
13047
                        pixelMouse = position.y - this.elementOffset.top - ( this._clickOffset ? this._clickOffset.top : 0 );
13048
                }
13049
13050
                percentMouse = ( pixelMouse / pixelTotal );
13051
                if ( percentMouse > 1 ) {
13052
                        percentMouse = 1;
13053
                }
13054
                if ( percentMouse < 0 ) {
13055
                        percentMouse = 0;
13056
                }
13057
                if ( this.orientation === "vertical" ) {
13058
                        percentMouse = 1 - percentMouse;
13059
                }
13060
13061
                valueTotal = this._valueMax() - this._valueMin();
13062
                valueMouse = this._valueMin() + percentMouse * valueTotal;
13063
13064
                return this._trimAlignValue( valueMouse );
13065
        },
13066
13067
        _start: function( event, index ) {
13068
                var uiHash = {
13069
                        handle: this.handles[ index ],
13070
                        value: this.value()
13071
                };
13072
                if ( this.options.values && this.options.values.length ) {
13073
                        uiHash.value = this.values( index );
13074
                        uiHash.values = this.values();
13075
                }
13076
                return this._trigger( "start", event, uiHash );
13077
        },
13078
13079
        _slide: function( event, index, newVal ) {
13080
                var otherVal,
13081
                        newValues,
13082
                        allowed;
13083
13084
                if ( this.options.values && this.options.values.length ) {
13085
                        otherVal = this.values( index ? 0 : 1 );
13086
13087
                        if ( ( this.options.values.length === 2 && this.options.range === true ) &&
13088
                                        ( ( index === 0 && newVal > otherVal) || ( index === 1 && newVal < otherVal ) )
13089
                                ) {
13090
                                newVal = otherVal;
13091
                        }
13092
13093
                        if ( newVal !== this.values( index ) ) {
13094
                                newValues = this.values();
13095
                                newValues[ index ] = newVal;
13096
                                // A slide can be canceled by returning false from the slide callback
13097
                                allowed = this._trigger( "slide", event, {
13098
                                        handle: this.handles[ index ],
13099
                                        value: newVal,
13100
                                        values: newValues
13101
                                } );
13102
                                otherVal = this.values( index ? 0 : 1 );
13103
                                if ( allowed !== false ) {
13104
                                        this.values( index, newVal );
13105
                                }
13106
                        }
13107
                } else {
13108
                        if ( newVal !== this.value() ) {
13109
                                // A slide can be canceled by returning false from the slide callback
13110
                                allowed = this._trigger( "slide", event, {
13111
                                        handle: this.handles[ index ],
13112
                                        value: newVal
13113
                                } );
13114
                                if ( allowed !== false ) {
13115
                                        this.value( newVal );
13116
                                }
13117
                        }
13118
                }
13119
        },
13120
13121
        _stop: function( event, index ) {
13122
                var uiHash = {
13123
                        handle: this.handles[ index ],
13124
                        value: this.value()
13125
                };
13126
                if ( this.options.values && this.options.values.length ) {
13127
                        uiHash.value = this.values( index );
13128
                        uiHash.values = this.values();
13129
                }
13130
13131
                this._trigger( "stop", event, uiHash );
13132
        },
13133
13134
        _change: function( event, index ) {
13135
                if ( !this._keySliding && !this._mouseSliding ) {
13136
                        var uiHash = {
13137
                                handle: this.handles[ index ],
13138
                                value: this.value()
13139
                        };
13140
                        if ( this.options.values && this.options.values.length ) {
13141
                                uiHash.value = this.values( index );
13142
                                uiHash.values = this.values();
13143
                        }
13144
13145
                        //store the last changed value index for reference when handles overlap
13146
                        this._lastChangedValue = index;
13147
13148
                        this._trigger( "change", event, uiHash );
13149
                }
13150
        },
13151
13152
        value: function( newValue ) {
13153
                if ( arguments.length ) {
13154
                        this.options.value = this._trimAlignValue( newValue );
13155
                        this._refreshValue();
13156
                        this._change( null, 0 );
13157
                        return;
13158
                }
13159
13160
                return this._value();
13161
        },
13162
13163
        values: function( index, newValue ) {
13164
                var vals,
13165
                        newValues,
13166
                        i;
13167
13168
                if ( arguments.length > 1 ) {
13169
                        this.options.values[ index ] = this._trimAlignValue( newValue );
13170
                        this._refreshValue();
13171
                        this._change( null, index );
13172
                        return;
13173
                }
13174
13175
                if ( arguments.length ) {
13176
                        if ( $.isArray( arguments[ 0 ] ) ) {
13177
                                vals = this.options.values;
13178
                                newValues = arguments[ 0 ];
13179
                                for ( i = 0; i < vals.length; i += 1 ) {
13180
                                        vals[ i ] = this._trimAlignValue( newValues[ i ] );
13181
                                        this._change( null, i );
13182
                                }
13183
                                this._refreshValue();
13184
                        } else {
13185
                                if ( this.options.values && this.options.values.length ) {
13186
                                        return this._values( index );
13187
                                } else {
13188
                                        return this.value();
13189
                                }
13190
                        }
13191
                } else {
13192
                        return this._values();
13193
                }
13194
        },
13195
13196
        _setOption: function( key, value ) {
13197
                var i,
13198
                        valsLength = 0;
13199
13200
                if ( key === "range" && this.options.range === true ) {
13201
                        if ( value === "min" ) {
13202
                                this.options.value = this._values( 0 );
13203
                                this.options.values = null;
13204
                        } else if ( value === "max" ) {
13205
                                this.options.value = this._values( this.options.values.length - 1 );
13206
                                this.options.values = null;
13207
                        }
13208
                }
13209
13210
                if ( $.isArray( this.options.values ) ) {
13211
                        valsLength = this.options.values.length;
13212
                }
13213
13214
                if ( key === "disabled" ) {
13215
                        this.element.toggleClass( "ui-state-disabled", !!value );
13216
                }
13217
13218
                this._super( key, value );
13219
13220
                switch ( key ) {
13221
                        case "orientation":
13222
                                this._detectOrientation();
13223
                                this.element
13224
                                        .removeClass( "ui-slider-horizontal ui-slider-vertical" )
13225
                                        .addClass( "ui-slider-" + this.orientation );
13226
                                this._refreshValue();
13227
13228
                                // Reset positioning from previous orientation
13229
                                this.handles.css( value === "horizontal" ? "bottom" : "left", "" );
13230
                                break;
13231
                        case "value":
13232
                                this._animateOff = true;
13233
                                this._refreshValue();
13234
                                this._change( null, 0 );
13235
                                this._animateOff = false;
13236
                                break;
13237
                        case "values":
13238
                                this._animateOff = true;
13239
                                this._refreshValue();
13240
                                for ( i = 0; i < valsLength; i += 1 ) {
13241
                                        this._change( null, i );
13242
                                }
13243
                                this._animateOff = false;
13244
                                break;
13245
                        case "step":
13246
                        case "min":
13247
                        case "max":
13248
                                this._animateOff = true;
13249
                                this._calculateNewMax();
13250
                                this._refreshValue();
13251
                                this._animateOff = false;
13252
                                break;
13253
                        case "range":
13254
                                this._animateOff = true;
13255
                                this._refresh();
13256
                                this._animateOff = false;
13257
                                break;
13258
                }
13259
        },
13260
13261
        //internal value getter
13262
        // _value() returns value trimmed by min and max, aligned by step
13263
        _value: function() {
13264
                var val = this.options.value;
13265
                val = this._trimAlignValue( val );
13266
13267
                return val;
13268
        },
13269
13270
        //internal values getter
13271
        // _values() returns array of values trimmed by min and max, aligned by step
13272
        // _values( index ) returns single value trimmed by min and max, aligned by step
13273
        _values: function( index ) {
13274
                var val,
13275
                        vals,
13276
                        i;
13277
13278
                if ( arguments.length ) {
13279
                        val = this.options.values[ index ];
13280
                        val = this._trimAlignValue( val );
13281
13282
                        return val;
13283
                } else if ( this.options.values && this.options.values.length ) {
13284
                        // .slice() creates a copy of the array
13285
                        // this copy gets trimmed by min and max and then returned
13286
                        vals = this.options.values.slice();
13287
                        for ( i = 0; i < vals.length; i += 1) {
13288
                                vals[ i ] = this._trimAlignValue( vals[ i ] );
13289
                        }
13290
13291
                        return vals;
13292
                } else {
13293
                        return [];
13294
                }
13295
        },
13296
13297
        // returns the step-aligned value that val is closest to, between (inclusive) min and max
13298
        _trimAlignValue: function( val ) {
13299
                if ( val <= this._valueMin() ) {
13300
                        return this._valueMin();
13301
                }
13302
                if ( val >= this._valueMax() ) {
13303
                        return this._valueMax();
13304
                }
13305
                var step = ( this.options.step > 0 ) ? this.options.step : 1,
13306
                        valModStep = (val - this._valueMin()) % step,
13307
                        alignValue = val - valModStep;
13308
13309
                if ( Math.abs(valModStep) * 2 >= step ) {
13310
                        alignValue += ( valModStep > 0 ) ? step : ( -step );
13311
                }
13312
13313
                // Since JavaScript has problems with large floats, round
13314
                // the final value to 5 digits after the decimal point (see #4124)
13315
                return parseFloat( alignValue.toFixed(5) );
13316
        },
13317
13318
        _calculateNewMax: function() {
13319
                var max = this.options.max,
13320
                        min = this._valueMin(),
13321
                        step = this.options.step,
13322
                        aboveMin = Math.floor( ( +( max - min ).toFixed( this._precision() ) ) / step ) * step;
13323
                max = aboveMin + min;
13324
                this.max = parseFloat( max.toFixed( this._precision() ) );
13325
        },
13326
13327
        _precision: function() {
13328
                var precision = this._precisionOf( this.options.step );
13329
                if ( this.options.min !== null ) {
13330
                        precision = Math.max( precision, this._precisionOf( this.options.min ) );
13331
                }
13332
                return precision;
13333
        },
13334
13335
        _precisionOf: function( num ) {
13336
                var str = num.toString(),
13337
                        decimal = str.indexOf( "." );
13338
                return decimal === -1 ? 0 : str.length - decimal - 1;
13339
        },
13340
13341
        _valueMin: function() {
13342
                return this.options.min;
13343
        },
13344
13345
        _valueMax: function() {
13346
                return this.max;
13347
        },
13348
13349
        _refreshValue: function() {
13350
                var lastValPercent, valPercent, value, valueMin, valueMax,
13351
                        oRange = this.options.range,
13352
                        o = this.options,
13353
                        that = this,
13354
                        animate = ( !this._animateOff ) ? o.animate : false,
13355
                        _set = {};
13356
13357
                if ( this.options.values && this.options.values.length ) {
13358
                        this.handles.each(function( i ) {
13359
                                valPercent = ( that.values(i) - that._valueMin() ) / ( that._valueMax() - that._valueMin() ) * 100;
13360
                                _set[ that.orientation === "horizontal" ? "left" : "bottom" ] = valPercent + "%";
13361
                                $( this ).stop( 1, 1 )[ animate ? "animate" : "css" ]( _set, o.animate );
13362
                                if ( that.options.range === true ) {
13363
                                        if ( that.orientation === "horizontal" ) {
13364
                                                if ( i === 0 ) {
13365
                                                        that.range.stop( 1, 1 )[ animate ? "animate" : "css" ]( { left: valPercent + "%" }, o.animate );
13366
                                                }
13367
                                                if ( i === 1 ) {
13368
                                                        that.range[ animate ? "animate" : "css" ]( { width: ( valPercent - lastValPercent ) + "%" }, { queue: false, duration: o.animate } );
13369
                                                }
13370
                                        } else {
13371
                                                if ( i === 0 ) {
13372
                                                        that.range.stop( 1, 1 )[ animate ? "animate" : "css" ]( { bottom: ( valPercent ) + "%" }, o.animate );
13373
                                                }
13374
                                                if ( i === 1 ) {
13375
                                                        that.range[ animate ? "animate" : "css" ]( { height: ( valPercent - lastValPercent ) + "%" }, { queue: false, duration: o.animate } );
13376
                                                }
13377
                                        }
13378
                                }
13379
                                lastValPercent = valPercent;
13380
                        });
13381
                } else {
13382
                        value = this.value();
13383
                        valueMin = this._valueMin();
13384
                        valueMax = this._valueMax();
13385
                        valPercent = ( valueMax !== valueMin ) ?
13386
                                        ( value - valueMin ) / ( valueMax - valueMin ) * 100 :
13387
                                        0;
13388
                        _set[ this.orientation === "horizontal" ? "left" : "bottom" ] = valPercent + "%";
13389
                        this.handle.stop( 1, 1 )[ animate ? "animate" : "css" ]( _set, o.animate );
13390
13391
                        if ( oRange === "min" && this.orientation === "horizontal" ) {
13392
                                this.range.stop( 1, 1 )[ animate ? "animate" : "css" ]( { width: valPercent + "%" }, o.animate );
13393
                        }
13394
                        if ( oRange === "max" && this.orientation === "horizontal" ) {
13395
                                this.range[ animate ? "animate" : "css" ]( { width: ( 100 - valPercent ) + "%" }, { queue: false, duration: o.animate } );
13396
                        }
13397
                        if ( oRange === "min" && this.orientation === "vertical" ) {
13398
                                this.range.stop( 1, 1 )[ animate ? "animate" : "css" ]( { height: valPercent + "%" }, o.animate );
13399
                        }
13400
                        if ( oRange === "max" && this.orientation === "vertical" ) {
13401
                                this.range[ animate ? "animate" : "css" ]( { height: ( 100 - valPercent ) + "%" }, { queue: false, duration: o.animate } );
13402
                        }
13403
                }
13404
        },
13405
13406
        _handleEvents: {
13407
                keydown: function( event ) {
13408
                        var allowed, curVal, newVal, step,
13409
                                index = $( event.target ).data( "ui-slider-handle-index" );
13410
13411
                        switch ( event.keyCode ) {
13412
                                case $.ui.keyCode.HOME:
13413
                                case $.ui.keyCode.END:
13414
                                case $.ui.keyCode.PAGE_UP:
13415
                                case $.ui.keyCode.PAGE_DOWN:
13416
                                case $.ui.keyCode.UP:
13417
                                case $.ui.keyCode.RIGHT:
13418
                                case $.ui.keyCode.DOWN:
13419
                                case $.ui.keyCode.LEFT:
13420
                                        event.preventDefault();
13421
                                        if ( !this._keySliding ) {
13422
                                                this._keySliding = true;
13423
                                                $( event.target ).addClass( "ui-state-active" );
13424
                                                allowed = this._start( event, index );
13425
                                                if ( allowed === false ) {
13426
                                                        return;
13427
                                                }
13428
                                        }
13429
                                        break;
13430
                        }
13431
13432
                        step = this.options.step;
13433
                        if ( this.options.values && this.options.values.length ) {
13434
                                curVal = newVal = this.values( index );
13435
                        } else {
13436
                                curVal = newVal = this.value();
13437
                        }
13438
13439
                        switch ( event.keyCode ) {
13440
                                case $.ui.keyCode.HOME:
13441
                                        newVal = this._valueMin();
13442
                                        break;
13443
                                case $.ui.keyCode.END:
13444
                                        newVal = this._valueMax();
13445
                                        break;
13446
                                case $.ui.keyCode.PAGE_UP:
13447
                                        newVal = this._trimAlignValue(
13448
                                                curVal + ( ( this._valueMax() - this._valueMin() ) / this.numPages )
13449
                                        );
13450
                                        break;
13451
                                case $.ui.keyCode.PAGE_DOWN:
13452
                                        newVal = this._trimAlignValue(
13453
                                                curVal - ( (this._valueMax() - this._valueMin()) / this.numPages ) );
13454
                                        break;
13455
                                case $.ui.keyCode.UP:
13456
                                case $.ui.keyCode.RIGHT:
13457
                                        if ( curVal === this._valueMax() ) {
13458
                                                return;
13459
                                        }
13460
                                        newVal = this._trimAlignValue( curVal + step );
13461
                                        break;
13462
                                case $.ui.keyCode.DOWN:
13463
                                case $.ui.keyCode.LEFT:
13464
                                        if ( curVal === this._valueMin() ) {
13465
                                                return;
13466
                                        }
13467
                                        newVal = this._trimAlignValue( curVal - step );
13468
                                        break;
13469
                        }
13470
13471
                        this._slide( event, index, newVal );
13472
                },
13473
                keyup: function( event ) {
13474
                        var index = $( event.target ).data( "ui-slider-handle-index" );
13475
13476
                        if ( this._keySliding ) {
13477
                                this._keySliding = false;
13478
                                this._stop( event, index );
13479
                                this._change( event, index );
13480
                                $( event.target ).removeClass( "ui-state-active" );
13481
                        }
13482
                }
13483
        }
13484
});
13485
13486
13487
/*!
13488
 * jQuery UI Sortable 1.11.4
13489
 * http://jqueryui.com
13490
 *
13491
 * Copyright jQuery Foundation and other contributors
13492
 * Released under the MIT license.
13493
 * http://jquery.org/license
13494
 *
13495
 * http://api.jqueryui.com/sortable/
13496
 */
13497
13498
13499
var sortable = $.widget("ui.sortable", $.ui.mouse, {
13500
        version: "1.11.4",
13501
        widgetEventPrefix: "sort",
13502
        ready: false,
13503
        options: {
13504
                appendTo: "parent",
13505
                axis: false,
13506
                connectWith: false,
13507
                containment: false,
13508
                cursor: "auto",
13509
                cursorAt: false,
13510
                dropOnEmpty: true,
13511
                forcePlaceholderSize: false,
13512
                forceHelperSize: false,
13513
                grid: false,
13514
                handle: false,
13515
                helper: "original",
13516
                items: "> *",
13517
                opacity: false,
13518
                placeholder: false,
13519
                revert: false,
13520
                scroll: true,
13521
                scrollSensitivity: 20,
13522
                scrollSpeed: 20,
13523
                scope: "default",
13524
                tolerance: "intersect",
13525
                zIndex: 1000,
13526
13527
                // callbacks
13528
                activate: null,
13529
                beforeStop: null,
13530
                change: null,
13531
                deactivate: null,
13532
                out: null,
13533
                over: null,
13534
                receive: null,
13535
                remove: null,
13536
                sort: null,
13537
                start: null,
13538
                stop: null,
13539
                update: null
13540
        },
13541
13542
        _isOverAxis: function( x, reference, size ) {
13543
                return ( x >= reference ) && ( x < ( reference + size ) );
13544
        },
13545
13546
        _isFloating: function( item ) {
13547
                return (/left|right/).test(item.css("float")) || (/inline|table-cell/).test(item.css("display"));
13548
        },
13549
13550
        _create: function() {
13551
                this.containerCache = {};
13552
                this.element.addClass("ui-sortable");
13553
13554
                //Get the items
13555
                this.refresh();
13556
13557
                //Let's determine the parent's offset
13558
                this.offset = this.element.offset();
13559
13560
                //Initialize mouse events for interaction
13561
                this._mouseInit();
13562
13563
                this._setHandleClassName();
13564
13565
                //We're ready to go
13566
                this.ready = true;
13567
13568
        },
13569
13570
        _setOption: function( key, value ) {
13571
                this._super( key, value );
13572
13573
                if ( key === "handle" ) {
13574
                        this._setHandleClassName();
13575
                }
13576
        },
13577
13578
        _setHandleClassName: function() {
13579
                this.element.find( ".ui-sortable-handle" ).removeClass( "ui-sortable-handle" );
13580
                $.each( this.items, function() {
13581
                        ( this.instance.options.handle ?
13582
                                this.item.find( this.instance.options.handle ) : this.item )
13583
                                .addClass( "ui-sortable-handle" );
13584
                });
13585
        },
13586
13587
        _destroy: function() {
13588
                this.element
13589
                        .removeClass( "ui-sortable ui-sortable-disabled" )
13590
                        .find( ".ui-sortable-handle" )
13591
                                .removeClass( "ui-sortable-handle" );
13592
                this._mouseDestroy();
13593
13594
                for ( var i = this.items.length - 1; i >= 0; i-- ) {
13595
                        this.items[i].item.removeData(this.widgetName + "-item");
13596
                }
13597
13598
                return this;
13599
        },
13600
13601
        _mouseCapture: function(event, overrideHandle) {
13602
                var currentItem = null,
13603
                        validHandle = false,
13604
                        that = this;
13605
13606
                if (this.reverting) {
13607
                        return false;
13608
                }
13609
13610
                if(this.options.disabled || this.options.type === "static") {
13611
                        return false;
13612
                }
13613
13614
                //We have to refresh the items data once first
13615
                this._refreshItems(event);
13616
13617
                //Find out if the clicked node (or one of its parents) is a actual item in this.items
13618
                $(event.target).parents().each(function() {
13619
                        if($.data(this, that.widgetName + "-item") === that) {
13620
                                currentItem = $(this);
13621
                                return false;
13622
                        }
13623
                });
13624
                if($.data(event.target, that.widgetName + "-item") === that) {
13625
                        currentItem = $(event.target);
13626
                }
13627
13628
                if(!currentItem) {
13629
                        return false;
13630
                }
13631
                if(this.options.handle && !overrideHandle) {
13632
                        $(this.options.handle, currentItem).find("*").addBack().each(function() {
13633
                                if(this === event.target) {
13634
                                        validHandle = true;
13635
                                }
13636
                        });
13637
                        if(!validHandle) {
13638
                                return false;
13639
                        }
13640
                }
13641
13642
                this.currentItem = currentItem;
13643
                this._removeCurrentsFromItems();
13644
                return true;
13645
13646
        },
13647
13648
        _mouseStart: function(event, overrideHandle, noActivation) {
13649
13650
                var i, body,
13651
                        o = this.options;
13652
13653
                this.currentContainer = this;
13654
13655
                //We only need to call refreshPositions, because the refreshItems call has been moved to mouseCapture
13656
                this.refreshPositions();
13657
13658
                //Create and append the visible helper
13659
                this.helper = this._createHelper(event);
13660
13661
                //Cache the helper size
13662
                this._cacheHelperProportions();
13663
13664
                /*
13665
                 * - Position generation -
13666
                 * This block generates everything position related - it's the core of draggables.
13667
                 */
13668
13669
                //Cache the margins of the original element
13670
                this._cacheMargins();
13671
13672
                //Get the next scrolling parent
13673
                this.scrollParent = this.helper.scrollParent();
13674
13675
                //The element's absolute position on the page minus margins
13676
                this.offset = this.currentItem.offset();
13677
                this.offset = {
13678
                        top: this.offset.top - this.margins.top,
13679
                        left: this.offset.left - this.margins.left
13680
                };
13681
13682
                $.extend(this.offset, {
13683
                        click: { //Where the click happened, relative to the element
13684
                                left: event.pageX - this.offset.left,
13685
                                top: event.pageY - this.offset.top
13686
                        },
13687
                        parent: this._getParentOffset(),
13688
                        relative: this._getRelativeOffset() //This is a relative to absolute position minus the actual position calculation - only used for relative positioned helper
13689
                });
13690
13691
                // Only after we got the offset, we can change the helper's position to absolute
13692
                // TODO: Still need to figure out a way to make relative sorting possible
13693
                this.helper.css("position", "absolute");
13694
                this.cssPosition = this.helper.css("position");
13695
13696
                //Generate the original position
13697
                this.originalPosition = this._generatePosition(event);
13698
                this.originalPageX = event.pageX;
13699
                this.originalPageY = event.pageY;
13700
13701
                //Adjust the mouse offset relative to the helper if "cursorAt" is supplied
13702
                (o.cursorAt && this._adjustOffsetFromHelper(o.cursorAt));
13703
13704
                //Cache the former DOM position
13705
                this.domPosition = { prev: this.currentItem.prev()[0], parent: this.currentItem.parent()[0] };
13706
13707
                //If the helper is not the original, hide the original so it's not playing any role during the drag, won't cause anything bad this way
13708
                if(this.helper[0] !== this.currentItem[0]) {
13709
                        this.currentItem.hide();
13710
                }
13711
13712
                //Create the placeholder
13713
                this._createPlaceholder();
13714
13715
                //Set a containment if given in the options
13716
                if(o.containment) {
13717
                        this._setContainment();
13718
                }
13719
13720
                if( o.cursor && o.cursor !== "auto" ) { // cursor option
13721
                        body = this.document.find( "body" );
13722
13723
                        // support: IE
13724
                        this.storedCursor = body.css( "cursor" );
13725
                        body.css( "cursor", o.cursor );
13726
13727
                        this.storedStylesheet = $( "<style>*{ cursor: "+o.cursor+" !important; }</style>" ).appendTo( body );
13728
                }
13729
13730
                if(o.opacity) { // opacity option
13731
                        if (this.helper.css("opacity")) {
13732
                                this._storedOpacity = this.helper.css("opacity");
13733
                        }
13734
                        this.helper.css("opacity", o.opacity);
13735
                }
13736
13737
                if(o.zIndex) { // zIndex option
13738
                        if (this.helper.css("zIndex")) {
13739
                                this._storedZIndex = this.helper.css("zIndex");
13740
                        }
13741
                        this.helper.css("zIndex", o.zIndex);
13742
                }
13743
13744
                //Prepare scrolling
13745
                if(this.scrollParent[0] !== this.document[0] && this.scrollParent[0].tagName !== "HTML") {
13746
                        this.overflowOffset = this.scrollParent.offset();
13747
                }
13748
13749
                //Call callbacks
13750
                this._trigger("start", event, this._uiHash());
13751
13752
                //Recache the helper size
13753
                if(!this._preserveHelperProportions) {
13754
                        this._cacheHelperProportions();
13755
                }
13756
13757
13758
                //Post "activate" events to possible containers
13759
                if( !noActivation ) {
13760
                        for ( i = this.containers.length - 1; i >= 0; i-- ) {
13761
                                this.containers[ i ]._trigger( "activate", event, this._uiHash( this ) );
13762
                        }
13763
                }
13764
13765
                //Prepare possible droppables
13766
                if($.ui.ddmanager) {
13767
                        $.ui.ddmanager.current = this;
13768
                }
13769
13770
                if ($.ui.ddmanager && !o.dropBehaviour) {
13771
                        $.ui.ddmanager.prepareOffsets(this, event);
13772
                }
13773
13774
                this.dragging = true;
13775
13776
                this.helper.addClass("ui-sortable-helper");
13777
                this._mouseDrag(event); //Execute the drag once - this causes the helper not to be visible before getting its correct position
13778
                return true;
13779
13780
        },
13781
13782
        _mouseDrag: function(event) {
13783
                var i, item, itemElement, intersection,
13784
                        o = this.options,
13785
                        scrolled = false;
13786
13787
                //Compute the helpers position
13788
                this.position = this._generatePosition(event);
13789
                this.positionAbs = this._convertPositionTo("absolute");
13790
13791
                if (!this.lastPositionAbs) {
13792
                        this.lastPositionAbs = this.positionAbs;
13793
                }
13794
13795
                //Do scrolling
13796
                if(this.options.scroll) {
13797
                        if(this.scrollParent[0] !== this.document[0] && this.scrollParent[0].tagName !== "HTML") {
13798
13799
                                if((this.overflowOffset.top + this.scrollParent[0].offsetHeight) - event.pageY < o.scrollSensitivity) {
13800
                                        this.scrollParent[0].scrollTop = scrolled = this.scrollParent[0].scrollTop + o.scrollSpeed;
13801
                                } else if(event.pageY - this.overflowOffset.top < o.scrollSensitivity) {
13802
                                        this.scrollParent[0].scrollTop = scrolled = this.scrollParent[0].scrollTop - o.scrollSpeed;
13803
                                }
13804
13805
                                if((this.overflowOffset.left + this.scrollParent[0].offsetWidth) - event.pageX < o.scrollSensitivity) {
13806
                                        this.scrollParent[0].scrollLeft = scrolled = this.scrollParent[0].scrollLeft + o.scrollSpeed;
13807
                                } else if(event.pageX - this.overflowOffset.left < o.scrollSensitivity) {
13808
                                        this.scrollParent[0].scrollLeft = scrolled = this.scrollParent[0].scrollLeft - o.scrollSpeed;
13809
                                }
13810
13811
                        } else {
13812
13813
                                if(event.pageY - this.document.scrollTop() < o.scrollSensitivity) {
13814
                                        scrolled = this.document.scrollTop(this.document.scrollTop() - o.scrollSpeed);
13815
                                } else if(this.window.height() - (event.pageY - this.document.scrollTop()) < o.scrollSensitivity) {
13816
                                        scrolled = this.document.scrollTop(this.document.scrollTop() + o.scrollSpeed);
13817
                                }
13818
13819
                                if(event.pageX - this.document.scrollLeft() < o.scrollSensitivity) {
13820
                                        scrolled = this.document.scrollLeft(this.document.scrollLeft() - o.scrollSpeed);
13821
                                } else if(this.window.width() - (event.pageX - this.document.scrollLeft()) < o.scrollSensitivity) {
13822
                                        scrolled = this.document.scrollLeft(this.document.scrollLeft() + o.scrollSpeed);
13823
                                }
13824
13825
                        }
13826
13827
                        if(scrolled !== false && $.ui.ddmanager && !o.dropBehaviour) {
13828
                                $.ui.ddmanager.prepareOffsets(this, event);
13829
                        }
13830
                }
13831
13832
                //Regenerate the absolute position used for position checks
13833
                this.positionAbs = this._convertPositionTo("absolute");
13834
13835
                //Set the helper position
13836
                if(!this.options.axis || this.options.axis !== "y") {
13837
                        this.helper[0].style.left = this.position.left+"px";
13838
                }
13839
                if(!this.options.axis || this.options.axis !== "x") {
13840
                        this.helper[0].style.top = this.position.top+"px";
13841
                }
13842
13843
                //Rearrange
13844
                for (i = this.items.length - 1; i >= 0; i--) {
13845
13846
                        //Cache variables and intersection, continue if no intersection
13847
                        item = this.items[i];
13848
                        itemElement = item.item[0];
13849
                        intersection = this._intersectsWithPointer(item);
13850
                        if (!intersection) {
13851
                                continue;
13852
                        }
13853
13854
                        // Only put the placeholder inside the current Container, skip all
13855
                        // items from other containers. This works because when moving
13856
                        // an item from one container to another the
13857
                        // currentContainer is switched before the placeholder is moved.
13858
                        //
13859
                        // Without this, moving items in "sub-sortables" can cause
13860
                        // the placeholder to jitter between the outer and inner container.
13861
                        if (item.instance !== this.currentContainer) {
13862
                                continue;
13863
                        }
13864
13865
                        // cannot intersect with itself
13866
                        // no useless actions that have been done before
13867
                        // no action if the item moved is the parent of the item checked
13868
                        if (itemElement !== this.currentItem[0] &&
13869
                                this.placeholder[intersection === 1 ? "next" : "prev"]()[0] !== itemElement &&
13870
                                !$.contains(this.placeholder[0], itemElement) &&
13871
                                (this.options.type === "semi-dynamic" ? !$.contains(this.element[0], itemElement) : true)
13872
                        ) {
13873
13874
                                this.direction = intersection === 1 ? "down" : "up";
13875
13876
                                if (this.options.tolerance === "pointer" || this._intersectsWithSides(item)) {
13877
                                        this._rearrange(event, item);
13878
                                } else {
13879
                                        break;
13880
                                }
13881
13882
                                this._trigger("change", event, this._uiHash());
13883
                                break;
13884
                        }
13885
                }
13886
13887
                //Post events to containers
13888
                this._contactContainers(event);
13889
13890
                //Interconnect with droppables
13891
                if($.ui.ddmanager) {
13892
                        $.ui.ddmanager.drag(this, event);
13893
                }
13894
13895
                //Call callbacks
13896
                this._trigger("sort", event, this._uiHash());
13897
13898
                this.lastPositionAbs = this.positionAbs;
13899
                return false;
13900
13901
        },
13902
13903
        _mouseStop: function(event, noPropagation) {
13904
13905
                if(!event) {
13906
                        return;
13907
                }
13908
13909
                //If we are using droppables, inform the manager about the drop
13910
                if ($.ui.ddmanager && !this.options.dropBehaviour) {
13911
                        $.ui.ddmanager.drop(this, event);
13912
                }
13913
13914
                if(this.options.revert) {
13915
                        var that = this,
13916
                                cur = this.placeholder.offset(),
13917
                                axis = this.options.axis,
13918
                                animation = {};
13919
13920
                        if ( !axis || axis === "x" ) {
13921
                                animation.left = cur.left - this.offset.parent.left - this.margins.left + (this.offsetParent[0] === this.document[0].body ? 0 : this.offsetParent[0].scrollLeft);
13922
                        }
13923
                        if ( !axis || axis === "y" ) {
13924
                                animation.top = cur.top - this.offset.parent.top - this.margins.top + (this.offsetParent[0] === this.document[0].body ? 0 : this.offsetParent[0].scrollTop);
13925
                        }
13926
                        this.reverting = true;
13927
                        $(this.helper).animate( animation, parseInt(this.options.revert, 10) || 500, function() {
13928
                                that._clear(event);
13929
                        });
13930
                } else {
13931
                        this._clear(event, noPropagation);
13932
                }
13933
13934
                return false;
13935
13936
        },
13937
13938
        cancel: function() {
13939
13940
                if(this.dragging) {
13941
13942
                        this._mouseUp({ target: null });
13943
13944
                        if(this.options.helper === "original") {
13945
                                this.currentItem.css(this._storedCSS).removeClass("ui-sortable-helper");
13946
                        } else {
13947
                                this.currentItem.show();
13948
                        }
13949
13950
                        //Post deactivating events to containers
13951
                        for (var i = this.containers.length - 1; i >= 0; i--){
13952
                                this.containers[i]._trigger("deactivate", null, this._uiHash(this));
13953
                                if(this.containers[i].containerCache.over) {
13954
                                        this.containers[i]._trigger("out", null, this._uiHash(this));
13955
                                        this.containers[i].containerCache.over = 0;
13956
                                }
13957
                        }
13958
13959
                }
13960
13961
                if (this.placeholder) {
13962
                        //$(this.placeholder[0]).remove(); would have been the jQuery way - unfortunately, it unbinds ALL events from the original node!
13963
                        if(this.placeholder[0].parentNode) {
13964
                                this.placeholder[0].parentNode.removeChild(this.placeholder[0]);
13965
                        }
13966
                        if(this.options.helper !== "original" && this.helper && this.helper[0].parentNode) {
13967
                                this.helper.remove();
13968
                        }
13969
13970
                        $.extend(this, {
13971
                                helper: null,
13972
                                dragging: false,
13973
                                reverting: false,
13974
                                _noFinalSort: null
13975
                        });
13976
13977
                        if(this.domPosition.prev) {
13978
                                $(this.domPosition.prev).after(this.currentItem);
13979
                        } else {
13980
                                $(this.domPosition.parent).prepend(this.currentItem);
13981
                        }
13982
                }
13983
13984
                return this;
13985
13986
        },
13987
13988
        serialize: function(o) {
13989
13990
                var items = this._getItemsAsjQuery(o && o.connected),
13991
                        str = [];
13992
                o = o || {};
13993
13994
                $(items).each(function() {
13995
                        var res = ($(o.item || this).attr(o.attribute || "id") || "").match(o.expression || (/(.+)[\-=_](.+)/));
13996
                        if (res) {
13997
                                str.push((o.key || res[1]+"[]")+"="+(o.key && o.expression ? res[1] : res[2]));
13998
                        }
13999
                });
14000
14001
                if(!str.length && o.key) {
14002
                        str.push(o.key + "=");
14003
                }
14004
14005
                return str.join("&");
14006
14007
        },
14008
14009
        toArray: function(o) {
14010
14011
                var items = this._getItemsAsjQuery(o && o.connected),
14012
                        ret = [];
14013
14014
                o = o || {};
14015
14016
                items.each(function() { ret.push($(o.item || this).attr(o.attribute || "id") || ""); });
14017
                return ret;
14018
14019
        },
14020
14021
        /* Be careful with the following core functions */
14022
        _intersectsWith: function(item) {
14023
14024
                var x1 = this.positionAbs.left,
14025
                        x2 = x1 + this.helperProportions.width,
14026
                        y1 = this.positionAbs.top,
14027
                        y2 = y1 + this.helperProportions.height,
14028
                        l = item.left,
14029
                        r = l + item.width,
14030
                        t = item.top,
14031
                        b = t + item.height,
14032
                        dyClick = this.offset.click.top,
14033
                        dxClick = this.offset.click.left,
14034
                        isOverElementHeight = ( this.options.axis === "x" ) || ( ( y1 + dyClick ) > t && ( y1 + dyClick ) < b ),
14035
                        isOverElementWidth = ( this.options.axis === "y" ) || ( ( x1 + dxClick ) > l && ( x1 + dxClick ) < r ),
14036
                        isOverElement = isOverElementHeight && isOverElementWidth;
14037
14038
                if ( this.options.tolerance === "pointer" ||
14039
                        this.options.forcePointerForContainers ||
14040
                        (this.options.tolerance !== "pointer" && this.helperProportions[this.floating ? "width" : "height"] > item[this.floating ? "width" : "height"])
14041
                ) {
14042
                        return isOverElement;
14043
                } else {
14044
14045
                        return (l < x1 + (this.helperProportions.width / 2) && // Right Half
14046
                                x2 - (this.helperProportions.width / 2) < r && // Left Half
14047
                                t < y1 + (this.helperProportions.height / 2) && // Bottom Half
14048
                                y2 - (this.helperProportions.height / 2) < b ); // Top Half
14049
14050
                }
14051
        },
14052
14053
        _intersectsWithPointer: function(item) {
14054
14055
                var isOverElementHeight = (this.options.axis === "x") || this._isOverAxis(this.positionAbs.top + this.offset.click.top, item.top, item.height),
14056
                        isOverElementWidth = (this.options.axis === "y") || this._isOverAxis(this.positionAbs.left + this.offset.click.left, item.left, item.width),
14057
                        isOverElement = isOverElementHeight && isOverElementWidth,
14058
                        verticalDirection = this._getDragVerticalDirection(),
14059
                        horizontalDirection = this._getDragHorizontalDirection();
14060
14061
                if (!isOverElement) {
14062
                        return false;
14063
                }
14064
14065
                return this.floating ?
14066
                        ( ((horizontalDirection && horizontalDirection === "right") || verticalDirection === "down") ? 2 : 1 )
14067
                        : ( verticalDirection && (verticalDirection === "down" ? 2 : 1) );
14068
14069
        },
14070
14071
        _intersectsWithSides: function(item) {
14072
14073
                var isOverBottomHalf = this._isOverAxis(this.positionAbs.top + this.offset.click.top, item.top + (item.height/2), item.height),
14074
                        isOverRightHalf = this._isOverAxis(this.positionAbs.left + this.offset.click.left, item.left + (item.width/2), item.width),
14075
                        verticalDirection = this._getDragVerticalDirection(),
14076
                        horizontalDirection = this._getDragHorizontalDirection();
14077
14078
                if (this.floating && horizontalDirection) {
14079
                        return ((horizontalDirection === "right" && isOverRightHalf) || (horizontalDirection === "left" && !isOverRightHalf));
14080
                } else {
14081
                        return verticalDirection && ((verticalDirection === "down" && isOverBottomHalf) || (verticalDirection === "up" && !isOverBottomHalf));
14082
                }
14083
14084
        },
14085
14086
        _getDragVerticalDirection: function() {
14087
                var delta = this.positionAbs.top - this.lastPositionAbs.top;
14088
                return delta !== 0 && (delta > 0 ? "down" : "up");
14089
        },
14090
14091
        _getDragHorizontalDirection: function() {
14092
                var delta = this.positionAbs.left - this.lastPositionAbs.left;
14093
                return delta !== 0 && (delta > 0 ? "right" : "left");
14094
        },
14095
14096
        refresh: function(event) {
14097
                this._refreshItems(event);
14098
                this._setHandleClassName();
14099
                this.refreshPositions();
14100
                return this;
14101
        },
14102
14103
        _connectWith: function() {
14104
                var options = this.options;
14105
                return options.connectWith.constructor === String ? [options.connectWith] : options.connectWith;
14106
        },
14107
14108
        _getItemsAsjQuery: function(connected) {
14109
14110
                var i, j, cur, inst,
14111
                        items = [],
14112
                        queries = [],
14113
                        connectWith = this._connectWith();
14114
14115
                if(connectWith && connected) {
14116
                        for (i = connectWith.length - 1; i >= 0; i--){
14117
                                cur = $(connectWith[i], this.document[0]);
14118
                                for ( j = cur.length - 1; j >= 0; j--){
14119
                                        inst = $.data(cur[j], this.widgetFullName);
14120
                                        if(inst && inst !== this && !inst.options.disabled) {
14121
                                                queries.push([$.isFunction(inst.options.items) ? inst.options.items.call(inst.element) : $(inst.options.items, inst.element).not(".ui-sortable-helper").not(".ui-sortable-placeholder"), inst]);
14122
                                        }
14123
                                }
14124
                        }
14125
                }
14126
14127
                queries.push([$.isFunction(this.options.items) ? this.options.items.call(this.element, null, { options: this.options, item: this.currentItem }) : $(this.options.items, this.element).not(".ui-sortable-helper").not(".ui-sortable-placeholder"), this]);
14128
14129
                function addItems() {
14130
                        items.push( this );
14131
                }
14132
                for (i = queries.length - 1; i >= 0; i--){
14133
                        queries[i][0].each( addItems );
14134
                }
14135
14136
                return $(items);
14137
14138
        },
14139
14140
        _removeCurrentsFromItems: function() {
14141
14142
                var list = this.currentItem.find(":data(" + this.widgetName + "-item)");
14143
14144
                this.items = $.grep(this.items, function (item) {
14145
                        for (var j=0; j < list.length; j++) {
14146
                                if(list[j] === item.item[0]) {
14147
                                        return false;
14148
                                }
14149
                        }
14150
                        return true;
14151
                });
14152
14153
        },
14154
14155
        _refreshItems: function(event) {
14156
14157
                this.items = [];
14158
                this.containers = [this];
14159
14160
                var i, j, cur, inst, targetData, _queries, item, queriesLength,
14161
                        items = this.items,
14162
                        queries = [[$.isFunction(this.options.items) ? this.options.items.call(this.element[0], event, { item: this.currentItem }) : $(this.options.items, this.element), this]],
14163
                        connectWith = this._connectWith();
14164
14165
                if(connectWith && this.ready) { //Shouldn't be run the first time through due to massive slow-down
14166
                        for (i = connectWith.length - 1; i >= 0; i--){
14167
                                cur = $(connectWith[i], this.document[0]);
14168
                                for (j = cur.length - 1; j >= 0; j--){
14169
                                        inst = $.data(cur[j], this.widgetFullName);
14170
                                        if(inst && inst !== this && !inst.options.disabled) {
14171
                                                queries.push([$.isFunction(inst.options.items) ? inst.options.items.call(inst.element[0], event, { item: this.currentItem }) : $(inst.options.items, inst.element), inst]);
14172
                                                this.containers.push(inst);
14173
                                        }
14174
                                }
14175
                        }
14176
                }
14177
14178
                for (i = queries.length - 1; i >= 0; i--) {
14179
                        targetData = queries[i][1];
14180
                        _queries = queries[i][0];
14181
14182
                        for (j=0, queriesLength = _queries.length; j < queriesLength; j++) {
14183
                                item = $(_queries[j]);
14184
14185
                                item.data(this.widgetName + "-item", targetData); // Data for target checking (mouse manager)
14186
14187
                                items.push({
14188
                                        item: item,
14189
                                        instance: targetData,
14190
                                        width: 0, height: 0,
14191
                                        left: 0, top: 0
14192
                                });
14193
                        }
14194
                }
14195
14196
        },
14197
14198
        refreshPositions: function(fast) {
14199
14200
                // Determine whether items are being displayed horizontally
14201
                this.floating = this.items.length ?
14202
                        this.options.axis === "x" || this._isFloating( this.items[ 0 ].item ) :
14203
                        false;
14204
14205
                //This has to be redone because due to the item being moved out/into the offsetParent, the offsetParent's position will change
14206
                if(this.offsetParent && this.helper) {
14207
                        this.offset.parent = this._getParentOffset();
14208
                }
14209
14210
                var i, item, t, p;
14211
14212
                for (i = this.items.length - 1; i >= 0; i--){
14213
                        item = this.items[i];
14214
14215
                        //We ignore calculating positions of all connected containers when we're not over them
14216
                        if(item.instance !== this.currentContainer && this.currentContainer && item.item[0] !== this.currentItem[0]) {
14217
                                continue;
14218
                        }
14219
14220
                        t = this.options.toleranceElement ? $(this.options.toleranceElement, item.item) : item.item;
14221
14222
                        if (!fast) {
14223
                                item.width = t.outerWidth();
14224
                                item.height = t.outerHeight();
14225
                        }
14226
14227
                        p = t.offset();
14228
                        item.left = p.left;
14229
                        item.top = p.top;
14230
                }
14231
14232
                if(this.options.custom && this.options.custom.refreshContainers) {
14233
                        this.options.custom.refreshContainers.call(this);
14234
                } else {
14235
                        for (i = this.containers.length - 1; i >= 0; i--){
14236
                                p = this.containers[i].element.offset();
14237
                                this.containers[i].containerCache.left = p.left;
14238
                                this.containers[i].containerCache.top = p.top;
14239
                                this.containers[i].containerCache.width = this.containers[i].element.outerWidth();
14240
                                this.containers[i].containerCache.height = this.containers[i].element.outerHeight();
14241
                        }
14242
                }
14243
14244
                return this;
14245
        },
14246
14247
        _createPlaceholder: function(that) {
14248
                that = that || this;
14249
                var className,
14250
                        o = that.options;
14251
14252
                if(!o.placeholder || o.placeholder.constructor === String) {
14253
                        className = o.placeholder;
14254
                        o.placeholder = {
14255
                                element: function() {
14256
14257
                                        var nodeName = that.currentItem[0].nodeName.toLowerCase(),
14258
                                                element = $( "<" + nodeName + ">", that.document[0] )
14259
                                                        .addClass(className || that.currentItem[0].className+" ui-sortable-placeholder")
14260
                                                        .removeClass("ui-sortable-helper");
14261
14262
                                        if ( nodeName === "tbody" ) {
14263
                                                that._createTrPlaceholder(
14264
                                                        that.currentItem.find( "tr" ).eq( 0 ),
14265
                                                        $( "<tr>", that.document[ 0 ] ).appendTo( element )
14266
                                                );
14267
                                        } else if ( nodeName === "tr" ) {
14268
                                                that._createTrPlaceholder( that.currentItem, element );
14269
                                        } else if ( nodeName === "img" ) {
14270
                                                element.attr( "src", that.currentItem.attr( "src" ) );
14271
                                        }
14272
14273
                                        if ( !className ) {
14274
                                                element.css( "visibility", "hidden" );
14275
                                        }
14276
14277
                                        return element;
14278
                                },
14279
                                update: function(container, p) {
14280
14281
                                        // 1. If a className is set as 'placeholder option, we don't force sizes - the class is responsible for that
14282
                                        // 2. The option 'forcePlaceholderSize can be enabled to force it even if a class name is specified
14283
                                        if(className && !o.forcePlaceholderSize) {
14284
                                                return;
14285
                                        }
14286
14287
                                        //If the element doesn't have a actual height by itself (without styles coming from a stylesheet), it receives the inline height from the dragged item
14288
                                        if(!p.height()) { p.height(that.currentItem.innerHeight() - parseInt(that.currentItem.css("paddingTop")||0, 10) - parseInt(that.currentItem.css("paddingBottom")||0, 10)); }
14289
                                        if(!p.width()) { p.width(that.currentItem.innerWidth() - parseInt(that.currentItem.css("paddingLeft")||0, 10) - parseInt(that.currentItem.css("paddingRight")||0, 10)); }
14290
                                }
14291
                        };
14292
                }
14293
14294
                //Create the placeholder
14295
                that.placeholder = $(o.placeholder.element.call(that.element, that.currentItem));
14296
14297
                //Append it after the actual current item
14298
                that.currentItem.after(that.placeholder);
14299
14300
                //Update the size of the placeholder (TODO: Logic to fuzzy, see line 316/317)
14301
                o.placeholder.update(that, that.placeholder);
14302
14303
        },
14304
14305
        _createTrPlaceholder: function( sourceTr, targetTr ) {
14306
                var that = this;
14307
14308
                sourceTr.children().each(function() {
14309
                        $( "<td>&#160;</td>", that.document[ 0 ] )
14310
                                .attr( "colspan", $( this ).attr( "colspan" ) || 1 )
14311
                                .appendTo( targetTr );
14312
                });
14313
        },
14314
14315
        _contactContainers: function(event) {
14316
                var i, j, dist, itemWithLeastDistance, posProperty, sizeProperty, cur, nearBottom, floating, axis,
14317
                        innermostContainer = null,
14318
                        innermostIndex = null;
14319
14320
                // get innermost container that intersects with item
14321
                for (i = this.containers.length - 1; i >= 0; i--) {
14322
14323
                        // never consider a container that's located within the item itself
14324
                        if($.contains(this.currentItem[0], this.containers[i].element[0])) {
14325
                                continue;
14326
                        }
14327
14328
                        if(this._intersectsWith(this.containers[i].containerCache)) {
14329
14330
                                // if we've already found a container and it's more "inner" than this, then continue
14331
                                if(innermostContainer && $.contains(this.containers[i].element[0], innermostContainer.element[0])) {
14332
                                        continue;
14333
                                }
14334
14335
                                innermostContainer = this.containers[i];
14336
                                innermostIndex = i;
14337
14338
                        } else {
14339
                                // container doesn't intersect. trigger "out" event if necessary
14340
                                if(this.containers[i].containerCache.over) {
14341
                                        this.containers[i]._trigger("out", event, this._uiHash(this));
14342
                                        this.containers[i].containerCache.over = 0;
14343
                                }
14344
                        }
14345
14346
                }
14347
14348
                // if no intersecting containers found, return
14349
                if(!innermostContainer) {
14350
                        return;
14351
                }
14352
14353
                // move the item into the container if it's not there already
14354
                if(this.containers.length === 1) {
14355
                        if (!this.containers[innermostIndex].containerCache.over) {
14356
                                this.containers[innermostIndex]._trigger("over", event, this._uiHash(this));
14357
                                this.containers[innermostIndex].containerCache.over = 1;
14358
                        }
14359
                } else {
14360
14361
                        //When entering a new container, we will find the item with the least distance and append our item near it
14362
                        dist = 10000;
14363
                        itemWithLeastDistance = null;
14364
                        floating = innermostContainer.floating || this._isFloating(this.currentItem);
14365
                        posProperty = floating ? "left" : "top";
14366
                        sizeProperty = floating ? "width" : "height";
14367
                        axis = floating ? "clientX" : "clientY";
14368
14369
                        for (j = this.items.length - 1; j >= 0; j--) {
14370
                                if(!$.contains(this.containers[innermostIndex].element[0], this.items[j].item[0])) {
14371
                                        continue;
14372
                                }
14373
                                if(this.items[j].item[0] === this.currentItem[0]) {
14374
                                        continue;
14375
                                }
14376
14377
                                cur = this.items[j].item.offset()[posProperty];
14378
                                nearBottom = false;
14379
                                if ( event[ axis ] - cur > this.items[ j ][ sizeProperty ] / 2 ) {
14380
                                        nearBottom = true;
14381
                                }
14382
14383
                                if ( Math.abs( event[ axis ] - cur ) < dist ) {
14384
                                        dist = Math.abs( event[ axis ] - cur );
14385
                                        itemWithLeastDistance = this.items[ j ];
14386
                                        this.direction = nearBottom ? "up": "down";
14387
                                }
14388
                        }
14389
14390
                        //Check if dropOnEmpty is enabled
14391
                        if(!itemWithLeastDistance && !this.options.dropOnEmpty) {
14392
                                return;
14393
                        }
14394
14395
                        if(this.currentContainer === this.containers[innermostIndex]) {
14396
                                if ( !this.currentContainer.containerCache.over ) {
14397
                                        this.containers[ innermostIndex ]._trigger( "over", event, this._uiHash() );
14398
                                        this.currentContainer.containerCache.over = 1;
14399
                                }
14400
                                return;
14401
                        }
14402
14403
                        itemWithLeastDistance ? this._rearrange(event, itemWithLeastDistance, null, true) : this._rearrange(event, null, this.containers[innermostIndex].element, true);
14404
                        this._trigger("change", event, this._uiHash());
14405
                        this.containers[innermostIndex]._trigger("change", event, this._uiHash(this));
14406
                        this.currentContainer = this.containers[innermostIndex];
14407
14408
                        //Update the placeholder
14409
                        this.options.placeholder.update(this.currentContainer, this.placeholder);
14410
14411
                        this.containers[innermostIndex]._trigger("over", event, this._uiHash(this));
14412
                        this.containers[innermostIndex].containerCache.over = 1;
14413
                }
14414
14415
14416
        },
14417
14418
        _createHelper: function(event) {
14419
14420
                var o = this.options,
14421
                        helper = $.isFunction(o.helper) ? $(o.helper.apply(this.element[0], [event, this.currentItem])) : (o.helper === "clone" ? this.currentItem.clone() : this.currentItem);
14422
14423
                //Add the helper to the DOM if that didn't happen already
14424
                if(!helper.parents("body").length) {
14425
                        $(o.appendTo !== "parent" ? o.appendTo : this.currentItem[0].parentNode)[0].appendChild(helper[0]);
14426
                }
14427
14428
                if(helper[0] === this.currentItem[0]) {
14429
                        this._storedCSS = { width: this.currentItem[0].style.width, height: this.currentItem[0].style.height, position: this.currentItem.css("position"), top: this.currentItem.css("top"), left: this.currentItem.css("left") };
14430
                }
14431
14432
                if(!helper[0].style.width || o.forceHelperSize) {
14433
                        helper.width(this.currentItem.width());
14434
                }
14435
                if(!helper[0].style.height || o.forceHelperSize) {
14436
                        helper.height(this.currentItem.height());
14437
                }
14438
14439
                return helper;
14440
14441
        },
14442
14443
        _adjustOffsetFromHelper: function(obj) {
14444
                if (typeof obj === "string") {
14445
                        obj = obj.split(" ");
14446
                }
14447
                if ($.isArray(obj)) {
14448
                        obj = {left: +obj[0], top: +obj[1] || 0};
14449
                }
14450
                if ("left" in obj) {
14451
                        this.offset.click.left = obj.left + this.margins.left;
14452
                }
14453
                if ("right" in obj) {
14454
                        this.offset.click.left = this.helperProportions.width - obj.right + this.margins.left;
14455
                }
14456
                if ("top" in obj) {
14457
                        this.offset.click.top = obj.top + this.margins.top;
14458
                }
14459
                if ("bottom" in obj) {
14460
                        this.offset.click.top = this.helperProportions.height - obj.bottom + this.margins.top;
14461
                }
14462
        },
14463
14464
        _getParentOffset: function() {
14465
14466
14467
                //Get the offsetParent and cache its position
14468
                this.offsetParent = this.helper.offsetParent();
14469
                var po = this.offsetParent.offset();
14470
14471
                // This is a special case where we need to modify a offset calculated on start, since the following happened:
14472
                // 1. The position of the helper is absolute, so it's position is calculated based on the next positioned parent
14473
                // 2. The actual offset parent is a child of the scroll parent, and the scroll parent isn't the document, which means that
14474
                //    the scroll is included in the initial calculation of the offset of the parent, and never recalculated upon drag
14475
                if(this.cssPosition === "absolute" && this.scrollParent[0] !== this.document[0] && $.contains(this.scrollParent[0], this.offsetParent[0])) {
14476
                        po.left += this.scrollParent.scrollLeft();
14477
                        po.top += this.scrollParent.scrollTop();
14478
                }
14479
14480
                // This needs to be actually done for all browsers, since pageX/pageY includes this information
14481
                // with an ugly IE fix
14482
                if( this.offsetParent[0] === this.document[0].body || (this.offsetParent[0].tagName && this.offsetParent[0].tagName.toLowerCase() === "html" && $.ui.ie)) {
14483
                        po = { top: 0, left: 0 };
14484
                }
14485
14486
                return {
14487
                        top: po.top + (parseInt(this.offsetParent.css("borderTopWidth"),10) || 0),
14488
                        left: po.left + (parseInt(this.offsetParent.css("borderLeftWidth"),10) || 0)
14489
                };
14490
14491
        },
14492
14493
        _getRelativeOffset: function() {
14494
14495
                if(this.cssPosition === "relative") {
14496
                        var p = this.currentItem.position();
14497
                        return {
14498
                                top: p.top - (parseInt(this.helper.css("top"),10) || 0) + this.scrollParent.scrollTop(),
14499
                                left: p.left - (parseInt(this.helper.css("left"),10) || 0) + this.scrollParent.scrollLeft()
14500
                        };
14501
                } else {
14502
                        return { top: 0, left: 0 };
14503
                }
14504
14505
        },
14506
14507
        _cacheMargins: function() {
14508
                this.margins = {
14509
                        left: (parseInt(this.currentItem.css("marginLeft"),10) || 0),
14510
                        top: (parseInt(this.currentItem.css("marginTop"),10) || 0)
14511
                };
14512
        },
14513
14514
        _cacheHelperProportions: function() {
14515
                this.helperProportions = {
14516
                        width: this.helper.outerWidth(),
14517
                        height: this.helper.outerHeight()
14518
                };
14519
        },
14520
14521
        _setContainment: function() {
14522
14523
                var ce, co, over,
14524
                        o = this.options;
14525
                if(o.containment === "parent") {
14526
                        o.containment = this.helper[0].parentNode;
14527
                }
14528
                if(o.containment === "document" || o.containment === "window") {
14529
                        this.containment = [
14530
                                0 - this.offset.relative.left - this.offset.parent.left,
14531
                                0 - this.offset.relative.top - this.offset.parent.top,
14532
                                o.containment === "document" ? this.document.width() : this.window.width() - this.helperProportions.width - this.margins.left,
14533
                                (o.containment === "document" ? this.document.width() : this.window.height() || this.document[0].body.parentNode.scrollHeight) - this.helperProportions.height - this.margins.top
14534
                        ];
14535
                }
14536
14537
                if(!(/^(document|window|parent)$/).test(o.containment)) {
14538
                        ce = $(o.containment)[0];
14539
                        co = $(o.containment).offset();
14540
                        over = ($(ce).css("overflow") !== "hidden");
14541
14542
                        this.containment = [
14543
                                co.left + (parseInt($(ce).css("borderLeftWidth"),10) || 0) + (parseInt($(ce).css("paddingLeft"),10) || 0) - this.margins.left,
14544
                                co.top + (parseInt($(ce).css("borderTopWidth"),10) || 0) + (parseInt($(ce).css("paddingTop"),10) || 0) - this.margins.top,
14545
                                co.left+(over ? Math.max(ce.scrollWidth,ce.offsetWidth) : ce.offsetWidth) - (parseInt($(ce).css("borderLeftWidth"),10) || 0) - (parseInt($(ce).css("paddingRight"),10) || 0) - this.helperProportions.width - this.margins.left,
14546
                                co.top+(over ? Math.max(ce.scrollHeight,ce.offsetHeight) : ce.offsetHeight) - (parseInt($(ce).css("borderTopWidth"),10) || 0) - (parseInt($(ce).css("paddingBottom"),10) || 0) - this.helperProportions.height - this.margins.top
14547
                        ];
14548
                }
14549
14550
        },
14551
14552
        _convertPositionTo: function(d, pos) {
14553
14554
                if(!pos) {
14555
                        pos = this.position;
14556
                }
14557
                var mod = d === "absolute" ? 1 : -1,
14558
                        scroll = this.cssPosition === "absolute" && !(this.scrollParent[0] !== this.document[0] && $.contains(this.scrollParent[0], this.offsetParent[0])) ? this.offsetParent : this.scrollParent,
14559
                        scrollIsRootNode = (/(html|body)/i).test(scroll[0].tagName);
14560
14561
                return {
14562
                        top: (
14563
                                pos.top        +                                                                                                                                // The absolute mouse position
14564
                                this.offset.relative.top * mod +                                                                                // Only for relative positioned nodes: Relative offset from element to offset parent
14565
                                this.offset.parent.top * mod -                                                                                        // The offsetParent's offset without borders (offset + border)
14566
                                ( ( this.cssPosition === "fixed" ? -this.scrollParent.scrollTop() : ( scrollIsRootNode ? 0 : scroll.scrollTop() ) ) * mod)
14567
                        ),
14568
                        left: (
14569
                                pos.left +                                                                                                                                // The absolute mouse position
14570
                                this.offset.relative.left * mod +                                                                                // Only for relative positioned nodes: Relative offset from element to offset parent
14571
                                this.offset.parent.left * mod        -                                                                                // The offsetParent's offset without borders (offset + border)
14572
                                ( ( this.cssPosition === "fixed" ? -this.scrollParent.scrollLeft() : scrollIsRootNode ? 0 : scroll.scrollLeft() ) * mod)
14573
                        )
14574
                };
14575
14576
        },
14577
14578
        _generatePosition: function(event) {
14579
14580
                var top, left,
14581
                        o = this.options,
14582
                        pageX = event.pageX,
14583
                        pageY = event.pageY,
14584
                        scroll = this.cssPosition === "absolute" && !(this.scrollParent[0] !== this.document[0] && $.contains(this.scrollParent[0], this.offsetParent[0])) ? this.offsetParent : this.scrollParent, scrollIsRootNode = (/(html|body)/i).test(scroll[0].tagName);
14585
14586
                // This is another very weird special case that only happens for relative elements:
14587
                // 1. If the css position is relative
14588
                // 2. and the scroll parent is the document or similar to the offset parent
14589
                // we have to refresh the relative offset during the scroll so there are no jumps
14590
                if(this.cssPosition === "relative" && !(this.scrollParent[0] !== this.document[0] && this.scrollParent[0] !== this.offsetParent[0])) {
14591
                        this.offset.relative = this._getRelativeOffset();
14592
                }
14593
14594
                /*
14595
                 * - Position constraining -
14596
                 * Constrain the position to a mix of grid, containment.
14597
                 */
14598
14599
                if(this.originalPosition) { //If we are not dragging yet, we won't check for options
14600
14601
                        if(this.containment) {
14602
                                if(event.pageX - this.offset.click.left < this.containment[0]) {
14603
                                        pageX = this.containment[0] + this.offset.click.left;
14604
                                }
14605
                                if(event.pageY - this.offset.click.top < this.containment[1]) {
14606
                                        pageY = this.containment[1] + this.offset.click.top;
14607
                                }
14608
                                if(event.pageX - this.offset.click.left > this.containment[2]) {
14609
                                        pageX = this.containment[2] + this.offset.click.left;
14610
                                }
14611
                                if(event.pageY - this.offset.click.top > this.containment[3]) {
14612
                                        pageY = this.containment[3] + this.offset.click.top;
14613
                                }
14614
                        }
14615
14616
                        if(o.grid) {
14617
                                top = this.originalPageY + Math.round((pageY - this.originalPageY) / o.grid[1]) * o.grid[1];
14618
                                pageY = this.containment ? ( (top - this.offset.click.top >= this.containment[1] && top - this.offset.click.top <= this.containment[3]) ? top : ((top - this.offset.click.top >= this.containment[1]) ? top - o.grid[1] : top + o.grid[1])) : top;
14619
14620
                                left = this.originalPageX + Math.round((pageX - this.originalPageX) / o.grid[0]) * o.grid[0];
14621
                                pageX = this.containment ? ( (left - this.offset.click.left >= this.containment[0] && left - this.offset.click.left <= this.containment[2]) ? left : ((left - this.offset.click.left >= this.containment[0]) ? left - o.grid[0] : left + o.grid[0])) : left;
14622
                        }
14623
14624
                }
14625
14626
                return {
14627
                        top: (
14628
                                pageY -                                                                                                                                // The absolute mouse position
14629
                                this.offset.click.top -                                                                                                        // Click offset (relative to the element)
14630
                                this.offset.relative.top        -                                                                                        // Only for relative positioned nodes: Relative offset from element to offset parent
14631
                                this.offset.parent.top +                                                                                                // The offsetParent's offset without borders (offset + border)
14632
                                ( ( this.cssPosition === "fixed" ? -this.scrollParent.scrollTop() : ( scrollIsRootNode ? 0 : scroll.scrollTop() ) ))
14633
                        ),
14634
                        left: (
14635
                                pageX -                                                                                                                                // The absolute mouse position
14636
                                this.offset.click.left -                                                                                                // Click offset (relative to the element)
14637
                                this.offset.relative.left        -                                                                                        // Only for relative positioned nodes: Relative offset from element to offset parent
14638
                                this.offset.parent.left +                                                                                                // The offsetParent's offset without borders (offset + border)
14639
                                ( ( this.cssPosition === "fixed" ? -this.scrollParent.scrollLeft() : scrollIsRootNode ? 0 : scroll.scrollLeft() ))
14640
                        )
14641
                };
14642
14643
        },
14644
14645
        _rearrange: function(event, i, a, hardRefresh) {
14646
14647
                a ? a[0].appendChild(this.placeholder[0]) : i.item[0].parentNode.insertBefore(this.placeholder[0], (this.direction === "down" ? i.item[0] : i.item[0].nextSibling));
14648
14649
                //Various things done here to improve the performance:
14650
                // 1. we create a setTimeout, that calls refreshPositions
14651
                // 2. on the instance, we have a counter variable, that get's higher after every append
14652
                // 3. on the local scope, we copy the counter variable, and check in the timeout, if it's still the same
14653
                // 4. this lets only the last addition to the timeout stack through
14654
                this.counter = this.counter ? ++this.counter : 1;
14655
                var counter = this.counter;
14656
14657
                this._delay(function() {
14658
                        if(counter === this.counter) {
14659
                                this.refreshPositions(!hardRefresh); //Precompute after each DOM insertion, NOT on mousemove
14660
                        }
14661
                });
14662
14663
        },
14664
14665
        _clear: function(event, noPropagation) {
14666
14667
                this.reverting = false;
14668
                // We delay all events that have to be triggered to after the point where the placeholder has been removed and
14669
                // everything else normalized again
14670
                var i,
14671
                        delayedTriggers = [];
14672
14673
                // We first have to update the dom position of the actual currentItem
14674
                // Note: don't do it if the current item is already removed (by a user), or it gets reappended (see #4088)
14675
                if(!this._noFinalSort && this.currentItem.parent().length) {
14676
                        this.placeholder.before(this.currentItem);
14677
                }
14678
                this._noFinalSort = null;
14679
14680
                if(this.helper[0] === this.currentItem[0]) {
14681
                        for(i in this._storedCSS) {
14682
                                if(this._storedCSS[i] === "auto" || this._storedCSS[i] === "static") {
14683
                                        this._storedCSS[i] = "";
14684
                                }
14685
                        }
14686
                        this.currentItem.css(this._storedCSS).removeClass("ui-sortable-helper");
14687
                } else {
14688
                        this.currentItem.show();
14689
                }
14690
14691
                if(this.fromOutside && !noPropagation) {
14692
                        delayedTriggers.push(function(event) { this._trigger("receive", event, this._uiHash(this.fromOutside)); });
14693
                }
14694
                if((this.fromOutside || this.domPosition.prev !== this.currentItem.prev().not(".ui-sortable-helper")[0] || this.domPosition.parent !== this.currentItem.parent()[0]) && !noPropagation) {
14695
                        delayedTriggers.push(function(event) { this._trigger("update", event, this._uiHash()); }); //Trigger update callback if the DOM position has changed
14696
                }
14697
14698
                // Check if the items Container has Changed and trigger appropriate
14699
                // events.
14700
                if (this !== this.currentContainer) {
14701
                        if(!noPropagation) {
14702
                                delayedTriggers.push(function(event) { this._trigger("remove", event, this._uiHash()); });
14703
                                delayedTriggers.push((function(c) { return function(event) { c._trigger("receive", event, this._uiHash(this)); };  }).call(this, this.currentContainer));
14704
                                delayedTriggers.push((function(c) { return function(event) { c._trigger("update", event, this._uiHash(this));  }; }).call(this, this.currentContainer));
14705
                        }
14706
                }
14707
14708
14709
                //Post events to containers
14710
                function delayEvent( type, instance, container ) {
14711
                        return function( event ) {
14712
                                container._trigger( type, event, instance._uiHash( instance ) );
14713
                        };
14714
                }
14715
                for (i = this.containers.length - 1; i >= 0; i--){
14716
                        if (!noPropagation) {
14717
                                delayedTriggers.push( delayEvent( "deactivate", this, this.containers[ i ] ) );
14718
                        }
14719
                        if(this.containers[i].containerCache.over) {
14720
                                delayedTriggers.push( delayEvent( "out", this, this.containers[ i ] ) );
14721
                                this.containers[i].containerCache.over = 0;
14722
                        }
14723
                }
14724
14725
                //Do what was originally in plugins
14726
                if ( this.storedCursor ) {
14727
                        this.document.find( "body" ).css( "cursor", this.storedCursor );
14728
                        this.storedStylesheet.remove();
14729
                }
14730
                if(this._storedOpacity) {
14731
                        this.helper.css("opacity", this._storedOpacity);
14732
                }
14733
                if(this._storedZIndex) {
14734
                        this.helper.css("zIndex", this._storedZIndex === "auto" ? "" : this._storedZIndex);
14735
                }
14736
14737
                this.dragging = false;
14738
14739
                if(!noPropagation) {
14740
                        this._trigger("beforeStop", event, this._uiHash());
14741
                }
14742
14743
                //$(this.placeholder[0]).remove(); would have been the jQuery way - unfortunately, it unbinds ALL events from the original node!
14744
                this.placeholder[0].parentNode.removeChild(this.placeholder[0]);
14745
14746
                if ( !this.cancelHelperRemoval ) {
14747
                        if ( this.helper[ 0 ] !== this.currentItem[ 0 ] ) {
14748
                                this.helper.remove();
14749
                        }
14750
                        this.helper = null;
14751
                }
14752
14753
                if(!noPropagation) {
14754
                        for (i=0; i < delayedTriggers.length; i++) {
14755
                                delayedTriggers[i].call(this, event);
14756
                        } //Trigger all delayed events
14757
                        this._trigger("stop", event, this._uiHash());
14758
                }
14759
14760
                this.fromOutside = false;
14761
                return !this.cancelHelperRemoval;
14762
14763
        },
14764
14765
        _trigger: function() {
14766
                if ($.Widget.prototype._trigger.apply(this, arguments) === false) {
14767
                        this.cancel();
14768
                }
14769
        },
14770
14771
        _uiHash: function(_inst) {
14772
                var inst = _inst || this;
14773
                return {
14774
                        helper: inst.helper,
14775
                        placeholder: inst.placeholder || $([]),
14776
                        position: inst.position,
14777
                        originalPosition: inst.originalPosition,
14778
                        offset: inst.positionAbs,
14779
                        item: inst.currentItem,
14780
                        sender: _inst ? _inst.element : null
14781
                };
14782
        }
14783
14784
});
14785
14786
14787
/*!
14788
 * jQuery UI Spinner 1.11.4
14789
 * http://jqueryui.com
14790
 *
14791
 * Copyright jQuery Foundation and other contributors
14792
 * Released under the MIT license.
14793
 * http://jquery.org/license
14794
 *
14795
 * http://api.jqueryui.com/spinner/
14796
 */
14797
14798
14799
function spinner_modifier( fn ) {
14800
        return function() {
14801
                var previous = this.element.val();
14802
                fn.apply( this, arguments );
14803
                this._refresh();
14804
                if ( previous !== this.element.val() ) {
14805
                        this._trigger( "change" );
14806
                }
14807
        };
14808
}
14809
14810
var spinner = $.widget( "ui.spinner", {
14811
        version: "1.11.4",
14812
        defaultElement: "<input>",
14813
        widgetEventPrefix: "spin",
14814
        options: {
14815
                culture: null,
14816
                icons: {
14817
                        down: "ui-icon-triangle-1-s",
14818
                        up: "ui-icon-triangle-1-n"
14819
                },
14820
                incremental: true,
14821
                max: null,
14822
                min: null,
14823
                numberFormat: null,
14824
                page: 10,
14825
                step: 1,
14826
14827
                change: null,
14828
                spin: null,
14829
                start: null,
14830
                stop: null
14831
        },
14832
14833
        _create: function() {
14834
                // handle string values that need to be parsed
14835
                this._setOption( "max", this.options.max );
14836
                this._setOption( "min", this.options.min );
14837
                this._setOption( "step", this.options.step );
14838
14839
                // Only format if there is a value, prevents the field from being marked
14840
                // as invalid in Firefox, see #9573.
14841
                if ( this.value() !== "" ) {
14842
                        // Format the value, but don't constrain.
14843
                        this._value( this.element.val(), true );
14844
                }
14845
14846
                this._draw();
14847
                this._on( this._events );
14848
                this._refresh();
14849
14850
                // turning off autocomplete prevents the browser from remembering the
14851
                // value when navigating through history, so we re-enable autocomplete
14852
                // if the page is unloaded before the widget is destroyed. #7790
14853
                this._on( this.window, {
14854
                        beforeunload: function() {
14855
                                this.element.removeAttr( "autocomplete" );
14856
                        }
14857
                });
14858
        },
14859
14860
        _getCreateOptions: function() {
14861
                var options = {},
14862
                        element = this.element;
14863
14864
                $.each( [ "min", "max", "step" ], function( i, option ) {
14865
                        var value = element.attr( option );
14866
                        if ( value !== undefined && value.length ) {
14867
                                options[ option ] = value;
14868
                        }
14869
                });
14870
14871
                return options;
14872
        },
14873
14874
        _events: {
14875
                keydown: function( event ) {
14876
                        if ( this._start( event ) && this._keydown( event ) ) {
14877
                                event.preventDefault();
14878
                        }
14879
                },
14880
                keyup: "_stop",
14881
                focus: function() {
14882
                        this.previous = this.element.val();
14883
                },
14884
                blur: function( event ) {
14885
                        if ( this.cancelBlur ) {
14886
                                delete this.cancelBlur;
14887
                                return;
14888
                        }
14889
14890
                        this._stop();
14891
                        this._refresh();
14892
                        if ( this.previous !== this.element.val() ) {
14893
                                this._trigger( "change", event );
14894
                        }
14895
                },
14896
                mousewheel: function( event, delta ) {
14897
                        if ( !delta ) {
14898
                                return;
14899
                        }
14900
                        if ( !this.spinning && !this._start( event ) ) {
14901
                                return false;
14902
                        }
14903
14904
                        this._spin( (delta > 0 ? 1 : -1) * this.options.step, event );
14905
                        clearTimeout( this.mousewheelTimer );
14906
                        this.mousewheelTimer = this._delay(function() {
14907
                                if ( this.spinning ) {
14908
                                        this._stop( event );
14909
                                }
14910
                        }, 100 );
14911
                        event.preventDefault();
14912
                },
14913
                "mousedown .ui-spinner-button": function( event ) {
14914
                        var previous;
14915
14916
                        // We never want the buttons to have focus; whenever the user is
14917
                        // interacting with the spinner, the focus should be on the input.
14918
                        // If the input is focused then this.previous is properly set from
14919
                        // when the input first received focus. If the input is not focused
14920
                        // then we need to set this.previous based on the value before spinning.
14921
                        previous = this.element[0] === this.document[0].activeElement ?
14922
                                this.previous : this.element.val();
14923
                        function checkFocus() {
14924
                                var isActive = this.element[0] === this.document[0].activeElement;
14925
                                if ( !isActive ) {
14926
                                        this.element.focus();
14927
                                        this.previous = previous;
14928
                                        // support: IE
14929
                                        // IE sets focus asynchronously, so we need to check if focus
14930
                                        // moved off of the input because the user clicked on the button.
14931
                                        this._delay(function() {
14932
                                                this.previous = previous;
14933
                                        });
14934
                                }
14935
                        }
14936
14937
                        // ensure focus is on (or stays on) the text field
14938
                        event.preventDefault();
14939
                        checkFocus.call( this );
14940
14941
                        // support: IE
14942
                        // IE doesn't prevent moving focus even with event.preventDefault()
14943
                        // so we set a flag to know when we should ignore the blur event
14944
                        // and check (again) if focus moved off of the input.
14945
                        this.cancelBlur = true;
14946
                        this._delay(function() {
14947
                                delete this.cancelBlur;
14948
                                checkFocus.call( this );
14949
                        });
14950
14951
                        if ( this._start( event ) === false ) {
14952
                                return;
14953
                        }
14954
14955
                        this._repeat( null, $( event.currentTarget ).hasClass( "ui-spinner-up" ) ? 1 : -1, event );
14956
                },
14957
                "mouseup .ui-spinner-button": "_stop",
14958
                "mouseenter .ui-spinner-button": function( event ) {
14959
                        // button will add ui-state-active if mouse was down while mouseleave and kept down
14960
                        if ( !$( event.currentTarget ).hasClass( "ui-state-active" ) ) {
14961
                                return;
14962
                        }
14963
14964
                        if ( this._start( event ) === false ) {
14965
                                return false;
14966
                        }
14967
                        this._repeat( null, $( event.currentTarget ).hasClass( "ui-spinner-up" ) ? 1 : -1, event );
14968
                },
14969
                // TODO: do we really want to consider this a stop?
14970
                // shouldn't we just stop the repeater and wait until mouseup before
14971
                // we trigger the stop event?
14972
                "mouseleave .ui-spinner-button": "_stop"
14973
        },
14974
14975
        _draw: function() {
14976
                var uiSpinner = this.uiSpinner = this.element
14977
                        .addClass( "ui-spinner-input" )
14978
                        .attr( "autocomplete", "off" )
14979
                        .wrap( this._uiSpinnerHtml() )
14980
                        .parent()
14981
                                // add buttons
14982
                                .append( this._buttonHtml() );
14983
14984
                this.element.attr( "role", "spinbutton" );
14985
14986
                // button bindings
14987
                this.buttons = uiSpinner.find( ".ui-spinner-button" )
14988
                        .attr( "tabIndex", -1 )
14989
                        .button()
14990
                        .removeClass( "ui-corner-all" );
14991
14992
                // IE 6 doesn't understand height: 50% for the buttons
14993
                // unless the wrapper has an explicit height
14994
                if ( this.buttons.height() > Math.ceil( uiSpinner.height() * 0.5 ) &&
14995
                                uiSpinner.height() > 0 ) {
14996
                        uiSpinner.height( uiSpinner.height() );
14997
                }
14998
14999
                // disable spinner if element was already disabled
15000
                if ( this.options.disabled ) {
15001
                        this.disable();
15002
                }
15003
        },
15004
15005
        _keydown: function( event ) {
15006
                var options = this.options,
15007
                        keyCode = $.ui.keyCode;
15008
15009
                switch ( event.keyCode ) {
15010
                case keyCode.UP:
15011
                        this._repeat( null, 1, event );
15012
                        return true;
15013
                case keyCode.DOWN:
15014
                        this._repeat( null, -1, event );
15015
                        return true;
15016
                case keyCode.PAGE_UP:
15017
                        this._repeat( null, options.page, event );
15018
                        return true;
15019
                case keyCode.PAGE_DOWN:
15020
                        this._repeat( null, -options.page, event );
15021
                        return true;
15022
                }
15023
15024
                return false;
15025
        },
15026
15027
        _uiSpinnerHtml: function() {
15028
                return "<span class='ui-spinner ui-widget ui-widget-content ui-corner-all'></span>";
15029
        },
15030
15031
        _buttonHtml: function() {
15032
                return "" +
15033
                        "<a class='ui-spinner-button ui-spinner-up ui-corner-tr'>" +
15034
                                "<span class='ui-icon " + this.options.icons.up + "'>&#9650;</span>" +
15035
                        "</a>" +
15036
                        "<a class='ui-spinner-button ui-spinner-down ui-corner-br'>" +
15037
                                "<span class='ui-icon " + this.options.icons.down + "'>&#9660;</span>" +
15038
                        "</a>";
15039
        },
15040
15041
        _start: function( event ) {
15042
                if ( !this.spinning && this._trigger( "start", event ) === false ) {
15043
                        return false;
15044
                }
15045
15046
                if ( !this.counter ) {
15047
                        this.counter = 1;
15048
                }
15049
                this.spinning = true;
15050
                return true;
15051
        },
15052
15053
        _repeat: function( i, steps, event ) {
15054
                i = i || 500;
15055
15056
                clearTimeout( this.timer );
15057
                this.timer = this._delay(function() {
15058
                        this._repeat( 40, steps, event );
15059
                }, i );
15060
15061
                this._spin( steps * this.options.step, event );
15062
        },
15063
15064
        _spin: function( step, event ) {
15065
                var value = this.value() || 0;
15066
15067
                if ( !this.counter ) {
15068
                        this.counter = 1;
15069
                }
15070
15071
                value = this._adjustValue( value + step * this._increment( this.counter ) );
15072
15073
                if ( !this.spinning || this._trigger( "spin", event, { value: value } ) !== false) {
15074
                        this._value( value );
15075
                        this.counter++;
15076
                }
15077
        },
15078
15079
        _increment: function( i ) {
15080
                var incremental = this.options.incremental;
15081
15082
                if ( incremental ) {
15083
                        return $.isFunction( incremental ) ?
15084
                                incremental( i ) :
15085
                                Math.floor( i * i * i / 50000 - i * i / 500 + 17 * i / 200 + 1 );
15086
                }
15087
15088
                return 1;
15089
        },
15090
15091
        _precision: function() {
15092
                var precision = this._precisionOf( this.options.step );
15093
                if ( this.options.min !== null ) {
15094
                        precision = Math.max( precision, this._precisionOf( this.options.min ) );
15095
                }
15096
                return precision;
15097
        },
15098
15099
        _precisionOf: function( num ) {
15100
                var str = num.toString(),
15101
                        decimal = str.indexOf( "." );
15102
                return decimal === -1 ? 0 : str.length - decimal - 1;
15103
        },
15104
15105
        _adjustValue: function( value ) {
15106
                var base, aboveMin,
15107
                        options = this.options;
15108
15109
                // make sure we're at a valid step
15110
                // - find out where we are relative to the base (min or 0)
15111
                base = options.min !== null ? options.min : 0;
15112
                aboveMin = value - base;
15113
                // - round to the nearest step
15114
                aboveMin = Math.round(aboveMin / options.step) * options.step;
15115
                // - rounding is based on 0, so adjust back to our base
15116
                value = base + aboveMin;
15117
15118
                // fix precision from bad JS floating point math
15119
                value = parseFloat( value.toFixed( this._precision() ) );
15120
15121
                // clamp the value
15122
                if ( options.max !== null && value > options.max) {
15123
                        return options.max;
15124
                }
15125
                if ( options.min !== null && value < options.min ) {
15126
                        return options.min;
15127
                }
15128
15129
                return value;
15130
        },
15131
15132
        _stop: function( event ) {
15133
                if ( !this.spinning ) {
15134
                        return;
15135
                }
15136
15137
                clearTimeout( this.timer );
15138
                clearTimeout( this.mousewheelTimer );
15139
                this.counter = 0;
15140
                this.spinning = false;
15141
                this._trigger( "stop", event );
15142
        },
15143
15144
        _setOption: function( key, value ) {
15145
                if ( key === "culture" || key === "numberFormat" ) {
15146
                        var prevValue = this._parse( this.element.val() );
15147
                        this.options[ key ] = value;
15148
                        this.element.val( this._format( prevValue ) );
15149
                        return;
15150
                }
15151
15152
                if ( key === "max" || key === "min" || key === "step" ) {
15153
                        if ( typeof value === "string" ) {
15154
                                value = this._parse( value );
15155
                        }
15156
                }
15157
                if ( key === "icons" ) {
15158
                        this.buttons.first().find( ".ui-icon" )
15159
                                .removeClass( this.options.icons.up )
15160
                                .addClass( value.up );
15161
                        this.buttons.last().find( ".ui-icon" )
15162
                                .removeClass( this.options.icons.down )
15163
                                .addClass( value.down );
15164
                }
15165
15166
                this._super( key, value );
15167
15168
                if ( key === "disabled" ) {
15169
                        this.widget().toggleClass( "ui-state-disabled", !!value );
15170
                        this.element.prop( "disabled", !!value );
15171
                        this.buttons.button( value ? "disable" : "enable" );
15172
                }
15173
        },
15174
15175
        _setOptions: spinner_modifier(function( options ) {
15176
                this._super( options );
15177
        }),
15178
15179
        _parse: function( val ) {
15180
                if ( typeof val === "string" && val !== "" ) {
15181
                        val = window.Globalize && this.options.numberFormat ?
15182
                                Globalize.parseFloat( val, 10, this.options.culture ) : +val;
15183
                }
15184
                return val === "" || isNaN( val ) ? null : val;
15185
        },
15186
15187
        _format: function( value ) {
15188
                if ( value === "" ) {
15189
                        return "";
15190
                }
15191
                return window.Globalize && this.options.numberFormat ?
15192
                        Globalize.format( value, this.options.numberFormat, this.options.culture ) :
15193
                        value;
15194
        },
15195
15196
        _refresh: function() {
15197
                this.element.attr({
15198
                        "aria-valuemin": this.options.min,
15199
                        "aria-valuemax": this.options.max,
15200
                        // TODO: what should we do with values that can't be parsed?
15201
                        "aria-valuenow": this._parse( this.element.val() )
15202
                });
15203
        },
15204
15205
        isValid: function() {
15206
                var value = this.value();
15207
15208
                // null is invalid
15209
                if ( value === null ) {
15210
                        return false;
15211
                }
15212
15213
                // if value gets adjusted, it's invalid
15214
                return value === this._adjustValue( value );
15215
        },
15216
15217
        // update the value without triggering change
15218
        _value: function( value, allowAny ) {
15219
                var parsed;
15220
                if ( value !== "" ) {
15221
                        parsed = this._parse( value );
15222
                        if ( parsed !== null ) {
15223
                                if ( !allowAny ) {
15224
                                        parsed = this._adjustValue( parsed );
15225
                                }
15226
                                value = this._format( parsed );
15227
                        }
15228
                }
15229
                this.element.val( value );
15230
                this._refresh();
15231
        },
15232
15233
        _destroy: function() {
15234
                this.element
15235
                        .removeClass( "ui-spinner-input" )
15236
                        .prop( "disabled", false )
15237
                        .removeAttr( "autocomplete" )
15238
                        .removeAttr( "role" )
15239
                        .removeAttr( "aria-valuemin" )
15240
                        .removeAttr( "aria-valuemax" )
15241
                        .removeAttr( "aria-valuenow" );
15242
                this.uiSpinner.replaceWith( this.element );
15243
        },
15244
15245
        stepUp: spinner_modifier(function( steps ) {
15246
                this._stepUp( steps );
15247
        }),
15248
        _stepUp: function( steps ) {
15249
                if ( this._start() ) {
15250
                        this._spin( (steps || 1) * this.options.step );
15251
                        this._stop();
15252
                }
15253
        },
15254
15255
        stepDown: spinner_modifier(function( steps ) {
15256
                this._stepDown( steps );
15257
        }),
15258
        _stepDown: function( steps ) {
15259
                if ( this._start() ) {
15260
                        this._spin( (steps || 1) * -this.options.step );
15261
                        this._stop();
15262
                }
15263
        },
15264
15265
        pageUp: spinner_modifier(function( pages ) {
15266
                this._stepUp( (pages || 1) * this.options.page );
15267
        }),
15268
15269
        pageDown: spinner_modifier(function( pages ) {
15270
                this._stepDown( (pages || 1) * this.options.page );
15271
        }),
15272
15273
        value: function( newVal ) {
15274
                if ( !arguments.length ) {
15275
                        return this._parse( this.element.val() );
15276
                }
15277
                spinner_modifier( this._value ).call( this, newVal );
15278
        },
15279
15280
        widget: function() {
15281
                return this.uiSpinner;
15282
        }
15283
});
15284
15285
15286
/*!
15287
 * jQuery UI Tabs 1.11.4
15288
 * http://jqueryui.com
15289
 *
15290
 * Copyright jQuery Foundation and other contributors
15291
 * Released under the MIT license.
15292
 * http://jquery.org/license
15293
 *
15294
 * http://api.jqueryui.com/tabs/
15295
 */
15296
15297
15298
var tabs = $.widget( "ui.tabs", {
15299
        version: "1.11.4",
15300
        delay: 300,
15301
        options: {
15302
                active: null,
15303
                collapsible: false,
15304
                event: "click",
15305
                heightStyle: "content",
15306
                hide: null,
15307
                show: null,
15308
15309
                // callbacks
15310
                activate: null,
15311
                beforeActivate: null,
15312
                beforeLoad: null,
15313
                load: null
15314
        },
15315
15316
        _isLocal: (function() {
15317
                var rhash = /#.*$/;
15318
15319
                return function( anchor ) {
15320
                        var anchorUrl, locationUrl;
15321
15322
                        // support: IE7
15323
                        // IE7 doesn't normalize the href property when set via script (#9317)
15324
                        anchor = anchor.cloneNode( false );
15325
15326
                        anchorUrl = anchor.href.replace( rhash, "" );
15327
                        locationUrl = location.href.replace( rhash, "" );
15328
15329
                        // decoding may throw an error if the URL isn't UTF-8 (#9518)
15330
                        try {
15331
                                anchorUrl = decodeURIComponent( anchorUrl );
15332
                        } catch ( error ) {}
15333
                        try {
15334
                                locationUrl = decodeURIComponent( locationUrl );
15335
                        } catch ( error ) {}
15336
15337
                        return anchor.hash.length > 1 && anchorUrl === locationUrl;
15338
                };
15339
        })(),
15340
15341
        _create: function() {
15342
                var that = this,
15343
                        options = this.options;
15344
15345
                this.running = false;
15346
15347
                this.element
15348
                        .addClass( "ui-tabs ui-widget ui-widget-content ui-corner-all" )
15349
                        .toggleClass( "ui-tabs-collapsible", options.collapsible );
15350
15351
                this._processTabs();
15352
                options.active = this._initialActive();
15353
15354
                // Take disabling tabs via class attribute from HTML
15355
                // into account and update option properly.
15356
                if ( $.isArray( options.disabled ) ) {
15357
                        options.disabled = $.unique( options.disabled.concat(
15358
                                $.map( this.tabs.filter( ".ui-state-disabled" ), function( li ) {
15359
                                        return that.tabs.index( li );
15360
                                })
15361
                        ) ).sort();
15362
                }
15363
15364
                // check for length avoids error when initializing empty list
15365
                if ( this.options.active !== false && this.anchors.length ) {
15366
                        this.active = this._findActive( options.active );
15367
                } else {
15368
                        this.active = $();
15369
                }
15370
15371
                this._refresh();
15372
15373
                if ( this.active.length ) {
15374
                        this.load( options.active );
15375
                }
15376
        },
15377
15378
        _initialActive: function() {
15379
                var active = this.options.active,
15380
                        collapsible = this.options.collapsible,
15381
                        locationHash = location.hash.substring( 1 );
15382
15383
                if ( active === null ) {
15384
                        // check the fragment identifier in the URL
15385
                        if ( locationHash ) {
15386
                                this.tabs.each(function( i, tab ) {
15387
                                        if ( $( tab ).attr( "aria-controls" ) === locationHash ) {
15388
                                                active = i;
15389
                                                return false;
15390
                                        }
15391
                                });
15392
                        }
15393
15394
                        // check for a tab marked active via a class
15395
                        if ( active === null ) {
15396
                                active = this.tabs.index( this.tabs.filter( ".ui-tabs-active" ) );
15397
                        }
15398
15399
                        // no active tab, set to false
15400
                        if ( active === null || active === -1 ) {
15401
                                active = this.tabs.length ? 0 : false;
15402
                        }
15403
                }
15404
15405
                // handle numbers: negative, out of range
15406
                if ( active !== false ) {
15407
                        active = this.tabs.index( this.tabs.eq( active ) );
15408
                        if ( active === -1 ) {
15409
                                active = collapsible ? false : 0;
15410
                        }
15411
                }
15412
15413
                // don't allow collapsible: false and active: false
15414
                if ( !collapsible && active === false && this.anchors.length ) {
15415
                        active = 0;
15416
                }
15417
15418
                return active;
15419
        },
15420
15421
        _getCreateEventData: function() {
15422
                return {
15423
                        tab: this.active,
15424
                        panel: !this.active.length ? $() : this._getPanelForTab( this.active )
15425
                };
15426
        },
15427
15428
        _tabKeydown: function( event ) {
15429
                var focusedTab = $( this.document[0].activeElement ).closest( "li" ),
15430
                        selectedIndex = this.tabs.index( focusedTab ),
15431
                        goingForward = true;
15432
15433
                if ( this._handlePageNav( event ) ) {
15434
                        return;
15435
                }
15436
15437
                switch ( event.keyCode ) {
15438
                        case $.ui.keyCode.RIGHT:
15439
                        case $.ui.keyCode.DOWN:
15440
                                selectedIndex++;
15441
                                break;
15442
                        case $.ui.keyCode.UP:
15443
                        case $.ui.keyCode.LEFT:
15444
                                goingForward = false;
15445
                                selectedIndex--;
15446
                                break;
15447
                        case $.ui.keyCode.END:
15448
                                selectedIndex = this.anchors.length - 1;
15449
                                break;
15450
                        case $.ui.keyCode.HOME:
15451
                                selectedIndex = 0;
15452
                                break;
15453
                        case $.ui.keyCode.SPACE:
15454
                                // Activate only, no collapsing
15455
                                event.preventDefault();
15456
                                clearTimeout( this.activating );
15457
                                this._activate( selectedIndex );
15458
                                return;
15459
                        case $.ui.keyCode.ENTER:
15460
                                // Toggle (cancel delayed activation, allow collapsing)
15461
                                event.preventDefault();
15462
                                clearTimeout( this.activating );
15463
                                // Determine if we should collapse or activate
15464
                                this._activate( selectedIndex === this.options.active ? false : selectedIndex );
15465
                                return;
15466
                        default:
15467
                                return;
15468
                }
15469
15470
                // Focus the appropriate tab, based on which key was pressed
15471
                event.preventDefault();
15472
                clearTimeout( this.activating );
15473
                selectedIndex = this._focusNextTab( selectedIndex, goingForward );
15474
15475
                // Navigating with control/command key will prevent automatic activation
15476
                if ( !event.ctrlKey && !event.metaKey ) {
15477
15478
                        // Update aria-selected immediately so that AT think the tab is already selected.
15479
                        // Otherwise AT may confuse the user by stating that they need to activate the tab,
15480
                        // but the tab will already be activated by the time the announcement finishes.
15481
                        focusedTab.attr( "aria-selected", "false" );
15482
                        this.tabs.eq( selectedIndex ).attr( "aria-selected", "true" );
15483
15484
                        this.activating = this._delay(function() {
15485
                                this.option( "active", selectedIndex );
15486
                        }, this.delay );
15487
                }
15488
        },
15489
15490
        _panelKeydown: function( event ) {
15491
                if ( this._handlePageNav( event ) ) {
15492
                        return;
15493
                }
15494
15495
                // Ctrl+up moves focus to the current tab
15496
                if ( event.ctrlKey && event.keyCode === $.ui.keyCode.UP ) {
15497
                        event.preventDefault();
15498
                        this.active.focus();
15499
                }
15500
        },
15501
15502
        // Alt+page up/down moves focus to the previous/next tab (and activates)
15503
        _handlePageNav: function( event ) {
15504
                if ( event.altKey && event.keyCode === $.ui.keyCode.PAGE_UP ) {
15505
                        this._activate( this._focusNextTab( this.options.active - 1, false ) );
15506
                        return true;
15507
                }
15508
                if ( event.altKey && event.keyCode === $.ui.keyCode.PAGE_DOWN ) {
15509
                        this._activate( this._focusNextTab( this.options.active + 1, true ) );
15510
                        return true;
15511
                }
15512
        },
15513
15514
        _findNextTab: function( index, goingForward ) {
15515
                var lastTabIndex = this.tabs.length - 1;
15516
15517
                function constrain() {
15518
                        if ( index > lastTabIndex ) {
15519
                                index = 0;
15520
                        }
15521
                        if ( index < 0 ) {
15522
                                index = lastTabIndex;
15523
                        }
15524
                        return index;
15525
                }
15526
15527
                while ( $.inArray( constrain(), this.options.disabled ) !== -1 ) {
15528
                        index = goingForward ? index + 1 : index - 1;
15529
                }
15530
15531
                return index;
15532
        },
15533
15534
        _focusNextTab: function( index, goingForward ) {
15535
                index = this._findNextTab( index, goingForward );
15536
                this.tabs.eq( index ).focus();
15537
                return index;
15538
        },
15539
15540
        _setOption: function( key, value ) {
15541
                if ( key === "active" ) {
15542
                        // _activate() will handle invalid values and update this.options
15543
                        this._activate( value );
15544
                        return;
15545
                }
15546
15547
                if ( key === "disabled" ) {
15548
                        // don't use the widget factory's disabled handling
15549
                        this._setupDisabled( value );
15550
                        return;
15551
                }
15552
15553
                this._super( key, value);
15554
15555
                if ( key === "collapsible" ) {
15556
                        this.element.toggleClass( "ui-tabs-collapsible", value );
15557
                        // Setting collapsible: false while collapsed; open first panel
15558
                        if ( !value && this.options.active === false ) {
15559
                                this._activate( 0 );
15560
                        }
15561
                }
15562
15563
                if ( key === "event" ) {
15564
                        this._setupEvents( value );
15565
                }
15566
15567
                if ( key === "heightStyle" ) {
15568
                        this._setupHeightStyle( value );
15569
                }
15570
        },
15571
15572
        _sanitizeSelector: function( hash ) {
15573
                return hash ? hash.replace( /[!"$%&'()*+,.\/:;<=>?@\[\]\^`{|}~]/g, "\\$&" ) : "";
15574
        },
15575
15576
        refresh: function() {
15577
                var options = this.options,
15578
                        lis = this.tablist.children( ":has(a[href])" );
15579
15580
                // get disabled tabs from class attribute from HTML
15581
                // this will get converted to a boolean if needed in _refresh()
15582
                options.disabled = $.map( lis.filter( ".ui-state-disabled" ), function( tab ) {
15583
                        return lis.index( tab );
15584
                });
15585
15586
                this._processTabs();
15587
15588
                // was collapsed or no tabs
15589
                if ( options.active === false || !this.anchors.length ) {
15590
                        options.active = false;
15591
                        this.active = $();
15592
                // was active, but active tab is gone
15593
                } else if ( this.active.length && !$.contains( this.tablist[ 0 ], this.active[ 0 ] ) ) {
15594
                        // all remaining tabs are disabled
15595
                        if ( this.tabs.length === options.disabled.length ) {
15596
                                options.active = false;
15597
                                this.active = $();
15598
                        // activate previous tab
15599
                        } else {
15600
                                this._activate( this._findNextTab( Math.max( 0, options.active - 1 ), false ) );
15601
                        }
15602
                // was active, active tab still exists
15603
                } else {
15604
                        // make sure active index is correct
15605
                        options.active = this.tabs.index( this.active );
15606
                }
15607
15608
                this._refresh();
15609
        },
15610
15611
        _refresh: function() {
15612
                this._setupDisabled( this.options.disabled );
15613
                this._setupEvents( this.options.event );
15614
                this._setupHeightStyle( this.options.heightStyle );
15615
15616
                this.tabs.not( this.active ).attr({
15617
                        "aria-selected": "false",
15618
                        "aria-expanded": "false",
15619
                        tabIndex: -1
15620
                });
15621
                this.panels.not( this._getPanelForTab( this.active ) )
15622
                        .hide()
15623
                        .attr({
15624
                                "aria-hidden": "true"
15625
                        });
15626
15627
                // Make sure one tab is in the tab order
15628
                if ( !this.active.length ) {
15629
                        this.tabs.eq( 0 ).attr( "tabIndex", 0 );
15630
                } else {
15631
                        this.active
15632
                                .addClass( "ui-tabs-active ui-state-active" )
15633
                                .attr({
15634
                                        "aria-selected": "true",
15635
                                        "aria-expanded": "true",
15636
                                        tabIndex: 0
15637
                                });
15638
                        this._getPanelForTab( this.active )
15639
                                .show()
15640
                                .attr({
15641
                                        "aria-hidden": "false"
15642
                                });
15643
                }
15644
        },
15645
15646
        _processTabs: function() {
15647
                var that = this,
15648
                        prevTabs = this.tabs,
15649
                        prevAnchors = this.anchors,
15650
                        prevPanels = this.panels;
15651
15652
                this.tablist = this._getList()
15653
                        .addClass( "ui-tabs-nav ui-helper-reset ui-helper-clearfix ui-widget-header ui-corner-all" )
15654
                        .attr( "role", "tablist" )
15655
15656
                        // Prevent users from focusing disabled tabs via click
15657
                        .delegate( "> li", "mousedown" + this.eventNamespace, function( event ) {
15658
                                if ( $( this ).is( ".ui-state-disabled" ) ) {
15659
                                        event.preventDefault();
15660
                                }
15661
                        })
15662
15663
                        // support: IE <9
15664
                        // Preventing the default action in mousedown doesn't prevent IE
15665
                        // from focusing the element, so if the anchor gets focused, blur.
15666
                        // We don't have to worry about focusing the previously focused
15667
                        // element since clicking on a non-focusable element should focus
15668
                        // the body anyway.
15669
                        .delegate( ".ui-tabs-anchor", "focus" + this.eventNamespace, function() {
15670
                                if ( $( this ).closest( "li" ).is( ".ui-state-disabled" ) ) {
15671
                                        this.blur();
15672
                                }
15673
                        });
15674
15675
                this.tabs = this.tablist.find( "> li:has(a[href])" )
15676
                        .addClass( "ui-state-default ui-corner-top" )
15677
                        .attr({
15678
                                role: "tab",
15679
                                tabIndex: -1
15680
                        });
15681
15682
                this.anchors = this.tabs.map(function() {
15683
                                return $( "a", this )[ 0 ];
15684
                        })
15685
                        .addClass( "ui-tabs-anchor" )
15686
                        .attr({
15687
                                role: "presentation",
15688
                                tabIndex: -1
15689
                        });
15690
15691
                this.panels = $();
15692
15693
                this.anchors.each(function( i, anchor ) {
15694
                        var selector, panel, panelId,
15695
                                anchorId = $( anchor ).uniqueId().attr( "id" ),
15696
                                tab = $( anchor ).closest( "li" ),
15697
                                originalAriaControls = tab.attr( "aria-controls" );
15698
15699
                        // inline tab
15700
                        if ( that._isLocal( anchor ) ) {
15701
                                selector = anchor.hash;
15702
                                panelId = selector.substring( 1 );
15703
                                panel = that.element.find( that._sanitizeSelector( selector ) );
15704
                        // remote tab
15705
                        } else {
15706
                                // If the tab doesn't already have aria-controls,
15707
                                // generate an id by using a throw-away element
15708
                                panelId = tab.attr( "aria-controls" ) || $( {} ).uniqueId()[ 0 ].id;
15709
                                selector = "#" + panelId;
15710
                                panel = that.element.find( selector );
15711
                                if ( !panel.length ) {
15712
                                        panel = that._createPanel( panelId );
15713
                                        panel.insertAfter( that.panels[ i - 1 ] || that.tablist );
15714
                                }
15715
                                panel.attr( "aria-live", "polite" );
15716
                        }
15717
15718
                        if ( panel.length) {
15719
                                that.panels = that.panels.add( panel );
15720
                        }
15721
                        if ( originalAriaControls ) {
15722
                                tab.data( "ui-tabs-aria-controls", originalAriaControls );
15723
                        }
15724
                        tab.attr({
15725
                                "aria-controls": panelId,
15726
                                "aria-labelledby": anchorId
15727
                        });
15728
                        panel.attr( "aria-labelledby", anchorId );
15729
                });
15730
15731
                this.panels
15732
                        .addClass( "ui-tabs-panel ui-widget-content ui-corner-bottom" )
15733
                        .attr( "role", "tabpanel" );
15734
15735
                // Avoid memory leaks (#10056)
15736
                if ( prevTabs ) {
15737
                        this._off( prevTabs.not( this.tabs ) );
15738
                        this._off( prevAnchors.not( this.anchors ) );
15739
                        this._off( prevPanels.not( this.panels ) );
15740
                }
15741
        },
15742
15743
        // allow overriding how to find the list for rare usage scenarios (#7715)
15744
        _getList: function() {
15745
                return this.tablist || this.element.find( "ol,ul" ).eq( 0 );
15746
        },
15747
15748
        _createPanel: function( id ) {
15749
                return $( "<div>" )
15750
                        .attr( "id", id )
15751
                        .addClass( "ui-tabs-panel ui-widget-content ui-corner-bottom" )
15752
                        .data( "ui-tabs-destroy", true );
15753
        },
15754
15755
        _setupDisabled: function( disabled ) {
15756
                if ( $.isArray( disabled ) ) {
15757
                        if ( !disabled.length ) {
15758
                                disabled = false;
15759
                        } else if ( disabled.length === this.anchors.length ) {
15760
                                disabled = true;
15761
                        }
15762
                }
15763
15764
                // disable tabs
15765
                for ( var i = 0, li; ( li = this.tabs[ i ] ); i++ ) {
15766
                        if ( disabled === true || $.inArray( i, disabled ) !== -1 ) {
15767
                                $( li )
15768
                                        .addClass( "ui-state-disabled" )
15769
                                        .attr( "aria-disabled", "true" );
15770
                        } else {
15771
                                $( li )
15772
                                        .removeClass( "ui-state-disabled" )
15773
                                        .removeAttr( "aria-disabled" );
15774
                        }
15775
                }
15776
15777
                this.options.disabled = disabled;
15778
        },
15779
15780
        _setupEvents: function( event ) {
15781
                var events = {};
15782
                if ( event ) {
15783
                        $.each( event.split(" "), function( index, eventName ) {
15784
                                events[ eventName ] = "_eventHandler";
15785
                        });
15786
                }
15787
15788
                this._off( this.anchors.add( this.tabs ).add( this.panels ) );
15789
                // Always prevent the default action, even when disabled
15790
                this._on( true, this.anchors, {
15791
                        click: function( event ) {
15792
                                event.preventDefault();
15793
                        }
15794
                });
15795
                this._on( this.anchors, events );
15796
                this._on( this.tabs, { keydown: "_tabKeydown" } );
15797
                this._on( this.panels, { keydown: "_panelKeydown" } );
15798
15799
                this._focusable( this.tabs );
15800
                this._hoverable( this.tabs );
15801
        },
15802
15803
        _setupHeightStyle: function( heightStyle ) {
15804
                var maxHeight,
15805
                        parent = this.element.parent();
15806
15807
                if ( heightStyle === "fill" ) {
15808
                        maxHeight = parent.height();
15809
                        maxHeight -= this.element.outerHeight() - this.element.height();
15810
15811
                        this.element.siblings( ":visible" ).each(function() {
15812
                                var elem = $( this ),
15813
                                        position = elem.css( "position" );
15814
15815
                                if ( position === "absolute" || position === "fixed" ) {
15816
                                        return;
15817
                                }
15818
                                maxHeight -= elem.outerHeight( true );
15819
                        });
15820
15821
                        this.element.children().not( this.panels ).each(function() {
15822
                                maxHeight -= $( this ).outerHeight( true );
15823
                        });
15824
15825
                        this.panels.each(function() {
15826
                                $( this ).height( Math.max( 0, maxHeight -
15827
                                        $( this ).innerHeight() + $( this ).height() ) );
15828
                        })
15829
                        .css( "overflow", "auto" );
15830
                } else if ( heightStyle === "auto" ) {
15831
                        maxHeight = 0;
15832
                        this.panels.each(function() {
15833
                                maxHeight = Math.max( maxHeight, $( this ).height( "" ).height() );
15834
                        }).height( maxHeight );
15835
                }
15836
        },
15837
15838
        _eventHandler: function( event ) {
15839
                var options = this.options,
15840
                        active = this.active,
15841
                        anchor = $( event.currentTarget ),
15842
                        tab = anchor.closest( "li" ),
15843
                        clickedIsActive = tab[ 0 ] === active[ 0 ],
15844
                        collapsing = clickedIsActive && options.collapsible,
15845
                        toShow = collapsing ? $() : this._getPanelForTab( tab ),
15846
                        toHide = !active.length ? $() : this._getPanelForTab( active ),
15847
                        eventData = {
15848
                                oldTab: active,
15849
                                oldPanel: toHide,
15850
                                newTab: collapsing ? $() : tab,
15851
                                newPanel: toShow
15852
                        };
15853
15854
                event.preventDefault();
15855
15856
                if ( tab.hasClass( "ui-state-disabled" ) ||
15857
                                // tab is already loading
15858
                                tab.hasClass( "ui-tabs-loading" ) ||
15859
                                // can't switch durning an animation
15860
                                this.running ||
15861
                                // click on active header, but not collapsible
15862
                                ( clickedIsActive && !options.collapsible ) ||
15863
                                // allow canceling activation
15864
                                ( this._trigger( "beforeActivate", event, eventData ) === false ) ) {
15865
                        return;
15866
                }
15867
15868
                options.active = collapsing ? false : this.tabs.index( tab );
15869
15870
                this.active = clickedIsActive ? $() : tab;
15871
                if ( this.xhr ) {
15872
                        this.xhr.abort();
15873
                }
15874
15875
                if ( !toHide.length && !toShow.length ) {
15876
                        $.error( "jQuery UI Tabs: Mismatching fragment identifier." );
15877
                }
15878
15879
                if ( toShow.length ) {
15880
                        this.load( this.tabs.index( tab ), event );
15881
                }
15882
                this._toggle( event, eventData );
15883
        },
15884
15885
        // handles show/hide for selecting tabs
15886
        _toggle: function( event, eventData ) {
15887
                var that = this,
15888
                        toShow = eventData.newPanel,
15889
                        toHide = eventData.oldPanel;
15890
15891
                this.running = true;
15892
15893
                function complete() {
15894
                        that.running = false;
15895
                        that._trigger( "activate", event, eventData );
15896
                }
15897
15898
                function show() {
15899
                        eventData.newTab.closest( "li" ).addClass( "ui-tabs-active ui-state-active" );
15900
15901
                        if ( toShow.length && that.options.show ) {
15902
                                that._show( toShow, that.options.show, complete );
15903
                        } else {
15904
                                toShow.show();
15905
                                complete();
15906
                        }
15907
                }
15908
15909
                // start out by hiding, then showing, then completing
15910
                if ( toHide.length && this.options.hide ) {
15911
                        this._hide( toHide, this.options.hide, function() {
15912
                                eventData.oldTab.closest( "li" ).removeClass( "ui-tabs-active ui-state-active" );
15913
                                show();
15914
                        });
15915
                } else {
15916
                        eventData.oldTab.closest( "li" ).removeClass( "ui-tabs-active ui-state-active" );
15917
                        toHide.hide();
15918
                        show();
15919
                }
15920
15921
                toHide.attr( "aria-hidden", "true" );
15922
                eventData.oldTab.attr({
15923
                        "aria-selected": "false",
15924
                        "aria-expanded": "false"
15925
                });
15926
                // If we're switching tabs, remove the old tab from the tab order.
15927
                // If we're opening from collapsed state, remove the previous tab from the tab order.
15928
                // If we're collapsing, then keep the collapsing tab in the tab order.
15929
                if ( toShow.length && toHide.length ) {
15930
                        eventData.oldTab.attr( "tabIndex", -1 );
15931
                } else if ( toShow.length ) {
15932
                        this.tabs.filter(function() {
15933
                                return $( this ).attr( "tabIndex" ) === 0;
15934
                        })
15935
                        .attr( "tabIndex", -1 );
15936
                }
15937
15938
                toShow.attr( "aria-hidden", "false" );
15939
                eventData.newTab.attr({
15940
                        "aria-selected": "true",
15941
                        "aria-expanded": "true",
15942
                        tabIndex: 0
15943
                });
15944
        },
15945
15946
        _activate: function( index ) {
15947
                var anchor,
15948
                        active = this._findActive( index );
15949
15950
                // trying to activate the already active panel
15951
                if ( active[ 0 ] === this.active[ 0 ] ) {
15952
                        return;
15953
                }
15954
15955
                // trying to collapse, simulate a click on the current active header
15956
                if ( !active.length ) {
15957
                        active = this.active;
15958
                }
15959
15960
                anchor = active.find( ".ui-tabs-anchor" )[ 0 ];
15961
                this._eventHandler({
15962
                        target: anchor,
15963
                        currentTarget: anchor,
15964
                        preventDefault: $.noop
15965
                });
15966
        },
15967
15968
        _findActive: function( index ) {
15969
                return index === false ? $() : this.tabs.eq( index );
15970
        },
15971
15972
        _getIndex: function( index ) {
15973
                // meta-function to give users option to provide a href string instead of a numerical index.
15974
                if ( typeof index === "string" ) {
15975
                        index = this.anchors.index( this.anchors.filter( "[href$='" + index + "']" ) );
15976
                }
15977
15978
                return index;
15979
        },
15980
15981
        _destroy: function() {
15982
                if ( this.xhr ) {
15983
                        this.xhr.abort();
15984
                }
15985
15986
                this.element.removeClass( "ui-tabs ui-widget ui-widget-content ui-corner-all ui-tabs-collapsible" );
15987
15988
                this.tablist
15989
                        .removeClass( "ui-tabs-nav ui-helper-reset ui-helper-clearfix ui-widget-header ui-corner-all" )
15990
                        .removeAttr( "role" );
15991
15992
                this.anchors
15993
                        .removeClass( "ui-tabs-anchor" )
15994
                        .removeAttr( "role" )
15995
                        .removeAttr( "tabIndex" )
15996
                        .removeUniqueId();
15997
15998
                this.tablist.unbind( this.eventNamespace );
15999
16000
                this.tabs.add( this.panels ).each(function() {
16001
                        if ( $.data( this, "ui-tabs-destroy" ) ) {
16002
                                $( this ).remove();
16003
                        } else {
16004
                                $( this )
16005
                                        .removeClass( "ui-state-default ui-state-active ui-state-disabled " +
16006
                                                "ui-corner-top ui-corner-bottom ui-widget-content ui-tabs-active ui-tabs-panel" )
16007
                                        .removeAttr( "tabIndex" )
16008
                                        .removeAttr( "aria-live" )
16009
                                        .removeAttr( "aria-busy" )
16010
                                        .removeAttr( "aria-selected" )
16011
                                        .removeAttr( "aria-labelledby" )
16012
                                        .removeAttr( "aria-hidden" )
16013
                                        .removeAttr( "aria-expanded" )
16014
                                        .removeAttr( "role" );
16015
                        }
16016
                });
16017
16018
                this.tabs.each(function() {
16019
                        var li = $( this ),
16020
                                prev = li.data( "ui-tabs-aria-controls" );
16021
                        if ( prev ) {
16022
                                li
16023
                                        .attr( "aria-controls", prev )
16024
                                        .removeData( "ui-tabs-aria-controls" );
16025
                        } else {
16026
                                li.removeAttr( "aria-controls" );
16027
                        }
16028
                });
16029
16030
                this.panels.show();
16031
16032
                if ( this.options.heightStyle !== "content" ) {
16033
                        this.panels.css( "height", "" );
16034
                }
16035
        },
16036
16037
        enable: function( index ) {
16038
                var disabled = this.options.disabled;
16039
                if ( disabled === false ) {
16040
                        return;
16041
                }
16042
16043
                if ( index === undefined ) {
16044
                        disabled = false;
16045
                } else {
16046
                        index = this._getIndex( index );
16047
                        if ( $.isArray( disabled ) ) {
16048
                                disabled = $.map( disabled, function( num ) {
16049
                                        return num !== index ? num : null;
16050
                                });
16051
                        } else {
16052
                                disabled = $.map( this.tabs, function( li, num ) {
16053
                                        return num !== index ? num : null;
16054
                                });
16055
                        }
16056
                }
16057
                this._setupDisabled( disabled );
16058
        },
16059
16060
        disable: function( index ) {
16061
                var disabled = this.options.disabled;
16062
                if ( disabled === true ) {
16063
                        return;
16064
                }
16065
16066
                if ( index === undefined ) {
16067
                        disabled = true;
16068
                } else {
16069
                        index = this._getIndex( index );
16070
                        if ( $.inArray( index, disabled ) !== -1 ) {
16071
                                return;
16072
                        }
16073
                        if ( $.isArray( disabled ) ) {
16074
                                disabled = $.merge( [ index ], disabled ).sort();
16075
                        } else {
16076
                                disabled = [ index ];
16077
                        }
16078
                }
16079
                this._setupDisabled( disabled );
16080
        },
16081
16082
        load: function( index, event ) {
16083
                index = this._getIndex( index );
16084
                var that = this,
16085
                        tab = this.tabs.eq( index ),
16086
                        anchor = tab.find( ".ui-tabs-anchor" ),
16087
                        panel = this._getPanelForTab( tab ),
16088
                        eventData = {
16089
                                tab: tab,
16090
                                panel: panel
16091
                        },
16092
                        complete = function( jqXHR, status ) {
16093
                                if ( status === "abort" ) {
16094
                                        that.panels.stop( false, true );
16095
                                }
16096
16097
                                tab.removeClass( "ui-tabs-loading" );
16098
                                panel.removeAttr( "aria-busy" );
16099
16100
                                if ( jqXHR === that.xhr ) {
16101
                                        delete that.xhr;
16102
                                }
16103
                        };
16104
16105
                // not remote
16106
                if ( this._isLocal( anchor[ 0 ] ) ) {
16107
                        return;
16108
                }
16109
16110
                this.xhr = $.ajax( this._ajaxSettings( anchor, event, eventData ) );
16111
16112
                // support: jQuery <1.8
16113
                // jQuery <1.8 returns false if the request is canceled in beforeSend,
16114
                // but as of 1.8, $.ajax() always returns a jqXHR object.
16115
                if ( this.xhr && this.xhr.statusText !== "canceled" ) {
16116
                        tab.addClass( "ui-tabs-loading" );
16117
                        panel.attr( "aria-busy", "true" );
16118
16119
                        this.xhr
16120
                                .done(function( response, status, jqXHR ) {
16121
                                        // support: jQuery <1.8
16122
                                        // http://bugs.jquery.com/ticket/11778
16123
                                        setTimeout(function() {
16124
                                                panel.html( response );
16125
                                                that._trigger( "load", event, eventData );
16126
16127
                                                complete( jqXHR, status );
16128
                                        }, 1 );
16129
                                })
16130
                                .fail(function( jqXHR, status ) {
16131
                                        // support: jQuery <1.8
16132
                                        // http://bugs.jquery.com/ticket/11778
16133
                                        setTimeout(function() {
16134
                                                complete( jqXHR, status );
16135
                                        }, 1 );
16136
                                });
16137
                }
16138
        },
16139
16140
        _ajaxSettings: function( anchor, event, eventData ) {
16141
                var that = this;
16142
                return {
16143
                        url: anchor.attr( "href" ),
16144
                        beforeSend: function( jqXHR, settings ) {
16145
                                return that._trigger( "beforeLoad", event,
16146
                                        $.extend( { jqXHR: jqXHR, ajaxSettings: settings }, eventData ) );
16147
                        }
16148
                };
16149
        },
16150
16151
        _getPanelForTab: function( tab ) {
16152
                var id = $( tab ).attr( "aria-controls" );
16153
                return this.element.find( this._sanitizeSelector( "#" + id ) );
16154
        }
16155
});
16156
16157
16158
/*!
16159
 * jQuery UI Tooltip 1.11.4
16160
 * http://jqueryui.com
16161
 *
16162
 * Copyright jQuery Foundation and other contributors
16163
 * Released under the MIT license.
16164
 * http://jquery.org/license
16165
 *
16166
 * http://api.jqueryui.com/tooltip/
16167
 */
16168
16169
16170
var tooltip = $.widget( "ui.tooltip", {
16171
        version: "1.11.4",
16172
        options: {
16173
                content: function() {
16174
                        // support: IE<9, Opera in jQuery <1.7
16175
                        // .text() can't accept undefined, so coerce to a string
16176
                        var title = $( this ).attr( "title" ) || "";
16177
                        // Escape title, since we're going from an attribute to raw HTML
16178
                        return $( "<a>" ).text( title ).html();
16179
                },
16180
                hide: true,
16181
                // Disabled elements have inconsistent behavior across browsers (#8661)
16182
                items: "[title]:not([disabled])",
16183
                position: {
16184
                        my: "left top+15",
16185
                        at: "left bottom",
16186
                        collision: "flipfit flip"
16187
                },
16188
                show: true,
16189
                tooltipClass: null,
16190
                track: false,
16191
16192
                // callbacks
16193
                close: null,
16194
                open: null
16195
        },
16196
16197
        _addDescribedBy: function( elem, id ) {
16198
                var describedby = (elem.attr( "aria-describedby" ) || "").split( /\s+/ );
16199
                describedby.push( id );
16200
                elem
16201
                        .data( "ui-tooltip-id", id )
16202
                        .attr( "aria-describedby", $.trim( describedby.join( " " ) ) );
16203
        },
16204
16205
        _removeDescribedBy: function( elem ) {
16206
                var id = elem.data( "ui-tooltip-id" ),
16207
                        describedby = (elem.attr( "aria-describedby" ) || "").split( /\s+/ ),
16208
                        index = $.inArray( id, describedby );
16209
16210
                if ( index !== -1 ) {
16211
                        describedby.splice( index, 1 );
16212
                }
16213
16214
                elem.removeData( "ui-tooltip-id" );
16215
                describedby = $.trim( describedby.join( " " ) );
16216
                if ( describedby ) {
16217
                        elem.attr( "aria-describedby", describedby );
16218
                } else {
16219
                        elem.removeAttr( "aria-describedby" );
16220
                }
16221
        },
16222
16223
        _create: function() {
16224
                this._on({
16225
                        mouseover: "open",
16226
                        focusin: "open"
16227
                });
16228
16229
                // IDs of generated tooltips, needed for destroy
16230
                this.tooltips = {};
16231
16232
                // IDs of parent tooltips where we removed the title attribute
16233
                this.parents = {};
16234
16235
                if ( this.options.disabled ) {
16236
                        this._disable();
16237
                }
16238
16239
                // Append the aria-live region so tooltips announce correctly
16240
                this.liveRegion = $( "<div>" )
16241
                        .attr({
16242
                                role: "log",
16243
                                "aria-live": "assertive",
16244
                                "aria-relevant": "additions"
16245
                        })
16246
                        .addClass( "ui-helper-hidden-accessible" )
16247
                        .appendTo( this.document[ 0 ].body );
16248
        },
16249
16250
        _setOption: function( key, value ) {
16251
                var that = this;
16252
16253
                if ( key === "disabled" ) {
16254
                        this[ value ? "_disable" : "_enable" ]();
16255
                        this.options[ key ] = value;
16256
                        // disable element style changes
16257
                        return;
16258
                }
16259
16260
                this._super( key, value );
16261
16262
                if ( key === "content" ) {
16263
                        $.each( this.tooltips, function( id, tooltipData ) {
16264
                                that._updateContent( tooltipData.element );
16265
                        });
16266
                }
16267
        },
16268
16269
        _disable: function() {
16270
                var that = this;
16271
16272
                // close open tooltips
16273
                $.each( this.tooltips, function( id, tooltipData ) {
16274
                        var event = $.Event( "blur" );
16275
                        event.target = event.currentTarget = tooltipData.element[ 0 ];
16276
                        that.close( event, true );
16277
                });
16278
16279
                // remove title attributes to prevent native tooltips
16280
                this.element.find( this.options.items ).addBack().each(function() {
16281
                        var element = $( this );
16282
                        if ( element.is( "[title]" ) ) {
16283
                                element
16284
                                        .data( "ui-tooltip-title", element.attr( "title" ) )
16285
                                        .removeAttr( "title" );
16286
                        }
16287
                });
16288
        },
16289
16290
        _enable: function() {
16291
                // restore title attributes
16292
                this.element.find( this.options.items ).addBack().each(function() {
16293
                        var element = $( this );
16294
                        if ( element.data( "ui-tooltip-title" ) ) {
16295
                                element.attr( "title", element.data( "ui-tooltip-title" ) );
16296
                        }
16297
                });
16298
        },
16299
16300
        open: function( event ) {
16301
                var that = this,
16302
                        target = $( event ? event.target : this.element )
16303
                                // we need closest here due to mouseover bubbling,
16304
                                // but always pointing at the same event target
16305
                                .closest( this.options.items );
16306
16307
                // No element to show a tooltip for or the tooltip is already open
16308
                if ( !target.length || target.data( "ui-tooltip-id" ) ) {
16309
                        return;
16310
                }
16311
16312
                if ( target.attr( "title" ) ) {
16313
                        target.data( "ui-tooltip-title", target.attr( "title" ) );
16314
                }
16315
16316
                target.data( "ui-tooltip-open", true );
16317
16318
                // kill parent tooltips, custom or native, for hover
16319
                if ( event && event.type === "mouseover" ) {
16320
                        target.parents().each(function() {
16321
                                var parent = $( this ),
16322
                                        blurEvent;
16323
                                if ( parent.data( "ui-tooltip-open" ) ) {
16324
                                        blurEvent = $.Event( "blur" );
16325
                                        blurEvent.target = blurEvent.currentTarget = this;
16326
                                        that.close( blurEvent, true );
16327
                                }
16328
                                if ( parent.attr( "title" ) ) {
16329
                                        parent.uniqueId();
16330
                                        that.parents[ this.id ] = {
16331
                                                element: this,
16332
                                                title: parent.attr( "title" )
16333
                                        };
16334
                                        parent.attr( "title", "" );
16335
                                }
16336
                        });
16337
                }
16338
16339
                this._registerCloseHandlers( event, target );
16340
                this._updateContent( target, event );
16341
        },
16342
16343
        _updateContent: function( target, event ) {
16344
                var content,
16345
                        contentOption = this.options.content,
16346
                        that = this,
16347
                        eventType = event ? event.type : null;
16348
16349
                if ( typeof contentOption === "string" ) {
16350
                        return this._open( event, target, contentOption );
16351
                }
16352
16353
                content = contentOption.call( target[0], function( response ) {
16354
16355
                        // IE may instantly serve a cached response for ajax requests
16356
                        // delay this call to _open so the other call to _open runs first
16357
                        that._delay(function() {
16358
16359
                                // Ignore async response if tooltip was closed already
16360
                                if ( !target.data( "ui-tooltip-open" ) ) {
16361
                                        return;
16362
                                }
16363
16364
                                // jQuery creates a special event for focusin when it doesn't
16365
                                // exist natively. To improve performance, the native event
16366
                                // object is reused and the type is changed. Therefore, we can't
16367
                                // rely on the type being correct after the event finished
16368
                                // bubbling, so we set it back to the previous value. (#8740)
16369
                                if ( event ) {
16370
                                        event.type = eventType;
16371
                                }
16372
                                this._open( event, target, response );
16373
                        });
16374
                });
16375
                if ( content ) {
16376
                        this._open( event, target, content );
16377
                }
16378
        },
16379
16380
        _open: function( event, target, content ) {
16381
                var tooltipData, tooltip, delayedShow, a11yContent,
16382
                        positionOption = $.extend( {}, this.options.position );
16383
16384
                if ( !content ) {
16385
                        return;
16386
                }
16387
16388
                // Content can be updated multiple times. If the tooltip already
16389
                // exists, then just update the content and bail.
16390
                tooltipData = this._find( target );
16391
                if ( tooltipData ) {
16392
                        tooltipData.tooltip.find( ".ui-tooltip-content" ).html( content );
16393
                        return;
16394
                }
16395
16396
                // if we have a title, clear it to prevent the native tooltip
16397
                // we have to check first to avoid defining a title if none exists
16398
                // (we don't want to cause an element to start matching [title])
16399
                //
16400
                // We use removeAttr only for key events, to allow IE to export the correct
16401
                // accessible attributes. For mouse events, set to empty string to avoid
16402
                // native tooltip showing up (happens only when removing inside mouseover).
16403
                if ( target.is( "[title]" ) ) {
16404
                        if ( event && event.type === "mouseover" ) {
16405
                                target.attr( "title", "" );
16406
                        } else {
16407
                                target.removeAttr( "title" );
16408
                        }
16409
                }
16410
16411
                tooltipData = this._tooltip( target );
16412
                tooltip = tooltipData.tooltip;
16413
                this._addDescribedBy( target, tooltip.attr( "id" ) );
16414
                tooltip.find( ".ui-tooltip-content" ).html( content );
16415
16416
                // Support: Voiceover on OS X, JAWS on IE <= 9
16417
                // JAWS announces deletions even when aria-relevant="additions"
16418
                // Voiceover will sometimes re-read the entire log region's contents from the beginning
16419
                this.liveRegion.children().hide();
16420
                if ( content.clone ) {
16421
                        a11yContent = content.clone();
16422
                        a11yContent.removeAttr( "id" ).find( "[id]" ).removeAttr( "id" );
16423
                } else {
16424
                        a11yContent = content;
16425
                }
16426
                $( "<div>" ).html( a11yContent ).appendTo( this.liveRegion );
16427
16428
                function position( event ) {
16429
                        positionOption.of = event;
16430
                        if ( tooltip.is( ":hidden" ) ) {
16431
                                return;
16432
                        }
16433
                        tooltip.position( positionOption );
16434
                }
16435
                if ( this.options.track && event && /^mouse/.test( event.type ) ) {
16436
                        this._on( this.document, {
16437
                                mousemove: position
16438
                        });
16439
                        // trigger once to override element-relative positioning
16440
                        position( event );
16441
                } else {
16442
                        tooltip.position( $.extend({
16443
                                of: target
16444
                        }, this.options.position ) );
16445
                }
16446
16447
                tooltip.hide();
16448
16449
                this._show( tooltip, this.options.show );
16450
                // Handle tracking tooltips that are shown with a delay (#8644). As soon
16451
                // as the tooltip is visible, position the tooltip using the most recent
16452
                // event.
16453
                if ( this.options.show && this.options.show.delay ) {
16454
                        delayedShow = this.delayedShow = setInterval(function() {
16455
                                if ( tooltip.is( ":visible" ) ) {
16456
                                        position( positionOption.of );
16457
                                        clearInterval( delayedShow );
16458
                                }
16459
                        }, $.fx.interval );
16460
                }
16461
16462
                this._trigger( "open", event, { tooltip: tooltip } );
16463
        },
16464
16465
        _registerCloseHandlers: function( event, target ) {
16466
                var events = {
16467
                        keyup: function( event ) {
16468
                                if ( event.keyCode === $.ui.keyCode.ESCAPE ) {
16469
                                        var fakeEvent = $.Event(event);
16470
                                        fakeEvent.currentTarget = target[0];
16471
                                        this.close( fakeEvent, true );
16472
                                }
16473
                        }
16474
                };
16475
16476
                // Only bind remove handler for delegated targets. Non-delegated
16477
                // tooltips will handle this in destroy.
16478
                if ( target[ 0 ] !== this.element[ 0 ] ) {
16479
                        events.remove = function() {
16480
                                this._removeTooltip( this._find( target ).tooltip );
16481
                        };
16482
                }
16483
16484
                if ( !event || event.type === "mouseover" ) {
16485
                        events.mouseleave = "close";
16486
                }
16487
                if ( !event || event.type === "focusin" ) {
16488
                        events.focusout = "close";
16489
                }
16490
                this._on( true, target, events );
16491
        },
16492
16493
        close: function( event ) {
16494
                var tooltip,
16495
                        that = this,
16496
                        target = $( event ? event.currentTarget : this.element ),
16497
                        tooltipData = this._find( target );
16498
16499
                // The tooltip may already be closed
16500
                if ( !tooltipData ) {
16501
16502
                        // We set ui-tooltip-open immediately upon open (in open()), but only set the
16503
                        // additional data once there's actually content to show (in _open()). So even if the
16504
                        // tooltip doesn't have full data, we always remove ui-tooltip-open in case we're in
16505
                        // the period between open() and _open().
16506
                        target.removeData( "ui-tooltip-open" );
16507
                        return;
16508
                }
16509
16510
                tooltip = tooltipData.tooltip;
16511
16512
                // disabling closes the tooltip, so we need to track when we're closing
16513
                // to avoid an infinite loop in case the tooltip becomes disabled on close
16514
                if ( tooltipData.closing ) {
16515
                        return;
16516
                }
16517
16518
                // Clear the interval for delayed tracking tooltips
16519
                clearInterval( this.delayedShow );
16520
16521
                // only set title if we had one before (see comment in _open())
16522
                // If the title attribute has changed since open(), don't restore
16523
                if ( target.data( "ui-tooltip-title" ) && !target.attr( "title" ) ) {
16524
                        target.attr( "title", target.data( "ui-tooltip-title" ) );
16525
                }
16526
16527
                this._removeDescribedBy( target );
16528
16529
                tooltipData.hiding = true;
16530
                tooltip.stop( true );
16531
                this._hide( tooltip, this.options.hide, function() {
16532
                        that._removeTooltip( $( this ) );
16533
                });
16534
16535
                target.removeData( "ui-tooltip-open" );
16536
                this._off( target, "mouseleave focusout keyup" );
16537
16538
                // Remove 'remove' binding only on delegated targets
16539
                if ( target[ 0 ] !== this.element[ 0 ] ) {
16540
                        this._off( target, "remove" );
16541
                }
16542
                this._off( this.document, "mousemove" );
16543
16544
                if ( event && event.type === "mouseleave" ) {
16545
                        $.each( this.parents, function( id, parent ) {
16546
                                $( parent.element ).attr( "title", parent.title );
16547
                                delete that.parents[ id ];
16548
                        });
16549
                }
16550
16551
                tooltipData.closing = true;
16552
                this._trigger( "close", event, { tooltip: tooltip } );
16553
                if ( !tooltipData.hiding ) {
16554
                        tooltipData.closing = false;
16555
                }
16556
        },
16557
16558
        _tooltip: function( element ) {
16559
                var tooltip = $( "<div>" )
16560
                                .attr( "role", "tooltip" )
16561
                                .addClass( "ui-tooltip ui-widget ui-corner-all ui-widget-content " +
16562
                                        ( this.options.tooltipClass || "" ) ),
16563
                        id = tooltip.uniqueId().attr( "id" );
16564
16565
                $( "<div>" )
16566
                        .addClass( "ui-tooltip-content" )
16567
                        .appendTo( tooltip );
16568
16569
                tooltip.appendTo( this.document[0].body );
16570
16571
                return this.tooltips[ id ] = {
16572
                        element: element,
16573
                        tooltip: tooltip
16574
                };
16575
        },
16576
16577
        _find: function( target ) {
16578
                var id = target.data( "ui-tooltip-id" );
16579
                return id ? this.tooltips[ id ] : null;
16580
        },
16581
16582
        _removeTooltip: function( tooltip ) {
16583
                tooltip.remove();
16584
                delete this.tooltips[ tooltip.attr( "id" ) ];
16585
        },
16586
16587
        _destroy: function() {
16588
                var that = this;
16589
16590
                // close open tooltips
16591
                $.each( this.tooltips, function( id, tooltipData ) {
16592
                        // Delegate to close method to handle common cleanup
16593
                        var event = $.Event( "blur" ),
16594
                                element = tooltipData.element;
16595
                        event.target = event.currentTarget = element[ 0 ];
16596
                        that.close( event, true );
16597
16598
                        // Remove immediately; destroying an open tooltip doesn't use the
16599
                        // hide animation
16600
                        $( "#" + id ).remove();
16601
16602
                        // Restore the title
16603
                        if ( element.data( "ui-tooltip-title" ) ) {
16604
                                // If the title attribute has changed since open(), don't restore
16605
                                if ( !element.attr( "title" ) ) {
16606
                                        element.attr( "title", element.data( "ui-tooltip-title" ) );
16607
                                }
16608
                                element.removeData( "ui-tooltip-title" );
16609
                        }
16610
                });
16611
                this.liveRegion.remove();
16612
        }
16613
});
16614
16615
16616
16617
}));