我正在做我的新项目,在这个项目中,我想展示一些杂乱无章的从20-30个小的png图像。所有的图像都会重复多次。最后,我想得到很多分散在csv形状容器周围的图像,它们会移动/抖动,可能是鼠标移动。
现在,我试图在javascript中遍历图像sprite,并在每个循环中更改sprite位置,但是.
由于覆盖变量,我在同一时间显示同一精灵的多个位置时有问题。即使我要做的事情是,与html5画布规范相关的动画会有一些问题,但我可能不得不保存每个图像的一些属性,因为画布只是绘制和忘记。有人能给我一些如何处理我的问题的建议吗?也许有人有类似的项目?我愿意接受任何建议和解决方案。
下面是我开始使用的一些代码,它是在我试图通过循环更改变量名之前编写的,因为它并没有使我更接近于解决方案。我添加了一些随机的雪碧从谷歌图形只是为了举例(现在没有原创)。
window.onload = function() {
var canvas = document.getElementById('canvas');
var context = canvas.getContext('2d');
for (i = 0; i < 10; i++) {
for (i = 0; i < 5; i++) {
var sprite = new Image();
var swidth = 260;
var sheight = 260;
var randomWidth = Math.random() * document.getElementById('canvas');
var randomHeight = Math.random() * document.getElementById('canvas');
var cx = randomWidth;
var cy = randomHeight;
var sx = i * 260;
var sy = 0;
sprite.onload = function() {
context.drawImage(sprite, sx, sy, swidth, sheight, cx, cy, 260, 213);
};
sprite.src = 'https://cdn.codeandweb.com/blog/2016/05/10/how-to-create-a-sprite-sheet/spritestrip.png';
}
}
}提前感谢
发布于 2017-07-01 18:51:50
只需使用数组或基于数组的列表对象。Javascript的对象非常灵活,您可以根据需要混合和匹配属性和行为。
在这个示例中,我创建了一个包含项目列表的list对象,一个添加项的函数(list.add),一个删除所有项(list.empty)的函数,以及一个处理所有项的迭代器(list.eachItem)。
然后是一个spriteSheet对象,它加载一个图像并创建一个雪碧表(当我获取您的图像并计算出动画与图像大小相关时,作弊)。雪碧工作表对象还创建sprite对象。
然后,我有一些对象来渲染和动画精灵,和一个原型对象(为了速度)来处理摆动的FX。
然后,从spriteSheet对象创建一个名为walkSprites的雪碧工作表,用于创建基本的精灵。
然后启动一个动画循环,调整画布的大小以适应页面和create 100精灵。我首先创建一个基本的sprite,它只具有x,y位置,比例和旋转,然后从摆动对象中添加行为。
都在片段里。点击并按住按钮给动画一个踢。
const ctx = canvas.getContext("2d");
const U = undefined;
const doFor = (count, callback) => {var i = 0; while (i < count && callback(i ++) !== true ); };
const randI = (min, max = min + (min = 0)) => (Math.random() * (max - min) + min) | 0;
const rand = (min, max = min + (min = 0)) => Math.random() * (max - min) + min;
const mouse = {x : 0, y : 0, button : false}
function mouseEvents(e){
mouse.x = e.pageX;
mouse.y = e.pageY;
mouse.button = e.type === "mousedown" ? true : e.type === "mouseup" ? false : mouse.button;
}
["down","up","move"].forEach(name => document.addEventListener("mouse"+name,mouseEvents));
const list = {
items : null,
add(item){ this.items.push(item); return item },
empty(){ this.items.length = 0 },
eachItem(callback){
var i;
for(i = 0; i < this.items.length; i++){
callback(this.items[i],i);
}
},
}
const spriteSheet = {
image : null,
ready : false,
load(URL){
this.image = new Image;
this.image.src = URL;
this.image.onload = ()=>{
const sprites = [];
const w = this.image.width;
const h = this.image.height;
for(var i = 0; i < w/h; i++){
sprites.push({x : i * h, y : 0, w : h,h : h});
}
this.image.sprites = sprites;
this.ready = true;
}
},
create(index,x,y,scale,rot){
return Object.assign({},sprite,{image:this.image,index,x,y,scale,rot} );
},
}
const sprite = {
draw(){
ctx.setTransform(this.scale,0,0,this.scale,this.x,this.y);
ctx.rotate(this.rot);
const spr = this.image.sprites[this.index];
const w = spr.w;
const h = spr.h;
ctx.drawImage(this.image,spr.x,spr.y,w,h,-w/2,-h/2,w,h);
}
}
function FChase(val,accel,drag){
this.val = val;
this.valChase = 0;
this.valReal = 0;
this.accel = accel;
this.drag = drag;
}
FChase.prototype = {
update(val = this.valReal){
this.valChase += (this.val-this.valReal) * this.accel;
this.valChase *= this.drag;
this.valReal += this.valChase;
return this.valReal;
},
kick(amount){
this.valChase += amount;
},
}
const accel = 0.15;
const drag = 0.7;
const animSpeed = 100;
const wobble = {
init(){
this.xc = new FChase(this.x,accel,drag);
this.yc = new FChase(this.y,accel,drag);
this.rc = new FChase(this.rot,accel,drag);
this.sc = new FChase(this.scale,accel,drag);
this.animSpeed = rand(0.2,1.2);
this.animOffset = rand(1);
return this;
},
update(){
this.index = (((globalTime / animSpeed) * this.animSpeed + this.animOffset * this.image.sprites.length) | 0) % this.image.sprites.length;
this.x = this.xc.update(this.x);
this.y = this.yc.update(this.x);
this.scale = this.sc.update(this.scale);
this.rot = this.rc.update(this.rot);
},
kick(){
this.xc.kick(rand(-40,40));
this.yc.kick(rand(-40,40));
// this.sc.kick(rand(-0.1,0.1));
this.rc.kick(rand(-1,1));
}
}
// create a sprite sheet and load media
const walkSprites = Object.assign({},spriteSheet);
walkSprites.load("https://cdn.codeandweb.com/blog/2016/05/10/how-to-create-a-sprite-sheet/spritestrip.png");
// create a list for the sprites.
const spriteList = Object.assign({},list,{items:[]});
// create 100 sprites
const spriteCount = 100;
function createSprites(){
spriteList.empty();
doFor(spriteCount,i=>{
var spr = spriteList.add(
walkSprites.create(
randI(6), // image may not have loaded, but I know how many sprites there are
randI(canvas.width ), // pos
randI(canvas.height),
rand(0.25,0.5), // scale
rand(Math.PI*2), // rotation
)
);
spr = Object.assign(spr,wobble).init();
})
}
// use requestAnimationFrame to animate
var w = canvas.width;
var h = canvas.height;
var cw = w / 2; // center
var ch = h / 2;
var globalTime;
var frameCount = 0;
// main update function
function update(timer){
globalTime = timer;
frameCount += 1;
ctx.setTransform(1,0,0,1,0,0); // reset transform
ctx.globalAlpha = 1; // reset alpha
if(w !== innerWidth || h !== innerHeight){
cw = (w = canvas.width = innerWidth) / 2;
ch = (h = canvas.height = innerHeight) / 2;
createSprites();
}else{
ctx.clearRect(0,0,w,h);
}
if(walkSprites.ready){
if(mouse.button){
spriteList.eachItem((sprite,i)=>{
if(frameCount % 20 === i % 20){
sprite.kick()
}
});
}
spriteList.eachItem(sprite=>sprite.update());
spriteList.eachItem(sprite=>sprite.draw());
}
requestAnimationFrame(update);
}
requestAnimationFrame(update);canvas, span{ position : absolute; top : 0px; left 0px; }
span {
z-index:10;
font-family:arial black;
font-size:34px;
color : yellow;
}<span >CLICK HOLD to KICK animations</span>
<canvas id="canvas"></canvas>
https://stackoverflow.com/questions/44863406
复制相似问题