@sarmal/core 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,373 @@
1
+ // src/engine.ts
2
+ var TWO_PI = Math.PI * 2;
3
+ var POINTS_PER_PERIOD_UNIT = 50;
4
+ var CircularBuffer = class {
5
+ constructor(capacity) {
6
+ this.head = 0;
7
+ this.count = 0;
8
+ this.capacity = capacity;
9
+ this.data = Array.from({ length: capacity }, () => ({ x: 0, y: 0 }));
10
+ }
11
+ /**
12
+ * Array elements are pre-allocated and `head` pointer is manually assigned,
13
+ * because AI said using `Array.shift` would be `O(n)` and this would be `O(1)`
14
+ * and I don't know any better.
15
+ */
16
+ push(x, y) {
17
+ this.data[this.head] = { x, y };
18
+ this.head = (this.head + 1) % this.capacity;
19
+ if (this.count < this.capacity) {
20
+ this.count++;
21
+ }
22
+ }
23
+ toArray() {
24
+ const result = new Array(this.count);
25
+ const start = this.count < this.capacity ? 0 : this.head;
26
+ for (let i = 0; i < this.count; i++) {
27
+ const index = (start + i) % this.capacity;
28
+ const p = this.data[index];
29
+ result[i] = { x: p.x, y: p.y };
30
+ }
31
+ return result;
32
+ }
33
+ clear() {
34
+ this.head = 0;
35
+ this.count = 0;
36
+ }
37
+ get length() {
38
+ return this.count;
39
+ }
40
+ };
41
+ function createEngine(curveDef, trailLength = 120) {
42
+ const curve = {
43
+ name: curveDef.name,
44
+ fn: curveDef.fn,
45
+ period: curveDef.period ?? TWO_PI,
46
+ speed: curveDef.speed ?? 1
47
+ };
48
+ const trail = new CircularBuffer(trailLength);
49
+ let t = 0;
50
+ let actualTime = 0;
51
+ return {
52
+ tick(deltaTime) {
53
+ t = (t + curve.speed * deltaTime) % curve.period;
54
+ actualTime += deltaTime;
55
+ const point = curve.fn(t, actualTime, {});
56
+ trail.push(point.x, point.y);
57
+ return trail.toArray();
58
+ },
59
+ reset() {
60
+ t = 0;
61
+ actualTime = 0;
62
+ trail.clear();
63
+ },
64
+ getSarmalSkeleton() {
65
+ const steps = Math.ceil(curve.period * POINTS_PER_PERIOD_UNIT);
66
+ const points = new Array(steps);
67
+ for (let i = 0; i < steps; i++) {
68
+ const sampleT = i / (steps - 1) * curve.period;
69
+ const point = curve.fn(sampleT, 0, {});
70
+ points[i] = point;
71
+ }
72
+ return points;
73
+ }
74
+ };
75
+ }
76
+
77
+ // src/renderer.ts
78
+ var DEFAULT_HEAD_RADIUS = 4;
79
+ var DEFAULT_GLOW_SIZE = 20;
80
+ var DEFAULT_SKELETON_COLOR = "#ffffff";
81
+ var DEFAULT_SKELETON_OPACITY = 0.15;
82
+ var FIT_PADDING = 0.1;
83
+ var TRAIL_BATCH_SIZE = 10;
84
+ var TRAIL_FADE_CURVE = 1.5;
85
+ var TRAIL_MAX_OPACITY = 0.88;
86
+ var TRAIL_MIN_WIDTH = 0.5;
87
+ var TRAIL_MAX_WIDTH = 2.5;
88
+ var GLOW_INNER_EDGE = 0.4;
89
+ var GLOW_FALLOFF_OPACITY = 0.53;
90
+ function hexToRgba(hex, alpha) {
91
+ const n = parseInt(hex.slice(1), 16);
92
+ return `rgba(${n >> 16},${n >> 8 & 255},${n & 255},${alpha})`;
93
+ }
94
+ function createRenderer(options) {
95
+ const canvas = options.canvas;
96
+ if (!canvas.getContext("2d")) {
97
+ throw new Error("Could not get 2d context from canvas");
98
+ }
99
+ const ctx = canvas.getContext("2d");
100
+ const engine = options.engine;
101
+ const opts = {
102
+ skeletonColor: options.skeletonColor ?? DEFAULT_SKELETON_COLOR,
103
+ trailColor: options.trailColor ?? "#ffffff",
104
+ headColor: options.headColor ?? "#ffffff",
105
+ headRadius: options.headRadius ?? DEFAULT_HEAD_RADIUS,
106
+ glowSize: options.glowSize ?? DEFAULT_GLOW_SIZE
107
+ };
108
+ let skeleton = [];
109
+ let trail = [];
110
+ let head = null;
111
+ let scale = 1;
112
+ let offsetX = 0;
113
+ let offsetY = 0;
114
+ let animationId = null;
115
+ let lastTime = 0;
116
+ function calculateBoundaries() {
117
+ if (skeleton.length === 0) {
118
+ return;
119
+ }
120
+ const first = skeleton[0];
121
+ let minX = first.x, maxX = first.x, minY = first.y, maxY = first.y;
122
+ for (const p of skeleton) {
123
+ if (p.x < minX) {
124
+ minX = p.x;
125
+ }
126
+ if (p.x > maxX) {
127
+ maxX = p.x;
128
+ }
129
+ if (p.y < minY) {
130
+ minY = p.y;
131
+ }
132
+ if (p.y > maxY) {
133
+ maxY = p.y;
134
+ }
135
+ }
136
+ const width = maxX - minX;
137
+ const height = maxY - minY;
138
+ const canvasWidth = canvas.width;
139
+ const canvasHeight = canvas.height;
140
+ const scaleX = canvasWidth / (width * (1 + FIT_PADDING * 2));
141
+ const scaleY = canvasHeight / (height * (1 + FIT_PADDING * 2));
142
+ scale = Math.min(scaleX, scaleY);
143
+ const boundsWidth = width * scale;
144
+ const boundsHeight = height * scale;
145
+ offsetX = (canvasWidth - boundsWidth) / 2 - minX * scale;
146
+ offsetY = (canvasHeight - boundsHeight) / 2 - minY * scale;
147
+ }
148
+ function transformCoordinateToPixel(p) {
149
+ return {
150
+ x: p.x * scale + offsetX,
151
+ y: p.y * scale + offsetY
152
+ };
153
+ }
154
+ function drawSkeleton() {
155
+ if (skeleton.length < 2) {
156
+ return;
157
+ }
158
+ ctx.strokeStyle = hexToRgba(opts.skeletonColor, DEFAULT_SKELETON_OPACITY);
159
+ ctx.lineWidth = 1.5;
160
+ ctx.beginPath();
161
+ const firstPixel = transformCoordinateToPixel(skeleton[0]);
162
+ ctx.moveTo(firstPixel.x, firstPixel.y);
163
+ for (let i = 1; i < skeleton.length; i++) {
164
+ const pixel = transformCoordinateToPixel(skeleton[i]);
165
+ ctx.lineTo(pixel.x, pixel.y);
166
+ }
167
+ ctx.stroke();
168
+ }
169
+ function drawTrail() {
170
+ if (trail.length < 2) {
171
+ return;
172
+ }
173
+ for (let b = 0; b < trail.length - 1; b += TRAIL_BATCH_SIZE) {
174
+ const bEnd = Math.min(b + TRAIL_BATCH_SIZE, trail.length - 1);
175
+ const progress = (b + bEnd) / 2 / (trail.length - 1);
176
+ const alpha = Math.pow(progress, TRAIL_FADE_CURVE) * TRAIL_MAX_OPACITY;
177
+ const lineWidth = TRAIL_MIN_WIDTH + progress * (TRAIL_MAX_WIDTH - TRAIL_MIN_WIDTH);
178
+ ctx.beginPath();
179
+ for (let i = b; i <= bEnd; i++) {
180
+ const pixel = transformCoordinateToPixel(trail[i]);
181
+ if (i === b) {
182
+ ctx.moveTo(pixel.x, pixel.y);
183
+ } else {
184
+ ctx.lineTo(pixel.x, pixel.y);
185
+ }
186
+ }
187
+ ctx.strokeStyle = hexToRgba(opts.trailColor, alpha);
188
+ ctx.lineWidth = lineWidth;
189
+ ctx.lineJoin = "round";
190
+ ctx.lineCap = "round";
191
+ ctx.stroke();
192
+ }
193
+ }
194
+ function drawHead() {
195
+ if (!head) {
196
+ return;
197
+ }
198
+ const { x, y } = transformCoordinateToPixel(head);
199
+ const gradient = ctx.createRadialGradient(x, y, 0, x, y, opts.glowSize);
200
+ gradient.addColorStop(0, opts.headColor);
201
+ gradient.addColorStop(GLOW_INNER_EDGE, hexToRgba(opts.headColor, GLOW_FALLOFF_OPACITY));
202
+ gradient.addColorStop(1, "transparent");
203
+ ctx.fillStyle = gradient;
204
+ ctx.beginPath();
205
+ ctx.arc(x, y, opts.glowSize, 0, Math.PI * 2);
206
+ ctx.fill();
207
+ ctx.fillStyle = opts.headColor;
208
+ ctx.beginPath();
209
+ ctx.arc(x, y, opts.headRadius, 0, Math.PI * 2);
210
+ ctx.fill();
211
+ }
212
+ function render() {
213
+ const now = performance.now();
214
+ const deltaTime = (now - lastTime) / 1e3;
215
+ lastTime = now;
216
+ trail = engine.tick(deltaTime);
217
+ head = trail.length > 0 ? trail[trail.length - 1] : null;
218
+ ctx.clearRect(0, 0, canvas.width, canvas.height);
219
+ drawSkeleton();
220
+ drawTrail();
221
+ drawHead();
222
+ animationId = requestAnimationFrame(render);
223
+ }
224
+ skeleton = engine.getSarmalSkeleton();
225
+ calculateBoundaries();
226
+ return {
227
+ start() {
228
+ if (animationId !== null) {
229
+ return;
230
+ }
231
+ lastTime = performance.now();
232
+ render();
233
+ },
234
+ stop() {
235
+ if (animationId === null) {
236
+ return;
237
+ }
238
+ cancelAnimationFrame(animationId);
239
+ animationId = null;
240
+ },
241
+ reset() {
242
+ engine.reset();
243
+ trail = [];
244
+ head = null;
245
+ },
246
+ destroy() {
247
+ if (animationId !== null) {
248
+ cancelAnimationFrame(animationId);
249
+ animationId = null;
250
+ }
251
+ }
252
+ };
253
+ }
254
+
255
+ // src/curves.ts
256
+ var TWO_PI2 = Math.PI * 2;
257
+ function epitrochoid7(t, _time, _params) {
258
+ const d = 1 + 0.55 * Math.sin(t * 0.5);
259
+ return {
260
+ x: 7 * Math.cos(t) - d * Math.cos(7 * t),
261
+ y: 7 * Math.sin(t) - d * Math.sin(7 * t)
262
+ };
263
+ }
264
+ function astroid(t, _time, _params) {
265
+ const c = Math.cos(t);
266
+ const s = Math.sin(t);
267
+ return {
268
+ x: c * c * c,
269
+ y: s * s * s
270
+ };
271
+ }
272
+ function deltoid(t, _time, _params) {
273
+ return {
274
+ x: 2 * Math.cos(t) + Math.cos(2 * t),
275
+ y: 2 * Math.sin(t) - Math.sin(2 * t)
276
+ };
277
+ }
278
+ function rose5(t, _time, _params) {
279
+ const r = Math.cos(5 * t);
280
+ return {
281
+ x: r * Math.cos(t),
282
+ y: r * Math.sin(t)
283
+ };
284
+ }
285
+ function rose3(t, _time, _params) {
286
+ const r = Math.cos(3 * t);
287
+ return {
288
+ x: r * Math.cos(t),
289
+ y: r * Math.sin(t)
290
+ };
291
+ }
292
+ function artemis2(t, _time, _params) {
293
+ const a = 0.35, b = 0.15, ox = 0.175;
294
+ const s = Math.sin(t), c = Math.cos(t);
295
+ const denom = 1 + s * s;
296
+ return {
297
+ x: c * (1 + a * c) / denom - ox,
298
+ y: s * c * (1 + b * c) / denom
299
+ };
300
+ }
301
+ var curves = {
302
+ epitrochoid7: {
303
+ name: "Epitrochoid",
304
+ fn: epitrochoid7,
305
+ period: TWO_PI2,
306
+ speed: 1.4
307
+ },
308
+ astroid: {
309
+ name: "Astroid",
310
+ fn: astroid,
311
+ period: TWO_PI2,
312
+ speed: 1.1
313
+ },
314
+ deltoid: {
315
+ name: "Deltoid",
316
+ fn: deltoid,
317
+ period: TWO_PI2,
318
+ speed: 0.9
319
+ },
320
+ rose5: {
321
+ name: "Rose (n=5)",
322
+ fn: rose5,
323
+ period: TWO_PI2,
324
+ speed: 1
325
+ },
326
+ rose3: {
327
+ name: "Rose (n=3)",
328
+ fn: rose3,
329
+ period: TWO_PI2,
330
+ speed: 1.15
331
+ },
332
+ artemis2: {
333
+ name: "Artemis II",
334
+ fn: artemis2,
335
+ period: TWO_PI2,
336
+ speed: 0.7
337
+ }
338
+ };
339
+
340
+ // src/index.ts
341
+ function createSarmal(canvas, curveDef, options) {
342
+ const { trailLength, ...rendererOpts } = options ?? {};
343
+ const engine = createEngine(curveDef, trailLength);
344
+ return createRenderer({ canvas, engine, ...rendererOpts });
345
+ }
346
+
347
+ // src/auto-init.ts
348
+ function init() {
349
+ const canvases = document.querySelectorAll("canvas[data-sarmal]");
350
+ canvases.forEach((canvas) => {
351
+ const curveName = canvas.getAttribute("data-sarmal");
352
+ if (curveName == null) {
353
+ return console.warn("[sarmal] curveName isrequried");
354
+ }
355
+ const curveDef = curves[curveName];
356
+ if (!curveDef) {
357
+ return console.error(`[sarmal] "${curveName}" is not a valid curve name`);
358
+ }
359
+ const sarmal = createSarmal(canvas, curveDef, {
360
+ ...canvas.dataset.trailColor && { trailColor: canvas.dataset.trailColor },
361
+ ...canvas.dataset.skeletonColor && { skeletonColor: canvas.dataset.skeletonColor },
362
+ ...canvas.dataset.headColor && { headColor: canvas.dataset.headColor }
363
+ });
364
+ sarmal.start();
365
+ });
366
+ }
367
+ if (document.readyState === "loading") {
368
+ document.addEventListener("DOMContentLoaded", init);
369
+ } else {
370
+ init();
371
+ }
372
+ //# sourceMappingURL=auto-init.js.map
373
+ //# sourceMappingURL=auto-init.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/engine.ts","../src/renderer.ts","../src/curves.ts","../src/index.ts","../src/auto-init.ts"],"names":["TWO_PI"],"mappings":"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","file":"auto-init.js","sourcesContent":["import type { CurveDef, Engine, Point } from \"./types\";\n\nconst TWO_PI = Math.PI * 2;\nconst POINTS_PER_PERIOD_UNIT = 50;\n\n/**\n * A fixed-size list of points with first in, last out method\n * The oldest entry is automatically discarded when the list is at capacity\n */\nclass CircularBuffer {\n private data: Array<Point>;\n private capacity: number;\n private head: number = 0;\n private count: number = 0;\n\n constructor(capacity: number) {\n this.capacity = capacity;\n this.data = Array.from({ length: capacity }, () => ({ x: 0, y: 0 }));\n }\n\n /**\n * Array elements are pre-allocated and `head` pointer is manually assigned,\n * because AI said using `Array.shift` would be `O(n)` and this would be `O(1)`\n * and I don't know any better.\n */\n push(x: number, y: number): void {\n this.data[this.head] = { x: x, y: y };\n this.head = (this.head + 1) % this.capacity;\n if (this.count < this.capacity) {\n this.count++;\n }\n }\n\n toArray(): Array<Point> {\n // oxlint-disable-next-line unicorn/no-new-array -- AI said it is pre-allocated for performance (runs every frame)\n const result: Array<Point> = new Array(this.count);\n const start = this.count < this.capacity ? 0 : this.head;\n for (let i = 0; i < this.count; i++) {\n const index = (start + i) % this.capacity;\n const p = this.data[index]!;\n result[i] = { x: p.x, y: p.y };\n }\n return result;\n }\n\n clear(): void {\n this.head = 0;\n this.count = 0;\n }\n\n get length(): number {\n return this.count;\n }\n}\n\n/**\n * Creates the core simulation engine for a sarmal\n *\n * it runs a clock (time `t`), asks the curve for the current Point position at that time,\n * and remembers the last N positions so the renderer can draw the trail\n *\n * The engine is only responsible for math coordinates,\n * so it is not responsible for drawing or colors\n *\n * @param curveDef A curve definition\n * @param trailLength default: `120`\n */\nexport function createEngine(curveDef: CurveDef, trailLength: number = 120): Engine {\n const curve = {\n name: curveDef.name,\n fn: curveDef.fn,\n period: curveDef.period ?? TWO_PI,\n speed: curveDef.speed ?? 1,\n };\n const trail = new CircularBuffer(trailLength);\n let t = 0;\n let actualTime = 0;\n\n return {\n tick(deltaTime: number): Array<Point> {\n t = (t + curve.speed * deltaTime) % curve.period;\n actualTime += deltaTime;\n const point = curve.fn(t, actualTime, {});\n trail.push(point.x, point.y);\n return trail.toArray();\n },\n\n reset(): void {\n t = 0;\n actualTime = 0;\n trail.clear();\n },\n\n getSarmalSkeleton(): Array<Point> {\n const steps = Math.ceil(curve.period * POINTS_PER_PERIOD_UNIT);\n // oxlint-disable-next-line unicorn/no-new-array -- array is pre-allocated, filled immediately below\n const points: Array<Point> = new Array(steps);\n for (let i = 0; i < steps; i++) {\n const sampleT = (i / (steps - 1)) * curve.period;\n const point = curve.fn(sampleT, 0, {});\n points[i] = point;\n }\n return points;\n },\n };\n}\n","import type { Point, RendererOptions, SarmalInstance } from \"./types\";\n\nconst DEFAULT_HEAD_RADIUS = 4;\nconst DEFAULT_GLOW_SIZE = 20;\nconst DEFAULT_SKELETON_COLOR = \"#ffffff\";\nconst DEFAULT_SKELETON_OPACITY = 0.15;\n\n/** Fraction of the bounding box added as padding when fitting the curve to the canvas */\nconst FIT_PADDING = 0.1;\n\n/**\n * The trail is drawn in batches of points\n * Each batch has lower opacity than the one that comes before it\n * (0 = oldest/tail, 1 = newest/head)\n */\nconst TRAIL_BATCH_SIZE = 10;\n/** Higher values = sharper fade near the tail, more of the trail appears faint */\nconst TRAIL_FADE_CURVE = 1.5;\nconst TRAIL_MAX_OPACITY = 0.88;\n/** Line width of tail */\nconst TRAIL_MIN_WIDTH = 0.5;\n/** Line width of head */\nconst TRAIL_MAX_WIDTH = 2.5;\n\nconst GLOW_INNER_EDGE = 0.4;\n/** Opacity at the inner edge of the glow falloff */\nconst GLOW_FALLOFF_OPACITY = 0.53;\n\n// TODO: might as well accept rgb/rgba directly too\nfunction hexToRgba(hex: string, alpha: number): string {\n const n = parseInt(hex.slice(1), 16);\n return `rgba(${n >> 16},${(n >> 8) & 255},${n & 255},${alpha})`;\n}\n\n/**\n * Creates a Canvas 2D renderer for sarmal animations\n * Renders the skeleton, the trail, and the glowing dot.\n */\nexport function createRenderer(options: RendererOptions): SarmalInstance {\n const canvas = options.canvas;\n if (!canvas.getContext(\"2d\")) {\n throw new Error(\"Could not get 2d context from canvas\");\n }\n const ctx = canvas.getContext(\"2d\")!;\n\n const engine = options.engine;\n const opts = {\n skeletonColor: options.skeletonColor ?? DEFAULT_SKELETON_COLOR,\n trailColor: options.trailColor ?? \"#ffffff\",\n headColor: options.headColor ?? \"#ffffff\",\n headRadius: options.headRadius ?? DEFAULT_HEAD_RADIUS,\n glowSize: options.glowSize ?? DEFAULT_GLOW_SIZE,\n };\n\n let skeleton: Array<Point> = [];\n let trail: Array<Point> = [];\n let head: Point | null = null;\n let scale = 1;\n let offsetX = 0;\n let offsetY = 0;\n let animationId: number | null = null;\n let lastTime = 0;\n\n /**\n * Computes how to map engine coordinates to canvas pixels\n *\n * Steps are roughly: curve fn -> coordinate point -> (scale + offset) -> pixel\n *\n * 1. Find the bounding box of the skeleton (min/max x/y in coordinates)\n * 2. Compute a scale factor within the bounds into the canvas with padding\n * 3. Compute offsets to center the curve in the canvas\n */\n function calculateBoundaries() {\n if (skeleton.length === 0) {\n return;\n }\n\n const first = skeleton[0]!;\n let minX = first.x,\n maxX = first.x,\n minY = first.y,\n maxY = first.y;\n for (const p of skeleton) {\n if (p.x < minX) {\n minX = p.x;\n }\n\n if (p.x > maxX) {\n maxX = p.x;\n }\n\n if (p.y < minY) {\n minY = p.y;\n }\n\n if (p.y > maxY) {\n maxY = p.y;\n }\n }\n\n const width = maxX - minX;\n const height = maxY - minY;\n const canvasWidth = canvas.width;\n const canvasHeight = canvas.height;\n\n const scaleX = canvasWidth / (width * (1 + FIT_PADDING * 2));\n const scaleY = canvasHeight / (height * (1 + FIT_PADDING * 2));\n scale = Math.min(scaleX, scaleY);\n\n const boundsWidth = width * scale;\n const boundsHeight = height * scale;\n offsetX = (canvasWidth - boundsWidth) / 2 - minX * scale;\n offsetY = (canvasHeight - boundsHeight) / 2 - minY * scale;\n }\n\n function transformCoordinateToPixel(p: Point) {\n return {\n x: p.x * scale + offsetX,\n y: p.y * scale + offsetY,\n };\n }\n\n function drawSkeleton() {\n if (skeleton.length < 2) {\n return;\n }\n\n ctx.strokeStyle = hexToRgba(opts.skeletonColor, DEFAULT_SKELETON_OPACITY);\n ctx.lineWidth = 1.5;\n ctx.beginPath();\n\n const firstPixel = transformCoordinateToPixel(skeleton[0]!);\n ctx.moveTo(firstPixel.x, firstPixel.y);\n\n for (let i = 1; i < skeleton.length; i++) {\n const pixel = transformCoordinateToPixel(skeleton[i]!);\n ctx.lineTo(pixel.x, pixel.y);\n }\n\n ctx.stroke();\n }\n\n function drawTrail() {\n if (trail.length < 2) {\n return;\n }\n\n for (let b = 0; b < trail.length - 1; b += TRAIL_BATCH_SIZE) {\n const bEnd = Math.min(b + TRAIL_BATCH_SIZE, trail.length - 1);\n /** Normalized position of this batch along the trail (0 = tail, 1 = head) */\n const progress = (b + bEnd) / 2 / (trail.length - 1);\n const alpha = Math.pow(progress, TRAIL_FADE_CURVE) * TRAIL_MAX_OPACITY;\n const lineWidth = TRAIL_MIN_WIDTH + progress * (TRAIL_MAX_WIDTH - TRAIL_MIN_WIDTH);\n\n ctx.beginPath();\n for (let i = b; i <= bEnd; i++) {\n const pixel = transformCoordinateToPixel(trail[i]!);\n if (i === b) {\n ctx.moveTo(pixel.x, pixel.y);\n } else {\n ctx.lineTo(pixel.x, pixel.y);\n }\n }\n\n ctx.strokeStyle = hexToRgba(opts.trailColor, alpha);\n ctx.lineWidth = lineWidth;\n ctx.lineJoin = \"round\";\n ctx.lineCap = \"round\";\n ctx.stroke();\n }\n }\n\n function drawHead() {\n if (!head) {\n return;\n }\n\n const { x, y } = transformCoordinateToPixel(head);\n\n const gradient = ctx.createRadialGradient(x, y, 0, x, y, opts.glowSize);\n gradient.addColorStop(0, opts.headColor);\n gradient.addColorStop(GLOW_INNER_EDGE, hexToRgba(opts.headColor, GLOW_FALLOFF_OPACITY));\n gradient.addColorStop(1, \"transparent\");\n\n ctx.fillStyle = gradient;\n ctx.beginPath();\n ctx.arc(x, y, opts.glowSize, 0, Math.PI * 2);\n ctx.fill();\n\n ctx.fillStyle = opts.headColor;\n ctx.beginPath();\n ctx.arc(x, y, opts.headRadius, 0, Math.PI * 2);\n ctx.fill();\n }\n\n function render() {\n const now = performance.now();\n const deltaTime = (now - lastTime) / 1000;\n lastTime = now;\n\n trail = engine.tick(deltaTime);\n head = trail.length > 0 ? trail[trail.length - 1]! : null;\n\n ctx.clearRect(0, 0, canvas.width, canvas.height);\n\n drawSkeleton();\n drawTrail();\n drawHead();\n\n animationId = requestAnimationFrame(render);\n }\n\n // Initialize skeleton on creation\n skeleton = engine.getSarmalSkeleton();\n calculateBoundaries();\n\n return {\n start() {\n if (animationId !== null) {\n return;\n }\n\n lastTime = performance.now();\n render();\n },\n\n stop() {\n if (animationId === null) {\n return;\n }\n cancelAnimationFrame(animationId);\n animationId = null;\n },\n\n reset() {\n engine.reset();\n trail = [];\n head = null;\n },\n\n destroy() {\n if (animationId !== null) {\n cancelAnimationFrame(animationId);\n animationId = null;\n }\n },\n };\n}\n","import type { CurveDef, Point } from \"./types\";\n\nconst TWO_PI = Math.PI * 2;\n\n/** 7-lobed epitrochoid with a breathing distance parameter — d oscillates with t, making the loops pulse in and out */\nfunction epitrochoid7(t: number, _time: number, _params: Record<string, number>): Point {\n const d = 1.0 + 0.55 * Math.sin(t * 0.5);\n return {\n x: 7 * Math.cos(t) - d * Math.cos(7 * t),\n y: 7 * Math.sin(t) - d * Math.sin(7 * t),\n };\n}\n\nfunction astroid(t: number, _time: number, _params: Record<string, number>): Point {\n const c = Math.cos(t);\n const s = Math.sin(t);\n return {\n x: c * c * c,\n y: s * s * s,\n };\n}\n\nfunction deltoid(t: number, _time: number, _params: Record<string, number>): Point {\n return {\n x: 2 * Math.cos(t) + Math.cos(2 * t),\n y: 2 * Math.sin(t) - Math.sin(2 * t),\n };\n}\n\nfunction rose5(t: number, _time: number, _params: Record<string, number>): Point {\n const r = Math.cos(5 * t);\n return {\n x: r * Math.cos(t),\n y: r * Math.sin(t),\n };\n}\n\nfunction rose3(t: number, _time: number, _params: Record<string, number>): Point {\n const r = Math.cos(3 * t);\n return {\n x: r * Math.cos(t),\n y: r * Math.sin(t),\n };\n}\n\n/**\n * Artemis II free-return lunar trajectory\n * @see https://www.nasa.gov/wp-content/uploads/2025/09/artemis-ii-map-508.pdf\n * a = x-axis asymmetry (widens one lobe),\n * b = y-axis asymmetry,\n * ox = horizontal offset to visually center the shape\n */\nfunction artemis2(t: number, _time: number, _params: Record<string, number>): Point {\n const a = 0.35,\n b = 0.15,\n ox = 0.175;\n const s = Math.sin(t),\n c = Math.cos(t);\n const denom = 1 + s * s;\n return {\n x: (c * (1 + a * c)) / denom - ox,\n y: (s * c * (1 + b * c)) / denom,\n };\n}\n\nexport const curves: Record<string, CurveDef> = {\n epitrochoid7: {\n name: \"Epitrochoid\",\n fn: epitrochoid7,\n period: TWO_PI,\n speed: 1.4,\n },\n astroid: {\n name: \"Astroid\",\n fn: astroid,\n period: TWO_PI,\n speed: 1.1,\n },\n deltoid: {\n name: \"Deltoid\",\n fn: deltoid,\n period: TWO_PI,\n speed: 0.9,\n },\n rose5: {\n name: \"Rose (n=5)\",\n fn: rose5,\n period: TWO_PI,\n speed: 1.0,\n },\n rose3: {\n name: \"Rose (n=3)\",\n fn: rose3,\n period: TWO_PI,\n speed: 1.15,\n },\n artemis2: {\n name: \"Artemis II\",\n fn: artemis2,\n period: TWO_PI,\n speed: 0.7,\n },\n};\n","export type {\n Point,\n CurveDef,\n Engine,\n SarmalInstance,\n RendererOptions,\n SarmalOptions,\n} from \"./types\";\n\nexport { createEngine } from \"./engine\";\nexport { createRenderer } from \"./renderer\";\nexport { curves } from \"./curves\";\n\nimport type { CurveDef, SarmalInstance, SarmalOptions } from \"./types\";\nimport { createEngine } from \"./engine\";\nimport { createRenderer } from \"./renderer\";\n\n/**\n * Creates a sarmal animation on a canvas element\n *\n * @example\n * ```ts\n * import { createSarmal, curves } from 'sarmal'\n * const sarmal = createSarmal(canvas, curves.artemis2)\n * sarmal.start()\n * ```\n */\nexport function createSarmal(\n canvas: HTMLCanvasElement,\n curveDef: CurveDef,\n options?: SarmalOptions,\n): SarmalInstance {\n const { trailLength, ...rendererOpts } = options ?? {};\n const engine = createEngine(curveDef, trailLength);\n\n return createRenderer({ canvas, engine, ...rendererOpts });\n}\n","/**\n * Scans for `<canvas data-sarmal=\"curveName\">` when DOMContentLoaded is triggered,\n * and creates a Sarmal instance for each one\n *\n * Usage (CDN):\n * <script src=\"https://unpkg.com/sarmal/dist/auto-init.js\"></script>\n * <canvas data-sarmal=\"artemis2\" width=\"200\" height=\"200\"></canvas>\n *\n * Usage (ESM):\n * import 'sarmal/auto'\n */\nimport { createSarmal } from \"./index\";\nimport { curves } from \"./curves\";\n\nfunction init(): void {\n const canvases = document.querySelectorAll<HTMLCanvasElement>(\"canvas[data-sarmal]\");\n\n canvases.forEach((canvas) => {\n const curveName = canvas.getAttribute(\"data-sarmal\");\n if (curveName == null) {\n return console.warn(\"[sarmal] curveName isrequried\");\n }\n\n const curveDef = curves[curveName];\n if (!curveDef) {\n return console.error(`[sarmal] \"${curveName}\" is not a valid curve name`);\n }\n\n const sarmal = createSarmal(canvas, curveDef, {\n ...(canvas.dataset.trailColor && { trailColor: canvas.dataset.trailColor }),\n ...(canvas.dataset.skeletonColor && { skeletonColor: canvas.dataset.skeletonColor }),\n ...(canvas.dataset.headColor && { headColor: canvas.dataset.headColor }),\n });\n sarmal.start();\n });\n}\n\nif (document.readyState === \"loading\") {\n document.addEventListener(\"DOMContentLoaded\", init);\n} else {\n init();\n}\n"]}