996 lines
30 KiB
996 lines
30 KiB
/** |
|
* @author elmasse (Maximiliano Fierro) |
|
* @version 0.1 beta |
|
* |
|
* @usage: |
|
* <code> |
|
* var images = [ |
|
* {src: 'images/ardillitaMac.jpg'}, |
|
* {src: 'http://farm2.static.flickr.com/1380/1426855399_b4b8eccbdb.jpg?v=0'}, |
|
* {src: 'http://farm1.static.flickr.com/69/213130158_0d1aa23576_d.jpg'} |
|
* ]; |
|
* var myCoverFlow = new YAHOO.ext.CoverFlow('coverFlowTest', {height: 200, width: 800, images: images, bgColor: '#C0C0C0'}); |
|
* </code> |
|
* |
|
*/ |
|
|
|
|
|
YAHOO.namespace("ext"); |
|
|
|
//(function(){ |
|
|
|
/** |
|
* @class CoverFlow |
|
* @namespace YAHOO.ext |
|
* @constructor |
|
* @param el {String|HTMLElement} Reference to element where CoverFlow will be rendered. |
|
* @param config {Object} configuration object |
|
* config.height {Number} Element height. Optional. Default: CoverFlow.DEFAULT_HEIGHT. |
|
* config.width {Number} Element width. Optional. Default: CoverFlow.DEFAULT_WIDTH. |
|
* config.images {Array} Array of Images. [{src:}] |
|
* config.bgColor {String} Background color. Could be in the form #00000 or black. Optional. Default: CoverFlow.DEFAULT_BG_COLOR. |
|
* |
|
*/ |
|
YAHOO.ext.CoverFlow = function(el, userConfig){ |
|
if(el) |
|
this.init(el, userConfig || {}); |
|
}; |
|
|
|
//shortcuts |
|
var CoverFlow = YAHOO.ext.CoverFlow; |
|
var Dom = YAHOO.util.Dom; |
|
|
|
|
|
/** |
|
* Defaults |
|
*/ |
|
CoverFlow.DEFAULT_HEIGHT = 300; |
|
CoverFlow.DEFAULT_WIDTH = 800; |
|
CoverFlow.DEFAULT_BG_COLOR = '#000000'; |
|
CoverFlow.IMAGE_SEPARATION = 50; |
|
CoverFlow.RIGHT = 'right'; |
|
CoverFlow.LEFT = 'left'; |
|
CoverFlow.LABEL_CLASS = 'coverFlowLabel'; |
|
|
|
CoverFlow.prototype = { |
|
//Images array (it's a sort of transient var) |
|
images: [], |
|
//Items array {CoverFlowItem[]} |
|
coverFlowItems: [], |
|
|
|
remainingImages: 9999, |
|
|
|
element: null, |
|
labelElement: null, |
|
containerHeight: 0, |
|
containerWidth: 0, |
|
|
|
imageHeightRatio: 0.6, |
|
imageWidthRatio: 0.2, |
|
reflectionRatio: 0.6, // this causes: imageTotalHeightRatio = imageHeightRatio + imageHeightRatio*reflectionRatio |
|
topRatio: 0.1, |
|
sideRatio: 0.4, |
|
|
|
perspectiveAngle: 20, |
|
imageZIndex: 1000, |
|
selectedImageZIndex: 9999, |
|
selectedItem: 0, |
|
|
|
moveQueue: [], |
|
animationWorking: false, |
|
|
|
init: function(el, userConfig){ |
|
|
|
this.element = Dom.get(el); |
|
this.applyConfig(userConfig); |
|
|
|
if(userConfig.images) |
|
this.addImages(userConfig.images); |
|
|
|
this.attachEventListeners(); |
|
this.createLabelElement(); |
|
}, |
|
|
|
applyConfig: function(config){ |
|
this.containerHeight = config.height || CoverFlow.DEFAULT_HEIGHT; |
|
this.containerWidth = config.width || CoverFlow.DEFAULT_WIDTH; |
|
this.backgroundColor = config.bgColor || CoverFlow.DEFAULT_BG_COLOR; |
|
|
|
this.element.style.position = 'relative'; |
|
this.element.style.height = this.containerHeight + 'px'; |
|
this.element.style.width = this.containerWidth + 'px'; |
|
this.element.style.background = this.backgroundColor; |
|
this.element.style.overflow = 'hidden'; |
|
}, |
|
|
|
addImages: function(images){ |
|
this.images = []; |
|
this.remainingImages = images.length; |
|
|
|
for(var i=0; i < images.length; i++){ |
|
var img = images[i]; |
|
var image = new Image(); |
|
image.id = Dom.generateId(); |
|
image.index = i; |
|
image.onclick = img.onclick; |
|
image.label = img.label; |
|
|
|
//hide images |
|
image.style.visibility = 'hidden'; |
|
image.style.display = 'none'; |
|
//this is to maintain image order since image.onload will be called randomly |
|
this.element.appendChild(image); |
|
//a shortcut to not create another context to call onload |
|
var me = this; |
|
// image.onload = function(){ |
|
// CoverFlow.preloadImage(me, this); // this = image |
|
// }; |
|
YAHOO.util.Event.on(image, 'load', this.preloadImage, image, this); |
|
image.src = img.src; |
|
|
|
}; |
|
|
|
}, |
|
|
|
/** |
|
* @function preloadImage |
|
* @param event |
|
* @param image |
|
* @return void |
|
*/ |
|
preloadImage : function(e, image){ |
|
this.images.push(image); |
|
this.checkAllImagesLoaded(); |
|
}, |
|
|
|
checkAllImagesLoaded: function(){ |
|
this.remainingImages--; |
|
if(!this.remainingImages){ |
|
this.setup(); |
|
} |
|
}, |
|
|
|
setup: function(){ |
|
this.createCoverFlowItems(); |
|
this.sortCoverFlowItems(); |
|
this.initCoverFlow(); |
|
}, |
|
|
|
initCoverFlow: function(){ |
|
|
|
for(var i=0; i < this.coverFlowItems.length; i++){ |
|
var coverFlowItem = this.coverFlowItems[i]; |
|
|
|
var angle = 0; |
|
var direction; |
|
|
|
if(i==0){ |
|
coverFlowItem.setZIndex(this.selectedImageZIndex); |
|
coverFlowItem.setLeft(this.getCenter() - coverFlowItem.element.width/2); |
|
coverFlowItem.isSelected(true); |
|
this.selectedItem = 0; |
|
this.showLabel(coverFlowItem.getLabel()); |
|
}else{ |
|
angle = this.perspectiveAngle; |
|
direction = CoverFlow.LEFT; |
|
coverFlowItem.setZIndex(this.imageZIndex - i); |
|
coverFlowItem.setLeft( this.getRightStart()+ (i - 1)* CoverFlow.IMAGE_SEPARATION); |
|
coverFlowItem.isSelected(false); |
|
} |
|
coverFlowItem.setAngle(angle); |
|
coverFlowItem.drawInPerspective(direction); |
|
} |
|
}, |
|
|
|
createLabelElement: function(){ |
|
var label = document.createElement('div'); |
|
label.id = Dom.generateId(); |
|
label.style.position = 'absolute'; |
|
label.style.top = this.getFooterOffset() + 'px'; |
|
label.innerHTML = ' '; |
|
label.style.textAlign = 'center'; |
|
label.style.width = this.containerWidth + 'px'; |
|
label.style.zIndex = this.selectedImageZIndex + 10; |
|
label.className = CoverFlow.LABEL_CLASS; |
|
this.labelElement = this.element.appendChild(label); |
|
}, |
|
|
|
showLabel: function(text){ |
|
if(text) |
|
this.labelElement.innerHTML = text; |
|
else |
|
this.labelElement.innerHTML = ''; |
|
}, |
|
|
|
attachEventListeners: function(){ |
|
new YAHOO.util.KeyListener(this.element, { |
|
keys:39 |
|
}, |
|
{ |
|
fn:this.selectNext, |
|
scope:this, |
|
correctScope:true |
|
} ).enable(); |
|
|
|
new YAHOO.util.KeyListener(this.element, { |
|
keys:37 |
|
}, |
|
{ |
|
fn:this.selectPrevious, |
|
scope:this, |
|
correctScope:true |
|
} ).enable(); |
|
|
|
|
|
}, |
|
|
|
select: function(e,coverFlowItem){ |
|
var distance = this.selectedItem - coverFlowItem.index; |
|
if(distance < 0){ |
|
for(var i=0; i < -distance; i++) |
|
this.selectNext(); |
|
}else{ |
|
for(var i=0; i < distance; i++) |
|
this.selectPrevious(); |
|
} |
|
}, |
|
|
|
|
|
selectNext: function(){ |
|
if(this.animationWorking){ |
|
this.moveQueue.push('moveLeft'); |
|
return; |
|
} |
|
|
|
var animateItems = []; |
|
|
|
for(var i=0; i < this.coverFlowItems.length; i++){ |
|
var coverFlowItem = this.coverFlowItems[i]; |
|
var isLast = (this.selectedItem == this.coverFlowItems.length -1); |
|
if(!isLast){ |
|
var distance = i-this.selectedItem; |
|
|
|
if(distance == 0){// selected |
|
coverFlowItem.setZIndex(this.imageZIndex); |
|
coverFlowItem.isSelected(false); |
|
animateItems.push({ |
|
item: coverFlowItem, |
|
attribute:{ |
|
angle: { |
|
start: 0, |
|
end: this.perspectiveAngle |
|
} |
|
} |
|
}); |
|
|
|
coverFlowItem = this.coverFlowItems[++i]; |
|
coverFlowItem.isSelected(true); |
|
this.showLabel(coverFlowItem.getLabel()); |
|
animateItems.push({ |
|
item: coverFlowItem, |
|
attribute:{ |
|
angle: { |
|
start: this.perspectiveAngle, |
|
end: 0 |
|
} |
|
} |
|
}); |
|
|
|
}else{ |
|
animateItems.push({ |
|
item: coverFlowItem, |
|
attribute: { |
|
left: { |
|
start:coverFlowItem.getLeft(), |
|
end: coverFlowItem.getLeft() - CoverFlow.IMAGE_SEPARATION |
|
} |
|
} |
|
}); |
|
} |
|
} |
|
} |
|
|
|
var animation = new CoverFlowAnimation({ |
|
direction: CoverFlow.LEFT, |
|
center: this.getCenter(), |
|
startLeftPos: this.getLeftStart(), |
|
startRightPos: this.getRightStart() |
|
}, |
|
animateItems, 0.5); |
|
|
|
animation.onStart.subscribe(this.handleAnimationWorking, this); |
|
animation.onComplete.subscribe(this.handleQueuedMove, this); |
|
|
|
animation.animate(); |
|
|
|
if(this.selectedItem + 1 < this.coverFlowItems.length) |
|
this.selectedItem++; |
|
}, |
|
|
|
selectPrevious: function(){ |
|
if(this.animationWorking){ |
|
this.moveQueue.push('moveRight'); |
|
return; |
|
} |
|
|
|
var animateItems = []; |
|
|
|
for(var i=0; i < this.coverFlowItems.length; i++){ |
|
var coverFlowItem = this.coverFlowItems[i]; |
|
var isFirst = (this.selectedItem == 0); |
|
var distance = i-this.selectedItem; |
|
if(!isFirst){ |
|
if(distance == - 1){ |
|
coverFlowItem.setZIndex(this.selectedImageZIndex); |
|
coverFlowItem.isSelected(true); |
|
this.showLabel(coverFlowItem.getLabel()); |
|
animateItems.push({ |
|
item: coverFlowItem, |
|
attribute: { |
|
angle: { |
|
start: this.perspectiveAngle, |
|
end: 0 |
|
} |
|
} |
|
}); |
|
|
|
coverFlowItem = this.coverFlowItems[++i]; |
|
coverFlowItem.isSelected(false); |
|
coverFlowItem.setZIndex(this.imageZIndex); |
|
animateItems.push({ |
|
item: coverFlowItem, |
|
attribute: { |
|
angle: { |
|
start: 0, |
|
end: this.perspectiveAngle |
|
} |
|
} |
|
}); |
|
}else{ |
|
coverFlowItem.setZIndex(coverFlowItem.getZIndex() - 1); |
|
animateItems.push({ |
|
item: coverFlowItem, |
|
attribute: { |
|
left: { |
|
start:coverFlowItem.getLeft(), |
|
end: coverFlowItem.getLeft() + CoverFlow.IMAGE_SEPARATION |
|
} |
|
} |
|
}); |
|
} |
|
} |
|
} |
|
var animation = new CoverFlowAnimation({ |
|
direction: CoverFlow.RIGHT, |
|
center: this.getCenter(), |
|
startLeftPos: this.getLeftStart(), |
|
startRightPos: this.getRightStart() |
|
}, |
|
animateItems, 0.5); |
|
|
|
animation.onStart.subscribe(this.handleAnimationWorking, this); |
|
animation.onComplete.subscribe(this.handleQueuedMove, this); |
|
|
|
animation.animate(); |
|
|
|
if(this.selectedItem > 0) |
|
this.selectedItem--; |
|
}, |
|
|
|
handleAnimationWorking: function(a, b, cf){ |
|
cf.animationWorking = true; |
|
}, |
|
|
|
handleQueuedMove: function(msg, data, cf){ |
|
cf.animationWorking = false; |
|
|
|
var next = cf.moveQueue.pop(); |
|
if(next == 'moveLeft') |
|
cf.selectNext(); |
|
if(next == 'moveRight') |
|
cf.selectPrevious(); |
|
}, |
|
|
|
getCenter: function(){ |
|
return this.containerWidth / 2; |
|
}, |
|
|
|
getRightStart: function() { |
|
return this.containerWidth - this.sideRatio * this.containerWidth; |
|
}, |
|
|
|
getLeftStart: function() { |
|
return this.sideRatio * this.containerWidth; |
|
}, |
|
|
|
sortCoverFlowItems: function(){ |
|
function sortFunction(aCoverFlowItem, bCoverFlowItem){ |
|
return aCoverFlowItem.index - bCoverFlowItem.index; |
|
} |
|
|
|
this.coverFlowItems.sort(sortFunction); |
|
}, |
|
|
|
createCoverFlowItems: function(){ |
|
this.coverFlowItems = []; |
|
for(var i=0; i<this.images.length; i++){ |
|
var image = this.images[i]; |
|
var coverFlowItem = new CoverFlowItem(image, { |
|
scaledWidth: this.scaleWidth(image), |
|
scaledHeight: this.scaleHeight(image), |
|
reflectionRatio: this.reflectionRatio, |
|
bgColor: this.backgroundColor, |
|
onclick: { |
|
fn: this.select, |
|
scope: this |
|
} |
|
}); |
|
this.alignCenterHeight(coverFlowItem); |
|
this.coverFlowItems.push(coverFlowItem); |
|
}; |
|
delete this.images; |
|
}, |
|
|
|
alignCenterHeight: function(coverFlowItem){//review!!!!! |
|
coverFlowItem.element.style.position = 'absolute'; |
|
|
|
var imageHeight = coverFlowItem.canvas.height / (1 + this.reflectionRatio); |
|
var top = this.getMaxImageHeight() - imageHeight; |
|
top += this.topRatio * this.containerHeight; |
|
|
|
coverFlowItem.setTop(top); |
|
|
|
}, |
|
|
|
scaleHeight: function(image){ |
|
var height = 0; |
|
if(image.height <= this.getMaxImageHeight() && image.width <= this.getMaxImageWidth()){ |
|
height = image.height; |
|
} |
|
if(image.height > this.getMaxImageHeight() && image.width <= this.getMaxImageWidth()){ |
|
height = ((image.height / this.getMaxImageHeight())) * image.height; |
|
} |
|
if(image.height <= this.getMaxImageHeight() && image.width > this.getMaxImageWidth()){ |
|
height = ((image.width / this.getMaxImageWidth())) * image.height; |
|
} |
|
if(image.height > this.getMaxImageHeight() && image.width > this.getMaxImageWidth()){ |
|
if(image.height > image.width) |
|
height = ((this.getMaxImageHeight() / image.height)) * image.height; |
|
else |
|
height = ((this.getMaxImageWidth() / image.width)) * image.height; |
|
} |
|
return height; |
|
}, |
|
|
|
scaleWidth: function(image){ |
|
var width = 0; |
|
if(image.height <= this.getMaxImageHeight() && image.width <= this.getMaxImageWidth()){ |
|
width = image.width; |
|
} |
|
if(image.height > this.getMaxImageHeight() && image.width <= this.getMaxImageWidth()){ |
|
width = ((image.height / this.getMaxImageHeight())) * image.width; |
|
} |
|
if(image.height <= this.getMaxImageHeight() && image.width > this.getMaxImageWidth()){ |
|
width = ((image.width / this.getMaxImageWidth())) * image.width; |
|
} |
|
if(image.height > this.getMaxImageHeight() && image.width > this.getMaxImageWidth()){ |
|
if(image.height > image.width) |
|
width = ((this.getMaxImageHeight() / image.height)) * image.width; |
|
else |
|
width = ((this.getMaxImageWidth() / image.width)) * image.width; |
|
} |
|
return width; |
|
}, |
|
|
|
|
|
getMaxImageHeight: function(){ |
|
return (this.containerHeight * this.imageHeightRatio); |
|
}, |
|
|
|
getMaxImageWidth: function(){ |
|
return (this.containerWidth * this.imageWidthRatio); |
|
}, |
|
|
|
getTopOffset: function(){ |
|
return this.containerHeight * this.topRatio; |
|
}, |
|
|
|
getFooterOffset: function(){ |
|
return this.containerHeight * (this.topRatio + this.imageHeightRatio); |
|
} |
|
}; |
|
|
|
|
|
/** |
|
* @class CoverFlowItem |
|
* |
|
*/ |
|
CoverFlowItem = function(image, config){ |
|
if(image) |
|
this.init(image, config); |
|
}; |
|
|
|
CoverFlowItem.prototype = { |
|
canvas: null, |
|
element: null, |
|
index: null, |
|
id: null, |
|
angle: 0, |
|
selected: false, |
|
onclickFn: null, |
|
selectedOnclickFn: null, |
|
label: null, |
|
|
|
onSelected: null, |
|
|
|
init: function(image, config){ |
|
var scaledWidth = config.scaledWidth; |
|
var scaledHeight = config.scaledHeight; |
|
var reflectionRatio = config.reflectionRatio; |
|
var bgColor = config.bgColor; |
|
|
|
this.id = image.id; |
|
this.index = image.index; |
|
this.onclickFn = config.onclick; |
|
this.selectedOnclickFn = image.onclick; |
|
this.label = image.label; |
|
var parent = image.parentNode; |
|
this.canvas = this.createImageCanvas(image,scaledWidth,scaledHeight,reflectionRatio, bgColor); |
|
this.element = this.canvas.cloneNode(false); |
|
this.element.id = this.id; |
|
parent.replaceChild(this.element, image); |
|
|
|
this.onSelected = new YAHOO.util.CustomEvent('onSelected', this); |
|
this.onSelected.subscribe(this.handleOnclick); |
|
|
|
}, |
|
|
|
getLabel: function(){ |
|
return this.label; |
|
}, |
|
|
|
handleOnclick: function(){ |
|
YAHOO.util.Event.removeListener(this.element, 'click'); |
|
if(!this.selected){ |
|
YAHOO.util.Event.addListener(this.element, 'click', this.onclickFn.fn, this, this.onclickFn.scope); |
|
}else{ |
|
if(this.selectedOnclickFn && this.selectedOnclickFn.fn) |
|
YAHOO.util.Event.addListener(this.element, 'click', this.selectedOnclickFn.fn, this, this.selectedOnclickFn.scope); |
|
else |
|
YAHOO.util.Event.addListener(this.element, 'click', this.selectedOnclickFn); |
|
} |
|
}, |
|
|
|
isSelected: function(selected){ |
|
this.selected = selected; |
|
this.onSelected.fire(); |
|
}, |
|
|
|
setAngle: function(angle){ |
|
this.angle = angle; |
|
}, |
|
|
|
getAngle: function(){ |
|
return this.angle; |
|
}, |
|
|
|
setTop: function(top){ |
|
this.element.style.top = top + 'px'; |
|
}, |
|
|
|
setLeft: function(left){ |
|
this.element.style.left = left + 'px'; |
|
}, |
|
|
|
getLeft: function(){ |
|
var ret = this.element.style.left; |
|
return new Number(ret.replace("px", "")); |
|
}, |
|
|
|
getZIndex: function(){ |
|
return this.element.style.zIndex; |
|
}, |
|
|
|
setZIndex: function(zIndex){ |
|
this.element.style.zIndex = zIndex; |
|
}, |
|
|
|
createImageCanvas: function(image, sWidth, sHeight, reflectionRatio, bgColor){ |
|
|
|
var imageCanvas = document.createElement('canvas'); |
|
|
|
if(imageCanvas.getContext){ |
|
|
|
var scaledWidth = sWidth; |
|
var scaledHeight = sHeight; |
|
var reflectionHeight = scaledHeight * reflectionRatio; |
|
|
|
imageCanvas.height = scaledHeight + reflectionHeight; |
|
imageCanvas.width = scaledWidth; |
|
|
|
var ctx = imageCanvas.getContext('2d'); |
|
|
|
ctx.clearRect(0, 0, imageCanvas.width, imageCanvas.height); |
|
ctx.globalCompositeOperation = 'source-over'; |
|
ctx.fillStyle = 'rgba(0, 0, 0, 1)'; |
|
ctx.fillRect(0, 0, imageCanvas.width, imageCanvas.height); |
|
|
|
//draw the reflection image |
|
ctx.save(); |
|
ctx.translate(0, (2*scaledHeight)); |
|
ctx.scale(1, -1); |
|
ctx.drawImage(image, 0, 0, scaledWidth, scaledHeight); |
|
ctx.restore(); |
|
//create the gradient effect |
|
ctx.save(); |
|
ctx.translate(0, scaledHeight); |
|
ctx.globalCompositeOperation = 'destination-out'; |
|
var grad = ctx.createLinearGradient( 0, 0, 0, scaledHeight); |
|
grad.addColorStop(1, 'rgba(0, 0, 0, 1)'); |
|
grad.addColorStop(0, 'rgba(0, 0, 0, 0.75)'); |
|
ctx.fillStyle = grad; |
|
ctx.fillRect(0, 0, scaledWidth, scaledHeight); |
|
//apply the background color to the gradient |
|
ctx.globalCompositeOperation = 'destination-over'; |
|
ctx.fillStyle = bgColor; |
|
'#000'; |
|
ctx.globalAlpha = 0.8; |
|
ctx.fillRect(0, 0 , scaledWidth, scaledHeight); |
|
ctx.restore(); |
|
//draw the image |
|
ctx.save(); |
|
ctx.translate(0, 0); |
|
ctx.globalCompositeOperation = 'source-over'; |
|
ctx.drawImage(image, 0, 0, scaledWidth, scaledHeight); |
|
ctx.restore(); |
|
|
|
return imageCanvas; |
|
} |
|
}, |
|
|
|
drawInPerspective: function(direction, frameSize){ |
|
var canvas = this.element; |
|
var image = this.canvas; |
|
var angle = Math.ceil(this.angle); |
|
var ctx; |
|
var originalWidth = image.width; |
|
var originalHeight = image.height; |
|
var destinationWidth = destinationWidth || originalWidth; // for future use |
|
var destinationHeight = destinationHeight || originalHeight; // for future use |
|
|
|
var perspectiveCanvas = document.createElement('canvas'); |
|
perspectiveCanvas.height = destinationHeight; |
|
perspectiveCanvas.width = destinationWidth; |
|
var perspectiveCtx = perspectiveCanvas.getContext('2d'); |
|
|
|
var alpha = angle * Math.PI/180; // Math uses radian |
|
|
|
if(alpha > 0){ // if we have an angle greater than 0 then apply the perspective |
|
var right = (direction == CoverFlow.RIGHT); |
|
|
|
var initialX=0, finalX=0, initialY=0, finalY=0; |
|
|
|
frameSize = frameSize || 1; |
|
var xDes, yDes; |
|
var heightDes, widthDes; |
|
var perspectiveWidht = destinationWidth; |
|
|
|
var frameFactor = frameSize / originalWidth; |
|
var frames = Math.floor(originalWidth / frameSize); |
|
|
|
var widthSrc = frameSize ; |
|
var heightSrc = originalHeight; |
|
|
|
for(var i=0; i < frames; i++){ |
|
var xSrc = (i) * frameSize; |
|
var ySrc = 0; |
|
var betaTan = 0; |
|
width = destinationWidth * (i) * frameFactor; |
|
horizon = destinationHeight / 2; |
|
|
|
if(right){ |
|
betaTan = horizon/((Math.tan(alpha)*horizon) + width); |
|
xDes = (betaTan*width)/(Math.tan(alpha) + betaTan); |
|
yDes = Math.tan(alpha) * xDes; |
|
|
|
if(i == frames -1){ |
|
finalX=xDes; |
|
finalY=yDes; |
|
} |
|
}else{ |
|
betaTan = horizon/((Math.tan(alpha)*horizon) +(destinationWidth-width)); |
|
xDes = (Math.tan(alpha)*(destinationWidth) + (betaTan * width))/(Math.tan(alpha) + betaTan); |
|
yDes = -Math.tan(alpha)*xDes + (Math.tan(alpha)*(destinationWidth)); |
|
|
|
if(i == 0){ |
|
initialX = xDes; |
|
initialY = yDes; |
|
finalX = destinationWidth; |
|
finalY = 0; |
|
} |
|
} |
|
|
|
heightDes = destinationHeight - (2*yDes); |
|
widthDes = heightDes / destinationHeight * destinationWidth; |
|
|
|
perspectiveCtx.drawImage(image, xSrc, ySrc, widthSrc, heightSrc, xDes, yDes, widthDes, heightDes); |
|
|
|
} |
|
|
|
perspectiveWidth = finalX - initialX; |
|
originalCanvasWidth = destinationWidth; |
|
canvas.width = perspectiveWidth; |
|
|
|
ctx = canvas.getContext('2d'); |
|
|
|
//remove exceeded pixels |
|
ctx.beginPath(); |
|
if(right){ |
|
ctx.moveTo(0, 0); |
|
ctx.lineTo(finalX, finalY); |
|
ctx.lineTo(finalX, finalY + (destinationHeight - 2*finalY)); |
|
ctx.lineTo(0, destinationHeight); |
|
ctx.lineTo(0,0); |
|
}else{ |
|
var initialX1 = initialX - (originalCanvasWidth - perspectiveWidth); |
|
var finalX1 = finalX - (originalCanvasWidth - perspectiveWidth); |
|
ctx.moveTo(0, initialY); |
|
ctx.lineTo(finalX1, finalY); |
|
ctx.lineTo(finalX1, destinationHeight); |
|
ctx.lineTo(initialX1, initialY + (destinationHeight - 2*initialY)); |
|
ctx.lineTo(0, initialY); |
|
} |
|
ctx.closePath(); |
|
ctx.clip(); |
|
|
|
ctx.drawImage(perspectiveCanvas, initialX, 0, perspectiveWidth, destinationHeight, 0, 0, perspectiveWidth, destinationHeight); |
|
|
|
}else{ |
|
|
|
canvas.width = perspectiveCanvas.width; |
|
canvas.height = perspectiveCanvas.height; |
|
perspectiveCtx.drawImage(image, 0, 0, originalWidth, originalHeight, 0, 0, destinationWidth, destinationHeight); |
|
ctx = canvas.getContext('2d'); |
|
ctx.clearRect(0, 0, canvas.width, canvas.height); |
|
ctx.drawImage(perspectiveCanvas, 0, 0); |
|
} |
|
} |
|
|
|
}; |
|
|
|
/** |
|
* @class CoverFlowAnimation |
|
* @requires YAHOO.util.AnimMgr |
|
*/ |
|
CoverFlowAnimation = function(config, animationItems, duration){ |
|
this.init(config, animationItems, duration); |
|
}; |
|
|
|
CoverFlowAnimation.prototype = { |
|
direction: null, |
|
|
|
center: null, |
|
|
|
startLeftPos: null, |
|
|
|
startRightPos: null, |
|
|
|
animationItems: null, |
|
|
|
method : YAHOO.util.Easing.easeNone, |
|
|
|
animated: false, |
|
|
|
startTime: null, |
|
|
|
actualFrames : 0, |
|
|
|
useSeconds : true, // default to seconds |
|
|
|
currentFrame : 0, |
|
|
|
totalFrames : YAHOO.util.AnimMgr.fps, |
|
|
|
init: function(config, animationItems, duration){ |
|
this.direction = config.direction; |
|
this.center = config.center; |
|
this.startLeftPos = config.startLeftPos; |
|
this.startRightPos = config.startRightPos; |
|
this.animationItems = animationItems; |
|
this.duration = duration || 1; |
|
this.registerEvents(); |
|
}, |
|
|
|
registerEvents: function(){ |
|
/** |
|
* Custom event that fires after onStart, useful in subclassing |
|
* @private |
|
*/ |
|
this._onStart = new YAHOO.util.CustomEvent('_start', this, true); |
|
|
|
/** |
|
* Custom event that fires when animation begins |
|
* Listen via subscribe method (e.g. myAnim.onStart.subscribe(someFunction) |
|
* @event onStart |
|
*/ |
|
this.onStart = new YAHOO.util.CustomEvent('start', this); |
|
|
|
/** |
|
* Custom event that fires between each frame |
|
* Listen via subscribe method (e.g. myAnim.onTween.subscribe(someFunction) |
|
* @event onTween |
|
*/ |
|
this.onTween = new YAHOO.util.CustomEvent('tween', this); |
|
|
|
/** |
|
* Custom event that fires after onTween |
|
* @private |
|
*/ |
|
this._onTween = new YAHOO.util.CustomEvent('_tween', this, true); |
|
|
|
/** |
|
* Custom event that fires when animation ends |
|
* Listen via subscribe method (e.g. myAnim.onComplete.subscribe(someFunction) |
|
* @event onComplete |
|
*/ |
|
this.onComplete = new YAHOO.util.CustomEvent('complete', this); |
|
/** |
|
* Custom event that fires after onComplete |
|
* @private |
|
*/ |
|
this._onComplete = new YAHOO.util.CustomEvent('_complete', this, true); |
|
|
|
this._onStart.subscribe(this.doOnStart); |
|
this._onTween.subscribe(this.doOnTween); |
|
this._onComplete.subscribe(this.doOnComplete); |
|
|
|
}, |
|
|
|
isAnimated : function() { |
|
return this.animated; |
|
}, |
|
|
|
getStartTime : function() { |
|
return this.startTime; |
|
}, |
|
|
|
doMethod: function(start, end) { |
|
return this.method(this.currentFrame, start, end - start, this.totalFrames); |
|
}, |
|
|
|
animate : function() { |
|
if ( this.isAnimated() ) { |
|
return false; |
|
} |
|
|
|
this.currentFrame = 0; |
|
|
|
this.totalFrames = ( this.useSeconds ) ? Math.ceil(YAHOO.util.AnimMgr.fps * this.duration) : this.duration; |
|
|
|
if (this.duration === 0 && this.useSeconds) { // jump to last frame if zero second duration |
|
this.totalFrames = 1; |
|
} |
|
YAHOO.util.AnimMgr.registerElement(this); |
|
return true; |
|
}, |
|
|
|
stop : function(finish) { |
|
if (!this.isAnimated()) { // nothing to stop |
|
return false; |
|
} |
|
|
|
if (finish) { |
|
this.currentFrame = this.totalFrames; |
|
this._onTween.fire(); |
|
} |
|
YAHOO.util.AnimMgr.stop(this); |
|
}, |
|
|
|
doOnStart : function() { |
|
this.onStart.fire(); |
|
|
|
this.runtimeItems = []; |
|
for (var i=0; i<this.animationItems.length; i++) { |
|
this.setRuntimeItem(this.animationItems[i]); |
|
} |
|
|
|
this.animated = true; |
|
this.actualFrames = 0; |
|
this.startTime = new Date(); |
|
}, |
|
|
|
doOnTween : function() { |
|
var data = { |
|
duration: new Date() - this.getStartTime(), |
|
currentFrame: this.currentFrame |
|
}; |
|
|
|
data.toString = function() { |
|
return ( |
|
'duration: ' + data.duration + |
|
', currentFrame: ' + data.currentFrame |
|
); |
|
}; |
|
|
|
this.onTween.fire(data); |
|
|
|
this.actualFrames += 1; |
|
|
|
var runtimeItems = this.runtimeItems; |
|
|
|
for (var i=0; i < runtimeItems.length; i++) { |
|
this.setItemAttributes(runtimeItems[i]); |
|
} |
|
|
|
}, |
|
|
|
doOnComplete : function() { |
|
var actual_duration = (new Date() - this.getStartTime()) / 1000 ; |
|
|
|
var data = { |
|
duration: actual_duration, |
|
frames: this.actualFrames, |
|
fps: this.actualFrames / actual_duration |
|
}; |
|
|
|
data.toString = function() { |
|
return ( |
|
'duration: ' + data.duration + |
|
', frames: ' + data.frames + |
|
', fps: ' + data.fps |
|
); |
|
}; |
|
|
|
this.animated = false; |
|
this.actualFrames = 0; |
|
this.onComplete.fire(data); |
|
}, |
|
|
|
setRuntimeItem: function(item){ |
|
var runtimeItem = {}; |
|
runtimeItem.item = item.item; |
|
runtimeItem.attribute = {}; |
|
for(var attr in item.attribute){ |
|
runtimeItem.attribute[attr] = item.attribute[attr]; |
|
if(attr == 'angle'){ |
|
if(item.attribute[attr].start - item.attribute[attr].end > 0){ |
|
runtimeItem.attribute[attr].perspectiveDirection = this.direction; |
|
runtimeItem.attribute[attr].center = true; |
|
}else{ |
|
runtimeItem.attribute[attr].perspectiveDirection = this.direction == CoverFlow.RIGHT ? CoverFlow.LEFT : CoverFlow.RIGHT; |
|
runtimeItem.attribute[attr].center = false; |
|
} |
|
} |
|
} |
|
this.runtimeItems.push(runtimeItem); |
|
}, |
|
|
|
setItemAttributes: function(item){ |
|
|
|
for(var attr in item.attribute){ |
|
|
|
var value = Math.ceil(this.doMethod(item.attribute[attr].start, item.attribute[attr].end)); |
|
|
|
if(attr == 'angle'){ |
|
item.item.setAngle(value); |
|
var frameSize = Math.ceil(this.doMethod(3, 1)); |
|
item.item.drawInPerspective(item.attribute[attr].perspectiveDirection, frameSize); |
|
var left; |
|
if(item.attribute[attr].center){ |
|
left = this.doMethod(item.item.getLeft(), this.center - item.item.element.width/2); |
|
}else{ |
|
if(this.direction == CoverFlow.LEFT) |
|
left = this.doMethod(item.item.getLeft(), this.startLeftPos - item.item.element.width); |
|
else |
|
left = this.doMethod(item.item.getLeft(), this.startRightPos); |
|
} |
|
item.item.setLeft(Math.ceil(left)); |
|
|
|
}else{ |
|
item.item.setLeft(value); |
|
} |
|
} |
|
} |
|
}; |
|
|
|
//});
|