Merge branch 'MDL-51515-master' of git://github.com/junpataleta/moodle
[moodle.git] / theme / bootstrapbase / amd / src / bootstrap.js
1 /* ===================================================
2  * bootstrap-transition.js v2.3.2
3  * http://getbootstrap.com/2.3.2/javascript.html#transitions
4  * ===================================================
5  * Copyright 2013 Twitter, Inc.
6  *
7  * Licensed under the Apache License, Version 2.0 (the "License");
8  * you may not use this file except in compliance with the License.
9  * You may obtain a copy of the License at
10  *
11  * http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing, software
14  * distributed under the License is distributed on an "AS IS" BASIS,
15  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16  * See the License for the specific language governing permissions and
17  * limitations under the License.
18  * ========================================================== */
20 /* jshint ignore:start */
21 define(['jquery'], function($) {
22     $('.h1').hide();
25   "use strict"; // jshint ;_;
28   /* CSS TRANSITION SUPPORT (http://www.modernizr.com/)
29    * ======================================================= */
31   $(function () {
33     $.support.transition = (function () {
35       var transitionEnd = (function () {
37         var el = document.createElement('bootstrap')
38           , transEndEventNames = {
39                'WebkitTransition' : 'webkitTransitionEnd'
40             ,  'MozTransition'    : 'transitionend'
41             ,  'OTransition'      : 'oTransitionEnd otransitionend'
42             ,  'transition'       : 'transitionend'
43             }
44           , name
46         for (name in transEndEventNames){
47           if (el.style[name] !== undefined) {
48             return transEndEventNames[name]
49           }
50         }
52       }())
54       return transitionEnd && {
55         end: transitionEnd
56       }
58     })()
60   })
62 /* ==========================================================
63  * bootstrap-alert.js v2.3.2
64  * http://getbootstrap.com/2.3.2/javascript.html#alerts
65  * ==========================================================
66  * Copyright 2013 Twitter, Inc.
67  *
68  * Licensed under the Apache License, Version 2.0 (the "License");
69  * you may not use this file except in compliance with the License.
70  * You may obtain a copy of the License at
71  *
72  * http://www.apache.org/licenses/LICENSE-2.0
73  *
74  * Unless required by applicable law or agreed to in writing, software
75  * distributed under the License is distributed on an "AS IS" BASIS,
76  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
77  * See the License for the specific language governing permissions and
78  * limitations under the License.
79  * ========================================================== */
83   "use strict"; // jshint ;_;
86  /* ALERT CLASS DEFINITION
87   * ====================== */
89   var dismiss = '[data-dismiss="alert"]'
90     , Alert = function (el) {
91         $(el).on('click', dismiss, this.close)
92       }
94   Alert.prototype.close = function (e) {
95     var $this = $(this)
96       , selector = $this.attr('data-target')
97       , $parent
99     if (!selector) {
100       selector = $this.attr('href')
101       selector = selector && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
102     }
104     $parent = $(selector)
106     e && e.preventDefault()
108     $parent.length || ($parent = $this.hasClass('alert') ? $this : $this.parent())
110     $parent.trigger(e = $.Event('close'))
112     if (e.isDefaultPrevented()) return
114     $parent.removeClass('in')
116     function removeElement() {
117       $parent
118         .trigger('closed')
119         .remove()
120     }
122     $.support.transition && $parent.hasClass('fade') ?
123       $parent.on($.support.transition.end, removeElement) :
124       removeElement()
125   }
128  /* ALERT PLUGIN DEFINITION
129   * ======================= */
131   var old = $.fn.alert
133   $.fn.alert = function (option) {
134     return this.each(function () {
135       var $this = $(this)
136         , data = $this.data('alert')
137       if (!data) $this.data('alert', (data = new Alert(this)))
138       if (typeof option == 'string') data[option].call($this)
139     })
140   }
142   $.fn.alert.Constructor = Alert
145  /* ALERT NO CONFLICT
146   * ================= */
148   $.fn.alert.noConflict = function () {
149     $.fn.alert = old
150     return this
151   }
154  /* ALERT DATA-API
155   * ============== */
157   $(document).on('click.alert.data-api', dismiss, Alert.prototype.close)
159 /* ============================================================
160  * bootstrap-button.js v2.3.2
161  * http://getbootstrap.com/2.3.2/javascript.html#buttons
162  * ============================================================
163  * Copyright 2013 Twitter, Inc.
164  *
165  * Licensed under the Apache License, Version 2.0 (the "License");
166  * you may not use this file except in compliance with the License.
167  * You may obtain a copy of the License at
168  *
169  * http://www.apache.org/licenses/LICENSE-2.0
170  *
171  * Unless required by applicable law or agreed to in writing, software
172  * distributed under the License is distributed on an "AS IS" BASIS,
173  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
174  * See the License for the specific language governing permissions and
175  * limitations under the License.
176  * ============================================================ */
180   "use strict"; // jshint ;_;
183  /* BUTTON PUBLIC CLASS DEFINITION
184   * ============================== */
186   var Button = function (element, options) {
187     this.$element = $(element)
188     this.options = $.extend({}, $.fn.button.defaults, options)
189   }
191   Button.prototype.setState = function (state) {
192     var d = 'disabled'
193       , $el = this.$element
194       , data = $el.data()
195       , val = $el.is('input') ? 'val' : 'html'
197     state = state + 'Text'
198     data.resetText || $el.data('resetText', $el[val]())
200     $el[val](data[state] || this.options[state])
202     // push to event loop to allow forms to submit
203     setTimeout(function () {
204       state == 'loadingText' ?
205         $el.addClass(d).attr(d, d) :
206         $el.removeClass(d).removeAttr(d)
207     }, 0)
208   }
210   Button.prototype.toggle = function () {
211     var $parent = this.$element.closest('[data-toggle="buttons-radio"]')
213     $parent && $parent
214       .find('.active')
215       .removeClass('active')
217     this.$element.toggleClass('active')
218   }
221  /* BUTTON PLUGIN DEFINITION
222   * ======================== */
224   var old = $.fn.button
226   $.fn.button = function (option) {
227     return this.each(function () {
228       var $this = $(this)
229         , data = $this.data('button')
230         , options = typeof option == 'object' && option
231       if (!data) $this.data('button', (data = new Button(this, options)))
232       if (option == 'toggle') data.toggle()
233       else if (option) data.setState(option)
234     })
235   }
237   $.fn.button.defaults = {
238     loadingText: 'loading...'
239   }
241   $.fn.button.Constructor = Button
244  /* BUTTON NO CONFLICT
245   * ================== */
247   $.fn.button.noConflict = function () {
248     $.fn.button = old
249     return this
250   }
253  /* BUTTON DATA-API
254   * =============== */
256   $(document).on('click.button.data-api', '[data-toggle^=button]', function (e) {
257     var $btn = $(e.target)
258     if (!$btn.hasClass('btn')) $btn = $btn.closest('.btn')
259     $btn.button('toggle')
260   })
262 /* ==========================================================
263  * bootstrap-carousel.js v2.3.2
264  * http://getbootstrap.com/2.3.2/javascript.html#carousel
265  * ==========================================================
266  * Copyright 2013 Twitter, Inc.
267  *
268  * Licensed under the Apache License, Version 2.0 (the "License");
269  * you may not use this file except in compliance with the License.
270  * You may obtain a copy of the License at
271  *
272  * http://www.apache.org/licenses/LICENSE-2.0
273  *
274  * Unless required by applicable law or agreed to in writing, software
275  * distributed under the License is distributed on an "AS IS" BASIS,
276  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
277  * See the License for the specific language governing permissions and
278  * limitations under the License.
279  * ========================================================== */
283   "use strict"; // jshint ;_;
286  /* CAROUSEL CLASS DEFINITION
287   * ========================= */
289   var Carousel = function (element, options) {
290     this.$element = $(element)
291     this.$indicators = this.$element.find('.carousel-indicators')
292     this.options = options
293     this.options.pause == 'hover' && this.$element
294       .on('mouseenter', $.proxy(this.pause, this))
295       .on('mouseleave', $.proxy(this.cycle, this))
296   }
298   Carousel.prototype = {
300     cycle: function (e) {
301       if (!e) this.paused = false
302       if (this.interval) clearInterval(this.interval);
303       this.options.interval
304         && !this.paused
305         && (this.interval = setInterval($.proxy(this.next, this), this.options.interval))
306       return this
307     }
309   , getActiveIndex: function () {
310       this.$active = this.$element.find('.item.active')
311       this.$items = this.$active.parent().children()
312       return this.$items.index(this.$active)
313     }
315   , to: function (pos) {
316       var activeIndex = this.getActiveIndex()
317         , that = this
319       if (pos > (this.$items.length - 1) || pos < 0) return
321       if (this.sliding) {
322         return this.$element.one('slid', function () {
323           that.to(pos)
324         })
325       }
327       if (activeIndex == pos) {
328         return this.pause().cycle()
329       }
331       return this.slide(pos > activeIndex ? 'next' : 'prev', $(this.$items[pos]))
332     }
334   , pause: function (e) {
335       if (!e) this.paused = true
336       if (this.$element.find('.next, .prev').length && $.support.transition.end) {
337         this.$element.trigger($.support.transition.end)
338         this.cycle(true)
339       }
340       clearInterval(this.interval)
341       this.interval = null
342       return this
343     }
345   , next: function () {
346       if (this.sliding) return
347       return this.slide('next')
348     }
350   , prev: function () {
351       if (this.sliding) return
352       return this.slide('prev')
353     }
355   , slide: function (type, next) {
356       var $active = this.$element.find('.item.active')
357         , $next = next || $active[type]()
358         , isCycling = this.interval
359         , direction = type == 'next' ? 'left' : 'right'
360         , fallback  = type == 'next' ? 'first' : 'last'
361         , that = this
362         , e
364       this.sliding = true
366       isCycling && this.pause()
368       $next = $next.length ? $next : this.$element.find('.item')[fallback]()
370       e = $.Event('slide', {
371         relatedTarget: $next[0]
372       , direction: direction
373       })
375       if ($next.hasClass('active')) return
377       if (this.$indicators.length) {
378         this.$indicators.find('.active').removeClass('active')
379         this.$element.one('slid', function () {
380           var $nextIndicator = $(that.$indicators.children()[that.getActiveIndex()])
381           $nextIndicator && $nextIndicator.addClass('active')
382         })
383       }
385       if ($.support.transition && this.$element.hasClass('slide')) {
386         this.$element.trigger(e)
387         if (e.isDefaultPrevented()) return
388         $next.addClass(type)
389         $next[0].offsetWidth // force reflow
390         $active.addClass(direction)
391         $next.addClass(direction)
392         this.$element.one($.support.transition.end, function () {
393           $next.removeClass([type, direction].join(' ')).addClass('active')
394           $active.removeClass(['active', direction].join(' '))
395           that.sliding = false
396           setTimeout(function () { that.$element.trigger('slid') }, 0)
397         })
398       } else {
399         this.$element.trigger(e)
400         if (e.isDefaultPrevented()) return
401         $active.removeClass('active')
402         $next.addClass('active')
403         this.sliding = false
404         this.$element.trigger('slid')
405       }
407       isCycling && this.cycle()
409       return this
410     }
412   }
415  /* CAROUSEL PLUGIN DEFINITION
416   * ========================== */
418   var old = $.fn.carousel
420   $.fn.carousel = function (option) {
421     return this.each(function () {
422       var $this = $(this)
423         , data = $this.data('carousel')
424         , options = $.extend({}, $.fn.carousel.defaults, typeof option == 'object' && option)
425         , action = typeof option == 'string' ? option : options.slide
426       if (!data) $this.data('carousel', (data = new Carousel(this, options)))
427       if (typeof option == 'number') data.to(option)
428       else if (action) data[action]()
429       else if (options.interval) data.pause().cycle()
430     })
431   }
433   $.fn.carousel.defaults = {
434     interval: 5000
435   , pause: 'hover'
436   }
438   $.fn.carousel.Constructor = Carousel
441  /* CAROUSEL NO CONFLICT
442   * ==================== */
444   $.fn.carousel.noConflict = function () {
445     $.fn.carousel = old
446     return this
447   }
449  /* CAROUSEL DATA-API
450   * ================= */
452   $(document).on('click.carousel.data-api', '[data-slide], [data-slide-to]', function (e) {
453     var $this = $(this), href
454       , $target = $($this.attr('data-target') || (href = $this.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '')) //strip for ie7
455       , options = $.extend({}, $target.data(), $this.data())
456       , slideIndex
458     $target.carousel(options)
460     if (slideIndex = $this.attr('data-slide-to')) {
461       $target.data('carousel').pause().to(slideIndex).cycle()
462     }
464     e.preventDefault()
465   })
467 /* =============================================================
468  * bootstrap-collapse.js v2.3.2
469  * http://getbootstrap.com/2.3.2/javascript.html#collapse
470  * =============================================================
471  * Copyright 2013 Twitter, Inc.
472  *
473  * Licensed under the Apache License, Version 2.0 (the "License");
474  * you may not use this file except in compliance with the License.
475  * You may obtain a copy of the License at
476  *
477  * http://www.apache.org/licenses/LICENSE-2.0
478  *
479  * Unless required by applicable law or agreed to in writing, software
480  * distributed under the License is distributed on an "AS IS" BASIS,
481  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
482  * See the License for the specific language governing permissions and
483  * limitations under the License.
484  * ============================================================ */
488   "use strict"; // jshint ;_;
491  /* COLLAPSE PUBLIC CLASS DEFINITION
492   * ================================ */
494   var Collapse = function (element, options) {
495     this.$element = $(element)
496     this.options = $.extend({}, $.fn.collapse.defaults, options)
498     if (this.options.parent) {
499       this.$parent = $(this.options.parent)
500     }
502     this.options.toggle && this.toggle()
503   }
505   Collapse.prototype = {
507     constructor: Collapse
509   , dimension: function () {
510       var hasWidth = this.$element.hasClass('width')
511       return hasWidth ? 'width' : 'height'
512     }
514   , show: function () {
515       var dimension
516         , scroll
517         , actives
518         , hasData
520       if (this.transitioning || this.$element.hasClass('in')) return
522       dimension = this.dimension()
523       scroll = $.camelCase(['scroll', dimension].join('-'))
524       actives = this.$parent && this.$parent.find('> .accordion-group > .in')
526       if (actives && actives.length) {
527         hasData = actives.data('collapse')
528         if (hasData && hasData.transitioning) return
529         actives.collapse('hide')
530         hasData || actives.data('collapse', null)
531       }
533       this.$element[dimension](0)
534       this.transition('addClass', $.Event('show'), 'shown')
535       $.support.transition && this.$element[dimension](this.$element[0][scroll])
536     }
538   , hide: function () {
539       var dimension
540       if (this.transitioning || !this.$element.hasClass('in')) return
541       dimension = this.dimension()
542       this.reset(this.$element[dimension]())
543       this.transition('removeClass', $.Event('hide'), 'hidden')
544       this.$element[dimension](0)
545     }
547   , reset: function (size) {
548       var dimension = this.dimension()
550       this.$element
551         .removeClass('collapse')
552         [dimension](size || 'auto')
553         [0].offsetWidth
555       this.$element[size !== null ? 'addClass' : 'removeClass']('collapse')
557       return this
558     }
560   , transition: function (method, startEvent, completeEvent) {
561       var that = this
562         , complete = function () {
563             if (startEvent.type == 'show') that.reset()
564             that.transitioning = 0
565             that.$element.trigger(completeEvent)
566           }
568       this.$element.trigger(startEvent)
570       if (startEvent.isDefaultPrevented()) return
572       this.transitioning = 1
574       this.$element[method]('in')
576       $.support.transition && this.$element.hasClass('collapse') ?
577         this.$element.one($.support.transition.end, complete) :
578         complete()
579     }
581   , toggle: function () {
582       this[this.$element.hasClass('in') ? 'hide' : 'show']()
583     }
585   }
588  /* COLLAPSE PLUGIN DEFINITION
589   * ========================== */
591   var old = $.fn.collapse
593   $.fn.collapse = function (option) {
594     return this.each(function () {
595       var $this = $(this)
596         , data = $this.data('collapse')
597         , options = $.extend({}, $.fn.collapse.defaults, $this.data(), typeof option == 'object' && option)
598       if (!data) $this.data('collapse', (data = new Collapse(this, options)))
599       if (typeof option == 'string') data[option]()
600     })
601   }
603   $.fn.collapse.defaults = {
604     toggle: true
605   }
607   $.fn.collapse.Constructor = Collapse
610  /* COLLAPSE NO CONFLICT
611   * ==================== */
613   $.fn.collapse.noConflict = function () {
614     $.fn.collapse = old
615     return this
616   }
619  /* COLLAPSE DATA-API
620   * ================= */
622   $(document).on('click.collapse.data-api', '[data-toggle=collapse]', function (e) {
623     var $this = $(this), href
624       , target = $this.attr('data-target')
625         || e.preventDefault()
626         || (href = $this.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '') //strip for ie7
627       , option = $(target).data('collapse') ? 'toggle' : $this.data()
628     $this[$(target).hasClass('in') ? 'addClass' : 'removeClass']('collapsed')
629     $(target).collapse(option)
630   })
632 /* ============================================================
633  * bootstrap-dropdown.js v2.3.2
634  * http://getbootstrap.com/2.3.2/javascript.html#dropdowns
635  * ============================================================
636  * Copyright 2013 Twitter, Inc.
637  *
638  * Licensed under the Apache License, Version 2.0 (the "License");
639  * you may not use this file except in compliance with the License.
640  * You may obtain a copy of the License at
641  *
642  * http://www.apache.org/licenses/LICENSE-2.0
643  *
644  * Unless required by applicable law or agreed to in writing, software
645  * distributed under the License is distributed on an "AS IS" BASIS,
646  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
647  * See the License for the specific language governing permissions and
648  * limitations under the License.
649  * ============================================================ */
653   "use strict"; // jshint ;_;
656  /* DROPDOWN CLASS DEFINITION
657   * ========================= */
659   var toggle = '[data-toggle=dropdown]'
660     , Dropdown = function (element) {
661         var $el = $(element).on('click.dropdown.data-api', this.toggle)
662         $('html').on('click.dropdown.data-api', function () {
663           $el.parent().removeClass('open')
664         })
665       }
667   Dropdown.prototype = {
669     constructor: Dropdown
671   , toggle: function (e) {
672       var $this = $(this)
673         , $parent
674         , isActive
676       if ($this.is('.disabled, :disabled')) return
678       $parent = getParent($this)
680       isActive = $parent.hasClass('open')
682       clearMenus()
684       if (!isActive) {
685         if ('ontouchstart' in document.documentElement) {
686           // if mobile we we use a backdrop because click events don't delegate
687           $('<div class="dropdown-backdrop"/>').insertBefore($(this)).on('click', clearMenus)
688         }
689         $parent.toggleClass('open')
690       }
692       $this.focus()
694       return false
695     }
697   , keydown: function (e) {
698       var $this
699         , $items
700         , $active
701         , $parent
702         , isActive
703         , index
705       if (!/(38|40|27)/.test(e.keyCode)) return
707       $this = $(this)
709       e.preventDefault()
710       e.stopPropagation()
712       if ($this.is('.disabled, :disabled')) return
714       $parent = getParent($this)
716       isActive = $parent.hasClass('open')
718       if (!isActive || (isActive && e.keyCode == 27)) {
719         if (e.which == 27) $parent.find(toggle).focus()
720         return $this.click()
721       }
723       $items = $('[role=menu] li:not(.divider):visible a', $parent)
725       if (!$items.length) return
727       index = $items.index($items.filter(':focus'))
729       if (e.keyCode == 38 && index > 0) index--                                        // up
730       if (e.keyCode == 40 && index < $items.length - 1) index++                        // down
731       if (!~index) index = 0
733       $items
734         .eq(index)
735         .focus()
736     }
738   }
740   function clearMenus() {
741     $('.dropdown-backdrop').remove()
742     $(toggle).each(function () {
743       getParent($(this)).removeClass('open')
744     })
745   }
747   function getParent($this) {
748     var selector = $this.attr('data-target')
749       , $parent
751     if (!selector) {
752       selector = $this.attr('href')
753       selector = selector && /#/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
754     }
756     $parent = selector && $(selector)
758     if (!$parent || !$parent.length) $parent = $this.parent()
760     return $parent
761   }
764   /* DROPDOWN PLUGIN DEFINITION
765    * ========================== */
767   var old = $.fn.dropdown
769   $.fn.dropdown = function (option) {
770     return this.each(function () {
771       var $this = $(this)
772         , data = $this.data('dropdown')
773       if (!data) $this.data('dropdown', (data = new Dropdown(this)))
774       if (typeof option == 'string') data[option].call($this)
775     })
776   }
778   $.fn.dropdown.Constructor = Dropdown
781  /* DROPDOWN NO CONFLICT
782   * ==================== */
784   $.fn.dropdown.noConflict = function () {
785     $.fn.dropdown = old
786     return this
787   }
790   /* APPLY TO STANDARD DROPDOWN ELEMENTS
791    * =================================== */
793   $(document)
794     .on('click.dropdown.data-api', clearMenus)
795     .on('click.dropdown.data-api', '.dropdown form', function (e) { e.stopPropagation() })
796     .on('click.dropdown.data-api'  , toggle, Dropdown.prototype.toggle)
797     .on('keydown.dropdown.data-api', toggle + ', [role=menu]' , Dropdown.prototype.keydown)
800 /* =========================================================
801  * bootstrap-modal.js v2.3.2
802  * http://getbootstrap.com/2.3.2/javascript.html#modals
803  * =========================================================
804  * Copyright 2013 Twitter, Inc.
805  *
806  * Licensed under the Apache License, Version 2.0 (the "License");
807  * you may not use this file except in compliance with the License.
808  * You may obtain a copy of the License at
809  *
810  * http://www.apache.org/licenses/LICENSE-2.0
811  *
812  * Unless required by applicable law or agreed to in writing, software
813  * distributed under the License is distributed on an "AS IS" BASIS,
814  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
815  * See the License for the specific language governing permissions and
816  * limitations under the License.
817  * ========================================================= */
821   "use strict"; // jshint ;_;
824  /* MODAL CLASS DEFINITION
825   * ====================== */
827   var Modal = function (element, options) {
828     this.options = options
829     this.$element = $(element)
830       .delegate('[data-dismiss="modal"]', 'click.dismiss.modal', $.proxy(this.hide, this))
831     this.options.remote && this.$element.find('.modal-body').load(this.options.remote)
832   }
834   Modal.prototype = {
836       constructor: Modal
838     , toggle: function () {
839         return this[!this.isShown ? 'show' : 'hide']()
840       }
842     , show: function () {
843         var that = this
844           , e = $.Event('show')
846         this.$element.trigger(e)
848         if (this.isShown || e.isDefaultPrevented()) return
850         this.isShown = true
852         this.escape()
854         this.backdrop(function () {
855           var transition = $.support.transition && that.$element.hasClass('fade')
857           if (!that.$element.parent().length) {
858             that.$element.appendTo(document.body) //don't move modals dom position
859           }
861           that.$element.show()
863           if (transition) {
864             that.$element[0].offsetWidth // force reflow
865           }
867           that.$element
868             .addClass('in')
869             .attr('aria-hidden', false)
871           that.enforceFocus()
873           transition ?
874             that.$element.one($.support.transition.end, function () { that.$element.focus().trigger('shown') }) :
875             that.$element.focus().trigger('shown')
877         })
878       }
880     , hide: function (e) {
881         e && e.preventDefault()
883         var that = this
885         e = $.Event('hide')
887         this.$element.trigger(e)
889         if (!this.isShown || e.isDefaultPrevented()) return
891         this.isShown = false
893         this.escape()
895         $(document).off('focusin.modal')
897         this.$element
898           .removeClass('in')
899           .attr('aria-hidden', true)
901         $.support.transition && this.$element.hasClass('fade') ?
902           this.hideWithTransition() :
903           this.hideModal()
904       }
906     , enforceFocus: function () {
907         var that = this
908         $(document).on('focusin.modal', function (e) {
909           if (that.$element[0] !== e.target && !that.$element.has(e.target).length) {
910             that.$element.focus()
911           }
912         })
913       }
915     , escape: function () {
916         var that = this
917         if (this.isShown && this.options.keyboard) {
918           this.$element.on('keyup.dismiss.modal', function ( e ) {
919             e.which == 27 && that.hide()
920           })
921         } else if (!this.isShown) {
922           this.$element.off('keyup.dismiss.modal')
923         }
924       }
926     , hideWithTransition: function () {
927         var that = this
928           , timeout = setTimeout(function () {
929               that.$element.off($.support.transition.end)
930               that.hideModal()
931             }, 500)
933         this.$element.one($.support.transition.end, function () {
934           clearTimeout(timeout)
935           that.hideModal()
936         })
937       }
939     , hideModal: function () {
940         var that = this
941         this.$element.hide()
942         this.backdrop(function () {
943           that.removeBackdrop()
944           that.$element.trigger('hidden')
945         })
946       }
948     , removeBackdrop: function () {
949         this.$backdrop && this.$backdrop.remove()
950         this.$backdrop = null
951       }
953     , backdrop: function (callback) {
954         var that = this
955           , animate = this.$element.hasClass('fade') ? 'fade' : ''
957         if (this.isShown && this.options.backdrop) {
958           var doAnimate = $.support.transition && animate
960           this.$backdrop = $('<div class="modal-backdrop ' + animate + '" />')
961             .appendTo(document.body)
963           this.$backdrop.click(
964             this.options.backdrop == 'static' ?
965               $.proxy(this.$element[0].focus, this.$element[0])
966             : $.proxy(this.hide, this)
967           )
969           if (doAnimate) this.$backdrop[0].offsetWidth // force reflow
971           this.$backdrop.addClass('in')
973           if (!callback) return
975           doAnimate ?
976             this.$backdrop.one($.support.transition.end, callback) :
977             callback()
979         } else if (!this.isShown && this.$backdrop) {
980           this.$backdrop.removeClass('in')
982           $.support.transition && this.$element.hasClass('fade')?
983             this.$backdrop.one($.support.transition.end, callback) :
984             callback()
986         } else if (callback) {
987           callback()
988         }
989       }
990   }
993  /* MODAL PLUGIN DEFINITION
994   * ======================= */
996   var old = $.fn.modal
998   $.fn.modal = function (option) {
999     return this.each(function () {
1000       var $this = $(this)
1001         , data = $this.data('modal')
1002         , options = $.extend({}, $.fn.modal.defaults, $this.data(), typeof option == 'object' && option)
1003       if (!data) $this.data('modal', (data = new Modal(this, options)))
1004       if (typeof option == 'string') data[option]()
1005       else if (options.show) data.show()
1006     })
1007   }
1009   $.fn.modal.defaults = {
1010       backdrop: true
1011     , keyboard: true
1012     , show: true
1013   }
1015   $.fn.modal.Constructor = Modal
1018  /* MODAL NO CONFLICT
1019   * ================= */
1021   $.fn.modal.noConflict = function () {
1022     $.fn.modal = old
1023     return this
1024   }
1027  /* MODAL DATA-API
1028   * ============== */
1030   $(document).on('click.modal.data-api', '[data-toggle="modal"]', function (e) {
1031     var $this = $(this)
1032       , href = $this.attr('href')
1033       , $target = $($this.attr('data-target') || (href && href.replace(/.*(?=#[^\s]+$)/, ''))) //strip for ie7
1034       , option = $target.data('modal') ? 'toggle' : $.extend({ remote:!/#/.test(href) && href }, $target.data(), $this.data())
1036     e.preventDefault()
1038     $target
1039       .modal(option)
1040       .one('hide', function () {
1041         $this.focus()
1042       })
1043   })
1046 /* ===========================================================
1047  * bootstrap-tooltip.js v2.3.2
1048  * http://getbootstrap.com/2.3.2/javascript.html#tooltips
1049  * Inspired by the original jQuery.tipsy by Jason Frame
1050  * ===========================================================
1051  * Copyright 2013 Twitter, Inc.
1052  *
1053  * Licensed under the Apache License, Version 2.0 (the "License");
1054  * you may not use this file except in compliance with the License.
1055  * You may obtain a copy of the License at
1056  *
1057  * http://www.apache.org/licenses/LICENSE-2.0
1058  *
1059  * Unless required by applicable law or agreed to in writing, software
1060  * distributed under the License is distributed on an "AS IS" BASIS,
1061  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1062  * See the License for the specific language governing permissions and
1063  * limitations under the License.
1064  * ========================================================== */
1068   "use strict"; // jshint ;_;
1071  /* TOOLTIP PUBLIC CLASS DEFINITION
1072   * =============================== */
1074   var Tooltip = function (element, options) {
1075     this.init('tooltip', element, options)
1076   }
1078   Tooltip.prototype = {
1080     constructor: Tooltip
1082   , init: function (type, element, options) {
1083       var eventIn
1084         , eventOut
1085         , triggers
1086         , trigger
1087         , i
1089       this.type = type
1090       this.$element = $(element)
1091       this.options = this.getOptions(options)
1092       this.enabled = true
1094       triggers = this.options.trigger.split(' ')
1096       for (i = triggers.length; i--;) {
1097         trigger = triggers[i]
1098         if (trigger == 'click') {
1099           this.$element.on('click.' + this.type, this.options.selector, $.proxy(this.toggle, this))
1100         } else if (trigger != 'manual') {
1101           eventIn = trigger == 'hover' ? 'mouseenter' : 'focus'
1102           eventOut = trigger == 'hover' ? 'mouseleave' : 'blur'
1103           this.$element.on(eventIn + '.' + this.type, this.options.selector, $.proxy(this.enter, this))
1104           this.$element.on(eventOut + '.' + this.type, this.options.selector, $.proxy(this.leave, this))
1105         }
1106       }
1108       this.options.selector ?
1109         (this._options = $.extend({}, this.options, { trigger: 'manual', selector: '' })) :
1110         this.fixTitle()
1111     }
1113   , getOptions: function (options) {
1114       options = $.extend({}, $.fn[this.type].defaults, this.$element.data(), options)
1116       if (options.delay && typeof options.delay == 'number') {
1117         options.delay = {
1118           show: options.delay
1119         , hide: options.delay
1120         }
1121       }
1123       return options
1124     }
1126   , enter: function (e) {
1127       var defaults = $.fn[this.type].defaults
1128         , options = {}
1129         , self
1131       this._options && $.each(this._options, function (key, value) {
1132         if (defaults[key] != value) options[key] = value
1133       }, this)
1135       self = $(e.currentTarget)[this.type](options).data(this.type)
1137       if (!self.options.delay || !self.options.delay.show) return self.show()
1139       clearTimeout(this.timeout)
1140       self.hoverState = 'in'
1141       this.timeout = setTimeout(function() {
1142         if (self.hoverState == 'in') self.show()
1143       }, self.options.delay.show)
1144     }
1146   , leave: function (e) {
1147       var self = $(e.currentTarget)[this.type](this._options).data(this.type)
1149       if (this.timeout) clearTimeout(this.timeout)
1150       if (!self.options.delay || !self.options.delay.hide) return self.hide()
1152       self.hoverState = 'out'
1153       this.timeout = setTimeout(function() {
1154         if (self.hoverState == 'out') self.hide()
1155       }, self.options.delay.hide)
1156     }
1158   , show: function () {
1159       var $tip
1160         , pos
1161         , actualWidth
1162         , actualHeight
1163         , placement
1164         , tp
1165         , e = $.Event('show')
1167       if (this.hasContent() && this.enabled) {
1168         this.$element.trigger(e)
1169         if (e.isDefaultPrevented()) return
1170         $tip = this.tip()
1171         this.setContent()
1173         if (this.options.animation) {
1174           $tip.addClass('fade')
1175         }
1177         placement = typeof this.options.placement == 'function' ?
1178           this.options.placement.call(this, $tip[0], this.$element[0]) :
1179           this.options.placement
1181         $tip
1182           .detach()
1183           .css({ top: 0, left: 0, display: 'block' })
1185         this.options.container ? $tip.appendTo(this.options.container) : $tip.insertAfter(this.$element)
1187         pos = this.getPosition()
1189         actualWidth = $tip[0].offsetWidth
1190         actualHeight = $tip[0].offsetHeight
1192         switch (placement) {
1193           case 'bottom':
1194             tp = {top: pos.top + pos.height, left: pos.left + pos.width / 2 - actualWidth / 2}
1195             break
1196           case 'top':
1197             tp = {top: pos.top - actualHeight, left: pos.left + pos.width / 2 - actualWidth / 2}
1198             break
1199           case 'left':
1200             tp = {top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left - actualWidth}
1201             break
1202           case 'right':
1203             tp = {top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left + pos.width}
1204             break
1205         }
1207         this.applyPlacement(tp, placement)
1208         this.$element.trigger('shown')
1209       }
1210     }
1212   , applyPlacement: function(offset, placement){
1213       var $tip = this.tip()
1214         , width = $tip[0].offsetWidth
1215         , height = $tip[0].offsetHeight
1216         , actualWidth
1217         , actualHeight
1218         , delta
1219         , replace
1221       $tip
1222         .offset(offset)
1223         .addClass(placement)
1224         .addClass('in')
1226       actualWidth = $tip[0].offsetWidth
1227       actualHeight = $tip[0].offsetHeight
1229       if (placement == 'top' && actualHeight != height) {
1230         offset.top = offset.top + height - actualHeight
1231         replace = true
1232       }
1234       if (placement == 'bottom' || placement == 'top') {
1235         delta = 0
1237         if (offset.left < 0){
1238           delta = offset.left * -2
1239           offset.left = 0
1240           $tip.offset(offset)
1241           actualWidth = $tip[0].offsetWidth
1242           actualHeight = $tip[0].offsetHeight
1243         }
1245         this.replaceArrow(delta - width + actualWidth, actualWidth, 'left')
1246       } else {
1247         this.replaceArrow(actualHeight - height, actualHeight, 'top')
1248       }
1250       if (replace) $tip.offset(offset)
1251     }
1253   , replaceArrow: function(delta, dimension, position){
1254       this
1255         .arrow()
1256         .css(position, delta ? (50 * (1 - delta / dimension) + "%") : '')
1257     }
1259   , setContent: function () {
1260       var $tip = this.tip()
1261         , title = this.getTitle()
1263       $tip.find('.tooltip-inner')[this.options.html ? 'html' : 'text'](title)
1264       $tip.removeClass('fade in top bottom left right')
1265     }
1267   , hide: function () {
1268       var that = this
1269         , $tip = this.tip()
1270         , e = $.Event('hide')
1272       this.$element.trigger(e)
1273       if (e.isDefaultPrevented()) return
1275       $tip.removeClass('in')
1277       function removeWithAnimation() {
1278         var timeout = setTimeout(function () {
1279           $tip.off($.support.transition.end).detach()
1280         }, 500)
1282         $tip.one($.support.transition.end, function () {
1283           clearTimeout(timeout)
1284           $tip.detach()
1285         })
1286       }
1288       $.support.transition && this.$tip.hasClass('fade') ?
1289         removeWithAnimation() :
1290         $tip.detach()
1292       this.$element.trigger('hidden')
1294       return this
1295     }
1297   , fixTitle: function () {
1298       var $e = this.$element
1299       if ($e.attr('title') || typeof($e.attr('data-original-title')) != 'string') {
1300         $e.attr('data-original-title', $e.attr('title') || '').attr('title', '')
1301       }
1302     }
1304   , hasContent: function () {
1305       return this.getTitle()
1306     }
1308   , getPosition: function () {
1309       var el = this.$element[0]
1310       return $.extend({}, (typeof el.getBoundingClientRect == 'function') ? el.getBoundingClientRect() : {
1311         width: el.offsetWidth
1312       , height: el.offsetHeight
1313       }, this.$element.offset())
1314     }
1316   , getTitle: function () {
1317       var title
1318         , $e = this.$element
1319         , o = this.options
1321       title = $e.attr('data-original-title')
1322         || (typeof o.title == 'function' ? o.title.call($e[0]) :  o.title)
1324       return title
1325     }
1327   , tip: function () {
1328       return this.$tip = this.$tip || $(this.options.template)
1329     }
1331   , arrow: function(){
1332       return this.$arrow = this.$arrow || this.tip().find(".tooltip-arrow")
1333     }
1335   , validate: function () {
1336       if (!this.$element[0].parentNode) {
1337         this.hide()
1338         this.$element = null
1339         this.options = null
1340       }
1341     }
1343   , enable: function () {
1344       this.enabled = true
1345     }
1347   , disable: function () {
1348       this.enabled = false
1349     }
1351   , toggleEnabled: function () {
1352       this.enabled = !this.enabled
1353     }
1355   , toggle: function (e) {
1356       var self = e ? $(e.currentTarget)[this.type](this._options).data(this.type) : this
1357       self.tip().hasClass('in') ? self.hide() : self.show()
1358     }
1360   , destroy: function () {
1361       this.hide().$element.off('.' + this.type).removeData(this.type)
1362     }
1364   }
1367  /* TOOLTIP PLUGIN DEFINITION
1368   * ========================= */
1370   var old = $.fn.tooltip
1372   $.fn.tooltip = function ( option ) {
1373     return this.each(function () {
1374       var $this = $(this)
1375         , data = $this.data('tooltip')
1376         , options = typeof option == 'object' && option
1377       if (!data) $this.data('tooltip', (data = new Tooltip(this, options)))
1378       if (typeof option == 'string') data[option]()
1379     })
1380   }
1382   $.fn.tooltip.Constructor = Tooltip
1384   $.fn.tooltip.defaults = {
1385     animation: true
1386   , placement: 'top'
1387   , selector: false
1388   , template: '<div class="tooltip"><div class="tooltip-arrow"></div><div class="tooltip-inner"></div></div>'
1389   , trigger: 'hover focus'
1390   , title: ''
1391   , delay: 0
1392   , html: false
1393   , container: false
1394   }
1397  /* TOOLTIP NO CONFLICT
1398   * =================== */
1400   $.fn.tooltip.noConflict = function () {
1401     $.fn.tooltip = old
1402     return this
1403   }
1406 /* ===========================================================
1407  * bootstrap-popover.js v2.3.2
1408  * http://getbootstrap.com/2.3.2/javascript.html#popovers
1409  * ===========================================================
1410  * Copyright 2013 Twitter, Inc.
1411  *
1412  * Licensed under the Apache License, Version 2.0 (the "License");
1413  * you may not use this file except in compliance with the License.
1414  * You may obtain a copy of the License at
1415  *
1416  * http://www.apache.org/licenses/LICENSE-2.0
1417  *
1418  * Unless required by applicable law or agreed to in writing, software
1419  * distributed under the License is distributed on an "AS IS" BASIS,
1420  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1421  * See the License for the specific language governing permissions and
1422  * limitations under the License.
1423  * =========================================================== */
1427   "use strict"; // jshint ;_;
1430  /* POPOVER PUBLIC CLASS DEFINITION
1431   * =============================== */
1433   var Popover = function (element, options) {
1434     this.init('popover', element, options)
1435   }
1438   /* NOTE: POPOVER EXTENDS BOOTSTRAP-TOOLTIP.js
1439      ========================================== */
1441   Popover.prototype = $.extend({}, $.fn.tooltip.Constructor.prototype, {
1443     constructor: Popover
1445   , setContent: function () {
1446       var $tip = this.tip()
1447         , title = this.getTitle()
1448         , content = this.getContent()
1450       $tip.find('.popover-title')[this.options.html ? 'html' : 'text'](title)
1451       $tip.find('.popover-content')[this.options.html ? 'html' : 'text'](content)
1453       $tip.removeClass('fade top bottom left right in')
1454     }
1456   , hasContent: function () {
1457       return this.getTitle() || this.getContent()
1458     }
1460   , getContent: function () {
1461       var content
1462         , $e = this.$element
1463         , o = this.options
1465       content = (typeof o.content == 'function' ? o.content.call($e[0]) :  o.content)
1466         || $e.attr('data-content')
1468       return content
1469     }
1471   , tip: function () {
1472       if (!this.$tip) {
1473         this.$tip = $(this.options.template)
1474       }
1475       return this.$tip
1476     }
1478   , destroy: function () {
1479       this.hide().$element.off('.' + this.type).removeData(this.type)
1480     }
1482   })
1485  /* POPOVER PLUGIN DEFINITION
1486   * ======================= */
1488   var old = $.fn.popover
1490   $.fn.popover = function (option) {
1491     return this.each(function () {
1492       var $this = $(this)
1493         , data = $this.data('popover')
1494         , options = typeof option == 'object' && option
1495       if (!data) $this.data('popover', (data = new Popover(this, options)))
1496       if (typeof option == 'string') data[option]()
1497     })
1498   }
1500   $.fn.popover.Constructor = Popover
1502   $.fn.popover.defaults = $.extend({} , $.fn.tooltip.defaults, {
1503     placement: 'right'
1504   , trigger: 'click'
1505   , content: ''
1506   , template: '<div class="popover"><div class="arrow"></div><h3 class="popover-title"></h3><div class="popover-content"></div></div>'
1507   })
1510  /* POPOVER NO CONFLICT
1511   * =================== */
1513   $.fn.popover.noConflict = function () {
1514     $.fn.popover = old
1515     return this
1516   }
1519 /* =============================================================
1520  * bootstrap-scrollspy.js v2.3.2
1521  * http://getbootstrap.com/2.3.2/javascript.html#scrollspy
1522  * =============================================================
1523  * Copyright 2013 Twitter, Inc.
1524  *
1525  * Licensed under the Apache License, Version 2.0 (the "License");
1526  * you may not use this file except in compliance with the License.
1527  * You may obtain a copy of the License at
1528  *
1529  * http://www.apache.org/licenses/LICENSE-2.0
1530  *
1531  * Unless required by applicable law or agreed to in writing, software
1532  * distributed under the License is distributed on an "AS IS" BASIS,
1533  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1534  * See the License for the specific language governing permissions and
1535  * limitations under the License.
1536  * ============================================================== */
1540   "use strict"; // jshint ;_;
1543  /* SCROLLSPY CLASS DEFINITION
1544   * ========================== */
1546   function ScrollSpy(element, options) {
1547     var process = $.proxy(this.process, this)
1548       , $element = $(element).is('body') ? $(window) : $(element)
1549       , href
1550     this.options = $.extend({}, $.fn.scrollspy.defaults, options)
1551     this.$scrollElement = $element.on('scroll.scroll-spy.data-api', process)
1552     this.selector = (this.options.target
1553       || ((href = $(element).attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '')) //strip for ie7
1554       || '') + ' .nav li > a'
1555     this.$body = $('body')
1556     this.refresh()
1557     this.process()
1558   }
1560   ScrollSpy.prototype = {
1562       constructor: ScrollSpy
1564     , refresh: function () {
1565         var self = this
1566           , $targets
1568         this.offsets = $([])
1569         this.targets = $([])
1571         $targets = this.$body
1572           .find(this.selector)
1573           .map(function () {
1574             var $el = $(this)
1575               , href = $el.data('target') || $el.attr('href')
1576               , $href = /^#\w/.test(href) && $(href)
1577             return ( $href
1578               && $href.length
1579               && [[ $href.position().top + (!$.isWindow(self.$scrollElement.get(0)) && self.$scrollElement.scrollTop()), href ]] ) || null
1580           })
1581           .sort(function (a, b) { return a[0] - b[0] })
1582           .each(function () {
1583             self.offsets.push(this[0])
1584             self.targets.push(this[1])
1585           })
1586       }
1588     , process: function () {
1589         var scrollTop = this.$scrollElement.scrollTop() + this.options.offset
1590           , scrollHeight = this.$scrollElement[0].scrollHeight || this.$body[0].scrollHeight
1591           , maxScroll = scrollHeight - this.$scrollElement.height()
1592           , offsets = this.offsets
1593           , targets = this.targets
1594           , activeTarget = this.activeTarget
1595           , i
1597         if (scrollTop >= maxScroll) {
1598           return activeTarget != (i = targets.last()[0])
1599             && this.activate ( i )
1600         }
1602         for (i = offsets.length; i--;) {
1603           activeTarget != targets[i]
1604             && scrollTop >= offsets[i]
1605             && (!offsets[i + 1] || scrollTop <= offsets[i + 1])
1606             && this.activate( targets[i] )
1607         }
1608       }
1610     , activate: function (target) {
1611         var active
1612           , selector
1614         this.activeTarget = target
1616         $(this.selector)
1617           .parent('.active')
1618           .removeClass('active')
1620         selector = this.selector
1621           + '[data-target="' + target + '"],'
1622           + this.selector + '[href="' + target + '"]'
1624         active = $(selector)
1625           .parent('li')
1626           .addClass('active')
1628         if (active.parent('.dropdown-menu').length)  {
1629           active = active.closest('li.dropdown').addClass('active')
1630         }
1632         active.trigger('activate')
1633       }
1635   }
1638  /* SCROLLSPY PLUGIN DEFINITION
1639   * =========================== */
1641   var old = $.fn.scrollspy
1643   $.fn.scrollspy = function (option) {
1644     return this.each(function () {
1645       var $this = $(this)
1646         , data = $this.data('scrollspy')
1647         , options = typeof option == 'object' && option
1648       if (!data) $this.data('scrollspy', (data = new ScrollSpy(this, options)))
1649       if (typeof option == 'string') data[option]()
1650     })
1651   }
1653   $.fn.scrollspy.Constructor = ScrollSpy
1655   $.fn.scrollspy.defaults = {
1656     offset: 10
1657   }
1660  /* SCROLLSPY NO CONFLICT
1661   * ===================== */
1663   $.fn.scrollspy.noConflict = function () {
1664     $.fn.scrollspy = old
1665     return this
1666   }
1669  /* SCROLLSPY DATA-API
1670   * ================== */
1672   $(window).on('load', function () {
1673     $('[data-spy="scroll"]').each(function () {
1674       var $spy = $(this)
1675       $spy.scrollspy($spy.data())
1676     })
1677   })
1679 /* ========================================================
1680  * bootstrap-tab.js v2.3.2
1681  * http://getbootstrap.com/2.3.2/javascript.html#tabs
1682  * ========================================================
1683  * Copyright 2013 Twitter, Inc.
1684  *
1685  * Licensed under the Apache License, Version 2.0 (the "License");
1686  * you may not use this file except in compliance with the License.
1687  * You may obtain a copy of the License at
1688  *
1689  * http://www.apache.org/licenses/LICENSE-2.0
1690  *
1691  * Unless required by applicable law or agreed to in writing, software
1692  * distributed under the License is distributed on an "AS IS" BASIS,
1693  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1694  * See the License for the specific language governing permissions and
1695  * limitations under the License.
1696  * ======================================================== */
1700   "use strict"; // jshint ;_;
1703  /* TAB CLASS DEFINITION
1704   * ==================== */
1706   var Tab = function (element) {
1707     this.element = $(element)
1708   }
1710   Tab.prototype = {
1712     constructor: Tab
1714   , show: function () {
1715       var $this = this.element
1716         , $ul = $this.closest('ul:not(.dropdown-menu)')
1717         , selector = $this.attr('data-target')
1718         , previous
1719         , $target
1720         , e
1722       if (!selector) {
1723         selector = $this.attr('href')
1724         selector = selector && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
1725       }
1727       if ( $this.parent('li').hasClass('active') ) return
1729       previous = $ul.find('.active:last a')[0]
1731       e = $.Event('show', {
1732         relatedTarget: previous
1733       })
1735       $this.trigger(e)
1737       if (e.isDefaultPrevented()) return
1739       $target = $(selector)
1741       this.activate($this.parent('li'), $ul)
1742       this.activate($target, $target.parent(), function () {
1743         $this.trigger({
1744           type: 'shown'
1745         , relatedTarget: previous
1746         })
1747       })
1748     }
1750   , activate: function ( element, container, callback) {
1751       var $active = container.find('> .active')
1752         , transition = callback
1753             && $.support.transition
1754             && $active.hasClass('fade')
1756       function next() {
1757         $active
1758           .removeClass('active')
1759           .find('> .dropdown-menu > .active')
1760           .removeClass('active')
1762         element.addClass('active')
1764         if (transition) {
1765           element[0].offsetWidth // reflow for transition
1766           element.addClass('in')
1767         } else {
1768           element.removeClass('fade')
1769         }
1771         if ( element.parent('.dropdown-menu') ) {
1772           element.closest('li.dropdown').addClass('active')
1773         }
1775         callback && callback()
1776       }
1778       transition ?
1779         $active.one($.support.transition.end, next) :
1780         next()
1782       $active.removeClass('in')
1783     }
1784   }
1787  /* TAB PLUGIN DEFINITION
1788   * ===================== */
1790   var old = $.fn.tab
1792   $.fn.tab = function ( option ) {
1793     return this.each(function () {
1794       var $this = $(this)
1795         , data = $this.data('tab')
1796       if (!data) $this.data('tab', (data = new Tab(this)))
1797       if (typeof option == 'string') data[option]()
1798     })
1799   }
1801   $.fn.tab.Constructor = Tab
1804  /* TAB NO CONFLICT
1805   * =============== */
1807   $.fn.tab.noConflict = function () {
1808     $.fn.tab = old
1809     return this
1810   }
1813  /* TAB DATA-API
1814   * ============ */
1816   $(document).on('click.tab.data-api', '[data-toggle="tab"], [data-toggle="pill"]', function (e) {
1817     e.preventDefault()
1818     $(this).tab('show')
1819   })
1821 /* =============================================================
1822  * bootstrap-typeahead.js v2.3.2
1823  * http://getbootstrap.com/2.3.2/javascript.html#typeahead
1824  * =============================================================
1825  * Copyright 2013 Twitter, Inc.
1826  *
1827  * Licensed under the Apache License, Version 2.0 (the "License");
1828  * you may not use this file except in compliance with the License.
1829  * You may obtain a copy of the License at
1830  *
1831  * http://www.apache.org/licenses/LICENSE-2.0
1832  *
1833  * Unless required by applicable law or agreed to in writing, software
1834  * distributed under the License is distributed on an "AS IS" BASIS,
1835  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1836  * See the License for the specific language governing permissions and
1837  * limitations under the License.
1838  * ============================================================ */
1842   "use strict"; // jshint ;_;
1845  /* TYPEAHEAD PUBLIC CLASS DEFINITION
1846   * ================================= */
1848   var Typeahead = function (element, options) {
1849     this.$element = $(element)
1850     this.options = $.extend({}, $.fn.typeahead.defaults, options)
1851     this.matcher = this.options.matcher || this.matcher
1852     this.sorter = this.options.sorter || this.sorter
1853     this.highlighter = this.options.highlighter || this.highlighter
1854     this.updater = this.options.updater || this.updater
1855     this.source = this.options.source
1856     this.$menu = $(this.options.menu)
1857     this.shown = false
1858     this.listen()
1859   }
1861   Typeahead.prototype = {
1863     constructor: Typeahead
1865   , select: function () {
1866       var val = this.$menu.find('.active').attr('data-value')
1867       this.$element
1868         .val(this.updater(val))
1869         .change()
1870       return this.hide()
1871     }
1873   , updater: function (item) {
1874       return item
1875     }
1877   , show: function () {
1878       var pos = $.extend({}, this.$element.position(), {
1879         height: this.$element[0].offsetHeight
1880       })
1882       this.$menu
1883         .insertAfter(this.$element)
1884         .css({
1885           top: pos.top + pos.height
1886         , left: pos.left
1887         })
1888         .show()
1890       this.shown = true
1891       return this
1892     }
1894   , hide: function () {
1895       this.$menu.hide()
1896       this.shown = false
1897       return this
1898     }
1900   , lookup: function (event) {
1901       var items
1903       this.query = this.$element.val()
1905       if (!this.query || this.query.length < this.options.minLength) {
1906         return this.shown ? this.hide() : this
1907       }
1909       items = $.isFunction(this.source) ? this.source(this.query, $.proxy(this.process, this)) : this.source
1911       return items ? this.process(items) : this
1912     }
1914   , process: function (items) {
1915       var that = this
1917       items = $.grep(items, function (item) {
1918         return that.matcher(item)
1919       })
1921       items = this.sorter(items)
1923       if (!items.length) {
1924         return this.shown ? this.hide() : this
1925       }
1927       return this.render(items.slice(0, this.options.items)).show()
1928     }
1930   , matcher: function (item) {
1931       return ~item.toLowerCase().indexOf(this.query.toLowerCase())
1932     }
1934   , sorter: function (items) {
1935       var beginswith = []
1936         , caseSensitive = []
1937         , caseInsensitive = []
1938         , item
1940       while (item = items.shift()) {
1941         if (!item.toLowerCase().indexOf(this.query.toLowerCase())) beginswith.push(item)
1942         else if (~item.indexOf(this.query)) caseSensitive.push(item)
1943         else caseInsensitive.push(item)
1944       }
1946       return beginswith.concat(caseSensitive, caseInsensitive)
1947     }
1949   , highlighter: function (item) {
1950       var query = this.query.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g, '\\$&')
1951       return item.replace(new RegExp('(' + query + ')', 'ig'), function ($1, match) {
1952         return '<strong>' + match + '</strong>'
1953       })
1954     }
1956   , render: function (items) {
1957       var that = this
1959       items = $(items).map(function (i, item) {
1960         i = $(that.options.item).attr('data-value', item)
1961         i.find('a').html(that.highlighter(item))
1962         return i[0]
1963       })
1965       items.first().addClass('active')
1966       this.$menu.html(items)
1967       return this
1968     }
1970   , next: function (event) {
1971       var active = this.$menu.find('.active').removeClass('active')
1972         , next = active.next()
1974       if (!next.length) {
1975         next = $(this.$menu.find('li')[0])
1976       }
1978       next.addClass('active')
1979     }
1981   , prev: function (event) {
1982       var active = this.$menu.find('.active').removeClass('active')
1983         , prev = active.prev()
1985       if (!prev.length) {
1986         prev = this.$menu.find('li').last()
1987       }
1989       prev.addClass('active')
1990     }
1992   , listen: function () {
1993       this.$element
1994         .on('focus',    $.proxy(this.focus, this))
1995         .on('blur',     $.proxy(this.blur, this))
1996         .on('keypress', $.proxy(this.keypress, this))
1997         .on('keyup',    $.proxy(this.keyup, this))
1999       if (this.eventSupported('keydown')) {
2000         this.$element.on('keydown', $.proxy(this.keydown, this))
2001       }
2003       this.$menu
2004         .on('click', $.proxy(this.click, this))
2005         .on('mouseenter', 'li', $.proxy(this.mouseenter, this))
2006         .on('mouseleave', 'li', $.proxy(this.mouseleave, this))
2007     }
2009   , eventSupported: function(eventName) {
2010       var isSupported = eventName in this.$element
2011       if (!isSupported) {
2012         this.$element.setAttribute(eventName, 'return;')
2013         isSupported = typeof this.$element[eventName] === 'function'
2014       }
2015       return isSupported
2016     }
2018   , move: function (e) {
2019       if (!this.shown) return
2021       switch(e.keyCode) {
2022         case 9: // tab
2023         case 13: // enter
2024         case 27: // escape
2025           e.preventDefault()
2026           break
2028         case 38: // up arrow
2029           e.preventDefault()
2030           this.prev()
2031           break
2033         case 40: // down arrow
2034           e.preventDefault()
2035           this.next()
2036           break
2037       }
2039       e.stopPropagation()
2040     }
2042   , keydown: function (e) {
2043       this.suppressKeyPressRepeat = ~$.inArray(e.keyCode, [40,38,9,13,27])
2044       this.move(e)
2045     }
2047   , keypress: function (e) {
2048       if (this.suppressKeyPressRepeat) return
2049       this.move(e)
2050     }
2052   , keyup: function (e) {
2053       switch(e.keyCode) {
2054         case 40: // down arrow
2055         case 38: // up arrow
2056         case 16: // shift
2057         case 17: // ctrl
2058         case 18: // alt
2059           break
2061         case 9: // tab
2062         case 13: // enter
2063           if (!this.shown) return
2064           this.select()
2065           break
2067         case 27: // escape
2068           if (!this.shown) return
2069           this.hide()
2070           break
2072         default:
2073           this.lookup()
2074       }
2076       e.stopPropagation()
2077       e.preventDefault()
2078   }
2080   , focus: function (e) {
2081       this.focused = true
2082     }
2084   , blur: function (e) {
2085       this.focused = false
2086       if (!this.mousedover && this.shown) this.hide()
2087     }
2089   , click: function (e) {
2090       e.stopPropagation()
2091       e.preventDefault()
2092       this.select()
2093       this.$element.focus()
2094     }
2096   , mouseenter: function (e) {
2097       this.mousedover = true
2098       this.$menu.find('.active').removeClass('active')
2099       $(e.currentTarget).addClass('active')
2100     }
2102   , mouseleave: function (e) {
2103       this.mousedover = false
2104       if (!this.focused && this.shown) this.hide()
2105     }
2107   }
2110   /* TYPEAHEAD PLUGIN DEFINITION
2111    * =========================== */
2113   var old = $.fn.typeahead
2115   $.fn.typeahead = function (option) {
2116     return this.each(function () {
2117       var $this = $(this)
2118         , data = $this.data('typeahead')
2119         , options = typeof option == 'object' && option
2120       if (!data) $this.data('typeahead', (data = new Typeahead(this, options)))
2121       if (typeof option == 'string') data[option]()
2122     })
2123   }
2125   $.fn.typeahead.defaults = {
2126     source: []
2127   , items: 8
2128   , menu: '<ul class="typeahead dropdown-menu"></ul>'
2129   , item: '<li><a href="#"></a></li>'
2130   , minLength: 1
2131   }
2133   $.fn.typeahead.Constructor = Typeahead
2136  /* TYPEAHEAD NO CONFLICT
2137   * =================== */
2139   $.fn.typeahead.noConflict = function () {
2140     $.fn.typeahead = old
2141     return this
2142   }
2145  /* TYPEAHEAD DATA-API
2146   * ================== */
2148   $(document).on('focus.typeahead.data-api', '[data-provide="typeahead"]', function (e) {
2149     var $this = $(this)
2150     if ($this.data('typeahead')) return
2151     $this.typeahead($this.data())
2152   })
2155 /* ==========================================================
2156  * bootstrap-affix.js v2.3.2
2157  * http://getbootstrap.com/2.3.2/javascript.html#affix
2158  * ==========================================================
2159  * Copyright 2013 Twitter, Inc.
2160  *
2161  * Licensed under the Apache License, Version 2.0 (the "License");
2162  * you may not use this file except in compliance with the License.
2163  * You may obtain a copy of the License at
2164  *
2165  * http://www.apache.org/licenses/LICENSE-2.0
2166  *
2167  * Unless required by applicable law or agreed to in writing, software
2168  * distributed under the License is distributed on an "AS IS" BASIS,
2169  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
2170  * See the License for the specific language governing permissions and
2171  * limitations under the License.
2172  * ========================================================== */
2176   "use strict"; // jshint ;_;
2179  /* AFFIX CLASS DEFINITION
2180   * ====================== */
2182   var Affix = function (element, options) {
2183     this.options = $.extend({}, $.fn.affix.defaults, options)
2184     this.$window = $(window)
2185       .on('scroll.affix.data-api', $.proxy(this.checkPosition, this))
2186       .on('click.affix.data-api',  $.proxy(function () { setTimeout($.proxy(this.checkPosition, this), 1) }, this))
2187     this.$element = $(element)
2188     this.checkPosition()
2189   }
2191   Affix.prototype.checkPosition = function () {
2192     if (!this.$element.is(':visible')) return
2194     var scrollHeight = $(document).height()
2195       , scrollTop = this.$window.scrollTop()
2196       , position = this.$element.offset()
2197       , offset = this.options.offset
2198       , offsetBottom = offset.bottom
2199       , offsetTop = offset.top
2200       , reset = 'affix affix-top affix-bottom'
2201       , affix
2203     if (typeof offset != 'object') offsetBottom = offsetTop = offset
2204     if (typeof offsetTop == 'function') offsetTop = offset.top()
2205     if (typeof offsetBottom == 'function') offsetBottom = offset.bottom()
2207     affix = this.unpin != null && (scrollTop + this.unpin <= position.top) ?
2208       false    : offsetBottom != null && (position.top + this.$element.height() >= scrollHeight - offsetBottom) ?
2209       'bottom' : offsetTop != null && scrollTop <= offsetTop ?
2210       'top'    : false
2212     if (this.affixed === affix) return
2214     this.affixed = affix
2215     this.unpin = affix == 'bottom' ? position.top - scrollTop : null
2217     this.$element.removeClass(reset).addClass('affix' + (affix ? '-' + affix : ''))
2218   }
2221  /* AFFIX PLUGIN DEFINITION
2222   * ======================= */
2224   var old = $.fn.affix
2226   $.fn.affix = function (option) {
2227     return this.each(function () {
2228       var $this = $(this)
2229         , data = $this.data('affix')
2230         , options = typeof option == 'object' && option
2231       if (!data) $this.data('affix', (data = new Affix(this, options)))
2232       if (typeof option == 'string') data[option]()
2233     })
2234   }
2236   $.fn.affix.Constructor = Affix
2238   $.fn.affix.defaults = {
2239     offset: 0
2240   }
2243  /* AFFIX NO CONFLICT
2244   * ================= */
2246   $.fn.affix.noConflict = function () {
2247     $.fn.affix = old
2248     return this
2249   }
2252  /* AFFIX DATA-API
2253   * ============== */
2255   $(window).on('load', function () {
2256     $('[data-spy="affix"]').each(function () {
2257       var $spy = $(this)
2258         , data = $spy.data()
2260       data.offset = data.offset || {}
2262       data.offsetBottom && (data.offset.bottom = data.offsetBottom)
2263       data.offsetTop && (data.offset.top = data.offsetTop)
2265       $spy.affix(data)
2266     })
2267   })
2271 });
2272 /* jshint ignore:end */