////////////////////////////////////////////////////////////////////////////////// // CloudCarousel V1.0.5 // (c) 2011 by R Cecco. // MIT License // // Reflection code based on plugin by Christophe Beyls // // Please retain this copyright header in all versions of the software ////////////////////////////////////////////////////////////////////////////////// var matched, browser; jQuery.uaMatch = function( ua ) { ua = ua.toLowerCase(); var match = /(chrome)[ \/]([\w.]+)/.exec( ua ) || /(webkit)[ \/]([\w.]+)/.exec( ua ) || /(opera)(?:.*version|)[ \/]([\w.]+)/.exec( ua ) || /(msie) ([\w.]+)/.exec( ua ) || ua.indexOf("compatible") < 0 && /(mozilla)(?:.*? rv:([\w.]+)|)/.exec( ua ) || []; return { browser: match[ 1 ] || "", version: match[ 2 ] || "0" }; }; matched = jQuery.uaMatch( navigator.userAgent ); browser = {}; if ( matched.browser ) { browser[ matched.browser ] = true; browser.version = matched.version; } // Chrome is Webkit, but Webkit is also Safari. if ( browser.chrome ) { browser.webkit = true; } else if ( browser.webkit ) { browser.safari = true; } jQuery.browser = browser; (function($) { // START Reflection object. // Creates a reflection for underneath an image. // IE uses an image with IE specific filter properties, other browsers use the Canvas tag. // The position and size of the reflection gets updated by updateAll() in Controller. function Reflection(img, reflHeight, opacity) { var reflection, cntx, imageWidth = img.width, imageHeight = img.width, gradient, parent; parent = $(img.parentNode); this.element = reflection = parent.append("").find(':last')[0]; if ( !reflection.getContext && $.browser.msie) { this.element = reflection = parent.append("").find(':last')[0]; reflection.src = img.src; reflection.style.filter = "flipv progid:DXImageTransform.Microsoft.Alpha(opacity=" + (opacity * 100) + ", style=1, finishOpacity=0, startx=0, starty=0, finishx=0, finishy=" + (reflHeight / imageHeight * 100) + ")"; } else { cntx = reflection.getContext("2d"); try { $(reflection).attr({width: imageWidth, height: reflHeight}); cntx.save(); cntx.translate(0, imageHeight-1); cntx.scale(1, -1); cntx.drawImage(img, 0, 0, imageWidth, imageHeight); cntx.restore(); cntx.globalCompositeOperation = "destination-out"; gradient = cntx.createLinearGradient(0, 0, 0, reflHeight); gradient.addColorStop(0, "rgba(255, 255, 255, " + (1 - opacity) + ")"); gradient.addColorStop(1, "rgba(255, 255, 255, 1.0)"); cntx.fillStyle = gradient; cntx.fillRect(0, 0, imageWidth, reflHeight); } catch(e) { return; } } // Store a copy of the alt and title attrs into the reflection $(reflection).attr({ 'alt': $(img).attr('alt'), title: $(img).attr('title')} ); } //END Reflection object // START Item object. // A wrapper object for items within the carousel. var Item = function(imgIn, options) { this.orgWidth = imgIn.width; this.orgHeight = imgIn.height; this.image = imgIn; this.reflection = null; this.alt = imgIn.alt; this.title = imgIn.title; this.imageOK = false; this.options = options; this.imageOK = true; if (this.options.reflHeight > 0) { this.reflection = new Reflection(this.image, this.options.reflHeight, this.options.reflOpacity); } $(this.image).css('position','absolute'); // Bizarre. This seems to reset image width to 0 on webkit! };// END Item object // Controller object. // This handles moving all the items, dealing with mouse clicks etc. var Controller = function(container, images, options) { var items = [], funcSin = Math.sin, funcCos = Math.cos, ctx=this; this.controlTimer = 0; this.stopped = false; //this.imagesLoaded = 0; this.container = container; this.xRadius = options.xRadius; this.yRadius = options.yRadius; this.showFrontTextTimer = 0; this.autoRotateTimer = 0; if (options.xRadius === 0) { this.xRadius = ($(container).width()/2.3); } if (options.yRadius === 0) { this.yRadius = ($(container).height()/6); } this.xCentre = options.xPos; this.yCentre = options.yPos; this.frontIndex = 0; // Index of the item at the front // Start with the first item at the front. this.rotation = this.destRotation = Math.PI/2; this.timeDelay = 1000/options.FPS; // Turn on the infoBox if(options.altBox !== null) { $(options.altBox).css('display','block'); $(options.titleBox).css('display','block'); } // Turn on relative position for container to allow absolutely positioned elements // within it to work. $(container).css({ position:'relative', overflow:'hidden'} ); $(options.buttonLeft).css('display','inline'); $(options.buttonRight).css('display','inline'); // Setup the buttons. $(options.buttonLeft).bind('mouseup',this,function(event){ event.data.rotate(-1); return false; }); $(options.buttonRight).bind('mouseup',this,function(event){ event.data.rotate(1); return false; }); // START METAMAPS CODE // Add code that makes tab and shift+tab scroll through metacodes $('.new_topic').bind('keydown',this,function(event){ if (event.keyCode == 9 && event.shiftKey) { $(container).show() event.data.rotate(-1); event.preventDefault(); event.stopPropagation(); } else if (event.keyCode == 9) { $(container).show() event.data.rotate(1); event.preventDefault(); event.stopPropagation(); } }); // END METAMAPS CODE // You will need this plugin for the mousewheel to work: http://plugins.jquery.com/project/mousewheel if (options.mouseWheel) { // START METAMAPS CODE /*$('body').bind('mousewheel',this,function(event, delta) { if (Metamaps.Create.newTopic.beingCreated && !Metamaps.Create.isSwitchingSet && !Metamaps.Create.newTopic.pinned) { event.data.rotate(delta); return false; } });*/ // END METAMAPS CODE // ORIGINAL CODE // $(container).bind('mousewheel',this,function(event, delta) { // event.data.rotate(delta); // return false; // }); // } $(container).unbind('mouseover click').bind('mouseover click',this,function(event){ clearInterval(event.data.autoRotateTimer); // Stop auto rotation if mouse over. var text = $(event.target).attr('alt'); // If we have moved over a carousel item, then show the alt and title text. if ( text !== undefined && text !== null ) { clearTimeout(event.data.showFrontTextTimer); $(options.altBox).html( ($(event.target).attr('alt') )); //$(options.titleBox).html( ($(event.target).attr('title') )); if ( options.bringToFront && event.type == 'click' ) { $(options.titleBox).html( ($(event.target).attr('title') )); // START METAMAPS CODE Metamaps.Create.newTopic.metacode = $(event.target).attr('data-id'); // END METAMAPS CODE var idx = $(event.target).data('itemIndex'); var frontIndex = event.data.frontIndex; //var diff = idx - frontIndex; var diff = (idx - frontIndex) % images.length; if (Math.abs(diff) > images.length / 2) { diff += (diff > 0 ? -images.length : images.length); } event.data.rotate(-diff); } } }); // START METAMAPS CODE - initialize newTopic.metacode var first = $(this.container).find('img').get(0) Metamaps.Create.newTopic.metacode = $(first).data('id') // END METAMAPS CODE // If we have moved out of a carousel item (or the container itself), // restore the text of the front item in 1 second. $(container).bind('mouseout',this,function(event){ var context = event.data; clearTimeout(context.showFrontTextTimer); context.showFrontTextTimer = setTimeout( function(){context.showFrontText();},1000); context.autoRotate(); // Start auto rotation. }); // Prevent items from being selected as mouse is moved and clicked in the container. $(container).bind('mousedown',this,function(event){ event.data.container.focus(); return false; }); container.onselectstart = function () { return false; }; // For IE. this.innerWrapper = $(container).wrapInner('
').children()[0]; // Shows the text from the front most item. this.showFrontText = function() { if ( items[this.frontIndex] === undefined ) { return; } // Images might not have loaded yet. // METAMAPS CODE Metamaps.Create.newTopic.setMetacode($(items[this.frontIndex].image).attr('data-id')) // NOT METAMAPS CODE //$(options.titleBox).html( $(items[this.frontIndex].image).attr('title')); //$(options.altBox).html( $(items[this.frontIndex].image).attr('alt')); }; this.go = function() { if(this.controlTimer !== 0) { return; } var context = this; this.controlTimer = setTimeout( function(){context.updateAll();},this.timeDelay); }; this.stop = function() { clearTimeout(this.controlTimer); this.controlTimer = 0; // METAMAPS CODE $(container).hide() // END METAMAPS CODE }; // Starts the rotation of the carousel. Direction is the number (+-) of carousel items to rotate by. this.rotate = function(direction) { this.frontIndex -= direction; if (this.frontIndex == -1) this.frontIndex = items.length - 1; this.frontIndex %= items.length; this.destRotation += ( Math.PI / items.length ) * ( 2*direction ); this.showFrontText(); this.go(); }; this.autoRotate = function() { if ( options.autoRotate !== 'no' ) { var dir = (options.autoRotate === 'right')? 1 : -1; this.autoRotateTimer = setInterval( function(){ctx.rotate(dir); }, options.autoRotateDelay ); } }; // This is the main loop function that moves everything. this.updateAll = function() { var minScale = options.minScale; // This is the smallest scale applied to the furthest item. var smallRange = (1-minScale) * 0.5; var w,h,x,y,scale,item,sinVal; var change = (this.destRotation - this.rotation); var absChange = Math.abs(change); this.rotation += change * options.speed; if ( absChange < 0.001 ) { this.rotation = this.destRotation; } var itemsLen = items.length; var spacing = (Math.PI / itemsLen) * 2; //var wrapStyle = null; var radians = this.rotation; var isMSIE = $.browser.msie; // Turn off display. This can reduce repaints/reflows when making style and position changes in the loop. // See http://dev.opera.com/articles/view/efficient-javascript/?page=3 this.innerWrapper.style.display = 'none'; var style; var px = 'px', reflHeight; var context = this; for (var i = 0; i>0; // >>0 = Math.foor(). Firefox doesn't like fractional decimals in z-index. w = img.width = item.orgWidth * scale; h = img.height = item.orgHeight * scale; img.style.left = x + px ; img.style.top = y + px; if (item.reflection !== null) { reflHeight = options.reflHeight * scale; style = item.reflection.element.style; style.left = x + px; style.top = y + h + options.reflGap * scale + px; style.width = w + px; if (isMSIE) { style.filter.finishy = (reflHeight / h * 100); }else { style.height = reflHeight + px; } } } radians += spacing; } // Turn display back on. this.innerWrapper.style.display = 'block'; // If we have a preceptable change in rotation then loop again next frame. if ( absChange >= 0.001 ) { this.controlTimer = setTimeout( function(){context.updateAll();},this.timeDelay); }else { // Otherwise just stop completely. this.stop(); } }; // END updateAll // Create an Item object for each image // func = function(){return;ctx.updateAll();} ; // Check if images have loaded. We need valid widths and heights for the reflections. this.checkImagesLoaded = function() { var i; for(i=0;i