beautytips-8.x-1.x-dev/js/jquery.bt.js
js/jquery.bt.js
/* * @name BeautyTips * @desc a tooltips/baloon-help plugin for jQuery * * @author Jeff Robbins - Lullabot - http://www.lullabot.com * @version 0.9.5 release candidate 1 (5/20/2009) */ jQuery.bt = {version: '0.9.5-rc1'}; /* * @type jQuery * @cat Plugins/bt * @requires jQuery v1.2+ (not tested on versions prior to 1.2.6) * * Dual licensed under the MIT and GPL licenses: * http://www.opensource.org/licenses/mit-license.php * http://www.gnu.org/licenses/gpl.html * * Encourage development. If you use BeautyTips for anything cool * or on a site that people have heard of, please drop me a note. * - jeff ^at lullabot > com * * No guarantees, warranties, or promises of any kind * */ ;(function ($) { /** * @credit Inspired by Karl Swedberg's ClueTip * (http://plugins.learningjquery.com/cluetip/), which in turn was * inspired * by Cody Lindley's jTip (http://www.codylindley.com) * * @fileoverview * Beauty Tips is a jQuery tooltips plugin which uses the canvas drawing * element in the HTML5 spec in order to dynamically draw tooltip "talk * bubbles" around the descriptive help text associated with an item. * This is in many ways similar to Google Maps which both provides * similar talk-bubbles and uses the canvas element to draw them. * * The canvas element is supported in modern versions of FireFox, Safari, and * Opera. However, Internet Explorer needs a separate library called * ExplorerCanvas included on the page in order to support canvas drawing * functions. ExplorerCanvas was created by Google for use with their web * apps and you can find it here: http://excanvas.sourceforge.net/ * * Beauty Tips was written to be simple to use and pretty. All of its options * are documented at the bottom of this file and defaults can be overwritten * globally for the entire page, or individually on each call. * * By default each tooltip will be positioned on the side of the target * element which has the most free space. This is affected by the scroll * position and size of the current window, so each Beauty Tip is redrawn * each time it is displayed. It may appear above an element at the * bottom of the page, but when the page is scrolled down (and the * element is at the top of the page) it will then appear below it. * Additionally, positions can be forced or a preferred order can be * defined. See examples below. * * To fix z-index problems in IE6, include the bgiframe plugin on your page * http://plugins.jquery.com/project/bgiframe - BeautyTips will automatically * recognize it and use it. * * BeautyTips also works with the hoverIntent plugin * http://cherne.net/brian/resources/jquery.hoverIntent.html * see hoverIntent example below for usage * * Usage * The function can be called in a number of ways. * $(selector).bt(); * $(selector).bt('Content text'); * $(selector).bt('Content text', {option1: value, option2: value}); * $(selector).bt({option1: value, option2: value}); * * For more/better documentation and lots of examples, visit the demo page * included with the distribution * */ jQuery.fn.bt = function (content, options) { if (typeof content != 'string') { var contentSelect = true; options = content; content = false; } else { var contentSelect = false; } // if hoverIntent is installed, use that as default instead of hover if (jQuery.fn.hoverIntent && jQuery.bt.defaults.trigger == 'hover') { jQuery.bt.defaults.trigger = 'hoverIntent'; } return this.each(function (index) { var opts = jQuery.extend(false, jQuery.bt.defaults, jQuery.bt.options, options); // clean up the options opts.spikeLength = numb(opts.spikeLength); opts.spikeGirth = numb(opts.spikeGirth); opts.overlap = numb(opts.overlap); var ajaxTimeout = false; /** * This is sort of the "starting spot" for the this.each() * These are the init functions to handle the .bt() call */ if (opts.killTitle) { $(this).find('[title]').addBack().each(function () { if (!$(this).attr('bt-xTitle')) { $(this).attr('bt-xTitle', $(this).attr('title')).attr('title', ''); } }); } if (typeof opts.trigger == 'string') { opts.trigger = [opts.trigger]; } if (opts.trigger[0] == 'hoverIntent') { var hoverOpts = jQuery.extend(opts.hoverIntentOpts, { over: function () { this.btOn(); }, out: function () { this.btOff(); } }); $(this).hoverIntent(hoverOpts); } else if (opts.trigger[0] == 'hover') { $(this).hover( function () { this.btOn(); }, function () { this.btOff(); } ); } else if (opts.trigger[0] == 'now') { // toggle the on/off right now // note that 'none' gives more control (see below) if ($(this).hasClass('bt-active')) { this.btOff(); } else { this.btOn(); } } else if (opts.trigger[0] == 'none') { // initialize the tip with no event trigger // use javascript to turn on/off tip as follows: // $('#selector').btOn(); // $('#selector').btOff(); } else if (opts.trigger.length > 1 && opts.trigger[0] != opts.trigger[1]) { $(this) .bind(opts.trigger[0], function () { this.btOn(); }) .bind(opts.trigger[1], function () { this.btOff(); }); } else { // toggle using the same event $(this).bind(opts.trigger[0], function () { if ($(this).hasClass('bt-active')) { this.btOff(); } else { this.btOn(); } }); } /** * The BIG TURN ON * Any element that has been initiated */ this.btOn = function () { if (typeof $(this).data('bt-box') == 'object') { // if there's already a popup, remove it before creating a new one. this.btOff(); } // trigger preBuild function // preBuild has no argument since the box hasn't been built yet opts.preBuild.apply(this); // turn off other tips $(jQuery.bt.vars.closeWhenOpenStack).btOff(); // add the class to the target element (for hilighting, for example) // bt-active is always applied to all, but activeClass can apply another $(this).addClass('bt-active ' + opts.activeClass); if (contentSelect && opts.ajaxPath == null) { // bizarre, I know if (opts.killTitle) { // if we've killed the title attribute, it's been stored in // 'bt-xTitle' so get it.. $(this).attr('title', $(this).attr('bt-xTitle')); } // then evaluate the selector... title is now in place content = $.isFunction(opts.contentSelector) ? opts.contentSelector.apply(this) : eval(opts.contentSelector); if (opts.killTitle) { // now remove the title again, so we don't get double tips $(this).attr('title', ''); } } // ---------------------------------------------- // All the Ajax(ish) stuff is in this next bit... // ---------------------------------------------- if (opts.ajaxPath != null && content == false) { if (typeof opts.ajaxPath == 'object') { var url = eval(opts.ajaxPath[0]); url += opts.ajaxPath[1] ? ' ' + opts.ajaxPath[1] : ''; } else { var url = opts.ajaxPath; } var off = url.indexOf(" "); if (off >= 0) { var selector = url.slice(off, url.length); url = url.slice(0, off); } // load any data cached for the given ajax path var cacheData = opts.ajaxCache ? $(document.body).data('btCache-' + url.replace(/\./g, '')) : null; if (typeof cacheData == 'string') { content = selector ? $("<div/>").append(cacheData.replace(/<script(.|\s)*?\/script>/g, "")).find(selector) : cacheData; } else { var target = this; // set up the options var ajaxOpts = jQuery.extend(false, { type: opts.ajaxType, data: opts.ajaxData, cache: opts.ajaxCache, url: url, complete: function (XMLHttpRequest, textStatus) { if (textStatus == 'success' || textStatus == 'notmodified') { if (opts.ajaxCache) { $(document.body).data('btCache-' + url.replace(/\./g, ''), XMLHttpRequest.responseText); } ajaxTimeout = false; content = selector ? // Create a dummy div to hold the results $("<div/>") // inject the contents of the document in, removing // the scripts to avoid any 'Permission Denied' // errors in IE .append(XMLHttpRequest.responseText.replace(/<script(.|\s)*?\/script>/g, "")) // Locate the specified elements .find(selector) : // If not, just inject the full result XMLHttpRequest.responseText; } else { if (textStatus == 'timeout') { // if there was a timeout, we don't cache the result ajaxTimeout = true; } content = opts.ajaxError.replace(/%error/g, XMLHttpRequest.statusText); } // if the user rolls out of the target element before the // ajax request comes back, don't show it if ($(target).hasClass('bt-active')) { target.btOn(); } } }, opts.ajaxOpts); // do the ajax request jQuery.ajax(ajaxOpts); // load the throbber while the magic happens content = opts.ajaxLoading; } } // </ ajax stuff > // now we start actually figuring out where to place the tip // figure out how to compensate for the shadow, if present var shadowMarginX = 0; // extra added to width to compensate for shadow var shadowMarginY = 0; // extra added to height var shadowShiftX = 0; // amount to shift the tip horizontally to allow // for shadow var shadowShiftY = 0; // amount to shift vertical if (opts.shadow && !shadowSupport()) { // if browser doesn't support drop shadows, turn them off opts.shadow = false; // and bring in the noShadows options jQuery.extend(opts, opts.noShadowOpts); } if (opts.shadow) { // figure out horizontal placement if (opts.shadowBlur > Math.abs(opts.shadowOffsetX)) { shadowMarginX = opts.shadowBlur * 2; } else { shadowMarginX = opts.shadowBlur + Math.abs(opts.shadowOffsetX); } shadowShiftX = (opts.shadowBlur - opts.shadowOffsetX) > 0 ? opts.shadowBlur - opts.shadowOffsetX : 0; // now vertical if (opts.shadowBlur > Math.abs(opts.shadowOffsetY)) { shadowMarginY = opts.shadowBlur * 2; } else { shadowMarginY = opts.shadowBlur + Math.abs(opts.shadowOffsetY); } shadowShiftY = (opts.shadowBlur - opts.shadowOffsetY) > 0 ? opts.shadowBlur - opts.shadowOffsetY : 0; } if (opts.offsetParent) { // if offsetParent is defined by user var offsetParent = $(opts.offsetParent); var offsetParentPos = offsetParent.offset(); var pos = $(this).offset(); var top = numb(pos.top) - numb(offsetParentPos.top) + numb($(this).css('margin-top')) - shadowShiftY; // IE can return 'auto' for margins var left = numb(pos.left) - numb(offsetParentPos.left) + numb($(this).css('margin-left')) - shadowShiftX; } else { // if the target element is absolutely positioned, use its parent's // offsetParent instead of its own var offsetParent = ($(this).css('position') == 'absolute') ? $(this).parents().eq(0).offsetParent() : $(this).offsetParent(); var pos = $(this).btPosition(); var top = numb(pos.top) + numb($(this).css('margin-top')) - shadowShiftY; // IE can return 'auto' for margins var left = numb(pos.left) + numb($(this).css('margin-left')) - shadowShiftX; } var width = $(this).btOuterWidth(); var height = $(this).outerHeight(); if (typeof content == 'object') { // if content is a DOM object (as opposed to text) // use a clone, rather than removing the original element // and ensure that it's visible if (content == null) { return; } var original = content; var clone = $(original).clone(true).show(); // also store a reference to the original object in the clone data // and a reference to the clone in the original var origClones = $(original).data('bt-clones') || []; origClones.push(clone); $(original).data('bt-clones', origClones); $(clone).data('bt-orig', original); $(this).data('bt-content-orig', {original: original, clone: clone}); content = clone; } if (typeof content == 'null' || content == '') { // if content is empty, bail out... return; } // create the tip content div, populate it, and style it var $text = $('<div class="bt-content"></div>').append(content); $text.css({ padding: opts.padding, position: 'absolute', width: (opts.shrinkToFit ? 'auto' : opts.width), zIndex: opts.textzIndex, left: shadowShiftX, top: shadowShiftY }); $text.css(opts.cssStyles); // create the wrapping box which contains text and canvas // put the content in it, style it, and append it to the same offset // parent as the target var $box = $('<div class="bt-wrapper"></div>').append($text).addClass(opts.cssClass).css({ position: 'absolute', width: opts.width, zIndex: opts.wrapperzIndex, visibility: 'hidden' }).appendTo(offsetParent); // use bgiframe to get around z-index problems in IE6 // http://plugins.jquery.com/project/bgiframe if (jQuery.fn.bgiframe) { $text.bgiframe(); $box.bgiframe(); } $(this).data('bt-box', $box); // see if the text box will fit in the various positions var scrollTop = numb($(document).scrollTop()); var scrollLeft = numb($(document).scrollLeft()); var docWidth = numb($(window).width()); var docHeight = numb($(window).height()); var winRight = scrollLeft + docWidth; var winBottom = scrollTop + docHeight; var space = new Object(); var thisOffset = $(this).offset(); space.top = thisOffset.top - scrollTop; space.bottom = docHeight - ((thisOffset + height) - scrollTop); space.left = thisOffset.left - scrollLeft; space.right = docWidth - ((thisOffset.left + width) - scrollLeft); var textOutHeight = numb($text.outerHeight()); var textOutWidth = numb($text.btOuterWidth()); if (opts.positions.constructor == String) { opts.positions = opts.positions.replace(/ /, '').split(','); } if (opts.positions[0] == 'most') { // figure out which is the largest var position = 'top'; // prime the pump for (var pig in space) { // <------- pigs in space! position = space[pig] > space[position] ? pig : position; } } else { for (var x in opts.positions) { var position = opts.positions[x]; // @todo: acommodate shadow space in the following lines... if ((position == 'left' || position == 'right') && space[position] > textOutWidth + opts.spikeLength) { break; } else if ((position == 'top' || position == 'bottom') && space[position] > textOutHeight + opts.spikeLength) { break; } } } // horizontal (left) offset for the box var horiz = left + ((width - textOutWidth) * .5); // vertical (top) offset for the box var vert = top + ((height - textOutHeight) * .5); var points = new Array(); var textTop, textLeft, textRight, textBottom, textTopSpace, textBottomSpace, textLeftSpace, textRightSpace, crossPoint, textCenter, spikePoint; // Yes, yes, this next bit really could use to be condensed // each switch case is basically doing the same thing in slightly // different ways switch (position) { // =================== TOP ======================= case 'top': // spike on bottom $text.css('margin-bottom', opts.spikeLength + 'px'); $box.css({ top: (top - $text.outerHeight(true)) + opts.overlap, left: horiz }); // move text left/right if extends out of window textRightSpace = (winRight - opts.windowMargin) - ($text.offset().left + $text.btOuterWidth(true)); var xShift = shadowShiftX; if (textRightSpace < 0) { // shift it left $box.css('left', (numb($box.css('left')) + textRightSpace) + 'px'); xShift -= textRightSpace; } // we test left space second to ensure that left of box is visible textLeftSpace = ($text.offset().left + numb($text.css('margin-left'))) - (scrollLeft + opts.windowMargin); if (textLeftSpace < 0) { // shift it right $box.css('left', (numb($box.css('left')) - textLeftSpace) + 'px'); xShift += textLeftSpace; } textTop = $text.btPosition().top + numb($text.css('margin-top')); textLeft = $text.btPosition().left + numb($text.css('margin-left')); textRight = textLeft + $text.btOuterWidth(); textBottom = textTop + $text.outerHeight(); textCenter = { x: textLeft + ($text.btOuterWidth() * opts.centerPointX), y: textTop + ($text.outerHeight() * opts.centerPointY) }; // points[points.length] = {x: x, y: y}; points[points.length] = spikePoint = { y: textBottom + opts.spikeLength, x: ((textRight - textLeft) * .5) + xShift, type: 'spike' }; crossPoint = findIntersectX(spikePoint.x, spikePoint.y, textCenter.x, textCenter.y, textBottom); // make sure that the crossPoint is not outside of text box // boundaries crossPoint.x = crossPoint.x < textLeft + opts.spikeGirth / 2 + opts.cornerRadius ? textLeft + opts.spikeGirth / 2 + opts.cornerRadius : crossPoint.x; crossPoint.x = crossPoint.x > (textRight - opts.spikeGirth / 2) - opts.cornerRadius ? (textRight - opts.spikeGirth / 2) - opts.CornerRadius : crossPoint.x; points[points.length] = { x: crossPoint.x - (opts.spikeGirth / 2), y: textBottom, type: 'join' }; points[points.length] = { x: textLeft, y: textBottom, type: 'corner' }; // left bottom corner points[points.length] = {x: textLeft, y: textTop, type: 'corner'}; // left top corner points[points.length] = {x: textRight, y: textTop, type: 'corner'}; // right top corner points[points.length] = { x: textRight, y: textBottom, type: 'corner' }; // right bottom corner points[points.length] = { x: crossPoint.x + (opts.spikeGirth / 2), y: textBottom, type: 'join' }; points[points.length] = spikePoint; break; // =================== LEFT ======================= case 'left': // spike on right $text.css('margin-right', opts.spikeLength + 'px'); $box.css({ top: vert + 'px', left: ((left - $text.btOuterWidth(true)) + opts.overlap) + 'px' }); // move text up/down if extends out of window textBottomSpace = (winBottom - opts.windowMargin) - ($text.offset().top + $text.outerHeight(true)); var yShift = shadowShiftY; if (textBottomSpace < 0) { // shift it up $box.css('top', (numb($box.css('top')) + textBottomSpace) + 'px'); yShift -= textBottomSpace; } // we ensure top space second to ensure that top of box is visible textTopSpace = ($text.offset().top + numb($text.css('margin-top'))) - (scrollTop + opts.windowMargin); if (textTopSpace < 0) { // shift it down $box.css('top', (numb($box.css('top')) - textTopSpace) + 'px'); yShift += textTopSpace; } textTop = $text.btPosition().top + numb($text.css('margin-top')); textLeft = $text.btPosition().left + numb($text.css('margin-left')); textRight = textLeft + $text.btOuterWidth(); textBottom = textTop + $text.outerHeight(); textCenter = { x: textLeft + ($text.btOuterWidth() * opts.centerPointX), y: textTop + ($text.outerHeight() * opts.centerPointY) }; points[points.length] = spikePoint = { x: textRight + opts.spikeLength, y: ((textBottom - textTop) * .5) + yShift, type: 'spike' }; crossPoint = findIntersectY(spikePoint.x, spikePoint.y, textCenter.x, textCenter.y, textRight); // make sure that the crossPoint is not outside of text box // boundaries crossPoint.y = crossPoint.y < textTop + opts.spikeGirth / 2 + opts.cornerRadius ? textTop + opts.spikeGirth / 2 + opts.cornerRadius : crossPoint.y; crossPoint.y = crossPoint.y > (textBottom - opts.spikeGirth / 2) - opts.cornerRadius ? (textBottom - opts.spikeGirth / 2) - opts.cornerRadius : crossPoint.y; points[points.length] = { x: textRight, y: crossPoint.y + opts.spikeGirth / 2, type: 'join' }; points[points.length] = { x: textRight, y: textBottom, type: 'corner' }; // right bottom corner points[points.length] = { x: textLeft, y: textBottom, type: 'corner' }; // left bottom corner points[points.length] = {x: textLeft, y: textTop, type: 'corner'}; // left top corner points[points.length] = {x: textRight, y: textTop, type: 'corner'}; // right top corner points[points.length] = { x: textRight, y: crossPoint.y - opts.spikeGirth / 2, type: 'join' }; points[points.length] = spikePoint; break; // =================== BOTTOM ======================= case 'bottom': // spike on top $text.css('margin-top', opts.spikeLength + 'px'); $box.css({top: (top + height) - opts.overlap, left: horiz}); // move text up/down if extends out of window textRightSpace = (winRight - opts.windowMargin) - ($text.offset().left + $text.btOuterWidth(true)); var xShift = shadowShiftX; if (textRightSpace < 0) { // shift it left $box.css('left', (numb($box.css('left')) + textRightSpace) + 'px'); xShift -= textRightSpace; } // we ensure left space second to ensure that left of box is visible textLeftSpace = ($text.offset().left + numb($text.css('margin-left'))) - (scrollLeft + opts.windowMargin); if (textLeftSpace < 0) { // shift it right $box.css('left', (numb($box.css('left')) - textLeftSpace) + 'px'); xShift += textLeftSpace; } textTop = $text.btPosition().top + numb($text.css('margin-top')); textLeft = $text.btPosition().left + numb($text.css('margin-left')); textRight = textLeft + $text.btOuterWidth(); textBottom = textTop + $text.outerHeight(); textCenter = { x: textLeft + ($text.btOuterWidth() * opts.centerPointX), y: textTop + ($text.outerHeight() * opts.centerPointY) }; points[points.length] = spikePoint = { x: ((textRight - textLeft) * .5) + xShift, y: shadowShiftY, type: 'spike' }; crossPoint = findIntersectX(spikePoint.x, spikePoint.y, textCenter.x, textCenter.y, textTop); // make sure that the crossPoint is not outside of text box // boundaries crossPoint.x = crossPoint.x < textLeft + opts.spikeGirth / 2 + opts.cornerRadius ? textLeft + opts.spikeGirth / 2 + opts.cornerRadius : crossPoint.x; crossPoint.x = crossPoint.x > (textRight - opts.spikeGirth / 2) - opts.cornerRadius ? (textRight - opts.spikeGirth / 2) - opts.cornerRadius : crossPoint.x; points[points.length] = { x: crossPoint.x + opts.spikeGirth / 2, y: textTop, type: 'join' }; points[points.length] = {x: textRight, y: textTop, type: 'corner'}; // right top corner points[points.length] = { x: textRight, y: textBottom, type: 'corner' }; // right bottom corner points[points.length] = { x: textLeft, y: textBottom, type: 'corner' }; // left bottom corner points[points.length] = {x: textLeft, y: textTop, type: 'corner'}; // left top corner points[points.length] = { x: crossPoint.x - (opts.spikeGirth / 2), y: textTop, type: 'join' }; points[points.length] = spikePoint; break; // =================== RIGHT ======================= case 'right': // spike on left $text.css('margin-left', (opts.spikeLength + 'px')); $box.css({ top: vert + 'px', left: ((left + width) - opts.overlap) + 'px' }); // move text up/down if extends out of window textBottomSpace = (winBottom - opts.windowMargin) - ($text.offset().top + $text.outerHeight(true)); var yShift = shadowShiftY; if (textBottomSpace < 0) { // shift it up $box.css('top', (numb($box.css('top')) + textBottomSpace) + 'px'); yShift -= textBottomSpace; } // we ensure top space second to ensure that top of box is visible textTopSpace = ($text.offset().top + numb($text.css('margin-top'))) - (scrollTop + opts.windowMargin); if (textTopSpace < 0) { // shift it down $box.css('top', (numb($box.css('top')) - textTopSpace) + 'px'); yShift += textTopSpace; } textTop = $text.btPosition().top + numb($text.css('margin-top')); textLeft = $text.btPosition().left + numb($text.css('margin-left')); textRight = textLeft + $text.btOuterWidth(); textBottom = textTop + $text.outerHeight(); textCenter = { x: textLeft + ($text.btOuterWidth() * opts.centerPointX), y: textTop + ($text.outerHeight() * opts.centerPointY) }; points[points.length] = spikePoint = { x: shadowShiftX, y: ((textBottom - textTop) * .5) + yShift, type: 'spike' }; crossPoint = findIntersectY(spikePoint.x, spikePoint.y, textCenter.x, textCenter.y, textLeft); // make sure that the crossPoint is not outside of text box // boundaries crossPoint.y = crossPoint.y < textTop + opts.spikeGirth / 2 + opts.cornerRadius ? textTop + opts.spikeGirth / 2 + opts.cornerRadius : crossPoint.y; crossPoint.y = crossPoint.y > (textBottom - opts.spikeGirth / 2) - opts.cornerRadius ? (textBottom - opts.spikeGirth / 2) - opts.cornerRadius : crossPoint.y; points[points.length] = { x: textLeft, y: crossPoint.y - opts.spikeGirth / 2, type: 'join' }; points[points.length] = {x: textLeft, y: textTop, type: 'corner'}; // left top corner points[points.length] = {x: textRight, y: textTop, type: 'corner'}; // right top corner points[points.length] = { x: textRight, y: textBottom, type: 'corner' }; // right bottom corner points[points.length] = { x: textLeft, y: textBottom, type: 'corner' }; // left bottom corner points[points.length] = { x: textLeft, y: crossPoint.y + opts.spikeGirth / 2, type: 'join' }; points[points.length] = spikePoint; break; } // </ switch > var canvas = document.createElement('canvas'); $(canvas).attr('width', (numb($text.btOuterWidth(true)) + opts.strokeWidth * 2 + shadowMarginX)).attr('height', (numb($text.outerHeight(true)) + opts.strokeWidth * 2 + shadowMarginY)).appendTo($box).css({ position: 'absolute', zIndex: opts.boxzIndex }); // if excanvas is set up, we need to initialize the new canvas element if (typeof G_vmlCanvasManager != 'undefined') { canvas = G_vmlCanvasManager.initElement(canvas); } if (opts.cornerRadius > 0) { // round the corners! var newPoints = new Array(); var newPoint; for (var i = 0; i < points.length; i++) { if (points[i].type == 'corner') { // create two new arc points // find point between this and previous (using modulo in case of // ending) newPoint = betweenPoint(points[i], points[(i - 1) % points.length], opts.cornerRadius); newPoint.type = 'arcStart'; newPoints[newPoints.length] = newPoint; // the original corner point newPoints[newPoints.length] = points[i]; // find point between this and next newPoint = betweenPoint(points[i], points[(i + 1) % points.length], opts.cornerRadius); newPoint.type = 'arcEnd'; newPoints[newPoints.length] = newPoint; } else { newPoints[newPoints.length] = points[i]; } } // overwrite points with new version points = newPoints; } var ctx = canvas.getContext("2d"); if (opts.shadow && opts.shadowOverlap !== true) { var shadowOverlap = numb(opts.shadowOverlap); // keep the shadow (and canvas) from overlapping the target element switch (position) { case 'top': if (opts.shadowOffsetX + opts.shadowBlur - shadowOverlap > 0) { $box.css('top', (numb($box.css('top')) - (opts.shadowOffsetX + opts.shadowBlur - shadowOverlap))); } break; case 'right': if (shadowShiftX - shadowOverlap > 0) { $box.css('left', (numb($box.css('left')) + shadowShiftX - shadowOverlap)); } break; case 'bottom': if (shadowShiftY - shadowOverlap > 0) { $box.css('top', (numb($box.css('top')) + shadowShiftY - shadowOverlap)); } break; case 'left': if (opts.shadowOffsetY + opts.shadowBlur - shadowOverlap > 0) { $box.css('left', (numb($box.css('left')) - (opts.shadowOffsetY + opts.shadowBlur - shadowOverlap))); } break; } } drawIt.apply(ctx, [points], opts.strokeWidth); ctx.fillStyle = opts.fill; if (opts.shadow) { ctx.shadowOffsetX = opts.shadowOffsetX; ctx.shadowOffsetY = opts.shadowOffsetY; ctx.shadowBlur = opts.shadowBlur; ctx.shadowColor = opts.shadowColor; } ctx.closePath(); ctx.fill(); if (opts.strokeWidth > 0) { ctx.shadowColor = 'rgba(0, 0, 0, 0)'; //remove shadow from stroke ctx.lineWidth = opts.strokeWidth; ctx.strokeStyle = opts.strokeStyle; ctx.beginPath(); drawIt.apply(ctx, [points], opts.strokeWidth); ctx.closePath(); ctx.stroke(); } // trigger preShow function // function receives the box element (the balloon wrapper div) as an // argument opts.preShow.apply(this, [$box[0]]); // switch from visibility: hidden to display: none so we can run // animations $box.css({display: 'none', visibility: 'visible'}); // Here's where we show the tip opts.showTip.apply(this, [$box[0]]); if (opts.overlay) { // EXPERIMENTAL AND FOR TESTING ONLY!!!! var overlay = $('<div class="bt-overlay"></div>').css({ position: 'absolute', backgroundColor: 'blue', top: top, left: left, width: width, height: height, opacity: '.2' }).appendTo(offsetParent); $(this).data('overlay', overlay); } if ((opts.ajaxPath != null && opts.ajaxCache == false) || ajaxTimeout) { // if ajaxCache is not enabled or if there was a server timeout, // remove the content variable so it will be loaded again from server content = false; } // stick this element into the clickAnywhereToClose stack if (opts.clickAnywhereToClose) { jQuery.bt.vars.clickAnywhereStack.push(this); $(document).click(jQuery.bt.docClick); } // stick this element into the closeWhenOthersOpen stack if (opts.closeWhenOthersOpen) { jQuery.bt.vars.closeWhenOpenStack.push(this); } // trigger postShow function // function receives the box element (the balloon wrapper div) as an // argument opts.postShow.apply(this, [$box[0]]); // Allow trigger btContentHover to turn off on tip on moueout of actual // tip currentDiv = this; $(".bt-content").mouseout(function () { $(currentDiv).trigger('btContentHover'); }); }; // </ turnOn() > this.btOff = function () { var box = $(this).data('bt-box'); if (typeof box == 'undefined') { return; } // trigger preHide function // function receives the box element (the balloon wrapper div) as an // argument opts.preHide.apply(this, [box]); var i = this; // set up the stuff to happen AFTER the tip is hidden i.btCleanup = function () { var box = $(i).data('bt-box'); var contentOrig = $(i).data('bt-content-orig'); var overlay = $(i).data('bt-overlay'); if (typeof box == 'object') { $(box).remove(); $(i).removeData('bt-box'); } if (typeof contentOrig == 'object') { var clones = $(contentOrig.original).data('bt-clones'); $(contentOrig).data('bt-clones', arrayRemove(clones, contentOrig.clone)); } if (typeof overlay == 'object') { $(overlay).remove(); $(i).removeData('bt-overlay'); } // remove this from the stacks jQuery.bt.vars.clickAnywhereStack = arrayRemove(jQuery.bt.vars.clickAnywhereStack, i); jQuery.bt.vars.closeWhenOpenStack = arrayRemove(jQuery.bt.vars.closeWhenOpenStack, i); // remove the 'bt-active' and activeClass classes from target $(i).removeClass('bt-active ' + opts.activeClass); // trigger postHide function // no box argument since it has been removed from the DOM opts.postHide.apply(i); } opts.hideTip.apply(this, [box, i.btCleanup]); }; // </ turnOff() > var refresh = this.btRefresh = function () { this.btOff(); this.btOn(); }; }); // </ this.each() > function drawIt(points, strokeWidth) { this.moveTo(points[0].x, points[0].y); for (i = 1; i < points.length; i++) { if (points[i - 1].type == 'arcStart') { // if we're creating a rounded corner //ctx.arc(round5(points[i].x), round5(points[i].y), // points[i].startAngle, points[i].endAngle, opts.cornerRadius, // false); this.quadraticCurveTo(round5(points[i].x, strokeWidth), round5(points[i].y, strokeWidth), round5(points[(i + 1) % points.length].x, strokeWidth), round5(points[(i + 1) % points.length].y, strokeWidth)); i++; //ctx.moveTo(round5(points[i].x), round5(points[i].y)); } else { this.lineTo(round5(points[i].x, strokeWidth), round5(points[i].y, strokeWidth)); } } }; // </ drawIt() > /** * For odd stroke widths, round to the nearest .5 pixel to avoid * antialiasing * http://developer.mozilla.org/en/Canvas_tutorial/Applying_styles_and_colors */ function round5(num, strokeWidth) { var ret; strokeWidth = numb(strokeWidth); if (strokeWidth % 2) { ret = num; } else { ret = Math.round(num - .5) + .5; } return ret; }; // </ round5() > /** * Ensure that a number is a number... or zero */ function numb(num) { return parseInt(num) || 0; }; // </ numb() > /** * Remove an element from an array */ function arrayRemove(arr, elem) { var x, newArr = new Array(); for (x in arr) { if (arr[x] != elem) { newArr.push(arr[x]); } } return newArr; }; // </ arrayRemove() > /** * Does the current browser support canvas? * This is a variation of http://code.google.com/p/browser-canvas-support/ */ function canvasSupport() { var canvas_compatible = false; try { canvas_compatible = !!(document.createElement('canvas').getContext('2d')); // S60 } catch (e) { canvas_compatible = !!(document.createElement('canvas').getContext); // IE } return canvas_compatible; } /** * Does the current browser support canvas drop shadows? */ function shadowSupport() { // to test for drop shadow support in the current browser, uncomment the // next line return true; // until a good feature-detect is found, we have to look at user agents try { var userAgent = navigator.userAgent.toLowerCase(); if (/webkit/.test(userAgent)) { // WebKit.. let's go! return true; } else if (/gecko|mozilla/.test(userAgent) && parseFloat(userAgent.match(/firefox\/(\d+(?:\.\d+)+)/)[1]) >= 3.1) { // Mozilla 3.1 or higher return true; } } catch (err) { // if there's an error, just keep going, we'll assume that drop shadows // are not supported } return false; } // </ shadowSupport() > /** * Given two points, find a point which is dist pixels from point1 on a * line to point2 */ function betweenPoint(point1, point2, dist) { // figure out if we're horizontal or vertical var y, x; if (point1.x == point2.x) { // vertical y = point1.y < point2.y ? point1.y + dist : point1.y - dist; return {x: point1.x, y: y}; } else if (point1.y == point2.y) { // horizontal x = point1.x < point2.x ? point1.x + dist : point1.x - dist; return {x: x, y: point1.y}; } }; // </ betweenPoint() > function centerPoint(arcStart, corner, arcEnd) { var x = corner.x == arcStart.x ? arcEnd.x : arcStart.x; var y = corner.y == arcStart.y ? arcEnd.y : arcStart.y; var startAngle, endAngle; if (arcStart.x < arcEnd.x) { if (arcStart.y > arcEnd.y) { // arc is on upper left startAngle = (Math.PI / 180) * 180; endAngle = (Math.PI / 180) * 90; } else { // arc is on upper right startAngle = (Math.PI / 180) * 90; endAngle = 0; } } else { if (arcStart.y > arcEnd.y) { // arc is on lower left startAngle = (Math.PI / 180) * 270; endAngle = (Math.PI / 180) * 180; } else { // arc is on lower right startAngle = 0; endAngle = (Math.PI / 180) * 270; } } return { x: x, y: y, type: 'center', startAngle: startAngle, endAngle: endAngle }; }; // </ centerPoint() > /** * Find the intersection point of two lines, each defined by two points * arguments are x1, y1 and x2, y2 for r1 (line 1) and r2 (line 2) * It's like an algebra party!!! */ function findIntersect(r1x1, r1y1, r1x2, r1y2, r2x1, r2y1, r2x2, r2y2) { if (r2x1 == r2x2) { return findIntersectY(r1x1, r1y1, r1x2, r1y2, r2x1); } if (r2y1 == r2y2) { return findIntersectX(r1x1, r1y1, r1x2, r1y2, r2y1); } // m = (y1 - y2) / (x1 - x2) // <-- how to find the slope // y = mx + b // the 'classic' linear equation // b = y - mx // how to find b (the y-intersect) // x = (y - b)/m // how to find x var r1m = (r1y1 - r1y2) / (r1x1 - r1x2); var r1b = r1y1 - (r1m * r1x1); var r2m = (r2y1 - r2y2) / (r2x1 - r2x2); var r2b = r2y1 - (r2m * r2x1); var x = (r2b - r1b) / (r1m - r2m); var y = r1m * x + r1b; return {x: x, y: y}; }; // </ findIntersect() > /** * Find the y intersection point of a line and given x vertical */ function findIntersectY(r1x1, r1y1, r1x2, r1y2, x) { if (r1y1 == r1y2) { return {x: x, y: r1y1}; } var r1m = (r1y1 - r1y2) / (r1x1 - r1x2); var r1b = r1y1 - (r1m * r1x1); var y = r1m * x + r1b; return {x: x, y: y}; }; // </ findIntersectY() > /** * Find the x intersection point of a line and given y horizontal */ function findIntersectX(r1x1, r1y1, r1x2, r1y2, y) { if (r1x1 == r1x2) { return {x: r1x1, y: y}; } var r1m = (r1y1 - r1y2) / (r1x1 - r1x2); var r1b = r1y1 - (r1m * r1x1); // y = mx + b // your old friend, linear equation // x = (y - b)/m // linear equation solved for x var x = (y - r1b) / r1m; return {x: x, y: y}; }; // </ findIntersectX() > }; // </ jQuery.fn.bt() > /** * jQuery's compat.js (used in Drupal's jQuery upgrade module, overrides the * $().position() function this is a copy of that function to allow the * plugin to work when compat.js is present once compat.js is fixed to not * override existing functions, this function can be removed and .btPosion() * can be replaced with .position() above... */ jQuery.fn.btPosition = function () { function num(elem, prop) { return elem[0] && parseInt(jQuery(elem[0]).css(prop), 10) || 0; }; var left = 0, top = 0, results; if (this[0]) { // Get *real* offsetParent var offsetParent = this.offsetParent(), // Get correct offsets offset = this.offset(), parentOffset = /^body|html$/i.test(offsetParent[0].tagName) ? { top: 0, left: 0 } : offsetParent.offset(); // Subtract element margins // note: when an element has margin: auto the offsetLeft and marginLeft // are the same in Safari causing offset.left to incorrectly be 0 offset.top -= num(this, 'marginTop'); offset.left -= num(this, 'marginLeft'); // Add offsetParent borders parentOffset.top += num(offsetParent, 'borderTopWidth'); parentOffset.left += num(offsetParent, 'borderLeftWidth'); // Subtract the two offsets results = { top: offset.top - parentOffset.top, left: offset.left - parentOffset.left }; } return results; }; // </ jQuery.fn.btPosition() > /** * jQuery's dimensions.js overrides the $().btOuterWidth() function * this is a copy of original jQuery's outerWidth() function to * allow the plugin to work when dimensions.js is present */ jQuery.fn.btOuterWidth = function (margin) { function num(elem, prop) { return elem[0] && parseInt(jQuery(elem[0]).css(prop), 10) || 0; }; return this["innerWidth"]() + num(this, "borderLeftWidth") + num(this, "borderRightWidth") + (margin ? num(this, "marginLeft") + num(this, "marginRight") : 0); }; // </ jQuery.fn.btOuterWidth() > /** * A convenience function to run btOn() (if available) * for each selected item */ jQuery.fn.btOn = function () { return this.each(function (index) { if (jQuery.isFunction(this.btOn)) { this.btOn(); } }); }; // </ $().btOn() > /** * * A convenience function to run btOff() (if available) * for each selected item */ jQuery.fn.btOff = function () { return this.each(function (index) { if (jQuery.isFunction(this.btOff)) { this.btOff(); } }); }; // </ $().btOff() > jQuery.bt.vars = {clickAnywhereStack: [], closeWhenOpenStack: []}; /** * This function gets bound to the document's click event * It turns off all of the tips in the click-anywhere-to-close stack */ jQuery.bt.docClick = function (e) { if (!e) { var e = window.event; } ; // if clicked element is a child of neither a tip NOR a target // and there are tips in the stack if (!$(e.target).parents().addBack().filter('.bt-wrapper, .bt-active').length && jQuery.bt.vars.clickAnywhereStack.length) { // if clicked element isn't inside tip, close tips in stack $(jQuery.bt.vars.clickAnywhereStack).btOff(); $(document).unbind('click', jQuery.bt.docClick); } }; // </ docClick() > /** * Defaults for the beauty tips * * Note this is a variable definition and not a function. So defaults can be * written for an entire page by simply redefining attributes like so: * * jQuery.bt.options.width = 400; * * Be sure to use *jQuery.bt.options* and not jQuery.bt.defaults when * overriding * * This would make all Beauty Tips boxes 400px wide. * * Each of these options may also be overridden during * * Can be overriden globally or at time of call. * */ jQuery.bt.defaults = { trigger: 'hover', // trigger to show/hide tip // use [on, off] to define separate on/off // triggers also use space character to // allow multiple to trigger examples: // ['focus', 'blur'] // focus displays, // blur hides 'dblclick' // // dblclick toggles on/off ['focus // mouseover', 'blur mouseout'] // // multiple triggers 'now' // // shows/hides tip without event 'none' // // use $('#selector').btOn(); // and ...btOff(); 'hoverIntent' // hover using hoverIntent plugin (settings below) note: hoverIntent becomes default if available clickAnywhereToClose: true, // clicking anywhere outside of // the tip will close it closeWhenOthersOpen: false, // tip will be closed before // another opens - stop >= 2 tips // being on shrinkToFit: false, // should short single-line content get // a narrower balloon? width: '200px', // width of tooltip box padding: '10px', // padding for content (get more fine // grained with cssStyles) spikeGirth: 10, // width of spike spikeLength: 15, // length of spike overlap: 0, // spike overlap (px) onto target (can // cause problems with 'hover' trigger) overlay: false, // display overlay on target (use CSS to // style) -- BUGGY! killTitle: true, // kill title tags to avoid double // tooltips textzIndex: 9999, // z-index for the text boxzIndex: 9998, // z-index for the "talk" box (should // always be less than textzIndex) wrapperzIndex: 9997, offsetParent: null, // DOM node to append the tooltip // into. // Must be positioned relative or // absolute. Can be selector or object positions: ['most'], // preference of positions for tip (will // use first with available space) // possible values 'top', 'bottom', // 'left', 'right' as an array in order // of preference. Last value will be used // if others don't have enough space. or // use 'most' to use the area with the // most space fill: "rgb(255, 255, 102)", // fill color for the tooltip box, you can use // any CSS-style color definition method // http://www.w3.org/TR/css3-color/#numerical // - not all methods have been tested windowMargin: 10, // space (px) to leave between text // box and browser edge strokeWidth: 1, // width of stroke around box, **set to // 0 for no stroke** strokeStyle: "#000", // color/alpha of stroke cornerRadius: 5, // radius of corners (px), set to 0 // for square corners // following values are on a scale of 0 to 1 with .5 being centered centerPointX: .5, // the spike extends from center of // the target edge to this point centerPointY: .5, // defined by percentage horizontal // (x) and vertical (y) shadow: false, // use drop shadow? (only displays in Safari // and FF 3.1) - experimental shadowOffsetX: 2, // shadow offset x (px) shadowOffsetY: 2, // shadow offset y (px) shadowBlur: 3, // shadow blur (px) shadowColor: "#000", // shadow color/alpha shadowOverlap: false, // when shadows overlap the target // element it can cause problem with // hovering set this to true to // overlap or set to a numeric value // to define the amount of overlap noShadowOpts: {strokeStyle: '#999'}, // use this to define 'fall-back' // options for browsers which don't // support drop shadows cssClass: '', // CSS class to add to the box wrapper div // (of the TIP) cssStyles: {}, // styles to add the text box // example: {fontFamily: 'Georgia, // Times, serif', fontWeight: 'bold'} activeClass: 'bt-active', // class added to TARGET element when // its BeautyTip is active contentSelector: "$(this).attr('title')", // if there is no content // argument, use this selector to // retrieve the title // a function which returns the content may also be passed here ajaxPath: null, // if using ajax request for content, this // contains url and (opt) selector this // will override content and // contentSelector examples (see jQuery // load() function): '/demo.html' // '/help/ajax/snip' '/help/existing/full // div#content' // ajaxPath can also be defined as an array // in which case, the first value will be parsed as a jQuery selector // the result of which will be used as the ajaxPath // the second (optional) value is the content selector as above // examples: // ["$(this).attr('href')", 'div#content'] // ["$(this).parents('.wrapper').find('.title').attr('href')"] // ["$('#some-element').val()"] ajaxError: '<strong>ERROR:</strong> <em>%error</em>', // error text, use "%error" to insert error from server ajaxLoading: '<blink>Loading...</blink>', // yes folks, it's the blink tag! ajaxData: {}, // key/value pairs ajaxType: 'GET', // 'GET' or 'POST' ajaxCache: true, // cache ajax results and do not send // request to same url multiple times ajaxOpts: {}, // any other ajax options - timeout, // passwords, processing functions, etc... // see // http://docs.jquery.com/Ajax/jQuery.ajax#options preBuild: function () { }, // function to run before popup is built preShow: function (box) { }, // function to run before popup is displayed showTip: function (box) { $(box).show(); }, postShow: function (box) { }, // function to run after popup is built and displayed preHide: function (box) { }, // function to run before popup is removed hideTip: function (box, callback) { $(box).hide(); callback(); // you MUST call "callback" at the end of your animations }, postHide: function () { }, // function to run after popup is removed hoverIntentOpts: { // options for hoverIntent (if installed) interval: 300, // http://cherne.net/brian/resources/jquery.hoverIntent.html timeout: 500 } }; // </ jQuery.bt.defaults > jQuery.bt.options = {}; })(jQuery); // @todo // use larger canvas (extend to edge of page when windowMargin is active) // add options to shift position of tip vert/horiz and position of spike tip // create drawn (canvas) shadows // use overlay to allow overlap with hover // experiment with making tooltip a subelement of the target // handle non-canvas-capable browsers elegantly