Vector2.prototype = {
copy : function() { return new Vector2(this.x, this.y); },
length : function() { return Math.sqrt(this.x * this.x + this.y * this.y); },
sqrLength : function() { return this.x * this.x + this.y * this.y; },
normalize : function() { var inv = 1/this.length(); return new Vector2(this.x * inv, this.y * inv); },
negate : function() { return new Vector2(-this.x, -this.y); },
add : function(v) { return new Vector2(this.x + v.x, this.y + v.y); },
subtract : function(v) { return new Vector2(this.x - v.x, this.y - v.y); },
multiply : function(f) { return new Vector2(this.x * f, this.y * f); },
divide : function(f) { var invf = 1/f; return new Vector2(this.x * invf, this.y * invf); },
dot : function(v) { return this.x * v.x + this.y * v.y; }
};
Vector2.zero = new Vector2(0, 0);
然后,就可以用HTML5 Canvas去描繪模擬的過程:
代碼如下:
var position = new Vector2(10, 200);
var velocity = new Vector2(50, -50);
var acceleration = new Vector2(0, 10);
var dt = 0.1;
function step() {
position = position.add(velocity.multiply(dt));
velocity = velocity.add(acceleration.multiply(dt));
ctx.strokeStyle = "#000000";
ctx.fillStyle = "#FFFFFF";
ctx.beginPath();
ctx.arc(position.x, position.y, 5, 0, Math.PI*2, true);
ctx.closePath();
ctx.fill();
ctx.stroke();
}
start("kinematicsCancas", step);
修改代碼試試看 |
以下是本文例子里實現的粒子類:
代碼如下:// Particle.js
Particle = function(position, velocity, life, color, size) {
this.position = position;
this.velocity = velocity;
this.acceleration = Vector2.zero;
this.age = 0;
this.life = life;
this.color = color;
this.size = size;
};
游戲循環
粒子系統通常可分為三個周期:
發射粒子
模擬粒子(粒子老化、碰撞、運動學模擬等等)
渲染粒子
在游戲循環(game loop)中,需要對每個粒子系統執行以上的三個步驟。
生與死
在本文的例子里,用一個JavaScript數組particles儲存所有活的粒子。產生一個粒子只是把它加到數組末端。代碼片段如下:
代碼如下://ParticleSystem.js
function ParticleSystem() {
// Private fields
var that = this;
var particles = new Array();
// Public fields
this.gravity = new Vector2(0, 100);
this.effectors = new Array();
// Public methods
this.emit = function(particle) {
particles.push(particle);
};
// ...
}
粒子在初始化時,年齡(age)設為零,生命(life)則是固定的。年齡和生命的單位都是秒。每個模擬步,都會把粒子老化,即是把年齡增加\Delta t,年齡超過生命,就會死亡。代碼片段如下:
代碼如下:function ParticleSystem() {
// ...
this.simulate = function(dt) {
aging(dt);
applyGravity();
applyEffectors();
kinematics(dt);
};
// ...
// Private methods
function aging(dt) {
for (var i = 0; i < particles.length; ) {
var p = particles[i];
p.age += dt;
if (p.age >= p.life)
kill(i);
else
i++;
}
}
function kill(index) {
if (particles.length > 1)
particles[index] = particles[particles.length - 1];
particles.pop();
}
// ...
}
在函數kill()里,用了一個技巧。因為粒子在數組里的次序并不重要,要刪除中間一個粒子,只需要復制最末的粒子到那個元素,并用pop()移除最末的粒子就可以。這通常比直接刪除數組中間的元素快(在C++中使用數組或std::vector亦是)。
運動學模擬
把本文最重要的兩句運動學模擬代碼套用至所有粒子就可以。另外,每次模擬會先把引力加速度寫入粒子的加速度。這樣做是為了將來可以每次改變加速度(續篇會談這方面)。
代碼如下:function ParticleSystem() {
// ...
function applyGravity() {
for (var i in particles)
particles[i].acceleration = that.gravity;
}
function kinematics(dt) {
for (var i in particles) {
var p = particles[i];
p.position = p.position.add(p.velocity.multiply(dt));
p.velocity = p.velocity.add(p.acceleration.multiply(dt));
}
}
// ...
}
渲染
粒子可以用很多不同方式渲染,例如用圓形、線段(當前位置和之前位置)、影像、精靈等等。本文采用圓形,并按年齡生命比來控制圓形的透明度,代碼片段如下:
代碼如下:function ParticleSystem() {
// ...
this.render = function(ctx) {
for (var i in particles) {
var p = particles[i];
var alpha = 1 - p.age / p.life;
ctx.fillStyle = "rgba("
+ Math.floor(p.color.r * 255) + ","
+ Math.floor(p.color.g * 255) + ","
+ Math.floor(p.color.b * 255) + ","
+ alpha.toFixed(2) + ")";
ctx.beginPath();
ctx.arc(p.position.x, p.position.y, p.size, 0, Math.PI * 2, true);
ctx.closePath();
ctx.fill();
}
}
// ...
}
基本粒子系統完成
以下的例子里,每幀會發射一個粒子,其位置在畫布中間(200,200),發射方向是360度,速率為100,生命為1秒,紅色、半徑為5象素。
代碼如下:
var ps = new ParticleSystem();
var dt = 0.01;
function sampleDirection() {
var theta = Math.random() * 2 * Math.PI;
return new Vector2(Math.cos(theta), Math.sin(theta));
}
function step() {
ps.emit(new Particle(new Vector2(200, 200), sampleDirection().multiply(100), 1, Color.red, 5));
ps.simulate(dt);
clearCanvas();
ps.render(ctx);
}
start("basicParticleSystemCanvas", step);
修改代碼試試看 |
互動發射
最后一個例子加入互動功能,在鼠標位置發射粒子,粒子方向是按鼠標移動速度再加上一點噪音(noise)。粒子的大小和生命都加入了隨機性。
代碼如下:var ps = new ParticleSystem();
ps.effectors.push(new ChamberBox(0, 0, 400, 400));
var dt = 0.01;
var oldMousePosition = Vector2.zero, newMousePosition = Vector2.zero;
function sampleDirection(angle1, angle2) {
var t = Math.random();
var theta = angle1 * t + angle2 * (1 - t);
return new Vector2(Math.cos(theta), Math.sin(theta));
}
function sampleColor(color1, color2) {
var t = Math.random();
return color1.multiply(t).add(color2.multiply(1 - t));
}
function sampleNumber(value1, value2) {
var t = Math.random();
return value1 * t + value2 * (1 - t);
}
function step() {
var velocity = newMousePosition.subtract(oldMousePosition).multiply(10);
velocity = velocity.add(sampleDirection(0, Math.PI * 2).multiply(20));
var color = sampleColor(Color.red, Color.yellow);
var life = sampleNumber(1, 2);
var size = sampleNumber(2, 4);
ps.emit(new Particle(newMousePosition, velocity, life, color, size));
oldMousePosition = newMousePosition;
ps.simulate(dt);
ctx.fillStyle="rgba(0, 0, 0, 0.1)";
ctx.fillRect(0,0,canvas.width,canvas.height);
ps.render(ctx);
}
start("interactiveEmitCanvas", step);
canvas.onmousemove = function(e) {
if (e.layerX || e.layerX == 0) { // Firefox
e.target.style.position='relative';
newMousePosition = new Vector2(e.layerX, e.layerY);
}
else
newMousePosition = new Vector2(e.offsetX, e.offsetY);
};
總結
本文介紹了最簡單的運動學模擬,使用歐拉方法作數值積分,并以此法去實現一個有簡單碰撞的粒子系統。本文的精華其實只有兩條簡單公式(只有兩個加數和兩個乘數),希望讓讀者明白,其實物理模擬可以很簡單。雖然本文的例子是在二維空間,但這例子能擴展至三維空間,只須把Vector2換成Vector3。本文完整源代碼可下載。
續篇會談及在此基礎上加入其他物理現象,有機會再加入其他物理模擬課題。希望各位支持,并給本人更多意見。
聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com