bootstrap.js 65.2 KB
Newer Older
XhmikosR's avatar
XhmikosR committed
1001
1002
1003
        that.$element.trigger('focus').trigger(e)
    })
  }
1004

XhmikosR's avatar
XhmikosR committed
1005
1006
  Modal.prototype.hide = function (e) {
    if (e) e.preventDefault()
1007

XhmikosR's avatar
XhmikosR committed
1008
    e = $.Event('hide.bs.modal')
fat's avatar
build    
fat committed
1009

XhmikosR's avatar
XhmikosR committed
1010
    this.$element.trigger(e)
1011

XhmikosR's avatar
XhmikosR committed
1012
    if (!this.isShown || e.isDefaultPrevented()) return
1013

XhmikosR's avatar
XhmikosR committed
1014
    this.isShown = false
1015

XhmikosR's avatar
XhmikosR committed
1016
    this.escape()
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1017
    this.resize()
Chris Rebert's avatar
Chris Rebert committed
1018

XhmikosR's avatar
XhmikosR committed
1019
    $(document).off('focusin.bs.modal')
Chris Rebert's avatar
Chris Rebert committed
1020

XhmikosR's avatar
XhmikosR committed
1021
1022
1023
1024
    this.$element
      .removeClass('in')
      .attr('aria-hidden', true)
      .off('click.dismiss.bs.modal')
1025

XhmikosR's avatar
XhmikosR committed
1026
1027
1028
    $.support.transition && this.$element.hasClass('fade') ?
      this.$element
        .one('bsTransitionEnd', $.proxy(this.hideModal, this))
1029
        .emulateTransitionEnd(Modal.TRANSITION_DURATION) :
XhmikosR's avatar
XhmikosR committed
1030
1031
      this.hideModal()
  }
1032

XhmikosR's avatar
XhmikosR committed
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
  Modal.prototype.enforceFocus = function () {
    $(document)
      .off('focusin.bs.modal') // guard against infinite focus loop
      .on('focusin.bs.modal', $.proxy(function (e) {
        if (this.$element[0] !== e.target && !this.$element.has(e.target).length) {
          this.$element.trigger('focus')
        }
      }, this))
  }

  Modal.prototype.escape = function () {
    if (this.isShown && this.options.keyboard) {
Mark Otto's avatar
grunt    
Mark Otto committed
1045
      this.$element.on('keydown.dismiss.bs.modal', $.proxy(function (e) {
XhmikosR's avatar
XhmikosR committed
1046
1047
1048
        e.which == 27 && this.hide()
      }, this))
    } else if (!this.isShown) {
Mark Otto's avatar
grunt    
Mark Otto committed
1049
      this.$element.off('keydown.dismiss.bs.modal')
XhmikosR's avatar
XhmikosR committed
1050
1051
1052
    }
  }

Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1053
1054
  Modal.prototype.resize = function () {
    if (this.isShown) {
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1055
      $(window).on('resize.bs.modal', $.proxy(this.handleUpdate, this))
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1056
1057
1058
1059
1060
    } else {
      $(window).off('resize.bs.modal')
    }
  }

XhmikosR's avatar
XhmikosR committed
1061
1062
1063
1064
  Modal.prototype.hideModal = function () {
    var that = this
    this.$element.hide()
    this.backdrop(function () {
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1065
      that.$body.removeClass('modal-open')
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1066
      that.resetAdjustments()
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1067
      that.resetScrollbar()
XhmikosR's avatar
XhmikosR committed
1068
1069
1070
      that.$element.trigger('hidden.bs.modal')
    })
  }
1071

XhmikosR's avatar
XhmikosR committed
1072
1073
1074
1075
  Modal.prototype.removeBackdrop = function () {
    this.$backdrop && this.$backdrop.remove()
    this.$backdrop = null
  }
1076

XhmikosR's avatar
XhmikosR committed
1077
1078
1079
  Modal.prototype.backdrop = function (callback) {
    var that = this
    var animate = this.$element.hasClass('fade') ? 'fade' : ''
1080

XhmikosR's avatar
XhmikosR committed
1081
1082
    if (this.isShown && this.options.backdrop) {
      var doAnimate = $.support.transition && animate
1083

XhmikosR's avatar
XhmikosR committed
1084
      this.$backdrop = $('<div class="modal-backdrop ' + animate + '" />')
1085
1086
1087
1088
1089
1090
1091
        .prependTo(this.$element)
        .on('click.dismiss.bs.modal', $.proxy(function (e) {
          if (e.target !== e.currentTarget) return
          this.options.backdrop == 'static'
            ? this.$element[0].focus.call(this.$element[0])
            : this.hide.call(this)
        }, this))
1092

XhmikosR's avatar
XhmikosR committed
1093
      if (doAnimate) this.$backdrop[0].offsetWidth // force reflow
1094

XhmikosR's avatar
XhmikosR committed
1095
      this.$backdrop.addClass('in')
1096

XhmikosR's avatar
XhmikosR committed
1097
      if (!callback) return
1098

XhmikosR's avatar
XhmikosR committed
1099
1100
1101
      doAnimate ?
        this.$backdrop
          .one('bsTransitionEnd', callback)
1102
          .emulateTransitionEnd(Modal.BACKDROP_TRANSITION_DURATION) :
XhmikosR's avatar
XhmikosR committed
1103
        callback()
1104

XhmikosR's avatar
XhmikosR committed
1105
1106
    } else if (!this.isShown && this.$backdrop) {
      this.$backdrop.removeClass('in')
Chris Rebert's avatar
Chris Rebert committed
1107

XhmikosR's avatar
XhmikosR committed
1108
1109
1110
      var callbackRemove = function () {
        that.removeBackdrop()
        callback && callback()
Chris Rebert's avatar
Chris Rebert committed
1111
      }
XhmikosR's avatar
XhmikosR committed
1112
1113
1114
      $.support.transition && this.$element.hasClass('fade') ?
        this.$backdrop
          .one('bsTransitionEnd', callbackRemove)
1115
          .emulateTransitionEnd(Modal.BACKDROP_TRANSITION_DURATION) :
XhmikosR's avatar
XhmikosR committed
1116
        callbackRemove()
1117

XhmikosR's avatar
XhmikosR committed
1118
1119
    } else if (callback) {
      callback()
Chris Rebert's avatar
Chris Rebert committed
1120
    }
XhmikosR's avatar
XhmikosR committed
1121
  }
1122

Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1123
1124
1125
1126
1127
1128
1129
1130
  // these following methods are used to handle overflowing modals

  Modal.prototype.handleUpdate = function () {
    if (this.options.backdrop) this.adjustBackdrop()
    this.adjustDialog()
  }

  Modal.prototype.adjustBackdrop = function () {
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1131
1132
1133
1134
1135
    this.$backdrop
      .css('height', 0)
      .css('height', this.$element[0].scrollHeight)
  }

Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
  Modal.prototype.adjustDialog = function () {
    var modalIsOverflowing = this.$element[0].scrollHeight > document.documentElement.clientHeight

    this.$element.css({
      paddingLeft:  !this.bodyIsOverflowing && modalIsOverflowing ? this.scrollbarWidth : '',
      paddingRight: this.bodyIsOverflowing && !modalIsOverflowing ? this.scrollbarWidth : ''
    })
  }

  Modal.prototype.resetAdjustments = function () {
    this.$element.css({
      paddingLeft: '',
      paddingRight: ''
    })
  }

XhmikosR's avatar
XhmikosR committed
1152
  Modal.prototype.checkScrollbar = function () {
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1153
    this.bodyIsOverflowing = document.body.scrollHeight > document.documentElement.clientHeight
Mark Otto's avatar
grunt    
Mark Otto committed
1154
    this.scrollbarWidth = this.measureScrollbar()
XhmikosR's avatar
XhmikosR committed
1155
  }
fat's avatar
build    
fat committed
1156

XhmikosR's avatar
XhmikosR committed
1157
1158
  Modal.prototype.setScrollbar = function () {
    var bodyPad = parseInt((this.$body.css('padding-right') || 0), 10)
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1159
    if (this.bodyIsOverflowing) this.$body.css('padding-right', bodyPad + this.scrollbarWidth)
XhmikosR's avatar
XhmikosR committed
1160
  }
fat's avatar
build    
fat committed
1161

XhmikosR's avatar
XhmikosR committed
1162
1163
1164
  Modal.prototype.resetScrollbar = function () {
    this.$body.css('padding-right', '')
  }
fat's avatar
build    
fat committed
1165

XhmikosR's avatar
XhmikosR committed
1166
1167
1168
1169
1170
1171
1172
1173
  Modal.prototype.measureScrollbar = function () { // thx walsh
    var scrollDiv = document.createElement('div')
    scrollDiv.className = 'modal-scrollbar-measure'
    this.$body.append(scrollDiv)
    var scrollbarWidth = scrollDiv.offsetWidth - scrollDiv.clientWidth
    this.$body[0].removeChild(scrollDiv)
    return scrollbarWidth
  }
1174
1175


XhmikosR's avatar
XhmikosR committed
1176
1177
  // MODAL PLUGIN DEFINITION
  // =======================
fat's avatar
fat committed
1178

XhmikosR's avatar
XhmikosR committed
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
1189
  function Plugin(option, _relatedTarget) {
    return this.each(function () {
      var $this   = $(this)
      var data    = $this.data('bs.modal')
      var options = $.extend({}, Modal.DEFAULTS, $this.data(), typeof option == 'object' && option)

      if (!data) $this.data('bs.modal', (data = new Modal(this, options)))
      if (typeof option == 'string') data[option](_relatedTarget)
      else if (options.show) data.show(_relatedTarget)
    })
  }
1190

XhmikosR's avatar
XhmikosR committed
1191
  var old = $.fn.modal
Mark Otto's avatar
Mark Otto committed
1192

XhmikosR's avatar
XhmikosR committed
1193
1194
  $.fn.modal             = Plugin
  $.fn.modal.Constructor = Modal
1195
1196


XhmikosR's avatar
XhmikosR committed
1197
1198
  // MODAL NO CONFLICT
  // =================
1199

XhmikosR's avatar
XhmikosR committed
1200
1201
1202
1203
  $.fn.modal.noConflict = function () {
    $.fn.modal = old
    return this
  }
1204
1205


XhmikosR's avatar
XhmikosR committed
1206
1207
  // MODAL DATA-API
  // ==============
1208

XhmikosR's avatar
XhmikosR committed
1209
1210
1211
1212
1213
  $(document).on('click.bs.modal.data-api', '[data-toggle="modal"]', function (e) {
    var $this   = $(this)
    var href    = $this.attr('href')
    var $target = $($this.attr('data-target') || (href && href.replace(/.*(?=#[^\s]+$)/, ''))) // strip for ie7
    var option  = $target.data('bs.modal') ? 'toggle' : $.extend({ remote: !/#/.test(href) && href }, $target.data(), $this.data())
1214

XhmikosR's avatar
XhmikosR committed
1215
    if ($this.is('a')) e.preventDefault()
1216

XhmikosR's avatar
XhmikosR committed
1217
1218
1219
1220
    $target.one('show.bs.modal', function (showEvent) {
      if (showEvent.isDefaultPrevented()) return // only register focus restorer if modal will actually get shown
      $target.one('hidden.bs.modal', function () {
        $this.is(':visible') && $this.trigger('focus')
Mark Otto's avatar
grunt    
Mark Otto committed
1221
      })
Mark Otto's avatar
Mark Otto committed
1222
    })
XhmikosR's avatar
XhmikosR committed
1223
    Plugin.call($target, option, this)
Jacob Thornton's avatar
Jacob Thornton committed
1224
  })
1225

XhmikosR's avatar
XhmikosR committed
1226
}(jQuery);
1227

1228
/* ========================================================================
Mark Otto's avatar
Mark Otto committed
1229
 * Bootstrap: tooltip.js v3.3.2
Mark Otto's avatar
Mark Otto committed
1230
 * http://getbootstrap.com/javascript/#tooltip
1231
 * Inspired by the original jQuery.tipsy by Jason Frame
1232
 * ========================================================================
Bootstrap's Grunt bot's avatar
Bootstrap's Grunt bot committed
1233
 * Copyright 2011-2015 Twitter, Inc.
1234
 * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
1235
 * ======================================================================== */
1236
1237


XhmikosR's avatar
XhmikosR committed
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
+function ($) {
  'use strict';

  // TOOLTIP PUBLIC CLASS DEFINITION
  // ===============================

  var Tooltip = function (element, options) {
    this.type       =
    this.options    =
    this.enabled    =
    this.timeout    =
    this.hoverState =
    this.$element   = null

    this.init('tooltip', element, options)
  }

Mark Otto's avatar
Mark Otto committed
1255
  Tooltip.VERSION  = '3.3.2'
XhmikosR's avatar
XhmikosR committed
1256

1257
1258
  Tooltip.TRANSITION_DURATION = 150

XhmikosR's avatar
XhmikosR committed
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
  Tooltip.DEFAULTS = {
    animation: true,
    placement: 'top',
    selector: false,
    template: '<div class="tooltip" role="tooltip"><div class="tooltip-arrow"></div><div class="tooltip-inner"></div></div>',
    trigger: 'hover focus',
    title: '',
    delay: 0,
    html: false,
    container: false,
    viewport: {
      selector: 'body',
      padding: 0
    }
  }

  Tooltip.prototype.init = function (type, element, options) {
    this.enabled   = true
    this.type      = type
    this.$element  = $(element)
    this.options   = this.getOptions(options)
    this.$viewport = this.options.viewport && $(this.options.viewport.selector || this.options.viewport)

    var triggers = this.options.trigger.split(' ')

    for (var i = triggers.length; i--;) {
      var trigger = triggers[i]

      if (trigger == 'click') {
        this.$element.on('click.' + this.type, this.options.selector, $.proxy(this.toggle, this))
      } else if (trigger != 'manual') {
        var eventIn  = trigger == 'hover' ? 'mouseenter' : 'focusin'
        var eventOut = trigger == 'hover' ? 'mouseleave' : 'focusout'

        this.$element.on(eventIn  + '.' + this.type, this.options.selector, $.proxy(this.enter, this))
        this.$element.on(eventOut + '.' + this.type, this.options.selector, $.proxy(this.leave, this))
      }
    }
1297

XhmikosR's avatar
XhmikosR committed
1298
1299
1300
1301
    this.options.selector ?
      (this._options = $.extend({}, this.options, { trigger: 'manual', selector: '' })) :
      this.fixTitle()
  }
1302

XhmikosR's avatar
XhmikosR committed
1303
1304
1305
  Tooltip.prototype.getDefaults = function () {
    return Tooltip.DEFAULTS
  }
1306

XhmikosR's avatar
XhmikosR committed
1307
1308
  Tooltip.prototype.getOptions = function (options) {
    options = $.extend({}, this.getDefaults(), this.$element.data(), options)
1309

XhmikosR's avatar
XhmikosR committed
1310
1311
1312
1313
    if (options.delay && typeof options.delay == 'number') {
      options.delay = {
        show: options.delay,
        hide: options.delay
1314
1315
      }
    }
1316

XhmikosR's avatar
XhmikosR committed
1317
1318
    return options
  }
Jacob Thornton's avatar
Jacob Thornton committed
1319

XhmikosR's avatar
XhmikosR committed
1320
1321
1322
  Tooltip.prototype.getDelegateOptions = function () {
    var options  = {}
    var defaults = this.getDefaults()
1323

XhmikosR's avatar
XhmikosR committed
1324
1325
1326
    this._options && $.each(this._options, function (key, value) {
      if (defaults[key] != value) options[key] = value
    })
fat's avatar
rebuild    
fat committed
1327

XhmikosR's avatar
XhmikosR committed
1328
1329
    return options
  }
Jacob Thornton's avatar
Jacob Thornton committed
1330

XhmikosR's avatar
XhmikosR committed
1331
1332
1333
  Tooltip.prototype.enter = function (obj) {
    var self = obj instanceof this.constructor ?
      obj : $(obj.currentTarget).data('bs.' + this.type)
Mark Otto's avatar
Mark Otto committed
1334

XhmikosR's avatar
XhmikosR committed
1335
1336
1337
1338
1339
    if (self && self.$tip && self.$tip.is(':visible')) {
      self.hoverState = 'in'
      return
    }

XhmikosR's avatar
XhmikosR committed
1340
1341
1342
    if (!self) {
      self = new this.constructor(obj.currentTarget, this.getDelegateOptions())
      $(obj.currentTarget).data('bs.' + this.type, self)
Mark Otto's avatar
Mark Otto committed
1343
    }
1344

XhmikosR's avatar
XhmikosR committed
1345
    clearTimeout(self.timeout)
1346

XhmikosR's avatar
XhmikosR committed
1347
    self.hoverState = 'in'
1348

XhmikosR's avatar
XhmikosR committed
1349
    if (!self.options.delay || !self.options.delay.show) return self.show()
1350

XhmikosR's avatar
XhmikosR committed
1351
1352
1353
1354
    self.timeout = setTimeout(function () {
      if (self.hoverState == 'in') self.show()
    }, self.options.delay.show)
  }
1355

XhmikosR's avatar
XhmikosR committed
1356
1357
1358
  Tooltip.prototype.leave = function (obj) {
    var self = obj instanceof this.constructor ?
      obj : $(obj.currentTarget).data('bs.' + this.type)
Mark Otto's avatar
Mark Otto committed
1359

XhmikosR's avatar
XhmikosR committed
1360
1361
1362
    if (!self) {
      self = new this.constructor(obj.currentTarget, this.getDelegateOptions())
      $(obj.currentTarget).data('bs.' + this.type, self)
Mark Otto's avatar
Mark Otto committed
1363
    }
1364

XhmikosR's avatar
XhmikosR committed
1365
    clearTimeout(self.timeout)
fat's avatar
fat committed
1366

XhmikosR's avatar
XhmikosR committed
1367
    self.hoverState = 'out'
1368

XhmikosR's avatar
XhmikosR committed
1369
    if (!self.options.delay || !self.options.delay.hide) return self.hide()
1370

XhmikosR's avatar
XhmikosR committed
1371
1372
1373
1374
    self.timeout = setTimeout(function () {
      if (self.hoverState == 'out') self.hide()
    }, self.options.delay.hide)
  }
1375

XhmikosR's avatar
XhmikosR committed
1376
1377
  Tooltip.prototype.show = function () {
    var e = $.Event('show.bs.' + this.type)
Chris Rebert's avatar
Chris Rebert committed
1378

XhmikosR's avatar
XhmikosR committed
1379
1380
    if (this.hasContent() && this.enabled) {
      this.$element.trigger(e)
fat's avatar
fat committed
1381

Heinrich Fenkart's avatar
Heinrich Fenkart committed
1382
      var inDom = $.contains(this.$element[0].ownerDocument.documentElement, this.$element[0])
XhmikosR's avatar
XhmikosR committed
1383
1384
      if (e.isDefaultPrevented() || !inDom) return
      var that = this
1385

XhmikosR's avatar
XhmikosR committed
1386
      var $tip = this.tip()
fat's avatar
fat committed
1387

XhmikosR's avatar
XhmikosR committed
1388
      var tipId = this.getUID(this.type)
fat's avatar
fat committed
1389

XhmikosR's avatar
XhmikosR committed
1390
1391
1392
      this.setContent()
      $tip.attr('id', tipId)
      this.$element.attr('aria-describedby', tipId)
1393

XhmikosR's avatar
XhmikosR committed
1394
      if (this.options.animation) $tip.addClass('fade')
1395

XhmikosR's avatar
XhmikosR committed
1396
1397
1398
      var placement = typeof this.options.placement == 'function' ?
        this.options.placement.call(this, $tip[0], this.$element[0]) :
        this.options.placement
1399

XhmikosR's avatar
XhmikosR committed
1400
1401
1402
      var autoToken = /\s?auto?\s?/i
      var autoPlace = autoToken.test(placement)
      if (autoPlace) placement = placement.replace(autoToken, '') || 'top'
1403

XhmikosR's avatar
XhmikosR committed
1404
1405
1406
1407
1408
      $tip
        .detach()
        .css({ top: 0, left: 0, display: 'block' })
        .addClass(placement)
        .data('bs.' + this.type, this)
fat's avatar
fat committed
1409

XhmikosR's avatar
XhmikosR committed
1410
      this.options.container ? $tip.appendTo(this.options.container) : $tip.insertAfter(this.$element)
1411

XhmikosR's avatar
XhmikosR committed
1412
1413
1414
      var pos          = this.getPosition()
      var actualWidth  = $tip[0].offsetWidth
      var actualHeight = $tip[0].offsetHeight
Chris Rebert's avatar
Chris Rebert committed
1415

XhmikosR's avatar
XhmikosR committed
1416
1417
      if (autoPlace) {
        var orgPlacement = placement
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1418
1419
        var $container   = this.options.container ? $(this.options.container) : this.$element.parent()
        var containerDim = this.getPosition($container)
Chris Rebert's avatar
Chris Rebert committed
1420

Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1421
1422
1423
1424
        placement = placement == 'bottom' && pos.bottom + actualHeight > containerDim.bottom ? 'top'    :
                    placement == 'top'    && pos.top    - actualHeight < containerDim.top    ? 'bottom' :
                    placement == 'right'  && pos.right  + actualWidth  > containerDim.width  ? 'left'   :
                    placement == 'left'   && pos.left   - actualWidth  < containerDim.left   ? 'right'  :
XhmikosR's avatar
XhmikosR committed
1425
                    placement
fat's avatar
fat committed
1426

1427
        $tip
XhmikosR's avatar
XhmikosR committed
1428
          .removeClass(orgPlacement)
1429
          .addClass(placement)
XhmikosR's avatar
XhmikosR committed
1430
      }
1431

XhmikosR's avatar
XhmikosR committed
1432
      var calculatedOffset = this.getCalculatedOffset(placement, pos, actualWidth, actualHeight)
1433

XhmikosR's avatar
XhmikosR committed
1434
      this.applyPlacement(calculatedOffset, placement)
fat's avatar
fat committed
1435

XhmikosR's avatar
XhmikosR committed
1436
      var complete = function () {
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1437
        var prevHoverState = that.hoverState
XhmikosR's avatar
XhmikosR committed
1438
1439
        that.$element.trigger('shown.bs.' + that.type)
        that.hoverState = null
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1440
1441

        if (prevHoverState == 'out') that.leave(that)
XhmikosR's avatar
XhmikosR committed
1442
      }
fat's avatar
fat committed
1443

XhmikosR's avatar
XhmikosR committed
1444
1445
1446
      $.support.transition && this.$tip.hasClass('fade') ?
        $tip
          .one('bsTransitionEnd', complete)
1447
          .emulateTransitionEnd(Tooltip.TRANSITION_DURATION) :
XhmikosR's avatar
XhmikosR committed
1448
1449
1450
        complete()
    }
  }
fat's avatar
fat committed
1451

XhmikosR's avatar
XhmikosR committed
1452
1453
1454
1455
  Tooltip.prototype.applyPlacement = function (offset, placement) {
    var $tip   = this.tip()
    var width  = $tip[0].offsetWidth
    var height = $tip[0].offsetHeight
fat's avatar
fat committed
1456

XhmikosR's avatar
XhmikosR committed
1457
1458
1459
    // manually read margins because getBoundingClientRect includes difference
    var marginTop = parseInt($tip.css('margin-top'), 10)
    var marginLeft = parseInt($tip.css('margin-left'), 10)
fat's avatar
fat committed
1460

XhmikosR's avatar
XhmikosR committed
1461
1462
1463
    // we must check for NaN for ie 8/9
    if (isNaN(marginTop))  marginTop  = 0
    if (isNaN(marginLeft)) marginLeft = 0
fat's avatar
fat committed
1464

XhmikosR's avatar
XhmikosR committed
1465
1466
    offset.top  = offset.top  + marginTop
    offset.left = offset.left + marginLeft
fat's avatar
fat committed
1467

XhmikosR's avatar
XhmikosR committed
1468
1469
1470
1471
1472
1473
1474
1475
    // $.fn.offset doesn't round pixel values
    // so we use setOffset directly with our own function B-0
    $.offset.setOffset($tip[0], $.extend({
      using: function (props) {
        $tip.css({
          top: Math.round(props.top),
          left: Math.round(props.left)
        })
1476
      }
XhmikosR's avatar
XhmikosR committed
1477
    }, offset), 0)
1478

XhmikosR's avatar
XhmikosR committed
1479
    $tip.addClass('in')
fat's avatar
fat committed
1480

XhmikosR's avatar
XhmikosR committed
1481
1482
1483
    // check to see if placing tip in new offset caused the tip to resize itself
    var actualWidth  = $tip[0].offsetWidth
    var actualHeight = $tip[0].offsetHeight
fat's avatar
fat committed
1484

XhmikosR's avatar
XhmikosR committed
1485
1486
1487
    if (placement == 'top' && actualHeight != height) {
      offset.top = offset.top + height - actualHeight
    }
fat's avatar
fat committed
1488

XhmikosR's avatar
XhmikosR committed
1489
    var delta = this.getViewportAdjustedDelta(placement, offset, actualWidth, actualHeight)
fat's avatar
fat committed
1490

XhmikosR's avatar
XhmikosR committed
1491
1492
    if (delta.left) offset.left += delta.left
    else offset.top += delta.top
1493

Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1494
1495
1496
    var isVertical          = /top|bottom/.test(placement)
    var arrowDelta          = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight
    var arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight'
1497

XhmikosR's avatar
XhmikosR committed
1498
    $tip.offset(offset)
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1499
    this.replaceArrow(arrowDelta, $tip[0][arrowOffsetPosition], isVertical)
XhmikosR's avatar
XhmikosR committed
1500
  }
1501

Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1502
1503
1504
1505
  Tooltip.prototype.replaceArrow = function (delta, dimension, isHorizontal) {
    this.arrow()
      .css(isHorizontal ? 'left' : 'top', 50 * (1 - delta / dimension) + '%')
      .css(isHorizontal ? 'top' : 'left', '')
XhmikosR's avatar
XhmikosR committed
1506
  }
1507

XhmikosR's avatar
XhmikosR committed
1508
1509
1510
  Tooltip.prototype.setContent = function () {
    var $tip  = this.tip()
    var title = this.getTitle()
1511

XhmikosR's avatar
XhmikosR committed
1512
1513
1514
    $tip.find('.tooltip-inner')[this.options.html ? 'html' : 'text'](title)
    $tip.removeClass('fade in top bottom left right')
  }
1515

Heinrich Fenkart's avatar
Heinrich Fenkart committed
1516
  Tooltip.prototype.hide = function (callback) {
XhmikosR's avatar
XhmikosR committed
1517
1518
1519
    var that = this
    var $tip = this.tip()
    var e    = $.Event('hide.bs.' + this.type)
1520

XhmikosR's avatar
XhmikosR committed
1521
1522
    function complete() {
      if (that.hoverState != 'in') $tip.detach()
Heinrich Fenkart's avatar
Heinrich Fenkart committed
1523
1524
1525
      that.$element
        .removeAttr('aria-describedby')
        .trigger('hidden.bs.' + that.type)
Heinrich Fenkart's avatar
Heinrich Fenkart committed
1526
      callback && callback()
XhmikosR's avatar
XhmikosR committed
1527
    }
Jacob Thornton's avatar
Jacob Thornton committed
1528

XhmikosR's avatar
XhmikosR committed
1529
    this.$element.trigger(e)
1530

XhmikosR's avatar
XhmikosR committed
1531
    if (e.isDefaultPrevented()) return
1532

XhmikosR's avatar
XhmikosR committed
1533
    $tip.removeClass('in')
1534

XhmikosR's avatar
XhmikosR committed
1535
1536
1537
    $.support.transition && this.$tip.hasClass('fade') ?
      $tip
        .one('bsTransitionEnd', complete)
1538
        .emulateTransitionEnd(Tooltip.TRANSITION_DURATION) :
XhmikosR's avatar
XhmikosR committed
1539
      complete()
1540

XhmikosR's avatar
XhmikosR committed
1541
    this.hoverState = null
1542

XhmikosR's avatar
XhmikosR committed
1543
1544
    return this
  }
1545

XhmikosR's avatar
XhmikosR committed
1546
1547
1548
1549
  Tooltip.prototype.fixTitle = function () {
    var $e = this.$element
    if ($e.attr('title') || typeof ($e.attr('data-original-title')) != 'string') {
      $e.attr('data-original-title', $e.attr('title') || '').attr('title', '')
1550
    }
XhmikosR's avatar
XhmikosR committed
1551
  }
1552

XhmikosR's avatar
XhmikosR committed
1553
1554
1555
  Tooltip.prototype.hasContent = function () {
    return this.getTitle()
  }
1556

XhmikosR's avatar
XhmikosR committed
1557
1558
  Tooltip.prototype.getPosition = function ($element) {
    $element   = $element || this.$element
Mark Otto's avatar
grunt    
Mark Otto committed
1559

XhmikosR's avatar
XhmikosR committed
1560
1561
    var el     = $element[0]
    var isBody = el.tagName == 'BODY'
Mark Otto's avatar
grunt    
Mark Otto committed
1562

Mark Otto's avatar
grunt    
Mark Otto committed
1563
    var elRect    = el.getBoundingClientRect()
Heinrich Fenkart's avatar
Heinrich Fenkart committed
1564
1565
1566
1567
    if (elRect.width == null) {
      // width and height are missing in IE8, so compute them manually; see https://github.com/twbs/bootstrap/issues/14093
      elRect = $.extend({}, elRect, { width: elRect.right - elRect.left, height: elRect.bottom - elRect.top })
    }
Mark Otto's avatar
grunt    
Mark Otto committed
1568
1569
    var elOffset  = isBody ? { top: 0, left: 0 } : $element.offset()
    var scroll    = { scroll: isBody ? document.documentElement.scrollTop || document.body.scrollTop : $element.scrollTop() }
Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1570
    var outerDims = isBody ? { width: $(window).width(), height: $(window).height() } : null
Mark Otto's avatar
grunt    
Mark Otto committed
1571
1572

    return $.extend({}, elRect, scroll, outerDims, elOffset)
XhmikosR's avatar
XhmikosR committed
1573
  }
1574

XhmikosR's avatar
XhmikosR committed
1575
  Tooltip.prototype.getCalculatedOffset = function (placement, pos, actualWidth, actualHeight) {
Bootstrap's Grunt bot's avatar
Bootstrap's Grunt bot committed
1576
1577
    return placement == 'bottom' ? { top: pos.top + pos.height,   left: pos.left + pos.width / 2 - actualWidth / 2 } :
           placement == 'top'    ? { top: pos.top - actualHeight, left: pos.left + pos.width / 2 - actualWidth / 2 } :
XhmikosR's avatar
XhmikosR committed
1578
           placement == 'left'   ? { top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left - actualWidth } :
Bootstrap's Grunt bot's avatar
Bootstrap's Grunt bot committed
1579
        /* placement == 'right' */ { top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left + pos.width }
1580

XhmikosR's avatar
XhmikosR committed
1581
  }
1582

XhmikosR's avatar
XhmikosR committed
1583
1584
1585
  Tooltip.prototype.getViewportAdjustedDelta = function (placement, pos, actualWidth, actualHeight) {
    var delta = { top: 0, left: 0 }
    if (!this.$viewport) return delta
1586

XhmikosR's avatar
XhmikosR committed
1587
1588
    var viewportPadding = this.options.viewport && this.options.viewport.padding || 0
    var viewportDimensions = this.getPosition(this.$viewport)
1589

XhmikosR's avatar
XhmikosR committed
1590
1591
1592
1593
1594
1595
1596
1597
1598
1599
1600
1601
1602
1603
1604
    if (/right|left/.test(placement)) {
      var topEdgeOffset    = pos.top - viewportPadding - viewportDimensions.scroll
      var bottomEdgeOffset = pos.top + viewportPadding - viewportDimensions.scroll + actualHeight
      if (topEdgeOffset < viewportDimensions.top) { // top overflow
        delta.top = viewportDimensions.top - topEdgeOffset
      } else if (bottomEdgeOffset > viewportDimensions.top + viewportDimensions.height) { // bottom overflow
        delta.top = viewportDimensions.top + viewportDimensions.height - bottomEdgeOffset
      }
    } else {
      var leftEdgeOffset  = pos.left - viewportPadding
      var rightEdgeOffset = pos.left + viewportPadding + actualWidth
      if (leftEdgeOffset < viewportDimensions.left) { // left overflow
        delta.left = viewportDimensions.left - leftEdgeOffset
      } else if (rightEdgeOffset > viewportDimensions.width) { // right overflow
        delta.left = viewportDimensions.left + viewportDimensions.width - rightEdgeOffset
1605
      }
Chris Rebert's avatar
Chris Rebert committed
1606
    }
fat's avatar
fat committed
1607

XhmikosR's avatar
XhmikosR committed
1608
1609
    return delta
  }
1610

XhmikosR's avatar
XhmikosR committed
1611
1612
1613
1614
  Tooltip.prototype.getTitle = function () {
    var title
    var $e = this.$element
    var o  = this.options
1615

XhmikosR's avatar
XhmikosR committed
1616
1617
    title = $e.attr('data-original-title')
      || (typeof o.title == 'function' ? o.title.call($e[0]) :  o.title)
1618

XhmikosR's avatar
XhmikosR committed
1619
1620
    return title
  }
1621

XhmikosR's avatar
XhmikosR committed
1622
1623
1624
1625
1626
  Tooltip.prototype.getUID = function (prefix) {
    do prefix += ~~(Math.random() * 1000000)
    while (document.getElementById(prefix))
    return prefix
  }
1627

XhmikosR's avatar
XhmikosR committed
1628
1629
1630
  Tooltip.prototype.tip = function () {
    return (this.$tip = this.$tip || $(this.options.template))
  }
1631

XhmikosR's avatar
XhmikosR committed
1632
1633
1634
  Tooltip.prototype.arrow = function () {
    return (this.$arrow = this.$arrow || this.tip().find('.tooltip-arrow'))
  }
1635

XhmikosR's avatar
XhmikosR committed
1636
1637
1638
  Tooltip.prototype.enable = function () {
    this.enabled = true
  }
1639

XhmikosR's avatar
XhmikosR committed
1640
1641
1642
  Tooltip.prototype.disable = function () {
    this.enabled = false
  }
Mark Otto's avatar
Mark Otto committed
1643

XhmikosR's avatar
XhmikosR committed
1644
1645
1646
  Tooltip.prototype.toggleEnabled = function () {
    this.enabled = !this.enabled
  }
Chris Rebert's avatar
Chris Rebert committed
1647

XhmikosR's avatar
XhmikosR committed
1648
1649
1650
1651
1652
1653
1654
1655
  Tooltip.prototype.toggle = function (e) {
    var self = this
    if (e) {
      self = $(e.currentTarget).data('bs.' + this.type)
      if (!self) {
        self = new this.constructor(e.currentTarget, this.getDelegateOptions())
        $(e.currentTarget).data('bs.' + this.type, self)
      }
Mark Otto's avatar
Mark Otto committed
1656
1657
    }

XhmikosR's avatar
XhmikosR committed
1658
1659
    self.tip().hasClass('in') ? self.leave(self) : self.enter(self)
  }
1660

XhmikosR's avatar
XhmikosR committed
1661
  Tooltip.prototype.destroy = function () {
Heinrich Fenkart's avatar
Heinrich Fenkart committed
1662
    var that = this
XhmikosR's avatar
XhmikosR committed
1663
    clearTimeout(this.timeout)
Heinrich Fenkart's avatar
Heinrich Fenkart committed
1664
1665
1666
    this.hide(function () {
      that.$element.off('.' + that.type).removeData('bs.' + that.type)
    })
XhmikosR's avatar
XhmikosR committed
1667
  }
1668
1669


XhmikosR's avatar
XhmikosR committed
1670
1671
  // TOOLTIP PLUGIN DEFINITION
  // =========================
1672

XhmikosR's avatar
XhmikosR committed
1673
1674
  function Plugin(option) {
    return this.each(function () {
Bootstrap's Grunt bot's avatar
Bootstrap's Grunt bot committed
1675
1676
1677
      var $this   = $(this)
      var data    = $this.data('bs.tooltip')
      var options = typeof option == 'object' && option
fat's avatar
fat committed
1678

XhmikosR's avatar
XhmikosR committed
1679
      if (!data && option == 'destroy') return
Bootstrap's Grunt bot's avatar
Bootstrap's Grunt bot committed
1680
      if (!data) $this.data('bs.tooltip', (data = new Tooltip(this, options)))
XhmikosR's avatar
XhmikosR committed
1681
1682
1683
      if (typeof option == 'string') data[option]()
    })
  }
1684

XhmikosR's avatar
XhmikosR committed
1685
  var old = $.fn.tooltip
Mark Otto's avatar
Mark Otto committed
1686

XhmikosR's avatar
XhmikosR committed
1687
1688
  $.fn.tooltip             = Plugin
  $.fn.tooltip.Constructor = Tooltip
1689

1690

XhmikosR's avatar
XhmikosR committed
1691
1692
  // TOOLTIP NO CONFLICT
  // ===================
1693

XhmikosR's avatar
XhmikosR committed
1694
1695
1696
1697
  $.fn.tooltip.noConflict = function () {
    $.fn.tooltip = old
    return this
  }
1698

XhmikosR's avatar
XhmikosR committed
1699
}(jQuery);
1700

1701
/* ========================================================================
Mark Otto's avatar
Mark Otto committed
1702
 * Bootstrap: popover.js v3.3.2
Mark Otto's avatar
Mark Otto committed
1703
 * http://getbootstrap.com/javascript/#popovers
1704
 * ========================================================================
Bootstrap's Grunt bot's avatar
Bootstrap's Grunt bot committed
1705
 * Copyright 2011-2015 Twitter, Inc.
1706
 * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
1707
 * ======================================================================== */
1708
1709


XhmikosR's avatar
XhmikosR committed
1710
1711
+function ($) {
  'use strict';
Mark Otto's avatar
grunt    
Mark Otto committed
1712

XhmikosR's avatar
XhmikosR committed
1713
1714
  // POPOVER PUBLIC CLASS DEFINITION
  // ===============================
1715

XhmikosR's avatar
XhmikosR committed
1716
1717
1718
  var Popover = function (element, options) {
    this.init('popover', element, options)
  }
fat's avatar
fat committed
1719

XhmikosR's avatar
XhmikosR committed
1720
  if (!$.fn.tooltip) throw new Error('Popover requires tooltip.js')
1721

Mark Otto's avatar
Mark Otto committed
1722
  Popover.VERSION  = '3.3.2'
1723

XhmikosR's avatar
XhmikosR committed
1724
1725
1726
1727
1728
1729
  Popover.DEFAULTS = $.extend({}, $.fn.tooltip.Constructor.DEFAULTS, {
    placement: 'right',
    trigger: 'click',
    content: '',
    template: '<div class="popover" role="tooltip"><div class="arrow"></div><h3 class="popover-title"></h3><div class="popover-content"></div></div>'
  })
fat's avatar
fat committed
1730
1731


XhmikosR's avatar
XhmikosR committed
1732
1733
  // NOTE: POPOVER EXTENDS tooltip.js
  // ================================
1734

XhmikosR's avatar
XhmikosR committed
1735
  Popover.prototype = $.extend({}, $.fn.tooltip.Constructor.prototype)
fat's avatar
fat committed
1736

XhmikosR's avatar
XhmikosR committed
1737
  Popover.prototype.constructor = Popover
1738

XhmikosR's avatar
XhmikosR committed
1739
1740
1741
  Popover.prototype.getDefaults = function () {
    return Popover.DEFAULTS
  }
1742

XhmikosR's avatar
XhmikosR committed
1743
1744
1745
1746
  Popover.prototype.setContent = function () {
    var $tip    = this.tip()
    var title   = this.getTitle()
    var content = this.getContent()
1747

XhmikosR's avatar
XhmikosR committed
1748
    $tip.find('.popover-title')[this.options.html ? 'html' : 'text'](title)
Mark Otto's avatar
grunt    
Mark Otto committed
1749
    $tip.find('.popover-content').children().detach().end()[ // we use append for html objects to maintain js events
XhmikosR's avatar
XhmikosR committed
1750
1751
      this.options.html ? (typeof content == 'string' ? 'html' : 'append') : 'text'
    ](content)
1752

XhmikosR's avatar
XhmikosR committed
1753
    $tip.removeClass('fade top bottom left right in')
1754

XhmikosR's avatar
XhmikosR committed
1755
1756
1757
1758
    // IE8 doesn't accept hiding via the `:empty` pseudo selector, we have to do
    // this manually by checking the contents.
    if (!$tip.find('.popover-title').html()) $tip.find('.popover-title').hide()
  }
1759

XhmikosR's avatar
XhmikosR committed
1760
1761
1762
  Popover.prototype.hasContent = function () {
    return this.getTitle() || this.getContent()
  }
1763

XhmikosR's avatar
XhmikosR committed
1764
1765
1766
  Popover.prototype.getContent = function () {
    var $e = this.$element
    var o  = this.options
1767

XhmikosR's avatar
XhmikosR committed
1768
1769
1770
1771
1772
    return $e.attr('data-content')
      || (typeof o.content == 'function' ?
            o.content.call($e[0]) :
            o.content)
  }
fat's avatar
fat committed
1773

XhmikosR's avatar
XhmikosR committed
1774
1775
1776
  Popover.prototype.arrow = function () {
    return (this.$arrow = this.$arrow || this.tip().find('.arrow'))
  }
1777

XhmikosR's avatar
XhmikosR committed
1778
1779
1780
1781
  Popover.prototype.tip = function () {
    if (!this.$tip) this.$tip = $(this.options.template)
    return this.$tip
  }
1782

1783

XhmikosR's avatar
XhmikosR committed
1784
1785
  // POPOVER PLUGIN DEFINITION
  // =========================
1786

XhmikosR's avatar
XhmikosR committed
1787
1788
  function Plugin(option) {
    return this.each(function () {
Bootstrap's Grunt bot's avatar
Bootstrap's Grunt bot committed
1789
1790
1791
      var $this   = $(this)
      var data    = $this.data('bs.popover')
      var options = typeof option == 'object' && option
Chris Rebert's avatar
Chris Rebert committed
1792

XhmikosR's avatar
XhmikosR committed
1793
      if (!data && option == 'destroy') return
Bootstrap's Grunt bot's avatar
Bootstrap's Grunt bot committed
1794
      if (!data) $this.data('bs.popover', (data = new Popover(this, options)))
XhmikosR's avatar
XhmikosR committed
1795
1796
1797
      if (typeof option == 'string') data[option]()
    })
  }
1798

XhmikosR's avatar
XhmikosR committed
1799
  var old = $.fn.popover
Mark Otto's avatar
Mark Otto committed
1800

XhmikosR's avatar
XhmikosR committed
1801
1802
  $.fn.popover             = Plugin
  $.fn.popover.Constructor = Popover
fat's avatar
fat committed
1803

1804

XhmikosR's avatar
XhmikosR committed
1805
1806
  // POPOVER NO CONFLICT
  // ===================
1807

XhmikosR's avatar
XhmikosR committed
1808
1809
1810
1811
  $.fn.popover.noConflict = function () {
    $.fn.popover = old
    return this
  }
1812

XhmikosR's avatar
XhmikosR committed
1813
}(jQuery);
1814

1815
/* ========================================================================
Mark Otto's avatar
Mark Otto committed
1816
 * Bootstrap: scrollspy.js v3.3.2
Mark Otto's avatar
Mark Otto committed
1817
 * http://getbootstrap.com/javascript/#scrollspy
1818
 * ========================================================================
Bootstrap's Grunt bot's avatar
Bootstrap's Grunt bot committed
1819
 * Copyright 2011-2015 Twitter, Inc.
1820
 * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
1821
 * ======================================================================== */
1822
1823


XhmikosR's avatar
XhmikosR committed
1824
1825
+function ($) {
  'use strict';
1826

XhmikosR's avatar
XhmikosR committed
1827
1828
  // SCROLLSPY CLASS DEFINITION
  // ==========================
1829

XhmikosR's avatar
XhmikosR committed
1830
1831
  function ScrollSpy(element, options) {
    var process  = $.proxy(this.process, this)
1832

XhmikosR's avatar
XhmikosR committed
1833
1834
1835
1836
1837
1838
1839
1840
    this.$body          = $('body')
    this.$scrollElement = $(element).is('body') ? $(window) : $(element)
    this.options        = $.extend({}, ScrollSpy.DEFAULTS, options)
    this.selector       = (this.options.target || '') + ' .nav li > a'
    this.offsets        = []
    this.targets        = []
    this.activeTarget   = null
    this.scrollHeight   = 0
1841

XhmikosR's avatar
XhmikosR committed
1842
1843
1844
1845
    this.$scrollElement.on('scroll.bs.scrollspy', process)
    this.refresh()
    this.process()
  }
1846

Mark Otto's avatar
Mark Otto committed
1847
  ScrollSpy.VERSION  = '3.3.2'
Mark Otto's avatar
grunt    
Mark Otto committed
1848

XhmikosR's avatar
XhmikosR committed
1849
1850
1851
  ScrollSpy.DEFAULTS = {
    offset: 10
  }
1852

XhmikosR's avatar
XhmikosR committed
1853
1854
1855
  ScrollSpy.prototype.getScrollHeight = function () {
    return this.$scrollElement[0].scrollHeight || Math.max(this.$body[0].scrollHeight, document.documentElement.scrollHeight)
  }
Mark Otto's avatar
grunt    
Mark Otto committed
1856

XhmikosR's avatar
XhmikosR committed
1857
1858
1859
  ScrollSpy.prototype.refresh = function () {
    var offsetMethod = 'offset'
    var offsetBase   = 0
Mark Otto's avatar
grunt    
Mark Otto committed
1860

XhmikosR's avatar
XhmikosR committed
1861
1862
1863
1864
    if (!$.isWindow(this.$scrollElement[0])) {
      offsetMethod = 'position'
      offsetBase   = this.$scrollElement.scrollTop()
    }
Mark Otto's avatar
grunt    
Mark Otto committed
1865

XhmikosR's avatar
XhmikosR committed
1866
1867
1868
    this.offsets = []
    this.targets = []
    this.scrollHeight = this.getScrollHeight()
fat's avatar
fat committed
1869

XhmikosR's avatar
XhmikosR committed
1870
    var self     = this
XhmikosR's avatar
XhmikosR committed
1871

XhmikosR's avatar
XhmikosR committed
1872
1873
1874
1875
1876
1877
    this.$body
      .find(this.selector)
      .map(function () {
        var $el   = $(this)
        var href  = $el.data('target') || $el.attr('href')
        var $href = /^#./.test(href) && $(href)
1878

XhmikosR's avatar
XhmikosR committed
1879
1880
1881
1882
1883
1884
1885
1886
1887
1888
1889
1890
1891
1892
1893
1894
1895
1896
1897
1898
1899
1900
1901
        return ($href
          && $href.length
          && $href.is(':visible')
          && [[$href[offsetMethod]().top + offsetBase, href]]) || null
      })
      .sort(function (a, b) { return a[0] - b[0] })
      .each(function () {
        self.offsets.push(this[0])
        self.targets.push(this[1])
      })
  }

  ScrollSpy.prototype.process = function () {
    var scrollTop    = this.$scrollElement.scrollTop() + this.options.offset
    var scrollHeight = this.getScrollHeight()
    var maxScroll    = this.options.offset + scrollHeight - this.$scrollElement.height()
    var offsets      = this.offsets
    var targets      = this.targets
    var activeTarget = this.activeTarget
    var i

    if (this.scrollHeight != scrollHeight) {
      this.refresh()
Mark Otto's avatar
grunt    
Mark Otto committed
1902
1903
    }

XhmikosR's avatar
XhmikosR committed
1904
1905
1906
    if (scrollTop >= maxScroll) {
      return activeTarget != (i = targets[targets.length - 1]) && this.activate(i)
    }
1907

Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1908
1909
1910
    if (activeTarget && scrollTop < offsets[0]) {
      this.activeTarget = null
      return this.clear()
Chris Rebert's avatar
Chris Rebert committed
1911
    }
1912

XhmikosR's avatar
XhmikosR committed
1913
1914
1915
1916
1917
1918
1919
    for (i = offsets.length; i--;) {
      activeTarget != targets[i]
        && scrollTop >= offsets[i]
        && (!offsets[i + 1] || scrollTop <= offsets[i + 1])
        && this.activate(targets[i])
    }
  }
1920

XhmikosR's avatar
XhmikosR committed
1921
1922
  ScrollSpy.prototype.activate = function (target) {
    this.activeTarget = target
1923

Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1924
    this.clear()
1925

XhmikosR's avatar
XhmikosR committed
1926
1927
1928
    var selector = this.selector +
        '[data-target="' + target + '"],' +
        this.selector + '[href="' + target + '"]'
1929

XhmikosR's avatar
XhmikosR committed
1930
1931
1932
    var active = $(selector)
      .parents('li')
      .addClass('active')
1933

XhmikosR's avatar
XhmikosR committed
1934
1935
1936
1937
    if (active.parent('.dropdown-menu').length) {
      active = active
        .closest('li.dropdown')
        .addClass('active')
Chris Rebert's avatar
Chris Rebert committed
1938
    }
1939

XhmikosR's avatar
XhmikosR committed
1940
1941
    active.trigger('activate.bs.scrollspy')
  }
1942

Heinrich Fenkart's avatar
grunt    
Heinrich Fenkart committed
1943
1944
1945
1946
1947
1948
  ScrollSpy.prototype.clear = function () {
    $(this.selector)
      .parentsUntil(this.options.target, '.active')
      .removeClass('active')
  }

1949

XhmikosR's avatar
XhmikosR committed
1950
1951
  // SCROLLSPY PLUGIN DEFINITION
  // ===========================
1952

XhmikosR's avatar
XhmikosR committed
1953
1954
1955
1956
1957
  function Plugin(option) {
    return this.each(function () {
      var $this   = $(this)
      var data    = $this.data('bs.scrollspy')
      var options = typeof option == 'object' && option
Mark Otto's avatar
Mark Otto committed
1958

XhmikosR's avatar
XhmikosR committed
1959
1960
1961
1962
      if (!data) $this.data('bs.scrollspy', (data = new ScrollSpy(this, options)))
      if (typeof option == 'string') data[option]()
    })
  }
1963

XhmikosR's avatar
XhmikosR committed
1964
  var old = $.fn.scrollspy
1965

XhmikosR's avatar
XhmikosR committed
1966
1967
  $.fn.scrollspy             = Plugin
  $.fn.scrollspy.Constructor = ScrollSpy
1968

1969

XhmikosR's avatar
XhmikosR committed
1970
1971
  // SCROLLSPY NO CONFLICT
  // =====================
1972

XhmikosR's avatar
XhmikosR committed
1973
1974
1975
1976
  $.fn.scrollspy.noConflict = function () {
    $.fn.scrollspy = old
    return this
  }
1977

Chris Rebert's avatar
Chris Rebert committed
1978

XhmikosR's avatar
XhmikosR committed
1979
1980
  // SCROLLSPY DATA-API
  // ==================
Chris Rebert's avatar
Chris Rebert committed
1981

XhmikosR's avatar
XhmikosR committed
1982
1983
1984
1985
1986
  $(window).on('load.bs.scrollspy.data-api', function () {
    $('[data-spy="scroll"]').each(function () {
      var $spy = $(this)
      Plugin.call($spy, $spy.data())
    })
1987
  })
1988

XhmikosR's avatar
XhmikosR committed
1989
}(jQuery);
1990

1991
/* ========================================================================
Mark Otto's avatar
Mark Otto committed
1992
 * Bootstrap: tab.js v3.3.2
Mark Otto's avatar
Mark Otto committed
1993
 * http://getbootstrap.com/javascript/#tabs
1994
 * ========================================================================
Bootstrap's Grunt bot's avatar
Bootstrap's Grunt bot committed
1995
 * Copyright 2011-2015 Twitter, Inc.
1996
 * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
1997
 * ======================================================================== */
1998

1999

XhmikosR's avatar
XhmikosR committed
2000
+function ($) {
For faster browsing, not all history is shown. View entire blame