<component lightWeight="true">
<attach event="onpropertychange" onevent="handlePropertychange()" />
<attach event="ondetach" onevent="restore()" />
<attach event="onresize" for="window" onevent="handleResize()" />
<script type="text/javascript">

var rsrc = /url\(["']?(.*?)["']?\)/,
	positions = {
		top: 0,
		left: 0,
		bottom: 1,
		right: 1,
		center: .5
	},
	doc = element.document;

init();

// remove the background-image and emulate it with a wrapped <img/>
function init() {
	var wrapper = doc.createElement( "div" ),
		img = doc.createElement( "img" ),
		expando,
		pos;

	wrapper.style.position = "absolute";
	wrapper.style.zIndex = -1;
	wrapper.style.top = 0;
	wrapper.style.right = 0;
	wrapper.style.left = 0;
	wrapper.style.bottom = 0;
	wrapper.style.overflow = "hidden";

	img.style.position = "absolute";
	img.style.width = img.style.width = "auto";

	wrapper.appendChild( img );

	element.insertBefore( wrapper, element.firstChild );

	pos = [
		element.currentStyle.backgroundPositionX,
		element.currentStyle.backgroundPositionY
	];

	// save useful data for quick access
	element.bgsExpando = expando = {
		wrapper: wrapper,
		img: img,
		backgroundSize: element.currentStyle['background-size'],
		// Only keywords or percentage values are supported
		backgroundPositionX: positions[ pos[0] ] || parseFloat( pos[0] ) / 100,
		backgroundPositionY: positions[ pos[1] ] || parseFloat( pos[1] ) / 100
	};

	// This is the part where we mess with the existing DOM
	// to make sure that the background image is correctly zIndexed
	if ( element.currentStyle.zIndex == "auto" ) {
		element.style.zIndex = 0;
	}
	if ( element.currentStyle.position == "static" ) {
		element.style.position = "relative";
	}

	if ( refreshDisplay( element, expando ) ) {
		refreshDimensions( element, expando );
		refreshBackgroundImage( element, expando, function() {
			updateBackground( element, expando );
		});
	}
}

function refreshDisplay( element, expando ) {
	var display = element.currentStyle.display;

	if ( display != expando.display ) {
		expando.display = display;
		expando.somethingChanged = true;
	}

	return display != "none";
}

function refreshDimensions( element, expando ) {
	var innerWidth = element.offsetWidth
			- ( parseFloat( element.currentStyle.borderLeftWidth ) || 0 )
			- ( parseFloat( element.currentStyle.borderRightWidth ) || 0 ),
		innerHeight = element.offsetHeight
			- ( parseFloat( element.currentStyle.borderTopWidth ) || 0 )
			- ( parseFloat( element.currentStyle.borderBottomWidth ) || 0 );

	if ( innerWidth != expando.innerWidth || innerHeight != expando.innerHeight ) {
		expando.innerWidth = innerWidth;
		expando.innerHeight = innerHeight;
		expando.somethingChanged = true;
	}
}

function refreshBackgroundImage( element, expando, callback ) {
	var img = expando.img,
		src = ( rsrc.exec( element.currentStyle.backgroundImage ) || [] )[1];

	if ( src && src != expando.backgroundSrc ) {
		expando.backgroundSrc = src;
		expando.somethingChanged = true;

		img.onload = function() {
			var width = img.width,
				height = img.height;

			// ignore onload on the proxy image
			if ( width == 1 && height == 1 ) { return; }

			expando.imgWidth = width;
			expando.imgHeight = height;
			expando.constrain = false;

			callback();

			img.style.visibility = "visible";
			img.onload = null;
		};

		img.style.visibility = "hidden";
		img.src = expando.backgroundSrc;

		// force onload execution
		if ( img.readyState || img.complete ) {
			img.src = "";
			img.src = expando.backgroundSrc;
		}

		expando.ignoreNextPropertyChange = true;
		element.style.backgroundImage = "none";
	
	} else {
		// the callback should be exacuted in all cases
		callback();
	}
}

function updateBackground( element, expando ) {
	if ( !expando.somethingChanged ) { return; }

	var img = expando.img,
		elemRatio = expando.innerWidth / expando.innerHeight,
		imgRatio = expando.imgWidth / expando.imgHeight,
		prevConstrain = expando.constrain,
		curConstrain,
		delta;

	if ( expando.backgroundSize == "contain" ) {
		if ( imgRatio > elemRatio ) {
			expando.constrain = curConstrain = "width";

			delta = Math.floor(
				( expando.innerHeight - expando.innerWidth / imgRatio )
				* expando.backgroundPositionY
			);

			img.style.top = delta + "px";

			// when switching from height to width constraint,
			// make sure to release constraint on height and reset left
			if ( curConstrain != prevConstrain ) {
				img.style.width = "100%";
				img.style.height = "auto";
				img.style.left = 0;
			}
		
		// elemRatio > imgRatio
		} else {
			expando.constrain = curConstrain = "height";

			delta = Math.floor(
				( expando.innerWidth - expando.innerHeight * imgRatio )
				* expando.backgroundPositionX
			);

			img.style.left = delta + "px";

			if ( curConstrain != prevConstrain ) {
				img.style.width = "auto";
				img.style.height = "100%";
				img.style.top = 0;
			}
		}

	} else if ( expando.backgroundSize == "cover" ) {
		if ( imgRatio > elemRatio ) {
			expando.constrain = curConstrain = "height";

			delta = Math.floor(
				( expando.innerHeight * imgRatio - expando.innerWidth )
				* expando.backgroundPositionX
			);

			img.style.left = -delta + "px";

			// when switching from height to width constraint,
			// make sure to release constraint on height and reset left
			if ( curConstrain != prevConstrain ) {
				img.style.width = "auto";
				img.style.height = "100%";
				img.style.top = 0;
			}
		
		// elemRatio > imgRatio
		} else {
			expando.constrain = curConstrain = "width";

			delta = Math.floor(
				( expando.innerWidth / imgRatio - expando.innerHeight )
				* expando.backgroundPositionY
			);

			img.style.top = -delta + "px";

			if ( curConstrain != prevConstrain ) {
				img.style.width = "100%";
				img.style.height = "auto";
				img.style.left = 0;
			}
		}
	}

	expando.somethingChanged = false;
}

// handle different style changes
function handlePropertychange() {
	var expando = element.bgsExpando;

	// this prevents handling propertychange events caused by this script
	// TODO: make it reliable
	if ( expando.ignoreNextPropertyChange ) {
		expando.ignoreNextPropertyChange = false;
		return;
	}

	if ( refreshDisplay( element, expando ) ) {
		refreshDimensions( element, expando );
		refreshBackgroundImage( element, expando, function() {
			updateBackground( element, expando );
		});
	}
}

function handleResize() {
	// TODO: throttle resize events

	var expando = element.bgsExpando;

	if ( expando.display != "none" ) {
		refreshDimensions( element, expando );
		updateBackground( element, expando );
	}
}

function restore() {
	var expando = element.bgsExpando;

	try {
		element.style.backgroundImage = "url('" + expando.backgroundSrc + "')";
		element.removeChild( expando.wrapper );
		element.bgsExpando = null;
	} catch(e) {}
}

</script>