This is an expansion on the particle system I posted a few hours ago. I wanted to add movement to the particles within the circle and have them reflect off of the boundary as you would expect them to (see the bad-*** while
loop in the move
method.)
Check it out
If you resize your browser and refresh, the circular boundary will redraw to a slightly different size - makes for a different effect.
As before the FLA contains no source-code - simply the document class definition. The other 2 files are the ParticleSystem.as and Particle.as files.
ParticleSystem.as
package {
import Particle;
import flash.events.Event;
import flash.events.MouseEvent;
import flash.display.Sprite;
import flash.display.MovieClip;
import flash.display.Stage;
import flash.display.StageAlign;
import flash.display.StageScaleMode;
public class ParticleSpace extends MovieClip {
// Configuration Variables
var num_particles:Number = 20; // The number of particles to create on each draw
var distance:Number = 100; // The maximum distance for connected particles
var vx_max:Number = 3; // The maximum horizontal velocity
var vx_min:Number = -3; // The minimum horizontal velocity
var vy_max:Number = 3; // The maximum veritcal velocity
var vy_min:Number = -3; // The minimum vertical velocity
// Variables
var particles:Array = [];
var radius:Number;
var connections:Sprite;
var circle:Sprite;
/**
* Constructor
* Initialises the drawing and sets a listener to redraw on every mouse click
*/
public function ParticleSpace() {
// Prepare
stage.align = StageAlign.TOP_LEFT;
stage.scaleMode = StageScaleMode.NO_SCALE;
// Begin
this.draw();
}
/**
* Event handler
* @param event <Event> The event
*/
function enter_frame(event:Event):void {
this.move();
this.drawConnections(distance);
}
/**
* Move the balls around with their constraints
*/
function move():void {
var i:Number = 0;
var radius_sq:Number = radius * radius;
for (i = 0; i < particles.length; i++) {
var particle:Particle = particles*;
// Target destination
var tx:Number = particle.x + particle.vx;
var ty:Number = particle.y + particle.vy;
// Watch for those borders ...
while (tx * tx + ty * ty >= radius_sq) {
// Reference: http://mathworld.wolfram.com/Circle-LineIntersection.html
// Point 1: (ox, oy)
// Point 2: (tx, ty)
var ox:Number = particle.x;
var oy:Number = particle.y;
var dx:Number = tx - ox;
var dy:Number = ty - oy;
var d:Number = dx * dx + dy * dy;
var D:Number = ox * ty - tx * oy;
var disc:Number = radius_sq * d - D * D;
if (disc >= 0) {
disc = Math.sqrt(disc);
// Work out which intersection point to use (still works when there's only 1)
var ix:Number = (D * dy + sign(dy) * dx * disc) / d;
var iy:Number;
if (sign(ix - ox) != sign(tx - ox)) {
ix = (D * dy - sign(dy) * dx * disc) / d;
iy = ( -D * dx - Math.abs(dy) * disc) / d;
} else {
iy = ( -D * dx + Math.abs(dy) * disc) / d;
}
// Calculate the angle to bounce away from the intersection at
var a:Number = 2 * Math.atan2( -iy, -ix) - Math.atan2( -dy, -dx);
var ca:Number = Math.cos(a);
var sa:Number = Math.sin(a);
dx = tx - ix;
dy = ty - iy;
d = Math.sqrt(dx * dx + dy * dy);
tx = ix + d * ca;
ty = iy + d * sa;
var v:Number = particle.v;
particle.vx = v * ca;
particle.vy = v * sa;
}
}
// Reposition the particle to the newly calculated position
particle.x = tx;
particle.y = ty;
}
}
/**
* Returns the sign of n
* @param n
*/
function sign(n:Number) {
if (n < 0) return -1;
return 1;
}
/**
* Calculates the dimensions for drawing and calls the worker functions
* Begins the cycles
*/
function draw():void {
// Calculate the dimensions
radius = Math.min(stage.stageHeight, stage.stageWidth) / 2;
this.x = stage.stageWidth / 2;
this.y = stage.stageHeight / 2;
// Representation of limitations
this.addChild(this.circle = new Sprite());
this.circle.graphics.lineStyle(5, 0x333333, 1);
this.circle.graphics.drawCircle(0, 0, radius);
// Prepare
this.addChild(this.connections = new Sprite());
this.drawParticles(num_particles, radius);
// Continue
this.addEventListener(Event.ENTER_FRAME, this.enter_frame);
}
/**
* Creates the particles on the stage
* @param n <Number> The number of particles to create
* @param radius <Number> The radius of the circle within which to distribute the particles
* @param center_x <Number> The centre of the circle on the horizontal axis
* @param center_y <Number> The centre of the circle of the vertical axis
*/
function drawParticles(n:Number, radius:Number):void {
var i:Number;
// Create n particles
for (i = 0; i < n; i++) {
var particle:Particle = new Particle(this);
particles.push(particle);
// Disribute uniformly throughout the circle
var r:Number = Math.sqrt(Math.random()) * radius;
var a:Number = (Math.random() * 2 - 1) * Math.PI;
particle.x = Math.sin(a) * r;
particle.y = Math.cos(a) * r;
// Define a random velocity
particle.vx = Math.random() * (vx_max - vx_min) + vx_min;
particle.vy = Math.random() * (vy_max - vy_min) + vy_min;
}
}
/**
* Draws connecting lines between close-enough particles
* @param dist <Number> The maximum distance between particles for them to be connected
*/
function drawConnections(dist:Number):void {
var i:Number;
var j:Number;
// This method requires less recursion through less use of the Math.sqrt function
this.connections.graphics.clear();
var distance_sq:Number = distance * distance;
// Loop through each PAIR of particles
for (i = 0; i < particles.length - 1; i++) {
var particle_a:Particle = particles*;
for (j = i + 1; j < particles.length; j++) {
var particle_b:Particle = particles[j];
// Calculate the square of their distance apart
var dx:Number = particle_a.x - particle_b.x;
var dy:Number = particle_a.y - particle_b.y;
var d:Number = dx * dx + dy * dy;
if (d < distance_sq) {
// Draw a line if they're close enough
var alpha:Number = (distance - Math.sqrt(d)) / distance;
this.connections.graphics.lineStyle(1, 0xFFFFFF, alpha);
this.connections.graphics.moveTo(particle_a.x, particle_a.y);
this.connections.graphics.lineTo(particle_b.x, particle_b.y);
}
}
}
}
/**
* Empties the stage of all particles and graphics
* Pretty much redundant
*/
function clear():void {
for (var i:Number = 0; i < particles.length; i++) {
particles*.parent.removeChild(particles*);
}
this.particles = [];
this.connections.graphics.clear();
}
}
}
Particle.as
package {
import flash.display.Sprite;
class Particle extends Sprite {
// Variables
var v:Number;
var _vx:Number;
var _vy:Number;
public function Particle(parent:Sprite) {
// Draw the particle
this.graphics.beginFill(0xFFFFFF);
this.graphics.drawCircle(0, 0, 3);
this.graphics.endFill();
// Add the particle to the particle space
parent.addChild(this);
}
public function get vx():Number {
return this._vx;
}
public function set vx(n:Number):void {
this._vx = n;
this.v = Math.sqrt(this.vx * this.vx + this.vy * this.vy);
}
public function get vy():Number {
return this._vy;
}
public function set vy(n:Number):void {
this._vy = n;
this.v = Math.sqrt(this.vx * this.vx + this.vy * this.vy);
}
}
}
Critiques of course welcome.
Thanks