-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathParticle.ts
153 lines (126 loc) · 3.37 KB
/
Particle.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
import { Vector2D as Vector } from "@coderosh/vector";
import type Heax from "./Heax";
class Particle {
/**
* Current position of particle
*/
public position: Vector;
/**
* Position of particle right before it achieved current position
*/
public oldPosition: Vector;
/**
* Radius of the particle
*/
public radius: number;
/**
* Color of the particle
*/
public color: string;
/**
* Mass of the particle
*/
public mass: number;
/**
* If true the particle will be pinned to the current position.
*/
public pinned: boolean;
/**
* If true the particle will not be rendered on the canvas
*/
public hidden: boolean;
/**
* Velocity of the particle (current position - old position)
*/
public velocity: Vector;
/**
* Instance of heax class
*/
public heax: Heax;
/**
* @param position Current position of particle
* @param oldPosition Position of particle right before it achieved current position
* @param heax Instance of heax class
* @param info Info about the particle
*/
constructor(
position: Vector,
oldPosition: Vector,
heax: Heax,
info: {
radius?: number;
color?: string;
mass?: number;
pinned?: boolean;
hidden?: boolean;
} = {}
) {
this.heax = heax;
this.position = position || new Vector();
this.oldPosition = oldPosition || new Vector();
this.radius = info.radius || 5;
this.color = info.color || "black";
this.mass = info.mass || 1;
this.hidden = info.hidden || false;
this.pinned = info.pinned || false;
this.velocity = this.position
.copy()
.sub(this.oldPosition)
.add(this.heax.gravity);
}
/**
* Renders the particle to the canvas
*/
render() {
if (this.hidden) return;
const { ctx } = this.heax;
ctx.beginPath();
ctx.fillStyle = this.color;
ctx.arc(this.position.x, this.position.y, this.radius, 0, Math.PI * 2);
ctx.fill();
ctx.closePath();
return this;
}
/**
* Update the position and velocity of the particle
*/
update() {
if (this.pinned) return;
this.velocity = this.position
.copy()
.sub(this.oldPosition)
.scale(this.heax.friction);
this.oldPosition.x = this.position.x;
this.oldPosition.y = this.position.y;
this.velocity.add(this.heax.gravity);
if (
this.position.y >= this.heax.height - this.radius &&
this.velocity.lenSqr() > 0
)
this.velocity.x *= this.heax.groundFriction;
this.position.add(this.velocity);
return this;
}
/**
* Constrain the particle within canvas height and width
*/
constrain() {
const { bounce, width, height } = this.heax;
if (this.position.x > width - this.radius) {
this.position.x = width - this.radius;
this.oldPosition.x = this.position.x + this.velocity.x * bounce;
} else if (this.position.x < this.radius) {
this.position.x = this.radius;
this.oldPosition.x = this.position.x + this.velocity.x * bounce;
}
if (this.position.y > height - this.radius) {
this.position.y = height - this.radius;
this.oldPosition.y = this.position.y + this.velocity.y * bounce;
} else if (this.position.y < this.radius) {
this.position.y = this.radius;
this.oldPosition.y = this.position.y + this.velocity.y * bounce;
}
return this;
}
}
export default Particle;