首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >循环浏览并在多个位置显示一个精灵或多个重复图像

循环浏览并在多个位置显示一个精灵或多个重复图像
EN

Stack Overflow用户
提问于 2017-07-01 16:57:30
回答 1查看 2K关注 0票数 0

我正在做我的新项目,在这个项目中,我想展示一些杂乱无章的从20-30个小的png图像。所有的图像都会重复多次。最后,我想得到很多分散在csv形状容器周围的图像,它们会移动/抖动,可能是鼠标移动。

现在,我试图在javascript中遍历图像sprite,并在每个循环中更改sprite位置,但是.

由于覆盖变量,我在同一时间显示同一精灵的多个位置时有问题。即使我要做的事情是,与html5画布规范相关的动画会有一些问题,但我可能不得不保存每个图像的一些属性,因为画布只是绘制和忘记。有人能给我一些如何处理我的问题的建议吗?也许有人有类似的项目?我愿意接受任何建议和解决方案。

下面是我开始使用的一些代码,它是在我试图通过循环更改变量名之前编写的,因为它并没有使我更接近于解决方案。我添加了一些随机的雪碧从谷歌图形只是为了举例(现在没有原创)。

代码语言:javascript
运行
复制
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';
  }
 }
}

提前感谢

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2017-07-01 18:51:50

只需使用数组或基于数组的列表对象。Javascript的对象非常灵活,您可以根据需要混合和匹配属性和行为。

在这个示例中,我创建了一个包含项目列表的list对象,一个添加项的函数(list.add),一个删除所有项(list.empty)的函数,以及一个处理所有项的迭代器(list.eachItem)。

然后是一个spriteSheet对象,它加载一个图像并创建一个雪碧表(当我获取您的图像并计算出动画与图像大小相关时,作弊)。雪碧工作表对象还创建sprite对象。

然后,我有一些对象来渲染和动画精灵,和一个原型对象(为了速度)来处理摆动的FX。

然后,从spriteSheet对象创建一个名为walkSprites的雪碧工作表,用于创建基本的精灵。

然后启动一个动画循环,调整画布的大小以适应页面和create 100精灵。我首先创建一个基本的sprite,它只具有x,y位置,比例和旋转,然后从摆动对象中添加行为。

都在片段里。点击并按住按钮给动画一个踢。

代码语言:javascript
运行
复制
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);
代码语言:javascript
运行
复制
canvas, span{ position : absolute; top : 0px; left 0px; }
span {
  z-index:10;
  font-family:arial black;
  font-size:34px;
  color : yellow;
 }
代码语言:javascript
运行
复制
<span >CLICK HOLD to KICK animations</span>
<canvas id="canvas"></canvas>

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/44863406

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档