2014-04-17 05:54:21 +08:00
|
|
|
define([
|
|
|
|
"./core",
|
|
|
|
"./var/strundefined",
|
|
|
|
"./core/access",
|
|
|
|
"./css/var/rnumnonpx",
|
|
|
|
"./css/curCSS",
|
|
|
|
"./css/addGetHookIf",
|
|
|
|
"./css/support",
|
|
|
|
|
|
|
|
"./core/init",
|
|
|
|
"./css",
|
|
|
|
"./selector" // contains
|
|
|
|
], function( jQuery, strundefined, access, rnumnonpx, curCSS, addGetHookIf, support ) {
|
|
|
|
|
|
|
|
var docElem = window.document.documentElement;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a window from an element
|
|
|
|
*/
|
|
|
|
function getWindow( elem ) {
|
2014-05-17 23:27:45 +08:00
|
|
|
return jQuery.isWindow( elem ) ? elem : elem.nodeType === 9 && elem.defaultView;
|
2014-04-17 05:54:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
jQuery.offset = {
|
|
|
|
setOffset: function( elem, options, i ) {
|
|
|
|
var curPosition, curLeft, curCSSTop, curTop, curOffset, curCSSLeft, calculatePosition,
|
|
|
|
position = jQuery.css( elem, "position" ),
|
|
|
|
curElem = jQuery( elem ),
|
|
|
|
props = {};
|
|
|
|
|
2014-05-17 23:27:45 +08:00
|
|
|
// Set position first, in-case top/left are set even on static elem
|
2014-04-17 05:54:21 +08:00
|
|
|
if ( position === "static" ) {
|
|
|
|
elem.style.position = "relative";
|
|
|
|
}
|
|
|
|
|
|
|
|
curOffset = curElem.offset();
|
|
|
|
curCSSTop = jQuery.css( elem, "top" );
|
|
|
|
curCSSLeft = jQuery.css( elem, "left" );
|
|
|
|
calculatePosition = ( position === "absolute" || position === "fixed" ) &&
|
2014-05-17 23:27:45 +08:00
|
|
|
( curCSSTop + curCSSLeft ).indexOf("auto") > -1;
|
2014-04-17 05:54:21 +08:00
|
|
|
|
2014-05-17 23:27:45 +08:00
|
|
|
// Need to be able to calculate position if either top or left is auto and position is either absolute or fixed
|
2014-04-17 05:54:21 +08:00
|
|
|
if ( calculatePosition ) {
|
|
|
|
curPosition = curElem.position();
|
|
|
|
curTop = curPosition.top;
|
|
|
|
curLeft = curPosition.left;
|
2014-05-17 23:27:45 +08:00
|
|
|
|
2014-04-17 05:54:21 +08:00
|
|
|
} else {
|
|
|
|
curTop = parseFloat( curCSSTop ) || 0;
|
|
|
|
curLeft = parseFloat( curCSSLeft ) || 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( jQuery.isFunction( options ) ) {
|
|
|
|
options = options.call( elem, i, curOffset );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( options.top != null ) {
|
|
|
|
props.top = ( options.top - curOffset.top ) + curTop;
|
|
|
|
}
|
|
|
|
if ( options.left != null ) {
|
|
|
|
props.left = ( options.left - curOffset.left ) + curLeft;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( "using" in options ) {
|
|
|
|
options.using.call( elem, props );
|
2014-05-17 23:27:45 +08:00
|
|
|
|
2014-04-17 05:54:21 +08:00
|
|
|
} else {
|
|
|
|
curElem.css( props );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
jQuery.fn.extend({
|
|
|
|
offset: function( options ) {
|
|
|
|
if ( arguments.length ) {
|
|
|
|
return options === undefined ?
|
|
|
|
this :
|
|
|
|
this.each(function( i ) {
|
|
|
|
jQuery.offset.setOffset( this, options, i );
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
var docElem, win,
|
|
|
|
elem = this[ 0 ],
|
2014-05-17 23:27:45 +08:00
|
|
|
box = { top: 0, left: 0 },
|
2014-04-17 05:54:21 +08:00
|
|
|
doc = elem && elem.ownerDocument;
|
|
|
|
|
|
|
|
if ( !doc ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
docElem = doc.documentElement;
|
|
|
|
|
|
|
|
// Make sure it's not a disconnected DOM node
|
|
|
|
if ( !jQuery.contains( docElem, elem ) ) {
|
|
|
|
return box;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we don't have gBCR, just use 0,0 rather than error
|
|
|
|
// BlackBerry 5, iOS 3 (original iPhone)
|
|
|
|
if ( typeof elem.getBoundingClientRect !== strundefined ) {
|
|
|
|
box = elem.getBoundingClientRect();
|
|
|
|
}
|
|
|
|
win = getWindow( doc );
|
|
|
|
return {
|
2014-05-17 23:27:45 +08:00
|
|
|
top: box.top + win.pageYOffset - docElem.clientTop,
|
|
|
|
left: box.left + win.pageXOffset - docElem.clientLeft
|
2014-04-17 05:54:21 +08:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
position: function() {
|
|
|
|
if ( !this[ 0 ] ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var offsetParent, offset,
|
2014-05-17 23:27:45 +08:00
|
|
|
elem = this[ 0 ],
|
|
|
|
parentOffset = { top: 0, left: 0 };
|
2014-04-17 05:54:21 +08:00
|
|
|
|
2014-05-17 23:27:45 +08:00
|
|
|
// Fixed elements are offset from window (parentOffset = {top:0, left: 0}, because it is its only offset parent
|
2014-04-17 05:54:21 +08:00
|
|
|
if ( jQuery.css( elem, "position" ) === "fixed" ) {
|
2014-05-17 23:27:45 +08:00
|
|
|
// We assume that getBoundingClientRect is available when computed position is fixed
|
2014-04-17 05:54:21 +08:00
|
|
|
offset = elem.getBoundingClientRect();
|
2014-05-17 23:27:45 +08:00
|
|
|
|
2014-04-17 05:54:21 +08:00
|
|
|
} else {
|
|
|
|
// Get *real* offsetParent
|
|
|
|
offsetParent = this.offsetParent();
|
|
|
|
|
|
|
|
// Get correct offsets
|
|
|
|
offset = this.offset();
|
|
|
|
if ( !jQuery.nodeName( offsetParent[ 0 ], "html" ) ) {
|
|
|
|
parentOffset = offsetParent.offset();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add offsetParent borders
|
2014-05-17 23:27:45 +08:00
|
|
|
parentOffset.top += jQuery.css( offsetParent[ 0 ], "borderTopWidth", true );
|
2014-04-17 05:54:21 +08:00
|
|
|
parentOffset.left += jQuery.css( offsetParent[ 0 ], "borderLeftWidth", true );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Subtract parent offsets and element margins
|
|
|
|
return {
|
2014-05-17 23:27:45 +08:00
|
|
|
top: offset.top - parentOffset.top - jQuery.css( elem, "marginTop", true ),
|
|
|
|
left: offset.left - parentOffset.left - jQuery.css( elem, "marginLeft", true )
|
2014-04-17 05:54:21 +08:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
offsetParent: function() {
|
|
|
|
return this.map(function() {
|
|
|
|
var offsetParent = this.offsetParent || docElem;
|
|
|
|
|
|
|
|
while ( offsetParent && ( !jQuery.nodeName( offsetParent, "html" ) && jQuery.css( offsetParent, "position" ) === "static" ) ) {
|
|
|
|
offsetParent = offsetParent.offsetParent;
|
|
|
|
}
|
2014-05-17 23:27:45 +08:00
|
|
|
|
2014-04-17 05:54:21 +08:00
|
|
|
return offsetParent || docElem;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Create scrollLeft and scrollTop methods
|
|
|
|
jQuery.each( { scrollLeft: "pageXOffset", scrollTop: "pageYOffset" }, function( method, prop ) {
|
2014-05-17 23:27:45 +08:00
|
|
|
var top = "pageYOffset" === prop;
|
2014-04-17 05:54:21 +08:00
|
|
|
|
|
|
|
jQuery.fn[ method ] = function( val ) {
|
|
|
|
return access( this, function( elem, method, val ) {
|
|
|
|
var win = getWindow( elem );
|
|
|
|
|
|
|
|
if ( val === undefined ) {
|
2014-05-17 23:27:45 +08:00
|
|
|
return win ? win[ prop ] : elem[ method ];
|
2014-04-17 05:54:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( win ) {
|
|
|
|
win.scrollTo(
|
2014-05-17 23:27:45 +08:00
|
|
|
!top ? val : window.pageXOffset,
|
|
|
|
top ? val : window.pageYOffset
|
2014-04-17 05:54:21 +08:00
|
|
|
);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
elem[ method ] = val;
|
|
|
|
}
|
|
|
|
}, method, val, arguments.length, null );
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
// Add the top/left cssHooks using jQuery.fn.position
|
|
|
|
// Webkit bug: https://bugs.webkit.org/show_bug.cgi?id=29084
|
|
|
|
// getComputedStyle returns percent when specified for top/left/bottom/right
|
|
|
|
// rather than make the css module depend on the offset module, we just check for it here
|
|
|
|
jQuery.each( [ "top", "left" ], function( i, prop ) {
|
|
|
|
jQuery.cssHooks[ prop ] = addGetHookIf( support.pixelPosition,
|
|
|
|
function( elem, computed ) {
|
|
|
|
if ( computed ) {
|
|
|
|
computed = curCSS( elem, prop );
|
|
|
|
// if curCSS returns percentage, fallback to offset
|
|
|
|
return rnumnonpx.test( computed ) ?
|
|
|
|
jQuery( elem ).position()[ prop ] + "px" :
|
|
|
|
computed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
return jQuery;
|
|
|
|
});
|