首页 > 解决方案 > 在 p5.js 中,是否可以在 setup 或 preload 函数中加载 gif,然后在其他地方多次使用该 gif?

问题描述

我有一个班级(我们称之为“敌人”),我希望他们在足够近的时候攻击我(它会显示一个动画 gif,看起来像咬人)。

我已经完成了所有这些工作,除了我能弄清楚的唯一方法是将 loadImage("attack.gif") 放在类中。这真的很快,因为每次有敌人产生时,它都必须重新加载那个 gif。

我尝试在课堂上使用 setup() 中加载的 gif,但他们的所有攻击都是同步的。

还有另一种方法可以做到这一点吗?

标签: imagegifp5.js

解决方案


您可以预加载 gif(或多个 gif)并将它们存储在一个数组中,以后可以重复使用该数组在多个位置绘制(例如,在您的情况下是多个生成的敌人)。

这是一个演示的基本示例:

  • 将图像加载到数组中
  • 使用加载的图像在屏幕上(重新)生成对象

let images;
let enemies = [];

function preload(){
  // load images and store them in an array
  images = [
 loadImage(""),  loadImage(""),
     ];
}

function setup(){
  createCanvas(600, 600);
  imageMode(CENTER);
}

function draw(){
  background(255);
  // update + draw enemies
  for(let i = 0; i < enemies.length; i++){
    enemies[i].update();
  }
  text("click to spawn", 10, 15);
}

function mousePressed(){
  // pick a random image
  let randomImage = images[int(random(images.length))];
  // make a new enemy
  enemies.push(new Enemy(randomImage, mouseX, mouseY));
}

class Enemy {
  constructor(skin, x, y) {
    this.skin = skin;
    this.position = createVector(x, y);
    this.velocity = createVector(random(-1,1), random(-1,1));
  }
  update(){
    // add velocity
    this.position.add(this.velocity);
    // check borders and flip direction (roughly)
    if(this.position.x < 0 || this.position.x > width ||
       this.position.y < 0 || this.position.y > height){
       this.velocity.mult(-1);
    }
    // render
    push();
    blendMode(MULTIPLY);
    image(this.skin, this.position.x, this.position.y);
    pop();
  }
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.3.0/p5.min.js"></script>

在上面的示例中,我使用 base64 编码图像来避免 CORS 问题,但是您应该可以加载自定义图像。这个想法是全局存储图像,以便稍后在实例化新敌人时重新使用它们。(如果你有几个图像,每个图像都应该使用单独的变量,否则数组会更容易管理)

注意在Enemy构造函数中只是接收对先前加载的图像的引用:

// when declared
constructor(skin, x, y) {
    this.skin = skin;
// when intantiated
new Enemy(randomImage, mouseX, mouseY)

剩下要做的就是根据需要渲染图像:

image(this.skin, this.position.x, this.position.y);

更新基于评论和共享代码,每个敌人共享相同的 gif,该 gif 以相同的速率以相同的帧数同步更新。

一种选择是images用同一图像的多个副本填充数组,基本上每个敌人重新加载一次 gif,尽管这似乎很浪费。

不幸的是p5.Image.get(),返回当前帧的快照,并且没有神奇的函数来克隆加载的 gif,但是未记录的gifProperties保存了手动执行此操作所需的数据:

let images;
let enemies = [];

function preload(){
  // load images and store them in an array
  images = [
    loadImage(""),
  ];
}

function setup(){
  createCanvas(600, 600);
  imageMode(CENTER);
}

function draw(){
  background(255);
  // update + draw enemies
  for(let i = 0; i < enemies.length; i++){
    enemies[i].update();
  }
  text("click to spawn", 10, 15);
}

function mousePressed(){
  // pick a random image
  let randomImage = images[int(random(images.length))];
  // offset the start frame of each enemy by one
  let startFrame = enemies.length % randomImage.numFrames();
  // make a new enemy
  enemies.push(new Enemy(cloneGif(randomImage,startFrame), mouseX, mouseY));
}

function cloneGif(gif, startFrame){
  let gifClone = gif.get();
  // access original gif properties
  gp = gif.gifProperties;
  // make a new object for the clone
  gifClone.gifProperties = {
    displayIndex: gp.displayIndex,
    // we still point to the original array of frames
    frames: gp.frames,
    lastChangeTime: gp.lastChangeTime,
    loopCount: gp.loopCount,
    loopLimit: gp.loopLimit,
    numFrames: gp.numFrames,
    playing: gp.playing,
    timeDisplayed: gp.timeDisplayed
  };
  // optional tweak the start frame
  gifClone.setFrame(startFrame);
  
  return gifClone;
}

class Enemy {
  constructor(skin, x, y) {
    this.skin = skin;
    this.x = x;
    this.y = y;
  }
  update(){
    image(this.skin, this.x, this.y);
  }
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.3.0/p5.min.js"></script>

生成一个新对象(这cloneGif意味着新的起始帧等),但是它应该指向原始 gif 的frames(保存像素 / ImageData

现在每个敌人都从一个新的帧开始。

我注意到更改延迟似乎会影响所有敌人:一旦将延迟设置为 1,似乎底层图像数据会以相同的速率复制。

如果您需要不同的延迟时间,您仍然可以管理访问相同的 gif frame ImageData,但不是依靠p5.Image控制 gif 播放(例如setFrame()/ delay()),您需要手动管理它并渲染到 p5 的画布:

let images;
let enemies = [];

function preload(){
  // load images and store them in an array
  images = [
    loadImage(""),
  ];
}

function setup(){
  createCanvas(600, 600);
  imageMode(CENTER);
}

function draw(){
  background(255);
  // update + draw enemies
  for(let i = 0; i < enemies.length; i++){
    enemies[i].update();
  }
  text("click to spawn", 10, 15);
}

function mousePressed(){
  // pick a random image
  let randomImage = images[int(random(images.length))];
  // make a new enemy
  let enemy = new Enemy({frames: randomImage.gifProperties.frames, gifWidth: randomImage.width, gifH: randomImage.height}, mouseX, mouseY);
  // pick a random start frame
  enemy.currentFrame = int(random(images.length));
  // pick a random per gif frame delay (e.g. the larger the number the slower the gif will play)
  enemy.frameDelay = int(random(40, 240));
  enemies.push(enemy);
}

class Enemy {
  constructor(gifData, x, y) {
    this.frames = gifData.frames;
    this.offX = -int(gifData.gifWidth * 0.5);
    this.offY = -int(gifData.gifHeight * 0.5);
    this.currentFrame = 0;
    this.numFrames = this.frames.length;
    this.frameDelay = 100;
    this.lastFrameUpdate = millis();
    this.x = x;
    this.y = y;
  }
  
  update(){
    let millisNow = millis();
    // increment frame
    if(millisNow - this.lastFrameUpdate >= this.frameDelay){
      this.currentFrame = (this.currentFrame + 1) % this.numFrames;  
      this.lastFrameUpdate = millisNow;
    }
     // render directly to p5's canvas context
 drawingContext.putImageData(this.frames[this.currentFrame].image, this.x + this.offX, this.y + this.offY);
  }
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.3.0/p5.min.js"></script>

还要注意 p5.Image 在透明度方面做得很好(即使原始 gif 缺少它):如果采用这种较低级别的路线,您需要手动解决这些问题。


推荐阅读