
// The Offset Method
// Originally By Brandon Aaron, part of the Dimension Plugin
// http://jquery.com/plugins/project/dimensions
jQuery.fn.offset = function() {
	var left = 0, top = 0, elem = this[0], results;
	
	if ( elem ) with ( jQuery.browser ) {
		var parent       = elem.parentNode, 
		    offsetChild  = elem,
		    offsetParent = elem.offsetParent, 
		    doc          = elem.ownerDocument,
		    safari2      = safari && parseInt(version) < 522,
		    fixed        = jQuery.css(elem, "position") == "fixed";
	
		// Use getBoundingClientRect if available
		if ( elem.getBoundingClientRect ) {
			var box = elem.getBoundingClientRect();
		
			// Add the document scroll offsets
			add(box.left + Math.max(doc.documentElement.scrollLeft, doc.body.scrollLeft),
				box.top  + Math.max(doc.documentElement.scrollTop,  doc.body.scrollTop));
		
			// IE adds the HTML element's border, by default it is medium which is 2px
			// IE 6 and 7 quirks mode the border width is overwritable by the following css html { border: 0; }
			// IE 7 standards mode, the border is always 2px
			// This border/offset is typically represented by the clientLeft and clientTop properties
			// However, in IE6 and 7 quirks mode the clientLeft and clientTop properties are not updated when overwriting it via CSS
			// Therefore this method will be off by 2px in IE while in quirksmode
			add( -doc.documentElement.clientLeft, -doc.documentElement.clientTop );
	
		// Otherwise loop through the offsetParents and parentNodes
		} else {
		
			// Initial element offsets
			add( elem.offsetLeft, elem.offsetTop );
			
			// Get parent offsets
			while ( offsetParent ) {
				// Add offsetParent offsets
				add( offsetParent.offsetLeft, offsetParent.offsetTop );
			
				// Mozilla and Safari > 2 does not include the border on offset parents
				// However Mozilla adds the border for table or table cells
				if ( mozilla && !/^t(able|d|h)$/i.test(offsetParent.tagName) || safari && !safari2 )
					border( offsetParent );
					
				// Add the document scroll offsets if position is fixed on any offsetParent
				if ( !fixed && jQuery.css(offsetParent, "position") == "fixed" )
					fixed = true;
			
				// Set offsetChild to previous offsetParent unless it is the body element
				offsetChild  = /^body$/i.test(offsetParent.tagName) ? offsetChild : offsetParent;
				// Get next offsetParent
				offsetParent = offsetParent.offsetParent;
			}
		
			// Get parent scroll offsets
			while ( parent && parent.tagName && !/^body|html$/i.test(parent.tagName) ) {
				// Remove parent scroll UNLESS that parent is inline or a table to work around Opera inline/table scrollLeft/Top bug
				if ( !/^inline|table.*$/i.test(jQuery.css(parent, "display")) )
					// Subtract parent scroll offsets
					add( -parent.scrollLeft, -parent.scrollTop );
			
				// Mozilla does not add the border for a parent that has overflow != visible
				if ( mozilla && jQuery.css(parent, "overflow") != "visible" )
					border( parent );
			
				// Get next parent
				parent = parent.parentNode;
			}
		
			// Safari <= 2 doubles body offsets with a fixed position element/offsetParent or absolutely positioned offsetChild
			// Mozilla doubles body offsets with a non-absolutely positioned offsetChild
			if ( (safari2 && (fixed || jQuery.css(offsetChild, "position") == "absolute")) || 
				(mozilla && jQuery.css(offsetChild, "position") != "absolute") )
					add( -doc.body.offsetLeft, -doc.body.offsetTop );
			
			// Add the document scroll offsets if position is fixed
			if ( fixed )
				add(Math.max(doc.documentElement.scrollLeft, doc.body.scrollLeft),
					Math.max(doc.documentElement.scrollTop,  doc.body.scrollTop));
		}

		// Return an object with top and left properties
		results = { top: top, left: left };
	}

	function border(elem) {
		add( jQuery.curCSS(elem, "borderLeftWidth", true), jQuery.curCSS(elem, "borderTopWidth", true) );
	}

	function add(l, t) {
		left += parseInt(l) || 0;
		top += parseInt(t) || 0;
	}

	return results;
};

// Newsticker
jQuery.fn.ticker = function(settings) {
	settings = jQuery.extend(
		{
			travelocity: 0.01
		},
		settings
	);

	return this.each(
		function(){
			var tickerWidth = 20;
			var tickerList = jQuery( this );
			
			jQuery( '#newsticker' ).css( 'position', 'relative' );

			// div.mask um ul packen
			tickerList.wrap( '<div class="mask"></div>' );
			
			var container = jQuery( document.createElement( 'div' ) )
				.attr( 'id', tickerList.parent().parent().attr( 'id' ) + '-container' )
				.css( 'overflow', 'hidden' );
			
			tickerList.parent().wrap( container );
			
			var tickerParentWidth = tickerList.parent().parent().width();
			
			// warum auch immer, der ie moechte noch eine node hoeher gehen..
			if ( jQuery.browser.msie ) {
				tickerParentWidth = tickerList.parent().parent().parent().width();
			}

			tickerList
				.width( tickerList.parent().width() )
				.css({
					'margin': '0px',
					'list-style': 'none'
				})
				.parent()
				.css({
					'margin': '0px',
					'position': 'absolute',
					'white-space': 'nowrap',
					'left': tickerParentWidth,
					'padding': ' 0px',
					'position': 'absolute',
					'height': tickerList.height(),
					'padding': tickerList.css( 'padding-top' ) + 'px 0px ' + tickerList.css( 'padding-bottom' ) + 'px 0px',
					'margin': tickerList.css( 'margin-top' ) + 'px 0px ' + tickerList.css( 'margin-bottom' ) + 'px 0px'
				} )
				.parent()
				.css({
					'position': 'relative',
					'height': tickerList.height()
				});
			
			tickerList
				.find( 'li' )
				.css( 'height', tickerList.height() )
				.find( 'span' )
				.each(function( i ){
					if ( typeof this == 'object' ) {
						tickerWidth += parseInt( jQuery( this, i ).css({ 'float': 'left', 'padding-right': '10px' }).width() ) + 20;
					}
				})
				.end()
				.bind( 'mouseenter', function(){
					jQuery( this ).stop();
				})
				.bind( 'mouseleave', function(){
					var offset = jQuery( this ).offset();
					var residualSpace = offset.left + tickerList.parent().width();
					var residualTime = residualSpace / settings.travelocity;
			
					newsticker_scroll( residualSpace, residualTime );
				})
				.each(function( i ){
					if ( typeof this == 'object' ) {
						jQuery( this, i ).width( tickerWidth );
					}
				});
			
			var totalWidth = tickerWidth + tickerParentWidth;

			function newsticker_scroll( ){
				var tickerParentList = tickerList.parent();
				var left = parseInt( tickerParentList.css( 'left' ), 10 );
				
				if ( left < tickerWidth * -1 ) {
					left = tickerList.width();
				}
				
				window.setTimeout(function(){
					tickerParentList.css( 'left', left - 2 )
					newsticker_scroll();
				}, 50 );
			}
			
			tickerList
				.css( 'width', totalWidth )
				.css( 'visibility', 'visible' );
				
			newsticker_scroll( );
		}
	);
};
