web/res/metadataplayer/res/js/jquery.jqDock.js
changeset 99 6cb4d10f0b8b
child 133 9578d3ddce17
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/web/res/metadataplayer/res/js/jquery.jqDock.js	Wed Apr 06 16:26:16 2011 +0200
@@ -0,0 +1,1437 @@
+/** @preserve jquery.jqDock.js v1.6
+ */
+/*
+ * jqDock jQuery plugin
+ * Version : 1.6
+ * Author : Roger Barrett
+ * Date : August 2010
+ *
+ * Inspired by:
+ *   iconDock jQuery plugin
+ *   http://icon.cat/software/iconDock
+ *   version: 0.8 beta
+ *   date: 2/05/2007
+ *   Copyright (c) 2007 Isaac Roca & icon.cat (iroca@icon.cat)
+ *   Dual licensed under the MIT-LICENSE.txt and GPL-LICENSE.txt
+ *   http://www.opensource.org/licenses/mit-license.php
+ *   http://www.gnu.org/licenses/gpl.html
+ *
+ * Dual licensed under the MIT-LICENSE.txt and GPL-LICENSE.txt
+ * http://www.opensource.org/licenses/mit-license.php
+ * http://www.gnu.org/licenses/gpl.html
+ *
+ * Change Log :
+ * v1.6
+ *    - bugfix : when initially fading in, the 'Asleep' state wasn't being cleared (typo) which meant that the Dock would run onWake(), and possibly trigger dockwake, which it shouldn't
+ *    - new option, active (default -1), which is the index (zero-based) of an image required to be expanded on initial display
+ *    - new option, noBuffer (default false), which disables the buffering of the last mouse event while the dock is asleep
+ *    - added a 'destroy' command to jqDock() function - jqDock('destroy') - which removes jqDock from a menu
+ *    - added 'expand' and 'active' commands, which expand a selected image to full size, with/without animation respectively; NB dock gets frozen!
+ *    - expanded the (previously undocumented) 'get' command to return either the internal Dock object, or an internal image object, depending on the selector (v1.5 'get' only handled images)
+ *    - added listener for custom event - dockfreeze - on the original menu element (as a sub-function of docksleep), which can be triggered by the calling program to (try to) freeze the dock, ie. put it to sleep but without 'tidying up' first
+ *    - added a 'freeze' command, which does the same thing as triggering the new dockfreeze event (but synchronously)
+ *    - the onReady, onSleep and onWake hook functions are now each passed a single argument - 'ready', 'sleep' or 'freeze', and 'wake' or 'thaw' respectively
+ *    - the triggered custom events - dockshow, docksleep and dockwake - are now passed 1 extra parameter when being triggered - 'ready', 'sleep' or 'freeze', and 'wake' or 'thaw' respectively
+ *    - the setup of labels has changed slightly...
+ *      - the outer label container (div.jqDockLabel) now gets created, styled, and has its click handler bound, *before* setLabel() is called
+ *      - setLabel() gets passed an extra parameter: the DOM element, div.jqDockLabel
+ *      - setLabel can now return false to prevent jqDock doing anything further with the label; otherwise it is expected
+ *        to return an html string as before, which jqDock will create an inner container for - div.jqDockLabelText - and append to the outer container
+ *    - the decision of whether or not to 'show' labels is now solely dependent on the 'labels' option setting
+ *    - partial expansion/collapse times (on mouseenter/leave) are now equal - eg. going on then off the menu (before expansion has completed) will allocate the same time to the collapse as was used for the expansion (instead of always using the full 'duration')
+ *    - re-worked the timings and added a 'tidy-up' loop - previous versions were too dependent on receiving mousemove events to complete the animations (particularly noticeable with 'flow' enabled)
+ * v1.5
+ *    - bugfix : the label click handler was not returning false, so clicks on labels were being notified to links (not images) twice
+ *    - new option, setLabel (default false), as a function called when initialising the label contents for each menu item
+ *    - added an extra layer - div.jqDockLabelText - inside div.jqDockLabel to facilitate positional 'tweaking' of the label without having to resort to the setLabel option
+ *    - new option, flow (default false), allowing the auto-centering to be disabled and the dock wrapper element to auto-size to precisely contain the dock
+ *    - new option, idle (default 0), as the number of milliseconds of idle time after the mouse has left the menu before the dock goes to sleep and the docksleep event is triggered (on the original menu element)
+ *    - new option, onSleep, as a function which is called with scope (this) of the original menu element when an optional number of milliseconds (the idle option) has elapsed since the mouse left the menu; returning false will prevent the dock from going to sleep
+ *    - new option, onWake, as a function which is called with scope (this) of the original menu element when dock is 'nudged' awake, but only if dock was asleep at the time; returning false will prevent the dock waking up (stays asleep)
+ *    - new option, onReady, as a function which is called with scope (this) of the orginal menu element when dock has been initialised and is ready for display; returning false will prevent the dock being displayed
+ *    - new custom event, dockshow, which is triggered on the original menu element when the dock has been completely initialised; this won't be triggered if the onReady() call returns false
+ *    - new custom event, docksleep, which is triggered on the original menu element following the onSleep() call, unless the onSleep() call returns false
+ *    - new custom event, dockwake, which is triggered on the original menu element following the onWake() call, unless the onWake() call returns false
+ *    - added listener for custom event - docknudge - on the original menu element, which *has* to be triggered by the calling program in order to (try to) wake the dock from a sleep
+ *    - added listener for custom event - dockidle - on the original menu element, which can be triggered by the calling program to (try to) put the dock to sleep
+ *    - added 2 commands to jqDock() function - jqDock('nudge') and jqDock('idle') - which do the same thing as triggering the respective docknudge and dockidle events (but synchronously)
+ *    - jqDock no longer hides the original menu element, since most likely usage is to pre-hide it to prevent 'flicker'; also now copes with visibility:hidden (as well as display:none)
+ *    - labels no longer get jqDockMouseN class
+ * v1.4
+ *    - bugfix : in IE8, non-statically positioned child elements do not inherit opacity, so fadeIn did not work correctly
+ *    - new option, fadeLayer (default ''), allows the fade-in to be switched from the original menu element down to either the
+ *      div.jqDockWrap or div.jqDock layer
+ * v1.3
+ *    - new option, inactivity (default 0), allowing auto-collapse after a specified period (mouse on dock)
+ *    - new option, fadeIn (default 0), allowing initialised menu to be faded in over a specified period (as opposed to an instant show)
+ *    - new option, step (default 50), which is the interval between animation steps
+ *    - default size increased to 48 (from 36)
+ *    - default distance increased to 72 (from 54)
+ *    - default duration reduced to 300 ms (from 500 ms)
+ *    - better 'best guess' for maximum dimensions of Dock
+ *    - handle integer options being passed in as strings (eg. size:'48' instead of size:48)
+ *    - the wrapper div now has width, height, and a class
+ *    - all menu items are double-wrapped now in 2 divs 
+ *    - double-wrap resolves ie8 horizontal float problem
+ *    - dimensioning switched from image to innermost of the item's double-wrap
+ *    - labels now assigned per menu item instead of one for the entire dock
+ *    - labels within anchors so clicking activates anchor
+ *    - labels are always created, regardless of option setting
+ *    - default label position changed from 'tc' to 'tl' for any alignment except 'top' (labels='br') and 'left' (labels='tr')
+ *    - events switched from mouseover/out to mouseenter/leave
+ * v1.2
+ *    - Fixes for Opera v9.5 - many thanks to Rubel Mujica
+ * v1.1
+ *    - some speed optimisation within the functions called by the event handler
+ *    - added positioning of labels (top/middle/bottom and left/center/right)
+ *    - added click handler to label (triggers click event on related image)
+ *    - added jqDockLabel(Link|Image) class to label, depending on type of current image
+ *    - updated demo and documentation for label positioning and clicking on labels
+ */
+(function($, window){
+if(!$.jqDock){ //can't see why it should be, but it doesn't hurt to check
+	var TRBL = ['Top', 'Right', 'Bottom', 'Left']
+		, AXES = ['Major', 'Minor']
+		, MOUSEEVENTS = ['mouseenter','mousemove','mouseleave']
+		, CUSTOMEVENTS = ['docknudge','dockidle','dockfreeze']
+		, TIMERS = ['Idler','Inactive','Indock','Overdock','Offdock']
+		, VANILLA = [
+				'<div style="position:relative;padding:0;'
+			, 'margin:0;border:0 none;background-color:transparent;'
+			, '">'
+			]
+		, VERTHORZ = { //note : lead and trail are indexes into TRBL
+				v: { wh:'height', xy:1, tl:'top', lead:0, trail:2, inv:'h' } //Opts.align = left/center/right
+			, h: { wh:'width', xy:0, tl:'left', lead:3, trail:1, inv:'v' } //Opts.align = top/middle/bottom
+			}
+		, DOCKS = []
+		, XY = [0, 0] //mouse position from left, mouse position from top
+		, EMPTYFUNC = function(){}
+/** returns integer numeric of leading digits in string argument
+ * @private
+ * @param {string} x String representation of an integer
+ * @return {integer} Number
+ */
+		, AS_INTEGER = function(x){
+				var r = parseInt(x, 10);
+				return isNaN(r) ? 0 : r;
+			}
+//v1.6 : handles clearing all timers...
+/** clears a specified timeout timer, or all timers if the supplied index is less than zero
+ * @private
+ * @param {object} Dock Dock object
+ * @param {integer} x Index into TIMERS of timer to clear, or a negative number to clear all timers
+ */
+		, CLEAR_TIMER = function(Dock, x){
+				var y = TIMERS[x] ? x + 1 : TIMERS.length;
+				for( ; x < y && y--; ){
+					if(Dock[TIMERS[y]]){
+						window.clearTimeout(Dock[TIMERS[y]]);
+						Dock[TIMERS[y]] = null;
+					}
+				}
+			}
+/** returns a dock index as indicated by the numeric suffix to the element's id attribute
+ * @private
+ * @param {element} el Element to test
+ * @return {integer} Dock index, -1 if not found
+ */
+		, DOCK_INDEX_FROM_ID = function(el){
+				return el ? 1 * ( (el.id || '').match(/^jqDock(\d+)$/) || [0,-1] )[1] : -1;
+			}
+//v1.6 : moved this out of initDock (used to be var callback) and corrected typo (Dock.Asleep instead of Dock.Sleep!)...
+/** from an initial fade-in of a menu, this clears filters (for IE) and notifies readiness
+ * @private
+ * @this {element} The element that was initially faded in
+ */
+		, FADEIN_COMPLETE = function(){
+				var Dock = DOCKS[ DOCK_INDEX_FROM_ID( $('.jqDockFilter', this).add(this)
+					//remove any filters...
+					.css({filter:''}).removeClass('jqDockFilter').filter('.jqDock')[0] ) ];
+				if(Dock){
+					//clear Asleep so that a docknudge won't do it's wake-up routine
+					Dock.Asleep = false;
+					//trigger dockshow (for the calling script) and docknudge (for me)...
+					Dock.Menu.trigger('dockshow', ['ready']).trigger(CUSTOMEVENTS[0]);
+				}
+			}
+/** finds a given IMG's entry within the Elem arrays, across all Docks
+ * @private
+ * @param {element} el Element to search for
+ * @param {boolean|integer} indices Switch asking for return of an array of the index values rather than an object
+ * @return {object|boolean|array} False if not found
+ */
+		, FIND_IMAGE = function(el, indices){
+				var cont = true
+					, id = DOCKS.length
+					, idx;
+				while(el && cont && id-- && DOCKS[id].Elem){
+					idx = DOCKS[id].Elem.length;
+					while(cont && idx--){
+						cont = DOCKS[id].Elem[idx].Img[0] !== el;
+					}
+				}
+				return cont ? !cont : (indices ? [id, idx] : DOCKS[id].Elem[idx]);
+			}
+//v1.6 : new timestamping function - if you go on then off the menu (before the expansion has completed),
+//       it makes the time period for the collapse the same as the time used for the partial expansion
+/** returns a timestamp; if Dock is supplied and there is anything left of a previous timestamp - when
+ *  duration is added - then subtract that remainder from the new timestamp
+ * @private
+ * @param {object} [Dock] Dock object
+ * @return {integer} New timestamp
+ */
+		, GET_TIME = function(Dock){
+				var rtn = (new Date()).getTime()
+					, prevWillLapse = Dock ? Dock.Stamp + Dock.Opts.duration : 0;
+				if(prevWillLapse > rtn){
+					rtn -= prevWillLapse - rtn;
+				}
+				return rtn;
+			}
+/** the onload handler for images; stores width/height, and runs initDock() (on a timeout) if all images for a dock are loaded
+ * @private
+ * @this {element} The image element
+ * @param {object} ev jQuery event object
+ */
+		, IMAGE_ONLOAD = function(ev){
+				//store 'large' width and height...
+				var Dock = DOCKS[ev.data.id]
+					, el = Dock.Elem[ev.data.idx];
+				el.height = this.height;
+				el.width = this.width;
+				if(--Dock.Load <= 0){ //check to see if all images are loaded...
+					window.setTimeout(function(){ $.jqDock.initDock(ev.data.id); }, 0);
+				}
+			}
+/** returns an item index as indicated by the numeric suffix to the closest jqDockMouse-classed element
+ * @private
+ * @param {element} el Element to start from
+ * @param {element} context Element that the provider of the item index must be within
+ * @return {integer} Item index, -1 if not found
+ */
+		, ITEM_INDEX_FROM_CLASS = function(el, context){
+				var m;
+				while(el && el.ownerDocument && el !== context){
+					m = el.className.toString().match(/jqDockMouse(\d+)/);
+					if(m){
+						return 1 * m[1];
+					}
+					el = el.parentNode;
+				}
+				return -1;
+			}
+/** returns an object containing width and height, with the one NOT represented by 'dim'
+ * being calculated proportionately
+ * if horizontal menu then attenuation is along horizontal (x) axis, thereby setting the new
+ * dimension for width, so the one to keep in proportion is height; and vice versa for
+ * vertical menus, obviously!
+ * @private
+ * @param {object} el Element of Elem array
+ * @param {integer} dim Image dimension
+ * @param {string} vh Vertical or horizontal
+ * @return {object} The provided dimension and the proportioned dimension (width and height, but not necessarily respectively!)
+ */
+		, KEEP_PROPORTION = function(el, dim, vh){
+				var r = {}
+					, vhwh = VERTHORZ[vh].wh //convenience
+					, invwh = VERTHORZ[VERTHORZ[vh].inv].wh //convenience
+					;
+				r[vhwh] = dim;
+				r[invwh] = Math.round(dim * el[invwh] / el[vhwh]);
+				return r;
+			}
+/** a label click handler that triggers its related image's click handler
+ * @private
+ * @this {element} The DOM element (label) the handler was bound to
+ * @return {boolean} False
+ */
+		, LABEL_CLICK = function(){
+				$(this).prev('img').trigger('click');
+				return false;
+			}
+/** shows/hides a label
+ * @private
+ * @param {object} Dock Dock object
+ * @param {integer} [show] Show label
+ */
+		, LABEL_SHOW = function(Dock, show){
+				var item = Dock.Elem[Dock.Current];
+				if(item && Dock.Opts.labels){
+					item.Label.el[show ? 'show' : 'hide']();
+				}
+			}
+/** re-positions a label if needed
+ *  only labels with middle and/or center alignment need re-positioning because css handles the corners
+ * @private
+ * @param {object} Dock Dock object
+ * @param {integer|boolean} show Whether to show the label or not
+ */
+		, POSITION_LABEL = function(Dock, show){
+				var labels = Dock.Opts.labels
+					, VH = VERTHORZ[Dock.Opts.vh]
+					, el = Dock.Elem[Dock.Current]
+					, i, j, label, labelElem;
+				if(el && labels){
+					label = el.Label;
+					labelElem = label.el;
+					//check to see if the information required for a middle/centred label has already been gathered...
+					//note : middle/centred labels can not be set up while the dock is display:none
+					if(label.mc){
+						label.mc = 0;
+						//if labels are being aligned middle and/or centre then we need to find any user-styled padding and width/height, and
+						//store the overall dimensions (incl. padding) for this image's label, so that we don't need to do this next time...
+						for(i in VERTHORZ){
+							label[i] = labelElem[VERTHORZ[i].wh]();
+							for(j in {lead:1, trail:1}){
+								label[i] += AS_INTEGER(labelElem.css('padding' + TRBL[VERTHORZ[i][j]]));
+							}
+						}
+					}
+					//note: if vertically or horizontally centred then centre is based on the IMAGE only
+					//note : .xy is 0 on horizontal menus, 1 on vertical menus (and vice versa for [.inv].xy!)...
+					if(labels.charAt(0) == 'm'){
+						labelElem.css({top: Math.floor((el[AXES[VERTHORZ[VH.inv].xy]] - label.v) / 2)});
+					}
+					if(labels.charAt(1) == 'c'){
+						labelElem.css({left: Math.floor((el[AXES[VH.xy]] - label.h) / 2)});
+					}
+				}
+				if(show){
+					LABEL_SHOW(Dock, 1); //show
+				}
+			}
+//v1.6 : was DELTA_XY, but has been renamed and re-worked to not depend on .Current
+/** translates (without affecting) XY[0] or XY[1] into an offset within div.jqDock
+ * note: doing it this way means that all attenuation is against the initial (shrunken) image positions,
+ * but it saves having to find every image's offset() each time the cursor moves or an image changes size!
+ * @private
+ * @param {object} Dock Dock object
+ * @return {number} Translated mouse offset, or -1 if outside dock
+ */
+		, RELATIVE_XY = function(Dock){
+				var VH = VERTHORZ[Dock.Opts.vh] //convenience
+					, numElems = Dock.Elem.length
+					, rtn = -1
+					, i = 0
+					, el, padding, majorWidth
+						//distance into the menu from the leading edge of first element in menu...
+					, offset = XY[VH.xy] - Dock.Elem[0].Wrap.parent().offset()[VH.tl];
+				if(offset >= 0){
+					for( ; rtn < 0 && i < numElems; i++){
+						el = Dock.Elem[i];
+						padding = el.Pad[VH.lead] + el.Pad[VH.trail];
+						majorWidth = el.Major + padding;
+						if(offset < majorWidth){
+							//we've found the element that the mouse is currently on (which may or may not be the same as Current)
+							if(i != Dock.Current){ 
+								//if its not the same as Current, make sure the label is hidden and reset Current...
+								LABEL_SHOW(Dock); //hide
+								Dock.Current = i;
+							}
+							rtn = el.Offset + (offset * (el.Initial + padding) / majorWidth);
+						}else{
+							offset -= majorWidth;
+						}
+					}
+				}
+				return rtn;
+			}
+/** removes ALL text nodes from the menu, so that we don't get spacing issues between menu elements
+ * @private
+ * @param {element} el DOM Element
+ * @recursive
+ */
+		, REMOVE_TEXT = function(el){
+				var i = el.childNodes.length, j;
+				while(i){
+					j = el.childNodes[--i];
+					if(j.childNodes && j.childNodes.length){
+						REMOVE_TEXT(j);
+					}else if(j.nodeType == 3){
+						el.removeChild(j);
+					}
+				}
+			}
+/** initial display of the menu, copes with visibility:hidden as well as display:none
+ * @private
+ * @param {object} Dock Dock object
+ */
+		, REVEAL_MENU = function(Dock){
+				Dock.Menu.css({visibility:'visible'}).show();
+			}
+/** if appropriate, sets an idle timer to trigger a dockidle
+ * @private
+ * @param {object} Dock Dock object
+ */
+		, SET_IDLER = function(Dock){
+				var idleDelay = Dock.Opts.idle;
+				if(idleDelay){
+					CLEAR_TIMER(Dock, 0); //Idler
+					//set Idler timer...
+					Dock[TIMERS[0]] = window.setTimeout(function(){
+							Dock.Menu.trigger('dockidle');
+						}
+						, idleDelay);
+				}
+			}
+/** create and append the label; unless the label uses middle/center alignment, this is all the label setup required
+ *  any label setting involving middle/center gets handled in POSITION_LABEL()
+ * @private
+ * @param {object} Dock Dock object
+ * @param {object} item Menu item object
+ * @param {integer} indx Index of menu item within menu
+ */
+		, SET_LABEL = function(Dock, item, indx){
+				var op = Dock.Opts //convenience
+					, labels = op.labels //convenience
+					, label = item.Label //convenience
+					, posBottom, posRight, txt
+					;
+				//labels always get created, and get shown if they are enabled; however ...
+				// - prior to v1.6, enabled labels only got shown IF they had content, where 'content' meant a text string which
+				//   was provided by option.setLabel() *before* actually creating the label
+				// - as of v1.6, this restriction has been removed, and the creation of the label has been changed slightly, so that
+				//   the label is partially created, *then* option.setLabel is called and the result - if there is one - is used to
+				//   help create the rest of the label, eg
+				//     - create outer label container, div.jqDockLabel
+				//     - set the outer container's css and add a click handler
+				//     - call option.setLabel
+				//     - if setLabel returns false
+				//       - do nothing else to the label
+				//       - create the inner label container (div.jqDockLabelText) with the returned text inside it
+				//   this means that option.setLabel can use DOM manipulation if wants to, and return false to prevent anything further
+				//   being done to the label by jqDock, and labels being 'shown' only depends on labels being enabled!
+				//   NOTE : this means that Dock.Elem[n].Label.txt is now (as of v1.6) superfluous, and has been removed!!!
+				//   NOTE : if setLabel() returns false, then it (setLabel) is responsible for creating the inner container (if it still
+				//          needs it)
+
+				//create the label's *outer* container (div.jqDockLabel) and hide it...
+				label.el = $('<div class="jqDockLabel jqDockLabel' + item.Link + '" style="position:absolute;margin:0;"></div>')
+					.hide().insertAfter(item.Img); //insert after the image element
+				if(labels){
+					posBottom = labels.charAt(0) == 'b';
+					posRight = labels.charAt(1) == 'r';
+					//position the label and give a click handler...
+					label.el.css({
+							top:    posBottom ? 'auto' : 0
+						, left:   posRight  ? 'auto' : 0
+						, bottom: posBottom ? 0 : 'auto'
+						, right:  posRight  ? 0 : 'auto'
+						}).click(LABEL_CLICK); //NB: the click handler returns false!
+					//Note that the click handler is on div.jqDockLabel, not div.jqDockLabelText! This should mean that
+					//the calling script can put its own click handler(s) on div.jqDockLabelText or its contents without
+					//having to remove jqDock's LABEL_CLICK handler?
+				}
+				//get the content for the *inner* label container...
+				//NB: 4th parameter (DOM element, div.jqDockLabel) added as of v1.6
+				txt = op.setLabel.call(Dock.Menu[0], item.Title, indx, label.el[0]);
+				if(txt !== false){
+					//if there is label content (as an HTML string!) then insert it with the inner container...
+					$('<div class="jqDockLabelText">' + txt.toString() + '</div>').appendTo(label.el);
+				}
+			}
+/** calculates the image sizes according to the current (translated) position of the cursor within div.jqDock
+ * result stored in Final for each menu element
+ * @private
+ * @param {integer} id Dock index
+ * @param {number} [relxy] Translated cursor offset in main axis
+ */
+		, SET_SIZES = function(id, relxy){
+				var Dock = DOCKS[id] //convenience
+					, op = Dock.Opts //convenience
+					, wh = VERTHORZ[op.vh].wh //convenience
+					, i = Dock.Elem.length
+					, el, ab, newFinal, oscillate;
+				//if not forced, use current translated cursor position (main axis)...
+				relxy = relxy || relxy === 0 ? relxy : RELATIVE_XY(Dock);
+				for( ; i--; ){
+					el = Dock.Elem[i];
+					newFinal = el.Initial;
+					if(relxy >= 0){
+						ab = Math.abs(relxy - el.Centre);
+						//if we're smack on or beyond the attenuation distance then set to the min dim
+						//ensure Final ends up as an integer to avoid 'flutter'
+						if(ab < op.distance){
+							newFinal = el[wh] - Math.round((el[wh] - el.Initial) * Math.pow(ab, op.coefficient) / op.attenuation); 
+						}
+						//need to check for oscillation, where, for example, the Final dimension gets changed by a pixel,
+						//which changes the relative position within the dock by a pixel, which changes the Final dimension
+						//back by a pixel, which changes the relative position within the dock, which changes... etc, etc, etc!
+						//it doesn't happen very often but it does happen!
+						if(i == Dock.Current){
+							oscillate = [XY[VERTHORZ[op.vh].xy], Dock.Current, newFinal].join(',');
+							if(oscillate === Dock.ToFro[0] && newFinal !== Dock.ToFro[2]){
+								newFinal = Dock.ToFro[2];
+							}else{
+								Dock.ToFro = [Dock.ToFro[1], oscillate, newFinal];
+							}
+						}
+					}
+					el.Final = newFinal;
+				}
+			}
+/** dummy function, simply returns labelText (for when options.setLabel is not provided)
+ * @private
+ * @this {element} original menu element
+ * @param {string} labelText Current label text for menu option
+ * @param {integer} indx Index of the menu option within the menu
+ * @param {element} container DOM element div.jqDockLabelText
+ * @return {string} labelText
+ */
+		, TRANSFORM_LABEL = function(labelText, indx, container){
+				return labelText;
+			}
+/** sets the css for an individual image wrapper to effect its change in size
+ * 'dim' is the new value for the main axis dimension as specified in VERTHORZ[Opts.vh].wh, so
+ * the margin needs to be applied to the inverse dimension!
+ * note: 'force' is only set when called from initDock() to do the initial shrink
+ * @private
+ * @param {integer} id Dock index
+ * @param {integer} idx Image index
+ * @param {integer} dim Main axis dimension of image
+ * @param {boolean} force Force change even if no size difference
+ */
+		, CHANGE_SIZE = function(id, idx, dim, force){
+				var Dock = DOCKS[id] //convenience
+					, el = Dock.Elem[idx] //convenience
+					, op = Dock.Opts //convenience
+					, yard = Dock.Yard //convenience
+					, VH = VERTHORZ[op.vh] //convenience
+					, invVH = VERTHORZ[VH.inv] //convenience
+					, srcDiff = el.src != el.altsrc
+					, bdr, css, diff
+					;
+				if(force || el.Major != dim){
+					//horizontal menus in IE quirks mode require border widths (if any) of the Dock to be added to the Dock's main axis dimension...
+					bdr = ($.boxModel || op.vh == 'v') ? 0 : Dock.Border[VH.lead] + Dock.Border[VH.trail];
+					//switch image source to large, if (a) it's different to small source, and (b) this is the first step of an expansion...
+					if(srcDiff && !force && el.Major == el.Initial){
+						el.Img[0].src = el.altsrc;
+					}
+					Dock.Spread += dim - el.Major; //adjust main axis dimension of dock
+					css = KEEP_PROPORTION(el, dim, op.vh);
+					diff = op.size - css[invVH.wh];
+					//add minor axis margins according to alignment...
+					//note: where diff is an odd number of pixels, for 'middle' or 'center' alignment put the odd pixel in the 'lead' margin
+					if({top:1, left:1}[op.align]){ //set bottom/right margin
+						css['margin' + TRBL[invVH.trail]] = diff;
+					}else if({middle:1, center:1}[op.align]){ //set top/left and bottom/right margins
+						css['margin' + TRBL[invVH.lead]] = (diff + (diff % 2)) / 2;
+						css['margin' + TRBL[invVH.trail]] = (diff - (diff % 2)) / 2;
+					}else{ //set top/left margin (op.align = 'bottom' or 'right')
+						css['margin' + TRBL[invVH.lead]] = diff;
+					}
+					//set dock's main axis dimension (if it's changed, or if force and this is first menu item)...
+					if (dim != el.Major || (force && !idx)) {
+						if(op.flow){
+							//if we ARE running flow, then the wrapper dimensions must be set so as to precisely contain the dock...
+							yard.parent()[VH.wh](Dock.Spread + Dock.Border[VH.lead] + Dock.Border[VH.trail]);
+						}
+						yard[VH.wh](Dock.Spread + bdr);
+					}
+					//change image wrapper size and margins...
+					el.Wrap.css(css);
+					//set dock's main axis 'lead' offset (not negative!)...
+					if(!op.flow){
+						//if we are NOT running flow (which is the default) then the dock needs to be centered within its wrapper...
+						yard.css(VH.tl, Math.floor(Math.max(0, (Dock[VH.wh] - Dock.Spread) / 2)));
+					}
+					//reposition the label if need be...
+					if(Dock.OnDock){
+						POSITION_LABEL(Dock, !Dock.Stamp);
+					}
+					//store new dimensions...
+					el.Major = dim; //main axis
+					el.Minor = css[invVH.wh]; //minor axis
+					//switch image source to small, if (a) it's different to large source, and (b) this was the last step of a shrink...
+					if(srcDiff && !force && dim == el.Initial){
+						el.Img[0].src = el.src;
+					}
+					css = null;
+				}
+			}
+//v1.6 : re-worked to remove need for the 'revers' switch
+/** modifies the target sizes in proportion to 'duration' if still within the 'duration' period following a mouseenter/leave
+ * calls CHANGE_SIZE() for each menu element (if more than Opts.step ms since mouseenter/leave)
+ * @private
+ * @param {integer} id Dock index
+ */
+		, FACTOR_SIZES = function(id){
+				var Dock = DOCKS[id] //convenience
+					, op = Dock.Opts //convenience
+					, VH = VERTHORZ[op.vh]
+					, lapse = op.duration + op.step
+					, i = 0 //must go through the elements in logical order
+					, el, sz, stepsLeft;
+				if(Dock.Stamp){
+					lapse = GET_TIME() - Dock.Stamp;
+					//there's no point continually checking Date once op.duration has passed...
+					if(lapse >= op.duration){
+						Dock.Stamp = 0;
+					}
+				}
+				if(lapse >= op.step){ //only if Opts.step ms have passed since last mouseenter/leave
+					stepsLeft = (op.duration - lapse) / op.step;
+					for( ; i < Dock.Elem.length; i++){
+						el = Dock.Elem[i];
+						sz = el.Final - el.Major;
+						sz = (sz && stepsLeft > 1) ? el.Major + Math[sz < 0 ? 'floor' : 'ceil'](sz / stepsLeft) : el.Final;
+						CHANGE_SIZE(id, i, sz); //...will set .Major to sz
+					}
+					//tweak 'best guess':
+					//having changed all item sizes within the dock, if Spread is greater than main axis dimension, adjust wrap dimension...
+					if(Dock.Spread > Dock[VH.wh]){
+						Dock.Yard.parent()[VH.wh](Dock.Spread + Dock.Border[VH.lead] + Dock.Border[VH.trail]);
+						Dock[VH.wh] = Dock.Spread;
+					}
+				}
+			}
+//v1.6 : new function
+/** handles movement of the mouse within a dock, and tidies up after entry (and halt) into a dock
+ * clears its own timer, runs SET_SIZES, then if not complete, runs FACTOR_SIZES and then itself on a timer 
+ * @private
+ * @param {integer} id Dock index
+ * @param {number} [relxy] Translated cursor offset in main axis (when provided to OVER_DOCK)
+ */
+		, IN_DOCK = function(id, relxy){
+				var Dock = DOCKS[id], el = Dock.Elem, i = el.length;
+				CLEAR_TIMER(Dock, 2); //Indock
+				if(Dock.OnDock && !Dock.Stamp){
+					SET_SIZES(id, relxy);
+					while((i--) && el[i].Major == el[i].Final){}
+					if(i < 0){
+						LABEL_SHOW(Dock, 1); //show
+					}else{
+						FACTOR_SIZES(id);
+						//set Indock timer...
+						Dock[TIMERS[2]] = window.setTimeout(function(){ IN_DOCK(id, relxy); }, Dock.Opts.step);
+					}
+				}
+		}
+/** called when cursor goes outside menu, and checks for completed shrinking of all menu elements
+ * calls FACTOR_SIZES() (with revers set) on any menu element that has not finished shrinking
+ * calls itself on a timer to complete the shrinkage
+ * @private
+ * @param {integer} id Dock index
+ * @param {boolean} noIdle Can idler be set
+ */
+		, OFF_DOCK = function(id, noIdle){
+				var Dock = DOCKS[id] //convenience
+					, el = Dock.Elem
+					, i = el.length
+					;
+				if(!Dock.OnDock){
+					while((i--) && el[i].Major <= el[i].Initial){}
+					//this is here for no other reason than that early versions of Opera seem to leave 
+					//a 'shadow' residue of the expanded image unless/until this function is called!...
+					RELATIVE_XY(Dock);
+					if(i < 0){ //complete
+						//reset everything back to 'at rest' state...
+						Dock.Stamp = 0;
+						for(i = el.length; i--; ){
+							el[i].Major = el[i].Final = el[i].Initial;
+						}
+						Dock.Current = -1;
+						if(!noIdle){
+							SET_IDLER(Dock);
+						}
+					}else{
+						FACTOR_SIZES(id);
+						//set Offdock timer...
+						Dock[TIMERS[4]] = window.setTimeout(function(){ OFF_DOCK(id, noIdle); }, Dock.Opts.step);
+					}
+				}
+			}
+/** checks for completed expansion (if OnDock)
+ * runs SET_SIZES() then, if not completed, runs FACTOR_SIZES() and then itself on a timer
+ * @private
+ * @param {integer} id Dock index
+ * @param {number} [relxy] Translated cursor offset in main axis
+ */
+		, OVER_DOCK = function(id, relxy){
+				var Dock = DOCKS[id] //convenience
+					, el = Dock.Elem
+					, i = el.length;
+				if(Dock.OnDock){
+					SET_SIZES(id, relxy);
+					while((i--) && el[i].Major == el[i].Final){}
+					if(i < 0 || !Dock.Stamp){ //complete, or beyond 'duration'
+						Dock.Stamp = 0;
+						IN_DOCK(id, relxy);
+					}else{
+						FACTOR_SIZES(id);
+						//set Overdock timer...
+						Dock[TIMERS[3]] = window.setTimeout(function(){ OVER_DOCK(id, relxy); }, Dock.Opts.step);
+					}
+				}
+			}
+/** actions for any type of mouse event
+ * @private
+ * @param {integer} etype Type of event as index into MOUSEEVENTS array
+ * @param {integer} id Dock id
+ * @param {integer} idx Menu item id or -1
+ * @param {integer} fake Set - usually to 1 - if called as a result of inactivity or when faking a mouseenter
+ */
+		, DO_MOUSE = function(etype, id, idx, fake){
+				var Dock = DOCKS[id] //convenience
+					, el = Dock.Elem //convenience
+					, i = el.length;
+				//mouseenter...
+				if(etype === 0){
+					Dock.OnDock = 1;
+					if(Dock.Current >= 0 && Dock.Current !== idx){
+						LABEL_SHOW(Dock); //hide
+					}
+					Dock.Current = idx;
+					//if fake is set greater than 1 then timestamp is set to zero (no animation)...
+					Dock.Stamp = fake && fake > 1 ? 0 : GET_TIME(Dock);
+					OVER_DOCK(id, fake ? el[idx].Centre : null);
+				}
+				//mousemove...
+				if(etype === 1){
+					if(idx !== Dock.Current){ //mousemove from one item onto another
+						LABEL_SHOW(Dock); //hide
+						Dock.Current = idx;
+					}
+					IN_DOCK(id);
+				}
+				//mouseleave...
+				if(etype === 2){
+					CLEAR_TIMER(Dock, 1); //Inactive
+					Dock.OnDock = 0;
+					LABEL_SHOW(Dock); //hide
+					Dock.Stamp = GET_TIME(Dock);
+					while(i--){
+						el[i].Final = el[i].Initial;
+					}
+					OFF_DOCK(id, !!fake); //clears Current when complete
+				}
+			}
+/** handler for all bound mouse events (move/enter/leave)
+ * @private
+ * @this {element}
+ * @param {object} ev jQuery Event object
+ * @return {boolean} false
+ */
+		, MOUSE_HANDLER = function(ev){
+				var dockId = DOCK_INDEX_FROM_ID(this)
+					, Dock = DOCKS[dockId]
+					, idx = Dock ? ITEM_INDEX_FROM_CLASS(ev.target, this) : -1
+					, doMse = -1
+					, onDock
+					;
+				if(Dock){
+					if(Dock.Asleep){ //buffer it?...
+						if(!Dock.Opts.noBuffer){ //...yes...
+							Dock.Sleeper = {
+									target:ev.target
+								, type:ev.type
+								, pageX:ev.pageX
+								, pageY:ev.pageY
+								};
+						}
+					}else{
+						onDock = Dock.OnDock;
+						CLEAR_TIMER(Dock, 0); //Idler
+						XY = [ev.pageX, ev.pageY];
+						if(ev.type == MOUSEEVENTS[2]){//=mouseleave
+							if(onDock){
+								doMse = 2; //mouseleave
+							}else{
+								SET_IDLER(Dock);
+							}
+						}else{ //=mousemove or mouseenter...
+							if(Dock.Opts.inactivity){
+								CLEAR_TIMER(Dock, 1); //Inactive
+								//set Inactive timer...
+								Dock[TIMERS[1]] = window.setTimeout(function(){ 
+										DO_MOUSE(2, dockId, idx, 1); //mouseleave (faked)
+									}, Dock.Opts.inactivity);
+							}
+							if(ev.type == MOUSEEVENTS[1]){ //=mousemove
+								if(idx < 0){
+									if(onDock && Dock.Current >= 0){ //off of current
+										doMse = 2; //mouseleave
+									}
+								}else if(!onDock || Dock.Current < 0){ //instant re-entry or no current
+									doMse = 0; //mouseenter
+								}else{ //change of current or moving within current
+									doMse = 1; //mousemove
+								}
+							}else if(idx >= 0 && !onDock){ //mouseenter...
+								doMse = 0; //mouseenter
+							}
+						}
+						Dock.Sleeper = null;
+						if(doMse >= 0){
+							DO_MOUSE(doMse, dockId, idx);
+						}
+					}
+				}
+//v1.5 don't return false, otherwise handlers listening on docksleep and then, for example,
+//     checking a mouseover on div.jqDock in order to 'bring back' a hidden menu, would
+//     not receive notification of the mouseover because it would be blocked here
+//				return false;
+			}
+/** handler for the docknudge and dockidle events
+ * @private
+ * @this {element} The original menu DOM element
+ * @param {object} ev jQuery event object
+ */
+		, LISTENER = function(ev){
+				var el = $('.jqDock', this).get(0)
+					, dockId = DOCK_INDEX_FROM_ID(el)
+					, Dock = DOCKS[ dockId ]
+					, frosty = ev.type == CUSTOMEVENTS[2]
+					, param = frosty ? 'freeze' : 'sleep'
+					, stateChange;
+				if(Dock){
+					//attempts to 'nudge' the dock awake...
+					if(ev.type == CUSTOMEVENTS[0]){ //docknudge
+						param = Dock.Frozen ? 'thaw' : 'wake';
+						//if Asleep, check for onWake returning a false - to stay asleep - and
+						//trigger a dockwake event if not still asleep...
+						if(Dock.Asleep && !(Dock.Asleep = (Dock.Opts.onWake.call(this, param) === false))){
+							//always clear frozen...
+							Dock.Frozen = !$(this).trigger('dockwake', [param]);
+						}
+						if(!Dock.Asleep){
+							//start (or reset) idling now...
+							SET_IDLER(Dock);
+							//if we have buffered mouse event, run it...
+							if(Dock.Sleeper){
+								MOUSE_HANDLER.call(el, Dock.Sleeper);
+							}
+						}
+					//...must be dockidle or dockfreeze event type...
+					}else{ //attempts to send the dock to sleep...
+						CLEAR_TIMER(Dock, 0); //Idler : needed if triggered by the calling program
+						//NB: returning false from onSleep() prevents the dock going to sleep/freezing, but
+						//it does NOT reset the idle timer!
+
+						//onSleep will only get called - and docksleep only get triggered - if we have a change of state, ie...
+						// - if not already asleep, or
+						// - if dockfreeze and not already frozen
+						//this means that if you idle a non-sleeping dock, then freeze it, onSleep will get called twice (once for each);
+						//but if you freeze, then idle, then freeze, onSleep will only get called for the first freeze.
+						stateChange = !Dock.Asleep || (frosty && !Dock.Frozen);
+						if(!stateChange || Dock.Opts.onSleep.call(Dock.Menu[0], param) !== false){
+							Dock.Asleep = !CLEAR_TIMER(Dock, frosty ? -1 : 1); //Inactive, or all if freezing
+							Dock.Frozen = Dock.Frozen || frosty;
+							if(stateChange){
+								Dock.Menu.trigger('docksleep', [param]);
+							}
+							if(frosty){
+								//need to clear the timestamp in case the dock was frozen during an automatic expansion/collapse...
+								Dock.Stamp = Dock.OnDock = 0;
+							}else{
+								DO_MOUSE(2, dockId, 0, 1); //fake a mouseleave as if it were due to inactivity
+							}
+						}
+					}
+				}
+			}
+		;
+
+/**
+ * The main $.jqDock object
+ * @private
+ * @return {object}
+ */
+	$.jqDock = (function(){
+		return {
+				version : 1.6
+			, defaults : { //can be set at runtime, per menu
+					size : 48 //[px] maximum minor axis dimension of image (width or height depending on 'align' : vertical menu = width, horizontal = height)
+				, distance : 72 //[px] attenuation distance from cursor
+				, coefficient : 1.5 //attenuation coefficient
+				, duration : 300 //[ms] duration of initial expansion and off-menu shrinkage
+				, align : 'bottom' //[top/middle/bottom or left/center/right] fixes horizontal/vertical expansion axis
+				, labels : 0 //enable/disable display of a label on the current image; (true) to use default position, or string to specify
+				, source : 0 //function: given scope of relevant image element; passed index of image within menu; required to return image source path, or false to use original
+				, loader : 0 //overrides useJqLoader if set to 'image' or 'jquery'
+				, inactivity : 0 //[ms] duration of inactivity (no mouse movement) after which any expanded images will collapse; 0 (zero) disables the inactivity timeout
+				, fadeIn : 0 //[ms] duration of the fade-in 'reveal' of the jqDocked menu; set to zero for instant 'show'
+				, fadeLayer : '' //if fadeIn is set, this can change the element that is faded; the default is the entire original menu; alternatives are 'wrap' (.jqDockWrap element) or 'dock' (.jqDock element)
+				, step : 50 //[ms] the timer interval between each step of shrinkage/expansion
+//v1.5 : added setLabel, flow and idle options...
+				, setLabel : 0  //function for transforming label text (ie. title) when initially building the label;
+												//this is provided so that if the label requires HTML, the transform function can set 
+												//it rather than having to put it in the title field and thereby make the markup invalid.
+												//the called function will be given the scope (this) of the original menu element, and will be
+												//passed 4 arguments: 
+												// - the derived default text of the label (from the title of either the image or its parent anchor)
+												// - the (zero-based) index of the option within the menu
+												// - the outer DOM element of the target label, div.jqDockLabel
+												// - an array of HTML for creating the inner label container, div.jqDockLabelText (['<div class="jqDockLabelText">', '</div>'])
+												//the function should return either
+												// - the HTML string for the label, in which case jqDock will create the inner container and append the returned text to it
+												// - or false, in which case jqDock will do nothing further with the label setup
+				, flow : 0  //alters the default dock behaviour such that the dock is NOT auto-centered and the wrap 
+										//element (.jqDockWrap, which a relatively positioned) expands and collapses to precisely
+										//contain the dock (.jqDock); this allows elements positioned around the docked menu to
+										//adjust their own relative position according to the current state of the docked menu
+				, idle : 0 //[ms] duration of idle time after the mouse has left the menu (without re-entering, obviously!) before the docksleep event is triggered (on the original menu element)
+//v1.5 : added onReady, onSleep and onWake hooks...
+				, onReady : 0 //function: called with scope of original menu element when dock has been initialised but not yet revealed (ie. before being shown)
+											//NB: the onReady() function is passed a single argument, 'ready', and can return false to cancel the 'reveal' of the menu and put the dock to sleep
+				, onSleep : 0 //function: called with scope of original menu element when dock has been idle for the defined idle period and has therefore gone to sleep,
+											//or when either a sleep or freeze has been requested by the calling script (by triggering dockidle/dockfreeze, or commanding idle/freeze)
+											//NB: the onSleep() function is passed a single argument, 'sleep' or 'freeze', and can return false to cancel the sleep/freeze
+				, onWake : 0  //function: called with scope of original menu element when dock is 'nudged' awake, but only triggered if the dock was asleep (incl. frozen) prior to the' nudge'
+											//NB: the onWake() function is passed a single argument, 'wake' or 'thaw', and can return false to cancel the wake-up (dock stays asleep/frozen)
+//v1.6 : added noBuffer, active options...
+				, noBuffer : 0 //disables the buffering of the last mouse event while the dock is asleep
+				, active : -1 //index (zero-based) of the image required to be expanded on initial display
+				}
+			, useJqLoader : $.browser.opera || $.browser.safari //use jQuery method for loading images, rather than "new Image()" method
+
+/**
+ * initDock()
+ * ==========
+ * called by the image onload function, it stores and sets image height/width;
+ * once all images have been loaded, it completes the setup of the dock menu
+ * note: unless all images get loaded, the menu will stay hidden!
+ * @this {$.jqDock}
+ * @param {integer} id Dock index
+ */
+			, initDock : function(id){
+			//========================================
+					var Dock = DOCKS[id] //convenience
+						, op = Dock.Opts //convenience
+						, VH = VERTHORZ[op.vh] //convenience
+						, invVH = VERTHORZ[VH.inv] //convenience
+						, borders = Dock.Border //convenience
+						, numItems = Dock.Elem.length
+						, vanillaDiv = VANILLA.join('')
+						, offset = 0
+						, i = 0
+						, j, k, el, wh, acc, upad, wrap
+						, fadeLayer = op.fadeLayer //convenience
+						;
+					// things will screw up if we don't clear text nodes...
+					REMOVE_TEXT(Dock.Menu[0]);
+					//double wrap, and set some basic styles on the dock elements, otherwise it won't work
+					Dock.Menu.children()
+						.each(function(i, kid){
+								var wrap = Dock.Elem[i].Wrap = $(kid).wrap(vanillaDiv + vanillaDiv + '</div></div>').parent(); 
+								if(op.vh == 'h'){
+									wrap.parent().css('float', 'left');
+								}
+							})
+						.find('img').andSelf()
+						.css({
+								position: 'relative'
+							, padding: 0
+							, margin: 0
+							, borderWidth: 0
+							, borderStyle: 'none'
+							, verticalAlign: 'top'
+							, display: 'block'
+							, width: '100%'
+							, height: '100%'
+							});
+					//resize each image and store various settings wrt main axis...
+					while(i < numItems){
+						el = Dock.Elem[i++];
+						//resize the image wrapper to make the minor axis dimension meet the specified 'Opts.size'...
+						wh = KEEP_PROPORTION(el, op.size, VH.inv); //inverted!
+						el.Major = el.Final = el.Initial = wh[VH.wh];
+						el.Wrap.css(wh); //resize the image wrapper to its new shrunken setting
+						//remove titles, alt text...
+						el.Img.attr({alt:''}).parent('a').andSelf().removeAttr('title');
+						//use inverts because we're after the minor axis dimension...
+						Dock[invVH.wh] = Math.max(Dock[invVH.wh], op.size + el.Pad[invVH.lead] + el.Pad[invVH.trail]);
+
+						el.Offset = offset;
+						el.Centre = offset + el.Pad[VH.lead] + (el.Initial / 2);
+						offset += el.Initial + el.Pad[VH.lead] + el.Pad[VH.trail];
+					}
+
+					//'best guess' at calculating max 'spread' (main axis dimension - horizontal or vertical) of menu:
+					//for each img element of the menu, call SET_SIZES() with a forced cursor position of the centre of the image;
+					//SET_SIZES() will set each element's Final value, so tally them all, including user-applied padding, to give
+					//an overall width/height for this cursor position; set dock width/height to be the largest width/height found;
+					//repeat, with a forced cursor position of the leading edge of image
+					i = 0;
+					while(i < numItems){
+						el = Dock.Elem[i++];
+						upad = el.Pad[VH.lead] + el.Pad[VH.trail]; //user padding in main axis
+						//tally the minimum widths...
+						Dock.Spread += el.Initial + upad;
+
+						//for override cursor positions of Centre and Offset...
+						for(k in {Centre:1, Offset:1}){
+							//set sizes with an overridden cursor position...
+							SET_SIZES(id, el[k]);
+							//tally image widths/heights (plus padding)...
+							acc = 0; //accumulator for main axis image dimensions
+							for(j = numItems; j--; ){
+								//note that Final is an image dimension (in main axis) and does not include any user padding...
+								acc += Dock.Elem[j].Final + upad;
+							}
+							//keep largest main axis dock dimension...
+							if(acc > Dock[VH.wh]){ Dock[VH.wh] = acc; }
+						}
+					} //... i is now numItems
+					//reset Final for each image...
+					while(i){
+						el = Dock.Elem[--i];
+						el.Final = el.Initial;
+					} //... i is now 0
+					wrap = [
+							VANILLA[0], VANILLA[2] //this will be div.jqDockWrap, but I don't want margin, border or background
+						, '<div id="jqDock', id, '" class="jqDock" style="position:absolute;top:0;left:0;padding:0;margin:0;overflow:visible;'
+						, 'height:', Dock.height, 'px;width:', Dock.width, 'px;"></div></div>'
+						].join('');
+					Dock.Yard = $('div.jqDock', Dock.Menu.wrapInner(wrap));
+					//now that we have div.jqDock, let's see if the user has applied any css border styling to it...
+					for(j = 4; j--; ){
+						borders[j] = AS_INTEGER(Dock.Yard.css('border' + TRBL[j] + 'Width'));
+					}
+					Dock.Yard.parent().addClass('jqDockWrap')
+						.width(Dock.width + borders[1] + borders[3]) //Right and Left
+						.height(Dock.height + borders[0] + borders[2]); //Top and Bottom
+					//shrink all images down to 'at rest' size, and add appropriate identifying class...
+					for( ; i < numItems; i++){
+						el = Dock.Elem[i];
+						//apply the image's user-applied padding to the outer element wrapper...
+						upad = el.Wrap.parent();
+						for(j = 4; j--; ){
+							if(el.Pad[j]){
+								upad.css('padding' + TRBL[j], el.Pad[j]);
+							}
+						}
+						CHANGE_SIZE(id, i, el.Final, true); //force
+						//give a mouse class to both the image and the outer element wrapper (to handle any user padding)...
+						upad.add(el.Img).addClass('jqDockMouse'+i);
+						//create and append the label
+						SET_LABEL(Dock, el, i);
+					}
+					//bind dock listener events to the original menu element...
+					el = Dock.Menu.bind(CUSTOMEVENTS.join(' '), LISTENER);
+					//bind the mousehandler to the dock, and set filter:inherit on everything below the dock (see below)...
+					Dock.Yard.bind(MOUSEEVENTS.join(' '), MOUSE_HANDLER).find('*').css({filter:'inherit'});
+
+					//if we have a request for an 'active' image...
+					if(Dock.Elem[op.active]){
+						//fake a mouseeenter, with no timestamp so no animation...
+						DO_MOUSE(0, id, op.active, 2);
+					}
+
+/*v1.4 : bugfix : in IE8, non-statically positioned child elements do not inherit opacity; a way round this
+                  is to set filter:inherit on child elements
+  v1.5 : Further complications with IE's opacity handling :
+         When animating opacity (as opposed to doing a fadeIn) the alpha filter of the animated element *must*
+         be cleared (='' or ='inherit') on completion back to opacity 1. Otherwise, in IE7 the element will not allow
+         children (in this case, the images) to be visible beyond its bounds (ie. expanding a menu item gets the image
+         chopped off at the edge of jqDock); in IE8, the image does expand ok, but leaves 'shadows' when collapsing!
+         Another complication is that jQuery does not recognise that filter can contain anything other than an
+         'alpha(opacity=xxx)' value, so when the filter is set to 'inherit', jQuery animates opacity by *appending*
+         the 'alpha(...)' value to the current 'inherit' value (eg. filter:'inheritalpha(...)' 
+         So ...
+            ... on the assumption that nothing outside of jDock is going to want to individually fade 
+            anything below the .jqDock, I'm setting filter:inherit on all its children, for IE8's sake.
+            this is just in case anyone uses docksleep to perform a fade on .jqDock; if they do a fade
+            on either .jqDockWrap or the original menu element, then they may have to set (and probably
+            clear) filter:inherit on .jqDock, or .jqDock and .jqDockWrap (respectively) themselves!
+*/
+
+					//show the menu now?...
+					//if onReady returns false then the dock goes to sleep and will require a 'nudge' at some point to wake it up
+					if(!(Dock.Asleep = (op.onReady.call(Dock.Menu[0], 'ready') === false))){
+						if(fadeLayer){
+							//can only be 1 of menu/wrap/dock, and el is already set to Dock.Menu...
+							if(fadeLayer != 'menu'){ //either dock or wrap...
+								el = Dock.Yard;
+								if(fadeLayer == 'wrap'){
+									el = el.parent();
+								}
+							}
+							//.jqDockFilter is used so that I can ensure that only elements *below* .jqDock
+							//have filter:inherit set; this is so that if the calling program uses docksleep
+							//to fade out .jqDock I can at least ensure that it will work for IE8 (regardless
+							//of the other problems with animating IE's opacity!)
+							//Unfortunately, because of IE (grrr), we have to put the dock to sleep while the
+							//fade is taking place. This is because if the user were to mouse-over the menu 
+							//while it was still fading in, the menu element expansion would either be cut off
+							//at the jqDockWrap boundary (IE6/7) or would leave a 'shadow' trail effect beyond
+							//the jqDockWrap boundary as it shrank (IE8) ... due to the filters not being reset
+							//until the end of the animation.
+							Dock.Asleep = !!$('.jqDock,.jqDockWrap', el).addClass('jqDockFilter').css({filter:'inherit'});
+							el.css({opacity:0});
+							REVEAL_MENU(Dock);
+							el.animate({opacity:1}, op.fadeIn, FADEIN_COMPLETE);
+						}else{
+							REVEAL_MENU(Dock);
+							Dock.Menu.trigger('dockshow', ['ready']);
+							SET_IDLER(Dock);
+						}
+					}
+				} //end function initDock()
+
+			}; //end of return object
+		})(); //run the function to set up $.jqDock
+
+	/***************************************************************************************************
+	*  jQuery.fn.jqDock()
+	*  ==================
+	* STANDARD
+	* usage:      $(selector).jqDock(options);
+	* options:    see $.jqDock.defaults
+  * returns:    $(selector)
+  *
+  * ALTERNATE   ...provides a means for modifying image paths post-initialisation
+  * usage:      $(image-selector).jqDock(options);
+	* options:    object, with the following possible properties...
+	*               src: {string|function} Path to 'at rest' image, or function returning a path
+	*               altsrc: {string|function} Path to expanded image, or function returning a path
+  * returns:    $(image-selector)
+  * Note : image-selector *must* result in solely IMG element(s)
+  * 
+  * ALTERNATE2  ...provides a means for nudging a dock awake, or sending it to sleep
+	*                (see Advanced documentation)
+  * usage:      $(selector).jqDock('nudge'); //'nudges' dock awake
+  *             $(selector).jqDock('idle'); //sends dock to sleep
+  * returns:    $(selector)
+  * Note : selector should be (or contain) already initialised dock(s), ie. classed with 'jqDocked'
+	*
+  * ALTERNATE3  ...a 'getter', providing a means for retrieving either a Dock's internal object, or
+  *                an image's object from the Elem array
+  *                (undocumented, but used in example.js)
+  * usage:      $(menu-selector).jqDock('get');
+  *             $(image-selector).jqDock('get');
+  * returns:    {object} The object corresponding to the first (active) Dock in the $(menu-selector)
+  *                      colection; or the object which is the element of the Elem array corresponding
+  *                      to the first 'img' DOM element in the $(image-selector) collection
+	* 
+	* ALTERNATE4  ...provides a means for removing jqDock from a 'docked' element
+	*                (see Advanced documentation)
+	* usage:      $(selector).jqDock('destroy');
+  * returns:    $(selector)
+  * Note : selector should be (or contain) already initialised dock(s), ie. classed with 'jqDocked'
+	* 
+	* ALTERNATE5  ...provides a means for expanding (making active) an image
+	* usage:      $(image-selector).jqDock('expand'); //with animation
+	*             $(image-selector).jqDock('active'); //without animation
+	* returns:    $(image-selector)
+	* 
+	* note: the aim is to do as little processing as possible after setup, because everything is
+	* driven from the mousemove/enter/leave events and I don't want to kill the browser if I can help it!
+	* hence the code below, and in $.jqDock.initDock(), sets up and stores everything it possibly can
+	* which will reduce processing at runtime, and hopefully give as smooth animation as possible.
+	***************************************************************************************************/
+	$.fn.jqDock = function(opts){
+		/***************************************************************************************************
+		* ALTERNATE2:
+		* Accepts 'nudge', 'idle' or 'freeze'. Chainable.
+		* 
+		* Example:
+		*   $('#menu').jqDock('nudge'); //wake from sleep
+		*   $('#menu').jqDock('idle'); //send to sleep
+		*   $('#menu').jqDock('freeze'); //freeze the dock
+		***************************************************************************************************/
+		if(opts === 'nudge' || opts === 'idle' || opts === 'freeze'){ //alternate usage 3 (nudge/idle/freeze)
+			this.filter('.jqDocked').each(function(){ //only runs on an original menu element that has been docked
+					LISTENER.call(this, {type:'dock'+opts});
+				});
+		/***************************************************************************************************
+		* ALTERNATE4:
+		* Accepts 'destroy'. Chainable
+		* added v1.6
+		* 
+		* Example:
+		*   $('#menu').jqDock('destroy'); //remove jqDock functionality from the menu
+		***************************************************************************************************/
+		}else if(opts === 'destroy'){ //alternate usage 3 (destroy)
+			this.filter('.jqDocked').each(function(){
+					var dockId = DOCK_INDEX_FROM_ID( $('.jqDock', $(this).removeClass('jqDocked')).get(0) )
+						, Dock = DOCKS[dockId]
+						, i = MOUSEEVENTS.length
+						, j, el, imageEl;
+					if(Dock){
+						//clear any timers...
+						CLEAR_TIMER(Dock, -1);
+						//remove all the mouse and custom events...
+						for( ; i--; ){
+							Dock.Yard.unbind(MOUSEEVENTS[i], MOUSE_HANDLER);
+						}
+						//only remove the custom events that jqDock was listening for; if the calling
+						//script bound listeners for the other custom events (show/sleep/wake) then it
+						//is the calling script's responsibility to remove them (or not) as it wishes
+						for(i = CUSTOMEVENTS.length; i--; ){
+							Dock.Menu.unbind(CUSTOMEVENTS[i], LISTENER);
+						}
+						for(i = 0; i < Dock.Elem.length; i++){
+							el = Dock.Elem[i];
+							imageEl = el.Img;
+							//unbind the label's click handler and remove the label...
+							el.Label.el.unbind('click', LABEL_CLICK).remove();
+							//put the original attributes back onto the image, and remove the jqDockMouseN class...
+							imageEl.attr(el.Orig.i).removeClass('jqDockMouse' + i);
+							if(!el.Orig.i.style){ //if there was no inline style, might as well remove the style attribute
+								imageEl.removeAttr('style');
+							}
+							//put the original attributes back onto the parent anchor (if present)...
+							if(el.Link == 'Link'){
+								imageEl.parent().attr(el.Orig.a);
+								if(!el.Orig.a.style){ //if there was no inline style, might as well remove it
+									imageEl.parent().removeAttr('style');
+								}
+							}
+							//move the anchor/image back up to the original menu element...
+							Dock.Menu.append(el.Wrap.children());
+							//clear down...
+							imageEl = el.Label.el = el.Orig.i = el.Orig.a = null;
+							for(j in el){
+								el[j] = null;
+							}
+							el = null;
+						}
+						//remove the dock wrapper...
+						$('.jqDockWrap', Dock.Menu).remove();
+						//clear down...
+						for(i in Dock){
+							Dock[i] = null;
+						}
+						Dock = DOCKS[dockId] = null;
+					}
+					for(dockId = DOCKS.length; dockId-- && DOCKS[dockId] === null; ){}
+					if(dockId < 0){
+						DOCKS = [];
+					}
+				});
+		/***************************************************************************************************
+		* ALTERNATE5:
+		* Accepts 'active' or 'expand'. Chainable
+		* added v1.6
+		* 
+		* Example:
+		*   $('#menu img').eq(1).jqDock('active'); //set the 2nd image instantly to fully expanded
+		*   $('#menu img').last().jqDock('expand'); //animate the last image to fully expanded
+		***************************************************************************************************/
+		}else if(opts === 'active' || opts === 'expand'){
+			this.each(function(){
+					var found = FIND_IMAGE(this, 1) //...NB I want the indices (dock and element) returned instead of the Elem object
+						, Dock = found ? DOCKS[found[0]] : 0;
+					if(Dock){
+						//clear all timers...
+						CLEAR_TIMER(Dock, -1);
+						//if it wasn't already frozen, freeze it now and notify...
+						if(!Dock.Frozen){
+							Dock.Frozen = Dock.Asleep = !!Dock.Menu.trigger('docksleep', ['freeze']);
+						}
+						//fake a mouseenter, with animation dependent on value of opts ('active' = no animation)...
+						DO_MOUSE(0, found[0], found[1], opts == 'active' ? 2 : 1);
+					}
+				});
+		/***************************************************************************************************
+		* ALTERNATE3:
+		* Accepts 'get'. Not chainable, returns object/null
+		* 
+		* Example:
+		*   //to retrieve an item's original text used for the label (unmodified by setLabel option)...
+		*   var labelText = $('#menu img:eq(2)').jqDock('get').Title;
+		* Example:
+		*   //to retrieve a dock's options...
+		*   var options = $('#menu').jqDock('get').Opts;
+		***************************************************************************************************/
+		}else if(opts === 'get'){
+			var item = this.filter('.jqDocked');
+			//if we've got an active Dock, return that; otherwise, look for an image...
+			item = item.length ? DOCKS[DOCK_INDEX_FROM_ID($('.jqDock', item).get(0))] : FIND_IMAGE(this.get(0));
+			//since this is a getter, it does not support chaining and needs to cop out now
+			return item ? $.extend(true, {}, item) : null;
+		/***************************************************************************************************
+		* ALTERNATE:
+		* If a function is provided, it will be called with scope of the image DOM element, and 2 parameters:
+		* - current setting
+		* - settingType, eg. 'src' or 'altsrc'
+		*
+		* Example (with strings):
+		*   $('#menu img').eq(0).jqDock({src:'newpath.jpg', altsrc:'newexpanderpath.jpg'});
+		* Example (with functions):
+		*   fnChangePath = function(current, type){
+		*       //always change altsrc, but only change src if image has a class of 'changeExpanded'...
+		*       return type == 'altsrc' || $(this).hasClass('changeExpanded')
+		*         ? current.replace(/old\.png$/, 'new.png')
+		*         : current;
+		*     };
+		*   $('#menu img').jqDock({src:fnChangePath, altsrc:fnChangePath});
+		***************************************************************************************************/
+		}else if(this.length && !this.not('img').length){ //images only!!
+			this.each(function(n, el){
+					var item = FIND_IMAGE(el)
+						, src = 0
+						, atRest, str, v
+						;
+					opts = opts || {};
+					if(item){
+						atRest = item.Major == item.Initial;
+						for(v in {src:1, altsrc:1}){
+							if(opts[v]){
+								str = ($.isFunction(opts[v]) ? opts[v].call(el, item[v], v) : opts[v]).toString();
+								if(item[v] !== str){
+									item[v] = str;
+									src = (v == 'src' ? atRest : !atRest) ? v : src;
+								}
+							}
+						}
+						if(src){
+							$(el).attr('src', item[src]);
+						}
+					}
+				});
+		/***************************************************************************************************
+		* STANDARD:
+		* Chainable.
+		* 
+		* Example:
+		*   $('#menu').jqDock({align:'top'});
+		***************************************************************************************************/
+		}else{ //standard usage...
+			this.not('.jqDocked').filter(function(){
+					//check that no parents are already docked, and that all children are either images, or anchors containing only an image...
+					return !$(this).parents('.jqDocked').length && !$(this).children().not('img').filter(function(){
+							return $(this).filter('a').children('img').parent().children().length !== 1;
+						}).length;
+				}).addClass('jqDocked')
+				.each(function(){
+					var Self = $(this)
+						, id = DOCKS.length
+						, Dock, op, jqld, mc, i;
+					//add an object to the docks array for this new dock...
+					DOCKS[id] = { 
+							Elem : [] // an object per img menu option
+						, Menu : Self //jQuery of original containing element
+						, OnDock : 0 //indicates cursor over menu and initial sizes set
+						, Stamp : 0 //set on mouseenter/leave and used (within opts.duration) to proportion the menu element sizes
+						, width : 0 //width of div.jqDock container
+						, height : 0 //height of div.jqDock container
+						, Spread : 0 //main axis dimension (horizontal = width, vertical = height)
+						, Border : [] //border widths on div.jqDock, indexed as per TRBL
+						, Opts : $.extend({}, $.jqDock.defaults, opts||{}, $.metadata ? Self.metadata() : {}) //options; support metadata plugin
+						, Current : -1 //current image index
+						, Load : 0 //count of images to load
+						, ToFro : [ //a pain, but needed to prevent possible oscillation around a stationary cursor on the dock (see SET_SIZES)...
+								'' //previous-but-one, held as ... [ XY dimension, Dock.Current, newFinal dimension ].join(',')
+							, '' //previous, held as ... [ XY dimension, Dock.Current, newFinal dimension ].join(',')
+							, 0  //previous newFinal dimension
+							]
+/* these don't need to be explicitly set a this stage, either because their usage is by testing for [non]existence and
+ * then assigning a value, or because they are explicitly set during initDock()...
+						, Inactive : null //inactivity timer
+						, Idler : null //idle timer
+						, Indock : null //timer for IN_DOCK recursion
+						, Overdock : null //timer for OVER_DOCK recursion
+						, Offdock : null //timer for OFF_DOCK recursion
+						, Asleep : false //set to true when dock is put is to sleep following an idle period timeout
+						, Frozen : false //set to true when the dock is 'freeze'd (dock is also sent to sleep!)
+						, Sleeper : null //while Asleep, the most recent mouse event gets buffered for use on being nudged awake
+						, Yard : 0 //jQuery of div.jqDock
+*/
+						};
+					Dock = DOCKS[id]; //convenience
+					op = Dock.Opts; //convenience
+					//check some of the options...
+					jqld = (!op.loader && $.jqDock.useJqLoader) || op.loader === 'jquery';
+					for(i in {size:1, distance:1, duration:1, inactivity:1, fadeIn:1, step:1, idle:1, active:1}){
+						op[i] = AS_INTEGER(op[i]);
+					}
+					i = op.coefficient * 1;
+					op.coefficient = isNaN(i) ? 1.5 : i;
+					op.labels = (/^[tmb][lcr]$/).test(op.labels.toString()) ? op.labels : ( op.labels ? {top:'br',left:'tr'}[op.align] || 'tl' : '' );
+					op.setLabel = !!op.setLabel ? op.setLabel : TRANSFORM_LABEL;
+					op.fadeLayer = op.fadeIn ? (({dock:1,wrap:1}[op.fadeLayer]) ? op.fadeLayer : 'menu') : '';
+					for(i in {onSleep:1, onWake:1, onReady:1, onFreeze:1}){
+						if(!op[i]){
+							op[i] = EMPTYFUNC;
+						}
+					}
+					mc = (/^m|c$/).test(op.labels); //indicates the need for middle/centre label positioning information to be gathered
+					//set up some extra Opts now, just to save some computing power later...
+					op.attenuation = Math.pow(op.distance, op.coefficient); //straightforward, static calculation
+					op.vh = ({left:1, center:1, right:1}[op.align]) ? 'v' : 'h'; //vertical/horizontal orientation based on 'align' option
+
+					$('img', Self).each(function(n, el){
+							//add an object to the dock's elements array for each image...
+							var jself = $(el)
+								, linkParent = jself.parent('a')
+								, origAnchorTitle = linkParent.attr('title') || ''
+								, origImg = {}
+								, i;
+							for(i in {src:1, alt:1, title:1, style:1}){
+								origImg[i] = jself.attr(i) || '';
+							}
+							++Dock.Load;
+							Dock.Elem[n] = { 
+									Img : jself //jQuery of img element
+								, src : origImg.src  //image path, small
+								, altsrc: (op.source ? op.source.call(el, n) : '')  //image path, large
+									|| ((/\.(gif|jpg|jpeg|png)$/i).test(origImg.alt||'') ? origImg.alt : '') 
+									|| origImg.src
+								, Title : origImg.title || origAnchorTitle || '' //label text? (pre setLabel())
+								, Orig : {
+										i : $.extend({}, origImg)
+									, a : {title: origAnchorTitle, style:linkParent.attr('style') || ''}
+									}
+								, Label : {
+										mc: mc //if set, it gets removed the first time POSITION_LABEL is called for this label
+/* these don't need to be explicitly set at this stage: the first one is *always* set by SET_LABEL(); the other 2 are
+ * only set (and used) by POSITION_LABEL() *if* the labels are being positioned middle and/or center
+									, el: 0 //jqQuery of div.jqDockLabel
+									, v: 0 //the 'v' stands for vertical, so this is the label's overall height (ie. height + top/bottom padding)
+									, h: 0 //the 'h' stands for horizontal, so this is the label's overall width (ie. width + left/right padding)
+ */
+									}
+								, Pad : [] //user-applied padding, set up below and indexed as per TRBL
+								, Link : linkParent.length ? 'Link' : 'Image' //image-within-link or not
+/* these don't need to be explicitly set a this stage, either because their usage is by testing for [non]existence and
+ * then assigning a value, or because they are explicitly set during IMAGE_ONLOAD() or initDock()...
+								, width : 0 //original width of img element (the one that expands)
+								, height : 0 //original height of img element (the one that expands)
+								, Initial : 0 //width/height when fully shrunk; it's important to note that this is not necessarily the same as Opts.size!
+								, Major : 0 //transitory width/height (main axis)
+								, Minor : 0 //transitory width/height (minor axis)
+								, Final : 0 //target width/height
+								, Offset : 0 //offset of 'lead' edge of the image within div.jqDock (including user-padding)
+								, Centre : 0 //'Offset' + 'lead' user-padding + half 'Initial' dimension
+								, Wrap : 0 //jQuery of the menu element's immediate parent wrapper
+*/
+								};
+							for(i = 4; i--;){
+								Dock.Elem[n].Pad[i] = AS_INTEGER(jself.css('padding' + TRBL[i]));
+							}
+						});
+					//we have to run a 'loader' function for the images because the expanding image
+					//may not be part of the current DOM. what this means though, is that if you
+					//have a missing image in your dock, the entire dock will not be displayed!
+					//however I've had a few problems with certain browsers: for instance, IE does
+					//not like the jQuery method; and Opera was causing me problems with the native
+					//method when reloading the page; I've also heard rumours that Safari 2 might cope better with
+					//the jQuery method, but I cannot confirm since I no longer have Safari 2.
+					//
+					//anyway, I'm providing both methods. if anyone finds it doesn't work, try
+					//overriding with option.loader, and/or changing $.jqDock.useJqLoader for the 
+					//browser in question and let me know if that solves it.
+					$.each(Dock.Elem, function(i, v){
+							var pre, altsrc = v.altsrc;
+							if(jqld){ //jQuery method...
+								$('<img>').bind('load', {id:id, idx:i}, IMAGE_ONLOAD).attr({src:altsrc});
+							}else{ //native 'new Image()' method...
+								pre = new Image();
+								pre.onload = function(){
+										IMAGE_ONLOAD.call(this, {data:{id:id, idx:i}});
+										pre.onload = ''; //wipe out this onload function
+										pre = null;
+									};
+								pre.src = altsrc;
+							}
+						});
+				});
+		}
+		return this;
+	}; //end jQuery.fn.jqDock()
+} //end of if()
+})(jQuery, window);