bootstrap-source/bootstrap-3.0.3/js/tooltip.js
author stetrabby <info@trabucchi.de>
Fri, 20 Dec 2013 22:49:16 +0100
changeset 54 0ded9d7748b7
permissions -rwxr-xr-x
initial less based on the pymove3d.css
info@54
     1
/* ========================================================================
info@54
     2
 * Bootstrap: tooltip.js v3.0.3
info@54
     3
 * http://getbootstrap.com/javascript/#tooltip
info@54
     4
 * Inspired by the original jQuery.tipsy by Jason Frame
info@54
     5
 * ========================================================================
info@54
     6
 * Copyright 2013 Twitter, Inc.
info@54
     7
 *
info@54
     8
 * Licensed under the Apache License, Version 2.0 (the "License");
info@54
     9
 * you may not use this file except in compliance with the License.
info@54
    10
 * You may obtain a copy of the License at
info@54
    11
 *
info@54
    12
 * http://www.apache.org/licenses/LICENSE-2.0
info@54
    13
 *
info@54
    14
 * Unless required by applicable law or agreed to in writing, software
info@54
    15
 * distributed under the License is distributed on an "AS IS" BASIS,
info@54
    16
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
info@54
    17
 * See the License for the specific language governing permissions and
info@54
    18
 * limitations under the License.
info@54
    19
 * ======================================================================== */
info@54
    20
info@54
    21
info@54
    22
+function ($) { "use strict";
info@54
    23
info@54
    24
  // TOOLTIP PUBLIC CLASS DEFINITION
info@54
    25
  // ===============================
info@54
    26
info@54
    27
  var Tooltip = function (element, options) {
info@54
    28
    this.type       =
info@54
    29
    this.options    =
info@54
    30
    this.enabled    =
info@54
    31
    this.timeout    =
info@54
    32
    this.hoverState =
info@54
    33
    this.$element   = null
info@54
    34
info@54
    35
    this.init('tooltip', element, options)
info@54
    36
  }
info@54
    37
info@54
    38
  Tooltip.DEFAULTS = {
info@54
    39
    animation: true
info@54
    40
  , placement: 'top'
info@54
    41
  , selector: false
info@54
    42
  , template: '<div class="tooltip"><div class="tooltip-arrow"></div><div class="tooltip-inner"></div></div>'
info@54
    43
  , trigger: 'hover focus'
info@54
    44
  , title: ''
info@54
    45
  , delay: 0
info@54
    46
  , html: false
info@54
    47
  , container: false
info@54
    48
  }
info@54
    49
info@54
    50
  Tooltip.prototype.init = function (type, element, options) {
info@54
    51
    this.enabled  = true
info@54
    52
    this.type     = type
info@54
    53
    this.$element = $(element)
info@54
    54
    this.options  = this.getOptions(options)
info@54
    55
info@54
    56
    var triggers = this.options.trigger.split(' ')
info@54
    57
info@54
    58
    for (var i = triggers.length; i--;) {
info@54
    59
      var trigger = triggers[i]
info@54
    60
info@54
    61
      if (trigger == 'click') {
info@54
    62
        this.$element.on('click.' + this.type, this.options.selector, $.proxy(this.toggle, this))
info@54
    63
      } else if (trigger != 'manual') {
info@54
    64
        var eventIn  = trigger == 'hover' ? 'mouseenter' : 'focus'
info@54
    65
        var eventOut = trigger == 'hover' ? 'mouseleave' : 'blur'
info@54
    66
info@54
    67
        this.$element.on(eventIn  + '.' + this.type, this.options.selector, $.proxy(this.enter, this))
info@54
    68
        this.$element.on(eventOut + '.' + this.type, this.options.selector, $.proxy(this.leave, this))
info@54
    69
      }
info@54
    70
    }
info@54
    71
info@54
    72
    this.options.selector ?
info@54
    73
      (this._options = $.extend({}, this.options, { trigger: 'manual', selector: '' })) :
info@54
    74
      this.fixTitle()
info@54
    75
  }
info@54
    76
info@54
    77
  Tooltip.prototype.getDefaults = function () {
info@54
    78
    return Tooltip.DEFAULTS
info@54
    79
  }
info@54
    80
info@54
    81
  Tooltip.prototype.getOptions = function (options) {
info@54
    82
    options = $.extend({}, this.getDefaults(), this.$element.data(), options)
info@54
    83
info@54
    84
    if (options.delay && typeof options.delay == 'number') {
info@54
    85
      options.delay = {
info@54
    86
        show: options.delay
info@54
    87
      , hide: options.delay
info@54
    88
      }
info@54
    89
    }
info@54
    90
info@54
    91
    return options
info@54
    92
  }
info@54
    93
info@54
    94
  Tooltip.prototype.getDelegateOptions = function () {
info@54
    95
    var options  = {}
info@54
    96
    var defaults = this.getDefaults()
info@54
    97
info@54
    98
    this._options && $.each(this._options, function (key, value) {
info@54
    99
      if (defaults[key] != value) options[key] = value
info@54
   100
    })
info@54
   101
info@54
   102
    return options
info@54
   103
  }
info@54
   104
info@54
   105
  Tooltip.prototype.enter = function (obj) {
info@54
   106
    var self = obj instanceof this.constructor ?
info@54
   107
      obj : $(obj.currentTarget)[this.type](this.getDelegateOptions()).data('bs.' + this.type)
info@54
   108
info@54
   109
    clearTimeout(self.timeout)
info@54
   110
info@54
   111
    self.hoverState = 'in'
info@54
   112
info@54
   113
    if (!self.options.delay || !self.options.delay.show) return self.show()
info@54
   114
info@54
   115
    self.timeout = setTimeout(function () {
info@54
   116
      if (self.hoverState == 'in') self.show()
info@54
   117
    }, self.options.delay.show)
info@54
   118
  }
info@54
   119
info@54
   120
  Tooltip.prototype.leave = function (obj) {
info@54
   121
    var self = obj instanceof this.constructor ?
info@54
   122
      obj : $(obj.currentTarget)[this.type](this.getDelegateOptions()).data('bs.' + this.type)
info@54
   123
info@54
   124
    clearTimeout(self.timeout)
info@54
   125
info@54
   126
    self.hoverState = 'out'
info@54
   127
info@54
   128
    if (!self.options.delay || !self.options.delay.hide) return self.hide()
info@54
   129
info@54
   130
    self.timeout = setTimeout(function () {
info@54
   131
      if (self.hoverState == 'out') self.hide()
info@54
   132
    }, self.options.delay.hide)
info@54
   133
  }
info@54
   134
info@54
   135
  Tooltip.prototype.show = function () {
info@54
   136
    var e = $.Event('show.bs.'+ this.type)
info@54
   137
info@54
   138
    if (this.hasContent() && this.enabled) {
info@54
   139
      this.$element.trigger(e)
info@54
   140
info@54
   141
      if (e.isDefaultPrevented()) return
info@54
   142
info@54
   143
      var $tip = this.tip()
info@54
   144
info@54
   145
      this.setContent()
info@54
   146
info@54
   147
      if (this.options.animation) $tip.addClass('fade')
info@54
   148
info@54
   149
      var placement = typeof this.options.placement == 'function' ?
info@54
   150
        this.options.placement.call(this, $tip[0], this.$element[0]) :
info@54
   151
        this.options.placement
info@54
   152
info@54
   153
      var autoToken = /\s?auto?\s?/i
info@54
   154
      var autoPlace = autoToken.test(placement)
info@54
   155
      if (autoPlace) placement = placement.replace(autoToken, '') || 'top'
info@54
   156
info@54
   157
      $tip
info@54
   158
        .detach()
info@54
   159
        .css({ top: 0, left: 0, display: 'block' })
info@54
   160
        .addClass(placement)
info@54
   161
info@54
   162
      this.options.container ? $tip.appendTo(this.options.container) : $tip.insertAfter(this.$element)
info@54
   163
info@54
   164
      var pos          = this.getPosition()
info@54
   165
      var actualWidth  = $tip[0].offsetWidth
info@54
   166
      var actualHeight = $tip[0].offsetHeight
info@54
   167
info@54
   168
      if (autoPlace) {
info@54
   169
        var $parent = this.$element.parent()
info@54
   170
info@54
   171
        var orgPlacement = placement
info@54
   172
        var docScroll    = document.documentElement.scrollTop || document.body.scrollTop
info@54
   173
        var parentWidth  = this.options.container == 'body' ? window.innerWidth  : $parent.outerWidth()
info@54
   174
        var parentHeight = this.options.container == 'body' ? window.innerHeight : $parent.outerHeight()
info@54
   175
        var parentLeft   = this.options.container == 'body' ? 0 : $parent.offset().left
info@54
   176
info@54
   177
        placement = placement == 'bottom' && pos.top   + pos.height  + actualHeight - docScroll > parentHeight  ? 'top'    :
info@54
   178
                    placement == 'top'    && pos.top   - docScroll   - actualHeight < 0                         ? 'bottom' :
info@54
   179
                    placement == 'right'  && pos.right + actualWidth > parentWidth                              ? 'left'   :
info@54
   180
                    placement == 'left'   && pos.left  - actualWidth < parentLeft                               ? 'right'  :
info@54
   181
                    placement
info@54
   182
info@54
   183
        $tip
info@54
   184
          .removeClass(orgPlacement)
info@54
   185
          .addClass(placement)
info@54
   186
      }
info@54
   187
info@54
   188
      var calculatedOffset = this.getCalculatedOffset(placement, pos, actualWidth, actualHeight)
info@54
   189
info@54
   190
      this.applyPlacement(calculatedOffset, placement)
info@54
   191
      this.$element.trigger('shown.bs.' + this.type)
info@54
   192
    }
info@54
   193
  }
info@54
   194
info@54
   195
  Tooltip.prototype.applyPlacement = function(offset, placement) {
info@54
   196
    var replace
info@54
   197
    var $tip   = this.tip()
info@54
   198
    var width  = $tip[0].offsetWidth
info@54
   199
    var height = $tip[0].offsetHeight
info@54
   200
info@54
   201
    // manually read margins because getBoundingClientRect includes difference
info@54
   202
    var marginTop = parseInt($tip.css('margin-top'), 10)
info@54
   203
    var marginLeft = parseInt($tip.css('margin-left'), 10)
info@54
   204
info@54
   205
    // we must check for NaN for ie 8/9
info@54
   206
    if (isNaN(marginTop))  marginTop  = 0
info@54
   207
    if (isNaN(marginLeft)) marginLeft = 0
info@54
   208
info@54
   209
    offset.top  = offset.top  + marginTop
info@54
   210
    offset.left = offset.left + marginLeft
info@54
   211
info@54
   212
    $tip
info@54
   213
      .offset(offset)
info@54
   214
      .addClass('in')
info@54
   215
info@54
   216
    // check to see if placing tip in new offset caused the tip to resize itself
info@54
   217
    var actualWidth  = $tip[0].offsetWidth
info@54
   218
    var actualHeight = $tip[0].offsetHeight
info@54
   219
info@54
   220
    if (placement == 'top' && actualHeight != height) {
info@54
   221
      replace = true
info@54
   222
      offset.top = offset.top + height - actualHeight
info@54
   223
    }
info@54
   224
info@54
   225
    if (/bottom|top/.test(placement)) {
info@54
   226
      var delta = 0
info@54
   227
info@54
   228
      if (offset.left < 0) {
info@54
   229
        delta       = offset.left * -2
info@54
   230
        offset.left = 0
info@54
   231
info@54
   232
        $tip.offset(offset)
info@54
   233
info@54
   234
        actualWidth  = $tip[0].offsetWidth
info@54
   235
        actualHeight = $tip[0].offsetHeight
info@54
   236
      }
info@54
   237
info@54
   238
      this.replaceArrow(delta - width + actualWidth, actualWidth, 'left')
info@54
   239
    } else {
info@54
   240
      this.replaceArrow(actualHeight - height, actualHeight, 'top')
info@54
   241
    }
info@54
   242
info@54
   243
    if (replace) $tip.offset(offset)
info@54
   244
  }
info@54
   245
info@54
   246
  Tooltip.prototype.replaceArrow = function(delta, dimension, position) {
info@54
   247
    this.arrow().css(position, delta ? (50 * (1 - delta / dimension) + "%") : '')
info@54
   248
  }
info@54
   249
info@54
   250
  Tooltip.prototype.setContent = function () {
info@54
   251
    var $tip  = this.tip()
info@54
   252
    var title = this.getTitle()
info@54
   253
info@54
   254
    $tip.find('.tooltip-inner')[this.options.html ? 'html' : 'text'](title)
info@54
   255
    $tip.removeClass('fade in top bottom left right')
info@54
   256
  }
info@54
   257
info@54
   258
  Tooltip.prototype.hide = function () {
info@54
   259
    var that = this
info@54
   260
    var $tip = this.tip()
info@54
   261
    var e    = $.Event('hide.bs.' + this.type)
info@54
   262
info@54
   263
    function complete() {
info@54
   264
      if (that.hoverState != 'in') $tip.detach()
info@54
   265
    }
info@54
   266
info@54
   267
    this.$element.trigger(e)
info@54
   268
info@54
   269
    if (e.isDefaultPrevented()) return
info@54
   270
info@54
   271
    $tip.removeClass('in')
info@54
   272
info@54
   273
    $.support.transition && this.$tip.hasClass('fade') ?
info@54
   274
      $tip
info@54
   275
        .one($.support.transition.end, complete)
info@54
   276
        .emulateTransitionEnd(150) :
info@54
   277
      complete()
info@54
   278
info@54
   279
    this.$element.trigger('hidden.bs.' + this.type)
info@54
   280
info@54
   281
    return this
info@54
   282
  }
info@54
   283
info@54
   284
  Tooltip.prototype.fixTitle = function () {
info@54
   285
    var $e = this.$element
info@54
   286
    if ($e.attr('title') || typeof($e.attr('data-original-title')) != 'string') {
info@54
   287
      $e.attr('data-original-title', $e.attr('title') || '').attr('title', '')
info@54
   288
    }
info@54
   289
  }
info@54
   290
info@54
   291
  Tooltip.prototype.hasContent = function () {
info@54
   292
    return this.getTitle()
info@54
   293
  }
info@54
   294
info@54
   295
  Tooltip.prototype.getPosition = function () {
info@54
   296
    var el = this.$element[0]
info@54
   297
    return $.extend({}, (typeof el.getBoundingClientRect == 'function') ? el.getBoundingClientRect() : {
info@54
   298
      width: el.offsetWidth
info@54
   299
    , height: el.offsetHeight
info@54
   300
    }, this.$element.offset())
info@54
   301
  }
info@54
   302
info@54
   303
  Tooltip.prototype.getCalculatedOffset = function (placement, pos, actualWidth, actualHeight) {
info@54
   304
    return placement == 'bottom' ? { top: pos.top + pos.height,   left: pos.left + pos.width / 2 - actualWidth / 2  } :
info@54
   305
           placement == 'top'    ? { top: pos.top - actualHeight, left: pos.left + pos.width / 2 - actualWidth / 2  } :
info@54
   306
           placement == 'left'   ? { top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left - actualWidth } :
info@54
   307
        /* placement == 'right' */ { top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left + pos.width   }
info@54
   308
  }
info@54
   309
info@54
   310
  Tooltip.prototype.getTitle = function () {
info@54
   311
    var title
info@54
   312
    var $e = this.$element
info@54
   313
    var o  = this.options
info@54
   314
info@54
   315
    title = $e.attr('data-original-title')
info@54
   316
      || (typeof o.title == 'function' ? o.title.call($e[0]) :  o.title)
info@54
   317
info@54
   318
    return title
info@54
   319
  }
info@54
   320
info@54
   321
  Tooltip.prototype.tip = function () {
info@54
   322
    return this.$tip = this.$tip || $(this.options.template)
info@54
   323
  }
info@54
   324
info@54
   325
  Tooltip.prototype.arrow = function () {
info@54
   326
    return this.$arrow = this.$arrow || this.tip().find('.tooltip-arrow')
info@54
   327
  }
info@54
   328
info@54
   329
  Tooltip.prototype.validate = function () {
info@54
   330
    if (!this.$element[0].parentNode) {
info@54
   331
      this.hide()
info@54
   332
      this.$element = null
info@54
   333
      this.options  = null
info@54
   334
    }
info@54
   335
  }
info@54
   336
info@54
   337
  Tooltip.prototype.enable = function () {
info@54
   338
    this.enabled = true
info@54
   339
  }
info@54
   340
info@54
   341
  Tooltip.prototype.disable = function () {
info@54
   342
    this.enabled = false
info@54
   343
  }
info@54
   344
info@54
   345
  Tooltip.prototype.toggleEnabled = function () {
info@54
   346
    this.enabled = !this.enabled
info@54
   347
  }
info@54
   348
info@54
   349
  Tooltip.prototype.toggle = function (e) {
info@54
   350
    var self = e ? $(e.currentTarget)[this.type](this.getDelegateOptions()).data('bs.' + this.type) : this
info@54
   351
    self.tip().hasClass('in') ? self.leave(self) : self.enter(self)
info@54
   352
  }
info@54
   353
info@54
   354
  Tooltip.prototype.destroy = function () {
info@54
   355
    this.hide().$element.off('.' + this.type).removeData('bs.' + this.type)
info@54
   356
  }
info@54
   357
info@54
   358
info@54
   359
  // TOOLTIP PLUGIN DEFINITION
info@54
   360
  // =========================
info@54
   361
info@54
   362
  var old = $.fn.tooltip
info@54
   363
info@54
   364
  $.fn.tooltip = function (option) {
info@54
   365
    return this.each(function () {
info@54
   366
      var $this   = $(this)
info@54
   367
      var data    = $this.data('bs.tooltip')
info@54
   368
      var options = typeof option == 'object' && option
info@54
   369
info@54
   370
      if (!data) $this.data('bs.tooltip', (data = new Tooltip(this, options)))
info@54
   371
      if (typeof option == 'string') data[option]()
info@54
   372
    })
info@54
   373
  }
info@54
   374
info@54
   375
  $.fn.tooltip.Constructor = Tooltip
info@54
   376
info@54
   377
info@54
   378
  // TOOLTIP NO CONFLICT
info@54
   379
  // ===================
info@54
   380
info@54
   381
  $.fn.tooltip.noConflict = function () {
info@54
   382
    $.fn.tooltip = old
info@54
   383
    return this
info@54
   384
  }
info@54
   385
info@54
   386
}(jQuery);
Impressum Datenschutzerklärung