@jayf0x/fluidity-js 0.1.6 → 0.2.1
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.
- package/README.md +65 -42
- package/dist/globals.d.ts +4 -0
- package/dist/index.js +1410 -515
- package/package.json +4 -1
package/dist/index.js
CHANGED
|
@@ -1,18 +1,18 @@
|
|
|
1
|
-
var
|
|
2
|
-
var
|
|
3
|
-
var
|
|
4
|
-
if (!
|
|
5
|
-
throw TypeError("Cannot " +
|
|
1
|
+
var EI = Object.defineProperty;
|
|
2
|
+
var jI = (e, g, C) => g in e ? EI(e, g, { enumerable: !0, configurable: !0, writable: !0, value: C }) : e[g] = C;
|
|
3
|
+
var sI = (e, g, C) => (jI(e, typeof g != "symbol" ? g + "" : g, C), C), nI = (e, g, C) => {
|
|
4
|
+
if (!g.has(e))
|
|
5
|
+
throw TypeError("Cannot " + C);
|
|
6
6
|
};
|
|
7
|
-
var
|
|
8
|
-
if (
|
|
7
|
+
var I = (e, g, C) => (nI(e, g, "read from private field"), C ? C.call(e) : g.get(e)), S = (e, g, C) => {
|
|
8
|
+
if (g.has(e))
|
|
9
9
|
throw TypeError("Cannot add the same private member more than once");
|
|
10
|
-
|
|
11
|
-
}, l = (
|
|
12
|
-
var
|
|
13
|
-
import { jsx as
|
|
14
|
-
import { useRef as
|
|
15
|
-
const
|
|
10
|
+
g instanceof WeakSet ? g.add(e) : g.set(e, C);
|
|
11
|
+
}, l = (e, g, C, i) => (nI(e, g, "write to private field"), i ? i.call(e, C) : g.set(e, C), C);
|
|
12
|
+
var h = (e, g, C) => (nI(e, g, "access private method"), C);
|
|
13
|
+
import { jsx as UI } from "react/jsx-runtime";
|
|
14
|
+
import { useRef as Mg, useEffect as eg, forwardRef as kI, useImperativeHandle as xI } from "react";
|
|
15
|
+
const fI = {
|
|
16
16
|
densityDissipation: 0.992,
|
|
17
17
|
velocityDissipation: 0.93,
|
|
18
18
|
pressureIterations: 1,
|
|
@@ -26,8 +26,8 @@ const OI = {
|
|
|
26
26
|
glowColor: [0.7, 0.85, 1],
|
|
27
27
|
algorithm: "standard",
|
|
28
28
|
warpStrength: 0.015
|
|
29
|
-
},
|
|
30
|
-
...
|
|
29
|
+
}, HI = {
|
|
30
|
+
...fI,
|
|
31
31
|
densityDissipation: 0.99,
|
|
32
32
|
velocityDissipation: 0.98,
|
|
33
33
|
pressureIterations: 3,
|
|
@@ -38,23 +38,23 @@ const OI = {
|
|
|
38
38
|
specularExp: 1,
|
|
39
39
|
shine: 0.1,
|
|
40
40
|
glowColor: [0, 0.502, 1]
|
|
41
|
-
},
|
|
41
|
+
}, PI = { dpr: 1, sim: 0.5 }, mI = {
|
|
42
42
|
backgroundColor: "#0a0a0a",
|
|
43
43
|
backgroundSize: "cover",
|
|
44
44
|
isMouseEnabled: !0,
|
|
45
45
|
isWorkerEnabled: !0,
|
|
46
|
-
quality:
|
|
47
|
-
},
|
|
48
|
-
...
|
|
46
|
+
quality: PI
|
|
47
|
+
}, Bg = {
|
|
48
|
+
...mI,
|
|
49
49
|
effect: 0,
|
|
50
50
|
imageSize: "cover"
|
|
51
|
-
},
|
|
52
|
-
...
|
|
51
|
+
}, Sg = {
|
|
52
|
+
...mI,
|
|
53
53
|
fontSize: 100,
|
|
54
54
|
color: "#ffffff",
|
|
55
55
|
fontFamily: "sans-serif",
|
|
56
56
|
fontWeight: 900
|
|
57
|
-
},
|
|
57
|
+
}, _I = {
|
|
58
58
|
calm: {
|
|
59
59
|
densityDissipation: 0.999,
|
|
60
60
|
velocityDissipation: 0.98,
|
|
@@ -113,73 +113,143 @@ const OI = {
|
|
|
113
113
|
waterColor: [0.06, 0.06, 0.06]
|
|
114
114
|
}
|
|
115
115
|
};
|
|
116
|
-
function
|
|
117
|
-
return { ...
|
|
116
|
+
function qg(e = {}, g, C = fI) {
|
|
117
|
+
return { ...g ? { ...C, ..._I[g] } : C, ...e };
|
|
118
118
|
}
|
|
119
|
-
function
|
|
120
|
-
let
|
|
121
|
-
s === "cover" ?
|
|
122
|
-
const
|
|
123
|
-
return { x: (
|
|
119
|
+
function Qg(e, g, C, i, s = "cover") {
|
|
120
|
+
let t;
|
|
121
|
+
s === "cover" ? t = Math.max(C / e, i / g) : s === "contain" ? t = Math.min(C / e, i / g) : typeof s == "string" && s.endsWith("%") ? t = Math.min(C / e, i / g) * (parseFloat(s) / 100) : typeof s == "string" && s.endsWith("px") ? t = parseFloat(s) / Math.max(e, g) : typeof s == "number" ? t = s : t = Math.max(C / e, i / g);
|
|
122
|
+
const o = e * t, a = g * t;
|
|
123
|
+
return { x: (C - o) / 2, y: (i - a) / 2, drawW: o, drawH: a };
|
|
124
124
|
}
|
|
125
|
-
function
|
|
126
|
-
const { text:
|
|
127
|
-
((
|
|
125
|
+
function qI(e, g, C, i, s = null, t = "cover") {
|
|
126
|
+
const { text: o, fontSize: a, color: u, fontFamily: A = "sans-serif", fontWeight: p = 900 } = i, c = new OffscreenCanvas(g, C), d = c.getContext("2d");
|
|
127
|
+
((b) => {
|
|
128
128
|
if (s) {
|
|
129
|
-
d.clearRect(0, 0,
|
|
130
|
-
const { x:
|
|
129
|
+
d.clearRect(0, 0, g, C), d.fillStyle = "black", d.fillRect(0, 0, g, C);
|
|
130
|
+
const { x: n, y: r, drawW: R, drawH: B } = Qg(
|
|
131
131
|
s.width,
|
|
132
132
|
s.height,
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
133
|
+
g,
|
|
134
|
+
C,
|
|
135
|
+
t
|
|
136
136
|
);
|
|
137
|
-
d.drawImage(s,
|
|
137
|
+
d.drawImage(s, n, r, R, B);
|
|
138
138
|
} else
|
|
139
|
-
d.fillStyle = "black", d.fillRect(0, 0,
|
|
140
|
-
d.fillStyle =
|
|
141
|
-
})(
|
|
142
|
-
const
|
|
143
|
-
d.fillStyle = "black", d.fillRect(0, 0,
|
|
144
|
-
const
|
|
145
|
-
return { backgroundTex:
|
|
139
|
+
d.fillStyle = "black", d.fillRect(0, 0, g, C);
|
|
140
|
+
d.fillStyle = b, d.font = `${p} ${a}px ${A}`, d.textAlign = "center", d.textBaseline = "middle", d.fillText(o, g / 2, C / 2);
|
|
141
|
+
})(u);
|
|
142
|
+
const m = Pg(e, c);
|
|
143
|
+
d.fillStyle = "black", d.fillRect(0, 0, g, C), d.fillStyle = "white", d.font = `${p} ${a}px ${A}`, d.textAlign = "center", d.textBaseline = "middle", d.fillText(o, g / 2, C / 2);
|
|
144
|
+
const w = Pg(e, c);
|
|
145
|
+
return { backgroundTex: m, obstacleTex: w, coverageTex: w };
|
|
146
146
|
}
|
|
147
|
-
function
|
|
148
|
-
const
|
|
149
|
-
if (
|
|
147
|
+
function $I(e, g, C, i, s = 0, t = "cover", o = null, a = "cover") {
|
|
148
|
+
const u = new OffscreenCanvas(C, i), A = u.getContext("2d"), { x: p, y: c, drawW: d, drawH: Z } = Qg(g.width, g.height, C, i, t);
|
|
149
|
+
if (A.clearRect(0, 0, C, i), A.fillStyle = "black", A.fillRect(0, 0, C, i), o) {
|
|
150
150
|
const {
|
|
151
|
-
x:
|
|
152
|
-
y:
|
|
153
|
-
drawW:
|
|
154
|
-
drawH:
|
|
155
|
-
} =
|
|
156
|
-
|
|
151
|
+
x: n,
|
|
152
|
+
y: r,
|
|
153
|
+
drawW: R,
|
|
154
|
+
drawH: B
|
|
155
|
+
} = Qg(o.width, o.height, C, i, a);
|
|
156
|
+
A.filter = `brightness(${s}) blur(8px)`, A.drawImage(o, n, r, R, B), A.filter = "none";
|
|
157
157
|
}
|
|
158
|
-
|
|
159
|
-
const
|
|
160
|
-
|
|
161
|
-
const
|
|
162
|
-
|
|
163
|
-
Math.max(0,
|
|
164
|
-
Math.max(0,
|
|
165
|
-
Math.min(d,
|
|
166
|
-
Math.min(
|
|
158
|
+
A.drawImage(g, p, c, d, Z);
|
|
159
|
+
const m = Pg(e, u);
|
|
160
|
+
A.clearRect(0, 0, C, i), A.fillStyle = "black", A.fillRect(0, 0, C, i), A.filter = `brightness(${s}) blur(8px)`, A.drawImage(g, p, c, d, Z), A.filter = "none";
|
|
161
|
+
const w = Pg(e, u);
|
|
162
|
+
A.clearRect(0, 0, C, i), A.fillStyle = "black", A.fillRect(0, 0, C, i), A.fillStyle = "white", A.fillRect(
|
|
163
|
+
Math.max(0, p),
|
|
164
|
+
Math.max(0, c),
|
|
165
|
+
Math.min(d, C - Math.max(0, p)),
|
|
166
|
+
Math.min(Z, i - Math.max(0, c))
|
|
167
167
|
);
|
|
168
|
-
const
|
|
169
|
-
return { backgroundTex:
|
|
168
|
+
const b = Pg(e, u);
|
|
169
|
+
return { backgroundTex: m, obstacleTex: w, coverageTex: b };
|
|
170
|
+
}
|
|
171
|
+
function Pg(e, g) {
|
|
172
|
+
const C = e.createTexture();
|
|
173
|
+
return e.bindTexture(e.TEXTURE_2D, C), e.pixelStorei(e.UNPACK_FLIP_Y_WEBGL, !0), e.texImage2D(e.TEXTURE_2D, 0, e.RGBA, e.RGBA, e.UNSIGNED_BYTE, g), e.texParameteri(e.TEXTURE_2D, e.TEXTURE_MIN_FILTER, e.LINEAR), e.texParameteri(e.TEXTURE_2D, e.TEXTURE_MAG_FILTER, e.LINEAR), e.texParameteri(e.TEXTURE_2D, e.TEXTURE_WRAP_S, e.CLAMP_TO_EDGE), e.texParameteri(e.TEXTURE_2D, e.TEXTURE_WRAP_T, e.CLAMP_TO_EDGE), C;
|
|
174
|
+
}
|
|
175
|
+
function gC(e, g, C, i, s = null, t = "cover") {
|
|
176
|
+
const { text: o, fontSize: a, color: u, fontFamily: A = "sans-serif", fontWeight: p = 900 } = i, c = new OffscreenCanvas(g, C), d = c.getContext("2d");
|
|
177
|
+
((b) => {
|
|
178
|
+
if (s) {
|
|
179
|
+
d.clearRect(0, 0, g, C), d.fillStyle = "black", d.fillRect(0, 0, g, C);
|
|
180
|
+
const { x: n, y: r, drawW: R, drawH: B } = Qg(
|
|
181
|
+
s.width,
|
|
182
|
+
s.height,
|
|
183
|
+
g,
|
|
184
|
+
C,
|
|
185
|
+
t
|
|
186
|
+
);
|
|
187
|
+
d.drawImage(s, n, r, R, B);
|
|
188
|
+
} else
|
|
189
|
+
d.fillStyle = "black", d.fillRect(0, 0, g, C);
|
|
190
|
+
d.fillStyle = b, d.font = `${p} ${a}px ${A}`, d.textAlign = "center", d.textBaseline = "middle", d.fillText(o, g / 2, C / 2);
|
|
191
|
+
})(u);
|
|
192
|
+
const m = _g(e, c, g, C);
|
|
193
|
+
d.fillStyle = "black", d.fillRect(0, 0, g, C), d.fillStyle = "white", d.font = `${p} ${a}px ${A}`, d.textAlign = "center", d.textBaseline = "middle", d.fillText(o, g / 2, C / 2);
|
|
194
|
+
const w = _g(e, c, g, C);
|
|
195
|
+
return {
|
|
196
|
+
backgroundTex: m,
|
|
197
|
+
backgroundView: m.createView(),
|
|
198
|
+
obstacleTex: w,
|
|
199
|
+
obstacleView: w.createView(),
|
|
200
|
+
coverageTex: w,
|
|
201
|
+
coverageView: w.createView(),
|
|
202
|
+
sharedCoverage: !0
|
|
203
|
+
};
|
|
170
204
|
}
|
|
171
|
-
function
|
|
172
|
-
const i =
|
|
173
|
-
|
|
205
|
+
function IC(e, g, C, i, s = 0, t = "cover", o = null, a = "cover") {
|
|
206
|
+
const u = new OffscreenCanvas(C, i), A = u.getContext("2d"), { x: p, y: c, drawW: d, drawH: Z } = Qg(g.width, g.height, C, i, t);
|
|
207
|
+
if (A.clearRect(0, 0, C, i), A.fillStyle = "black", A.fillRect(0, 0, C, i), o) {
|
|
208
|
+
const { x: n, y: r, drawW: R, drawH: B } = Qg(
|
|
209
|
+
o.width,
|
|
210
|
+
o.height,
|
|
211
|
+
C,
|
|
212
|
+
i,
|
|
213
|
+
a
|
|
214
|
+
);
|
|
215
|
+
A.filter = `brightness(${s}) blur(8px)`, A.drawImage(o, n, r, R, B), A.filter = "none";
|
|
216
|
+
}
|
|
217
|
+
A.drawImage(g, p, c, d, Z);
|
|
218
|
+
const m = _g(e, u, C, i);
|
|
219
|
+
A.clearRect(0, 0, C, i), A.fillStyle = "black", A.fillRect(0, 0, C, i), A.filter = `brightness(${s}) blur(8px)`, A.drawImage(g, p, c, d, Z), A.filter = "none";
|
|
220
|
+
const w = _g(e, u, C, i);
|
|
221
|
+
A.clearRect(0, 0, C, i), A.fillStyle = "black", A.fillRect(0, 0, C, i), A.fillStyle = "white", A.fillRect(Math.max(0, p), Math.max(0, c), Math.min(d, C - Math.max(0, p)), Math.min(Z, i - Math.max(0, c)));
|
|
222
|
+
const b = _g(e, u, C, i);
|
|
223
|
+
return {
|
|
224
|
+
backgroundTex: m,
|
|
225
|
+
backgroundView: m.createView(),
|
|
226
|
+
obstacleTex: w,
|
|
227
|
+
obstacleView: w.createView(),
|
|
228
|
+
coverageTex: b,
|
|
229
|
+
coverageView: b.createView(),
|
|
230
|
+
sharedCoverage: !1
|
|
231
|
+
};
|
|
232
|
+
}
|
|
233
|
+
function _g(e, g, C, i) {
|
|
234
|
+
const s = e.createTexture({
|
|
235
|
+
size: [C, i],
|
|
236
|
+
format: "rgba8unorm",
|
|
237
|
+
usage: GPUTextureUsage.TEXTURE_BINDING | GPUTextureUsage.COPY_DST | GPUTextureUsage.RENDER_ATTACHMENT
|
|
238
|
+
});
|
|
239
|
+
return e.queue.copyExternalImageToTexture(
|
|
240
|
+
{ source: g },
|
|
241
|
+
{ texture: s },
|
|
242
|
+
[C, i]
|
|
243
|
+
), s;
|
|
174
244
|
}
|
|
175
|
-
async function
|
|
176
|
-
const
|
|
177
|
-
if (!
|
|
178
|
-
throw new Error(`Failed to fetch image: ${
|
|
179
|
-
const
|
|
180
|
-
return createImageBitmap(
|
|
245
|
+
async function KI(e) {
|
|
246
|
+
const g = await fetch(e);
|
|
247
|
+
if (!g.ok)
|
|
248
|
+
throw new Error(`Failed to fetch image: ${e} (${g.status})`);
|
|
249
|
+
const C = await g.blob();
|
|
250
|
+
return createImageBitmap(C);
|
|
181
251
|
}
|
|
182
|
-
const
|
|
252
|
+
const ag = (
|
|
183
253
|
/* glsl */
|
|
184
254
|
`
|
|
185
255
|
precision highp float;
|
|
@@ -196,7 +266,7 @@ const II = (
|
|
|
196
266
|
gl_Position = vec4(aPosition, 0.0, 1.0);
|
|
197
267
|
}
|
|
198
268
|
`
|
|
199
|
-
),
|
|
269
|
+
), CC = (
|
|
200
270
|
/* glsl */
|
|
201
271
|
`
|
|
202
272
|
precision highp float;
|
|
@@ -213,7 +283,7 @@ const II = (
|
|
|
213
283
|
gl_FragColor = dissipation * texture2D(uSource, coord);
|
|
214
284
|
}
|
|
215
285
|
`
|
|
216
|
-
),
|
|
286
|
+
), iC = (
|
|
217
287
|
/* glsl */
|
|
218
288
|
`
|
|
219
289
|
precision highp float;
|
|
@@ -228,7 +298,7 @@ const II = (
|
|
|
228
298
|
gl_FragColor = vec4(0.5 * (R - L + T - B), 0.0, 0.0, 1.0);
|
|
229
299
|
}
|
|
230
300
|
`
|
|
231
|
-
),
|
|
301
|
+
), eC = (
|
|
232
302
|
/* glsl */
|
|
233
303
|
`
|
|
234
304
|
precision highp float;
|
|
@@ -246,7 +316,7 @@ const II = (
|
|
|
246
316
|
gl_FragColor = vec4((L + R + B + T - div) * 0.25, 0.0, 0.0, 1.0);
|
|
247
317
|
}
|
|
248
318
|
`
|
|
249
|
-
),
|
|
319
|
+
), sC = (
|
|
250
320
|
/* glsl */
|
|
251
321
|
`
|
|
252
322
|
precision highp float;
|
|
@@ -264,7 +334,7 @@ const II = (
|
|
|
264
334
|
gl_FragColor = vec4(vel, 0.0, 1.0);
|
|
265
335
|
}
|
|
266
336
|
`
|
|
267
|
-
),
|
|
337
|
+
), tC = (
|
|
268
338
|
/* glsl */
|
|
269
339
|
`
|
|
270
340
|
precision highp float;
|
|
@@ -281,7 +351,7 @@ const II = (
|
|
|
281
351
|
gl_FragColor = vec4(texture2D(uTarget, vUv).xyz + splat, 1.0);
|
|
282
352
|
}
|
|
283
353
|
`
|
|
284
|
-
),
|
|
354
|
+
), lC = (
|
|
285
355
|
/* glsl */
|
|
286
356
|
`
|
|
287
357
|
precision highp float;
|
|
@@ -295,7 +365,7 @@ const II = (
|
|
|
295
365
|
gl_FragColor = vec4(0.5 * (R - L - T + B), 0.0, 0.0, 1.0);
|
|
296
366
|
}
|
|
297
367
|
`
|
|
298
|
-
),
|
|
368
|
+
), oC = (
|
|
299
369
|
/* glsl */
|
|
300
370
|
`
|
|
301
371
|
precision highp float;
|
|
@@ -316,7 +386,7 @@ const II = (
|
|
|
316
386
|
gl_FragColor = vec4(texture2D(uVelocity, vUv).xy + force * dt, 0.0, 1.0);
|
|
317
387
|
}
|
|
318
388
|
`
|
|
319
|
-
),
|
|
389
|
+
), AC = (
|
|
320
390
|
/* glsl */
|
|
321
391
|
`
|
|
322
392
|
precision highp float;
|
|
@@ -417,37 +487,54 @@ const II = (
|
|
|
417
487
|
}
|
|
418
488
|
`
|
|
419
489
|
);
|
|
420
|
-
function
|
|
421
|
-
const
|
|
422
|
-
let
|
|
423
|
-
const
|
|
424
|
-
|
|
425
|
-
const s =
|
|
426
|
-
return
|
|
427
|
-
|
|
428
|
-
|
|
490
|
+
function dC(e) {
|
|
491
|
+
const g = { alpha: !0, depth: !1, stencil: !1, antialias: !0, preserveDrawingBuffer: !1 };
|
|
492
|
+
let C = e.getContext("webgl2", g);
|
|
493
|
+
const i = !!C;
|
|
494
|
+
i || (C = e.getContext("webgl", g), C.getExtension("EXT_color_buffer_half_float"));
|
|
495
|
+
const s = i ? null : C.getExtension("OES_texture_half_float"), t = i ? C.HALF_FLOAT : s.HALF_FLOAT_OES;
|
|
496
|
+
return C.getExtension("EXT_color_buffer_float"), C.getExtension("OES_texture_half_float_linear"), {
|
|
497
|
+
type: i ? "webgl2" : "webgl1",
|
|
498
|
+
gl: C,
|
|
499
|
+
isWebGL2: i,
|
|
429
500
|
ext: {
|
|
430
|
-
internalFormat:
|
|
431
|
-
format:
|
|
432
|
-
type:
|
|
501
|
+
internalFormat: i ? C.RGBA16F : C.RGBA,
|
|
502
|
+
format: C.RGBA,
|
|
503
|
+
type: t
|
|
433
504
|
}
|
|
434
505
|
};
|
|
435
506
|
}
|
|
436
|
-
|
|
437
|
-
|
|
438
|
-
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
|
|
444
|
-
|
|
445
|
-
|
|
507
|
+
async function cC(e) {
|
|
508
|
+
if (typeof navigator > "u" || !navigator.gpu)
|
|
509
|
+
return null;
|
|
510
|
+
try {
|
|
511
|
+
const g = await navigator.gpu.requestAdapter();
|
|
512
|
+
if (!g)
|
|
513
|
+
return null;
|
|
514
|
+
const C = await g.requestDevice(), i = e.getContext("webgpu");
|
|
515
|
+
if (!i)
|
|
516
|
+
return null;
|
|
517
|
+
const s = navigator.gpu.getPreferredCanvasFormat();
|
|
518
|
+
return i.configure({ device: C, format: s, alphaMode: "premultiplied" }), { type: "webgpu", adapter: g, device: C, context: i, format: s };
|
|
519
|
+
} catch {
|
|
520
|
+
return null;
|
|
521
|
+
}
|
|
522
|
+
}
|
|
523
|
+
class pg {
|
|
524
|
+
constructor(g, C, i) {
|
|
525
|
+
sI(this, "program");
|
|
526
|
+
sI(this, "uniforms", {});
|
|
527
|
+
sI(this, "_gl");
|
|
528
|
+
this._gl = g, this.program = g.createProgram(), g.attachShader(this.program, this._compile(g.VERTEX_SHADER, C)), g.attachShader(this.program, this._compile(g.FRAGMENT_SHADER, i)), g.linkProgram(this.program);
|
|
529
|
+
const s = g.getProgramParameter(this.program, g.ACTIVE_UNIFORMS);
|
|
530
|
+
for (let t = 0; t < s; t++) {
|
|
531
|
+
const o = g.getActiveUniform(this.program, t).name;
|
|
532
|
+
this.uniforms[o] = g.getUniformLocation(this.program, o);
|
|
446
533
|
}
|
|
447
534
|
}
|
|
448
|
-
_compile(
|
|
449
|
-
const
|
|
450
|
-
return
|
|
535
|
+
_compile(g, C) {
|
|
536
|
+
const i = this._gl, s = i.createShader(g);
|
|
537
|
+
return i.shaderSource(s, C), i.compileShader(s), s;
|
|
451
538
|
}
|
|
452
539
|
bind() {
|
|
453
540
|
this._gl.useProgram(this.program);
|
|
@@ -456,119 +543,650 @@ class gI {
|
|
|
456
543
|
this._gl.deleteProgram(this.program);
|
|
457
544
|
}
|
|
458
545
|
}
|
|
459
|
-
function
|
|
546
|
+
function SC(e) {
|
|
547
|
+
return {
|
|
548
|
+
advection: new pg(e, ag, CC),
|
|
549
|
+
divergence: new pg(e, ag, iC),
|
|
550
|
+
pressure: new pg(e, ag, eC),
|
|
551
|
+
gradientSubtract: new pg(e, ag, sC),
|
|
552
|
+
splat: new pg(e, ag, tC),
|
|
553
|
+
curl: new pg(e, ag, lC),
|
|
554
|
+
vorticity: new pg(e, ag, oC),
|
|
555
|
+
display: new pg(e, ag, AC)
|
|
556
|
+
};
|
|
557
|
+
}
|
|
558
|
+
function oI(e, g, C, i) {
|
|
559
|
+
e.activeTexture(e.TEXTURE0);
|
|
560
|
+
const s = e.createTexture();
|
|
561
|
+
e.bindTexture(e.TEXTURE_2D, s), e.texParameteri(e.TEXTURE_2D, e.TEXTURE_MIN_FILTER, e.LINEAR), e.texParameteri(e.TEXTURE_2D, e.TEXTURE_MAG_FILTER, e.LINEAR), e.texParameteri(e.TEXTURE_2D, e.TEXTURE_WRAP_S, e.CLAMP_TO_EDGE), e.texParameteri(e.TEXTURE_2D, e.TEXTURE_WRAP_T, e.CLAMP_TO_EDGE), e.texImage2D(e.TEXTURE_2D, 0, g.internalFormat, C, i, 0, g.format, g.type, null);
|
|
562
|
+
const t = e.createFramebuffer();
|
|
563
|
+
return e.bindFramebuffer(e.FRAMEBUFFER, t), e.framebufferTexture2D(e.FRAMEBUFFER, e.COLOR_ATTACHMENT0, e.TEXTURE_2D, s, 0), { tex: s, fbo: t, width: C, height: i };
|
|
564
|
+
}
|
|
565
|
+
function GI(e, g, C, i) {
|
|
566
|
+
let s = oI(e, g, C, i), t = oI(e, g, C, i);
|
|
460
567
|
return {
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
|
|
468
|
-
|
|
568
|
+
get read() {
|
|
569
|
+
return s;
|
|
570
|
+
},
|
|
571
|
+
get write() {
|
|
572
|
+
return t;
|
|
573
|
+
},
|
|
574
|
+
swap() {
|
|
575
|
+
[s, t] = [t, s];
|
|
576
|
+
},
|
|
577
|
+
dispose() {
|
|
578
|
+
e.deleteTexture(s.tex), e.deleteFramebuffer(s.fbo), e.deleteTexture(t.tex), e.deleteFramebuffer(t.fbo);
|
|
579
|
+
}
|
|
580
|
+
};
|
|
581
|
+
}
|
|
582
|
+
function aC(e) {
|
|
583
|
+
const g = e.createBuffer();
|
|
584
|
+
return e.bindBuffer(e.ARRAY_BUFFER, g), e.bufferData(e.ARRAY_BUFFER, new Float32Array([-1, -1, -1, 1, 1, 1, 1, -1]), e.STATIC_DRAW), e.vertexAttribPointer(0, 2, e.FLOAT, !1, 0, 0), e.enableVertexAttribArray(0), function(i) {
|
|
585
|
+
e.bindFramebuffer(e.FRAMEBUFFER, i), e.drawArrays(e.TRIANGLE_FAN, 0, 4);
|
|
469
586
|
};
|
|
470
587
|
}
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
|
|
474
|
-
|
|
475
|
-
|
|
476
|
-
|
|
588
|
+
const wg = (
|
|
589
|
+
/* wgsl */
|
|
590
|
+
`
|
|
591
|
+
struct VSOut {
|
|
592
|
+
@builtin(position) pos : vec4f,
|
|
593
|
+
@location(0) uv : vec2f,
|
|
594
|
+
@location(1) vL : vec2f,
|
|
595
|
+
@location(2) vR : vec2f,
|
|
596
|
+
@location(3) vT : vec2f,
|
|
597
|
+
@location(4) vB : vec2f,
|
|
598
|
+
}`
|
|
599
|
+
), pC = (
|
|
600
|
+
/* wgsl */
|
|
601
|
+
`
|
|
602
|
+
${wg}
|
|
603
|
+
|
|
604
|
+
struct U {
|
|
605
|
+
texelSize : vec2f,
|
|
606
|
+
dt : f32,
|
|
607
|
+
dissipation: f32,
|
|
608
|
+
}
|
|
609
|
+
@group(0) @binding(0) var<uniform> u : U;
|
|
610
|
+
@group(0) @binding(1) var samp : sampler;
|
|
611
|
+
@group(0) @binding(2) var uVel : texture_2d<f32>;
|
|
612
|
+
@group(0) @binding(3) var uSrc : texture_2d<f32>;
|
|
613
|
+
@group(0) @binding(4) var uObs : texture_2d<f32>;
|
|
614
|
+
|
|
615
|
+
@vertex fn vs(@location(0) a: vec2f) -> VSOut {
|
|
616
|
+
var o: VSOut;
|
|
617
|
+
o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
|
|
618
|
+
o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
|
|
619
|
+
o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
|
|
620
|
+
o.vT = o.uv + vec2f(0.0, u.texelSize.y);
|
|
621
|
+
o.vB = o.uv - vec2f(0.0, u.texelSize.y);
|
|
622
|
+
o.pos = vec4f(a, 0.0, 1.0);
|
|
623
|
+
return o;
|
|
624
|
+
}
|
|
625
|
+
|
|
626
|
+
@fragment fn fs(i: VSOut) -> @location(0) vec4f {
|
|
627
|
+
// Sample all textures before branching — textureSample requires uniform control flow
|
|
628
|
+
let obs = textureSample(uObs, samp, i.uv).r;
|
|
629
|
+
let vel = textureSample(uVel, samp, i.uv).xy;
|
|
630
|
+
let coord = i.uv - u.dt * vel * u.texelSize;
|
|
631
|
+
let src = textureSample(uSrc, samp, coord);
|
|
632
|
+
if (obs > 0.5) { return vec4f(0.0); }
|
|
633
|
+
return u.dissipation * src;
|
|
634
|
+
}
|
|
635
|
+
`
|
|
636
|
+
), uC = (
|
|
637
|
+
/* wgsl */
|
|
638
|
+
`
|
|
639
|
+
${wg}
|
|
640
|
+
|
|
641
|
+
struct U { texelSize: vec2f, _pad: vec2f }
|
|
642
|
+
@group(0) @binding(0) var<uniform> u : U;
|
|
643
|
+
@group(0) @binding(1) var samp : sampler;
|
|
644
|
+
@group(0) @binding(2) var uVel : texture_2d<f32>;
|
|
645
|
+
@group(0) @binding(3) var uObs : texture_2d<f32>;
|
|
646
|
+
|
|
647
|
+
@vertex fn vs(@location(0) a: vec2f) -> VSOut {
|
|
648
|
+
var o: VSOut;
|
|
649
|
+
o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
|
|
650
|
+
o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
|
|
651
|
+
o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
|
|
652
|
+
o.vT = o.uv + vec2f(0.0, u.texelSize.y);
|
|
653
|
+
o.vB = o.uv - vec2f(0.0, u.texelSize.y);
|
|
654
|
+
o.pos = vec4f(a, 0.0, 1.0);
|
|
655
|
+
return o;
|
|
656
|
+
}
|
|
657
|
+
|
|
658
|
+
@fragment fn fs(i: VSOut) -> @location(0) vec4f {
|
|
659
|
+
let L = select(textureSample(uVel, samp, i.vL).x, 0.0, textureSample(uObs, samp, i.vL).r > 0.5);
|
|
660
|
+
let R = select(textureSample(uVel, samp, i.vR).x, 0.0, textureSample(uObs, samp, i.vR).r > 0.5);
|
|
661
|
+
let T = select(textureSample(uVel, samp, i.vT).y, 0.0, textureSample(uObs, samp, i.vT).r > 0.5);
|
|
662
|
+
let B = select(textureSample(uVel, samp, i.vB).y, 0.0, textureSample(uObs, samp, i.vB).r > 0.5);
|
|
663
|
+
return vec4f(0.5 * (R - L + T - B), 0.0, 0.0, 1.0);
|
|
664
|
+
}
|
|
665
|
+
`
|
|
666
|
+
), BC = (
|
|
667
|
+
/* wgsl */
|
|
668
|
+
`
|
|
669
|
+
${wg}
|
|
670
|
+
|
|
671
|
+
struct U { texelSize: vec2f, _pad: vec2f }
|
|
672
|
+
@group(0) @binding(0) var<uniform> u : U;
|
|
673
|
+
@group(0) @binding(1) var samp : sampler;
|
|
674
|
+
@group(0) @binding(2) var uPres: texture_2d<f32>;
|
|
675
|
+
@group(0) @binding(3) var uDiv : texture_2d<f32>;
|
|
676
|
+
@group(0) @binding(4) var uObs : texture_2d<f32>;
|
|
677
|
+
|
|
678
|
+
@vertex fn vs(@location(0) a: vec2f) -> VSOut {
|
|
679
|
+
var o: VSOut;
|
|
680
|
+
o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
|
|
681
|
+
o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
|
|
682
|
+
o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
|
|
683
|
+
o.vT = o.uv + vec2f(0.0, u.texelSize.y);
|
|
684
|
+
o.vB = o.uv - vec2f(0.0, u.texelSize.y);
|
|
685
|
+
o.pos = vec4f(a, 0.0, 1.0);
|
|
686
|
+
return o;
|
|
687
|
+
}
|
|
688
|
+
|
|
689
|
+
@fragment fn fs(i: VSOut) -> @location(0) vec4f {
|
|
690
|
+
let C = textureSample(uPres, samp, i.uv).x;
|
|
691
|
+
let L = select(textureSample(uPres, samp, i.vL).x, C, textureSample(uObs, samp, i.vL).r > 0.5);
|
|
692
|
+
let R = select(textureSample(uPres, samp, i.vR).x, C, textureSample(uObs, samp, i.vR).r > 0.5);
|
|
693
|
+
let T = select(textureSample(uPres, samp, i.vT).x, C, textureSample(uObs, samp, i.vT).r > 0.5);
|
|
694
|
+
let B = select(textureSample(uPres, samp, i.vB).x, C, textureSample(uObs, samp, i.vB).r > 0.5);
|
|
695
|
+
let dv = textureSample(uDiv, samp, i.uv).x;
|
|
696
|
+
return vec4f((L + R + B + T - dv) * 0.25, 0.0, 0.0, 1.0);
|
|
697
|
+
}
|
|
698
|
+
`
|
|
699
|
+
), ZC = (
|
|
700
|
+
/* wgsl */
|
|
701
|
+
`
|
|
702
|
+
${wg}
|
|
703
|
+
|
|
704
|
+
struct U { texelSize: vec2f, _pad: vec2f }
|
|
705
|
+
@group(0) @binding(0) var<uniform> u : U;
|
|
706
|
+
@group(0) @binding(1) var samp : sampler;
|
|
707
|
+
@group(0) @binding(2) var uPres: texture_2d<f32>;
|
|
708
|
+
@group(0) @binding(3) var uVel : texture_2d<f32>;
|
|
709
|
+
@group(0) @binding(4) var uObs : texture_2d<f32>;
|
|
710
|
+
|
|
711
|
+
@vertex fn vs(@location(0) a: vec2f) -> VSOut {
|
|
712
|
+
var o: VSOut;
|
|
713
|
+
o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
|
|
714
|
+
o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
|
|
715
|
+
o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
|
|
716
|
+
o.vT = o.uv + vec2f(0.0, u.texelSize.y);
|
|
717
|
+
o.vB = o.uv - vec2f(0.0, u.texelSize.y);
|
|
718
|
+
o.pos = vec4f(a, 0.0, 1.0);
|
|
719
|
+
return o;
|
|
720
|
+
}
|
|
721
|
+
|
|
722
|
+
@fragment fn fs(i: VSOut) -> @location(0) vec4f {
|
|
723
|
+
// Sample all textures before branching — textureSample requires uniform control flow
|
|
724
|
+
let obs = textureSample(uObs, samp, i.uv).r;
|
|
725
|
+
let L = textureSample(uPres, samp, i.vL).x;
|
|
726
|
+
let R = textureSample(uPres, samp, i.vR).x;
|
|
727
|
+
let T = textureSample(uPres, samp, i.vT).x;
|
|
728
|
+
let B = textureSample(uPres, samp, i.vB).x;
|
|
729
|
+
let vel = textureSample(uVel, samp, i.uv).xy - vec2f(R - L, T - B);
|
|
730
|
+
if (obs > 0.5) { return vec4f(0.0); }
|
|
731
|
+
return vec4f(vel, 0.0, 1.0);
|
|
732
|
+
}
|
|
733
|
+
`
|
|
734
|
+
), mC = (
|
|
735
|
+
/* wgsl */
|
|
736
|
+
`
|
|
737
|
+
${wg}
|
|
738
|
+
|
|
739
|
+
// texelSize occupies bytes 0-7 for the shared vertex stage; aspectRatio/radius fill the rest.
|
|
740
|
+
struct U {
|
|
741
|
+
texelSize : vec2f,
|
|
742
|
+
aspectRatio: f32,
|
|
743
|
+
radius : f32,
|
|
744
|
+
color : vec4f, // xyz = colour, w unused
|
|
745
|
+
point : vec2f,
|
|
746
|
+
_pad : vec2f,
|
|
747
|
+
}
|
|
748
|
+
@group(0) @binding(0) var<uniform> u : U;
|
|
749
|
+
@group(0) @binding(1) var samp : sampler;
|
|
750
|
+
@group(0) @binding(2) var uTgt : texture_2d<f32>;
|
|
751
|
+
|
|
752
|
+
@vertex fn vs(@location(0) a: vec2f) -> VSOut {
|
|
753
|
+
var o: VSOut;
|
|
754
|
+
o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
|
|
755
|
+
o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
|
|
756
|
+
o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
|
|
757
|
+
o.vT = o.uv + vec2f(0.0, u.texelSize.y);
|
|
758
|
+
o.vB = o.uv - vec2f(0.0, u.texelSize.y);
|
|
759
|
+
o.pos = vec4f(a, 0.0, 1.0);
|
|
760
|
+
return o;
|
|
761
|
+
}
|
|
762
|
+
|
|
763
|
+
@fragment fn fs(i: VSOut) -> @location(0) vec4f {
|
|
764
|
+
var p = i.uv - u.point;
|
|
765
|
+
p.x *= u.aspectRatio;
|
|
766
|
+
let sp = exp(-dot(p, p) / u.radius) * u.color.xyz;
|
|
767
|
+
return vec4f(textureSample(uTgt, samp, i.uv).xyz + sp, 1.0);
|
|
768
|
+
}
|
|
769
|
+
`
|
|
770
|
+
), nC = (
|
|
771
|
+
/* wgsl */
|
|
772
|
+
`
|
|
773
|
+
${wg}
|
|
774
|
+
|
|
775
|
+
struct U { texelSize: vec2f, _pad: vec2f }
|
|
776
|
+
@group(0) @binding(0) var<uniform> u : U;
|
|
777
|
+
@group(0) @binding(1) var samp : sampler;
|
|
778
|
+
@group(0) @binding(2) var uVel : texture_2d<f32>;
|
|
779
|
+
|
|
780
|
+
@vertex fn vs(@location(0) a: vec2f) -> VSOut {
|
|
781
|
+
var o: VSOut;
|
|
782
|
+
o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
|
|
783
|
+
o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
|
|
784
|
+
o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
|
|
785
|
+
o.vT = o.uv + vec2f(0.0, u.texelSize.y);
|
|
786
|
+
o.vB = o.uv - vec2f(0.0, u.texelSize.y);
|
|
787
|
+
o.pos = vec4f(a, 0.0, 1.0);
|
|
788
|
+
return o;
|
|
789
|
+
}
|
|
790
|
+
|
|
791
|
+
@fragment fn fs(i: VSOut) -> @location(0) vec4f {
|
|
792
|
+
let L = textureSample(uVel, samp, i.vL).y;
|
|
793
|
+
let R = textureSample(uVel, samp, i.vR).y;
|
|
794
|
+
let T = textureSample(uVel, samp, i.vT).x;
|
|
795
|
+
let B = textureSample(uVel, samp, i.vB).x;
|
|
796
|
+
return vec4f(0.5 * (R - L - T + B), 0.0, 0.0, 1.0);
|
|
797
|
+
}
|
|
798
|
+
`
|
|
799
|
+
), GC = (
|
|
800
|
+
/* wgsl */
|
|
801
|
+
`
|
|
802
|
+
${wg}
|
|
803
|
+
|
|
804
|
+
struct U {
|
|
805
|
+
texelSize: vec2f,
|
|
806
|
+
curl : f32,
|
|
807
|
+
dt : f32,
|
|
808
|
+
}
|
|
809
|
+
@group(0) @binding(0) var<uniform> u : U;
|
|
810
|
+
@group(0) @binding(1) var samp : sampler;
|
|
811
|
+
@group(0) @binding(2) var uVel : texture_2d<f32>;
|
|
812
|
+
@group(0) @binding(3) var uCrl : texture_2d<f32>;
|
|
813
|
+
|
|
814
|
+
@vertex fn vs(@location(0) a: vec2f) -> VSOut {
|
|
815
|
+
var o: VSOut;
|
|
816
|
+
o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
|
|
817
|
+
o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
|
|
818
|
+
o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
|
|
819
|
+
o.vT = o.uv + vec2f(0.0, u.texelSize.y);
|
|
820
|
+
o.vB = o.uv - vec2f(0.0, u.texelSize.y);
|
|
821
|
+
o.pos = vec4f(a, 0.0, 1.0);
|
|
822
|
+
return o;
|
|
823
|
+
}
|
|
824
|
+
|
|
825
|
+
@fragment fn fs(i: VSOut) -> @location(0) vec4f {
|
|
826
|
+
let L = textureSample(uCrl, samp, i.vL).x;
|
|
827
|
+
let R = textureSample(uCrl, samp, i.vR).x;
|
|
828
|
+
let T = textureSample(uCrl, samp, i.vT).x;
|
|
829
|
+
let B = textureSample(uCrl, samp, i.vB).x;
|
|
830
|
+
let C = textureSample(uCrl, samp, i.uv).x;
|
|
831
|
+
var force = 0.5 * vec2f(abs(T) - abs(B), abs(R) - abs(L));
|
|
832
|
+
force /= length(force) + 0.0001;
|
|
833
|
+
force *= u.curl * 30.0 * C;
|
|
834
|
+
let vel = textureSample(uVel, samp, i.uv).xy + force * u.dt;
|
|
835
|
+
return vec4f(vel, 0.0, 1.0);
|
|
836
|
+
}
|
|
837
|
+
`
|
|
838
|
+
), rC = (
|
|
839
|
+
/* wgsl */
|
|
840
|
+
`
|
|
841
|
+
${wg}
|
|
842
|
+
|
|
843
|
+
struct U {
|
|
844
|
+
texelSize : vec2f,
|
|
845
|
+
refraction : f32,
|
|
846
|
+
specularExp : f32,
|
|
847
|
+
waterColor : vec4f,
|
|
848
|
+
glowColor : vec4f,
|
|
849
|
+
shine : f32,
|
|
850
|
+
warpStrength: f32,
|
|
851
|
+
algorithm : i32,
|
|
852
|
+
_pad : f32,
|
|
853
|
+
}
|
|
854
|
+
@group(0) @binding(0) var<uniform> u : U;
|
|
855
|
+
@group(0) @binding(1) var samp : sampler;
|
|
856
|
+
@group(0) @binding(2) var uTex : texture_2d<f32>;
|
|
857
|
+
@group(0) @binding(3) var uObs : texture_2d<f32>;
|
|
858
|
+
@group(0) @binding(4) var uBg : texture_2d<f32>;
|
|
859
|
+
@group(0) @binding(5) var uCov : texture_2d<f32>;
|
|
860
|
+
@group(0) @binding(6) var uVel : texture_2d<f32>;
|
|
861
|
+
|
|
862
|
+
@vertex fn vs(@location(0) a: vec2f) -> VSOut {
|
|
863
|
+
var o: VSOut;
|
|
864
|
+
o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
|
|
865
|
+
o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
|
|
866
|
+
o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
|
|
867
|
+
o.vT = o.uv + vec2f(0.0, u.texelSize.y);
|
|
868
|
+
o.vB = o.uv - vec2f(0.0, u.texelSize.y);
|
|
869
|
+
o.pos = vec4f(a, 0.0, 1.0);
|
|
870
|
+
return o;
|
|
871
|
+
}
|
|
872
|
+
|
|
873
|
+
@fragment fn fs(i: VSOut) -> @location(0) vec4f {
|
|
874
|
+
let obs = textureSample(uObs, samp, i.uv).r;
|
|
875
|
+
let density = max(textureSample(uTex, samp, i.uv).r, 0.0) * (1.0 - step(0.5, obs));
|
|
876
|
+
let cov = textureSample(uCov, samp, i.uv).r;
|
|
877
|
+
|
|
878
|
+
let ts2 = u.texelSize * 2.0;
|
|
879
|
+
let dL = max(textureSample(uTex, samp, i.uv - vec2f(ts2.x, 0.0)).r, 0.0);
|
|
880
|
+
let dR = max(textureSample(uTex, samp, i.uv + vec2f(ts2.x, 0.0)).r, 0.0);
|
|
881
|
+
let dT = max(textureSample(uTex, samp, i.uv + vec2f(0.0, ts2.y)).r, 0.0);
|
|
882
|
+
let dB = max(textureSample(uTex, samp, i.uv - vec2f(0.0, ts2.y)).r, 0.0);
|
|
883
|
+
|
|
884
|
+
let norm = normalize(vec3f(dL - dR, dB - dT, 0.2));
|
|
885
|
+
let ldir = normalize(vec3f(0.5, 1.0, 0.5));
|
|
886
|
+
let halfV = normalize(ldir + vec3f(0.0, 0.0, 1.0));
|
|
887
|
+
let spec = pow(max(dot(norm, halfV), 0.0), u.specularExp) * u.shine * density;
|
|
888
|
+
|
|
889
|
+
let bgRaw = textureSample(uBg, samp, i.uv).rgb;
|
|
890
|
+
let wc = u.waterColor.rgb;
|
|
891
|
+
let gc = u.glowColor.rgb;
|
|
892
|
+
let bg = mix(wc, bgRaw, cov);
|
|
893
|
+
var color = bg;
|
|
894
|
+
|
|
895
|
+
if (u.algorithm == 1) {
|
|
896
|
+
let ruv = clamp(i.uv + norm.xy * u.refraction * density * 3.0, vec2f(0.0), vec2f(1.0));
|
|
897
|
+
let rbg = mix(wc, textureSample(uBg, samp, mix(i.uv, ruv, 1.0 - obs)).rgb, cov);
|
|
898
|
+
color = rbg + spec * gc * 2.5;
|
|
899
|
+
color = mix(color, bg * 0.6, obs * 0.3);
|
|
900
|
+
} else if (u.algorithm == 2) {
|
|
901
|
+
let inkD = min(density * 4.0, 1.0);
|
|
902
|
+
let ruv = clamp(i.uv + norm.xy * u.refraction * density * 0.4, vec2f(0.0), vec2f(1.0));
|
|
903
|
+
let rbg = mix(wc, textureSample(uBg, samp, mix(i.uv, ruv, 1.0 - obs)).rgb, cov);
|
|
904
|
+
color = mix(rbg, wc + spec * gc, inkD);
|
|
905
|
+
color = mix(color, bg * 0.5, obs * 0.15);
|
|
906
|
+
} else if (u.algorithm == 3) {
|
|
907
|
+
let vel = textureSample(uVel, samp, i.uv).xy;
|
|
908
|
+
let velMag = clamp(length(vel) * 20.0, 0.0, 1.0);
|
|
909
|
+
let wuv = clamp(i.uv + vel * u.warpStrength, vec2f(0.0), vec2f(1.0));
|
|
910
|
+
let wbg = mix(wc, textureSample(uBg, samp, wuv).rgb, cov);
|
|
911
|
+
color = mix(bg, wbg, velMag * (1.0 - obs));
|
|
912
|
+
color += spec * gc * velMag * 1.5;
|
|
913
|
+
color += wc * density * 0.3;
|
|
914
|
+
color = mix(color, bg * 0.5, obs * 0.2);
|
|
915
|
+
} else if (u.algorithm == 4) {
|
|
916
|
+
let ruv = clamp(i.uv + norm.xy * u.refraction * density * 6.0, vec2f(0.0), vec2f(1.0));
|
|
917
|
+
let rbg = mix(wc, textureSample(uBg, samp, mix(i.uv, ruv, 1.0 - obs)).rgb, cov);
|
|
918
|
+
let fres = pow(clamp(1.0 - dot(norm, vec3f(0.0, 0.0, 1.0)), 0.0, 1.0), 3.0) * density;
|
|
919
|
+
color = rbg;
|
|
920
|
+
color += fres * gc * 2.0;
|
|
921
|
+
color += spec * gc * density * 2.0;
|
|
922
|
+
color = mix(color, bg * 0.5, obs * 0.2);
|
|
923
|
+
} else {
|
|
924
|
+
let ruv = i.uv + norm.xy * u.refraction * density;
|
|
925
|
+
let rbg = mix(wc, textureSample(uBg, samp, mix(i.uv, ruv, 1.0 - obs)).rgb, cov);
|
|
926
|
+
color = mix(rbg, wc, min(density * 1.5, 0.8));
|
|
927
|
+
color += spec * gc;
|
|
928
|
+
color = mix(color, bg * 0.5, obs * 0.2);
|
|
929
|
+
}
|
|
930
|
+
|
|
931
|
+
let alpha = clamp(max(density * 1.5, cov), 0.0, 1.0);
|
|
932
|
+
return vec4f(color * alpha, alpha);
|
|
933
|
+
}
|
|
934
|
+
`
|
|
935
|
+
), LC = {
|
|
936
|
+
arrayStride: 8,
|
|
937
|
+
attributes: [{ shaderLocation: 0, offset: 0, format: "float32x2" }]
|
|
938
|
+
}, XI = new Float32Array([
|
|
939
|
+
-1,
|
|
940
|
+
-1,
|
|
941
|
+
-1,
|
|
942
|
+
1,
|
|
943
|
+
1,
|
|
944
|
+
-1,
|
|
945
|
+
1,
|
|
946
|
+
-1,
|
|
947
|
+
-1,
|
|
948
|
+
1,
|
|
949
|
+
1,
|
|
950
|
+
1
|
|
951
|
+
]);
|
|
952
|
+
function wC(e) {
|
|
953
|
+
const g = e.createBuffer({ size: XI.byteLength, usage: GPUBufferUsage.VERTEX | GPUBufferUsage.COPY_DST });
|
|
954
|
+
return e.queue.writeBuffer(g, 0, XI), g;
|
|
955
|
+
}
|
|
956
|
+
function AI(e, g, C, i) {
|
|
957
|
+
const s = e.createTexture({
|
|
958
|
+
size: [C, i],
|
|
959
|
+
format: g,
|
|
960
|
+
usage: GPUTextureUsage.TEXTURE_BINDING | GPUTextureUsage.RENDER_ATTACHMENT | GPUTextureUsage.COPY_SRC
|
|
961
|
+
});
|
|
962
|
+
return { tex: s, view: s.createView(), width: C, height: i };
|
|
477
963
|
}
|
|
478
|
-
function
|
|
479
|
-
let s =
|
|
964
|
+
function rI(e, g, C, i) {
|
|
965
|
+
let s = AI(e, g, C, i), t = AI(e, g, C, i);
|
|
480
966
|
return {
|
|
481
967
|
get read() {
|
|
482
968
|
return s;
|
|
483
969
|
},
|
|
484
970
|
get write() {
|
|
485
|
-
return
|
|
971
|
+
return t;
|
|
486
972
|
},
|
|
487
973
|
swap() {
|
|
488
|
-
[s,
|
|
974
|
+
[s, t] = [t, s];
|
|
489
975
|
},
|
|
490
976
|
dispose() {
|
|
491
|
-
|
|
977
|
+
s.tex.destroy(), t.tex.destroy();
|
|
492
978
|
}
|
|
493
979
|
};
|
|
494
980
|
}
|
|
495
|
-
function
|
|
496
|
-
const
|
|
497
|
-
return
|
|
498
|
-
|
|
981
|
+
function ug(e, g, C) {
|
|
982
|
+
const i = e.createShaderModule({ code: g });
|
|
983
|
+
return e.createRenderPipeline({
|
|
984
|
+
layout: "auto",
|
|
985
|
+
vertex: { module: i, entryPoint: "vs", buffers: [LC] },
|
|
986
|
+
fragment: { module: i, entryPoint: "fs", targets: [{ format: C }] },
|
|
987
|
+
primitive: { topology: "triangle-list" }
|
|
988
|
+
});
|
|
989
|
+
}
|
|
990
|
+
function bC(e, g) {
|
|
991
|
+
const C = "rgba16float";
|
|
992
|
+
return {
|
|
993
|
+
advection: ug(e, pC, C),
|
|
994
|
+
divergence: ug(e, uC, C),
|
|
995
|
+
pressure: ug(e, BC, C),
|
|
996
|
+
gradientSubtract: ug(e, ZC, C),
|
|
997
|
+
splat: ug(e, mC, C),
|
|
998
|
+
curl: ug(e, nC, C),
|
|
999
|
+
vorticity: ug(e, GC, C),
|
|
1000
|
+
display: ug(e, rC, g)
|
|
499
1001
|
};
|
|
500
1002
|
}
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
1003
|
+
function hC(e) {
|
|
1004
|
+
return e.createSampler({ magFilter: "linear", minFilter: "linear", addressModeU: "clamp-to-edge", addressModeV: "clamp-to-edge" });
|
|
1005
|
+
}
|
|
1006
|
+
function _(e, g) {
|
|
1007
|
+
return e.createBuffer({ size: g, usage: GPUBufferUsage.UNIFORM | GPUBufferUsage.COPY_DST });
|
|
1008
|
+
}
|
|
1009
|
+
function RI(e, g, C, i, s, t) {
|
|
1010
|
+
const o = new Float32Array([C, i, s, t]);
|
|
1011
|
+
e.queue.writeBuffer(g, 0, o);
|
|
1012
|
+
}
|
|
1013
|
+
function tI(e, g, C, i) {
|
|
1014
|
+
const s = new Float32Array([C, i, 0, 0]);
|
|
1015
|
+
e.queue.writeBuffer(g, 0, s);
|
|
1016
|
+
}
|
|
1017
|
+
function KC(e, g, C, i, s, t) {
|
|
1018
|
+
const o = new Float32Array([C, i, s, t]);
|
|
1019
|
+
e.queue.writeBuffer(g, 0, o);
|
|
1020
|
+
}
|
|
1021
|
+
function lI(e, g, C, i, s, t, o, a, u, A, p) {
|
|
1022
|
+
const c = new Float32Array(12);
|
|
1023
|
+
c[0] = C, c[1] = i, c[2] = s, c[3] = t, c[4] = o, c[5] = a, c[6] = u, c[7] = 0, c[8] = A, c[9] = p, c[10] = 0, c[11] = 0, e.queue.writeBuffer(g, 0, c);
|
|
1024
|
+
}
|
|
1025
|
+
function yC(e, g, C, i, s, t, o, a, u, A, p) {
|
|
1026
|
+
const c = new Float32Array(16), d = new Int32Array(c.buffer);
|
|
1027
|
+
c[0] = C, c[1] = i, c[2] = s, c[3] = t, c[4] = o[0], c[5] = o[1], c[6] = o[2], c[7] = 0, c[8] = a[0], c[9] = a[1], c[10] = a[2], c[11] = 0, c[12] = u, c[13] = A, d[14] = p, e.queue.writeBuffer(g, 0, c);
|
|
1028
|
+
}
|
|
1029
|
+
function N(e, g, C, i, s) {
|
|
1030
|
+
const t = e.beginRenderPass({
|
|
1031
|
+
colorAttachments: [{
|
|
1032
|
+
view: s,
|
|
1033
|
+
clearValue: [0, 0, 0, 0],
|
|
1034
|
+
loadOp: "clear",
|
|
1035
|
+
storeOp: "store"
|
|
1036
|
+
}]
|
|
1037
|
+
});
|
|
1038
|
+
t.setPipeline(g), t.setBindGroup(0, C), t.setVertexBuffer(0, i), t.draw(6), t.end();
|
|
1039
|
+
}
|
|
1040
|
+
function VC(e, g, C, i, s) {
|
|
1041
|
+
const t = e.beginRenderPass({
|
|
1042
|
+
colorAttachments: [{
|
|
1043
|
+
view: s,
|
|
1044
|
+
clearValue: [0, 0, 0, 0],
|
|
1045
|
+
loadOp: "clear",
|
|
1046
|
+
storeOp: "store"
|
|
1047
|
+
}]
|
|
1048
|
+
});
|
|
1049
|
+
t.setPipeline(g), t.setBindGroup(0, C), t.setVertexBuffer(0, i), t.draw(6), t.end();
|
|
1050
|
+
}
|
|
1051
|
+
const vI = typeof requestAnimationFrame < "u" ? requestAnimationFrame.bind(globalThis) : (e) => setTimeout(e, 1e3 / 60), HC = typeof cancelAnimationFrame < "u" ? cancelAnimationFrame.bind(globalThis) : clearTimeout, Og = 0.016, WI = { standard: 0, glass: 1, ink: 2, aurora: 3, ripple: 4 };
|
|
1052
|
+
var bg, Q, $g, hg, Fg, x, H, q, $, gg, sg, T, Ig, F, Dg, Kg, Tg, f, y, Zg, Cg, mg, z, yg, Vg, Hg, Xg, Rg, tg, lg, vg, Wg, Yg, V, X, O, E, D, Jg, og, J, ng, Ag, L, k, ig, Ug, gI, cI, MI, kg, Eg, II, LI, Gg, fg, xg, jg, CI, wI, iI, bI, SI, FI, aI, DI, pI, TI, uI, JI, BI, zI;
|
|
1053
|
+
const yI = class yI {
|
|
1054
|
+
// ── Constructor ─────────────────────────────────────────────────────────────
|
|
1055
|
+
constructor(g, C = {}, i = {}, s) {
|
|
1056
|
+
// ---------------------------------------------------------------------------
|
|
1057
|
+
// Private — GPU initialisation
|
|
1058
|
+
// ---------------------------------------------------------------------------
|
|
1059
|
+
S(this, cI);
|
|
505
1060
|
// ---------------------------------------------------------------------------
|
|
506
|
-
// Private helpers
|
|
1061
|
+
// Private — shared helpers
|
|
507
1062
|
// ---------------------------------------------------------------------------
|
|
508
|
-
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
|
|
514
|
-
|
|
515
|
-
|
|
516
|
-
|
|
517
|
-
|
|
518
|
-
|
|
519
|
-
|
|
520
|
-
|
|
521
|
-
|
|
522
|
-
|
|
523
|
-
|
|
524
|
-
|
|
525
|
-
|
|
526
|
-
|
|
527
|
-
|
|
528
|
-
|
|
529
|
-
|
|
530
|
-
|
|
531
|
-
|
|
532
|
-
|
|
533
|
-
|
|
534
|
-
|
|
535
|
-
//
|
|
536
|
-
|
|
537
|
-
|
|
538
|
-
|
|
539
|
-
|
|
540
|
-
|
|
541
|
-
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
|
|
546
|
-
|
|
547
|
-
|
|
548
|
-
|
|
1063
|
+
S(this, kg);
|
|
1064
|
+
S(this, II);
|
|
1065
|
+
S(this, Gg);
|
|
1066
|
+
S(this, xg);
|
|
1067
|
+
S(this, CI);
|
|
1068
|
+
S(this, iI);
|
|
1069
|
+
// ---------------------------------------------------------------------------
|
|
1070
|
+
// Private — frame dispatch
|
|
1071
|
+
// ---------------------------------------------------------------------------
|
|
1072
|
+
S(this, SI);
|
|
1073
|
+
// ---------------------------------------------------------------------------
|
|
1074
|
+
// Private — WebGPU simulation step
|
|
1075
|
+
// ---------------------------------------------------------------------------
|
|
1076
|
+
S(this, aI);
|
|
1077
|
+
// ---------------------------------------------------------------------------
|
|
1078
|
+
// Private — WebGPU direct splat
|
|
1079
|
+
// ---------------------------------------------------------------------------
|
|
1080
|
+
S(this, pI);
|
|
1081
|
+
// ---------------------------------------------------------------------------
|
|
1082
|
+
// Private — WebGL splat
|
|
1083
|
+
// ---------------------------------------------------------------------------
|
|
1084
|
+
S(this, uI);
|
|
1085
|
+
// ---------------------------------------------------------------------------
|
|
1086
|
+
// Private — WebGL simulation step (unchanged from original)
|
|
1087
|
+
// ---------------------------------------------------------------------------
|
|
1088
|
+
S(this, BI);
|
|
1089
|
+
S(this, bg, void 0);
|
|
1090
|
+
// ── WebGL path ──────────────────────────────────────────────────────────────
|
|
1091
|
+
S(this, Q, null);
|
|
1092
|
+
S(this, $g, null);
|
|
1093
|
+
S(this, hg, null);
|
|
1094
|
+
S(this, Fg, null);
|
|
1095
|
+
S(this, x, null);
|
|
1096
|
+
S(this, H, null);
|
|
1097
|
+
S(this, q, null);
|
|
1098
|
+
S(this, $, null);
|
|
1099
|
+
S(this, gg, null);
|
|
1100
|
+
S(this, sg, null);
|
|
1101
|
+
S(this, T, null);
|
|
1102
|
+
S(this, Ig, null);
|
|
1103
|
+
// ── WebGPU path ─────────────────────────────────────────────────────────────
|
|
1104
|
+
S(this, F, null);
|
|
1105
|
+
S(this, Dg, null);
|
|
1106
|
+
S(this, Kg, null);
|
|
1107
|
+
S(this, Tg, null);
|
|
1108
|
+
S(this, f, null);
|
|
1109
|
+
S(this, y, null);
|
|
1110
|
+
S(this, Zg, null);
|
|
1111
|
+
S(this, Cg, null);
|
|
1112
|
+
S(this, mg, null);
|
|
1113
|
+
S(this, z, null);
|
|
1114
|
+
// Pre-allocated uniform buffers (sizes: see gpu-utils writeXxx docs)
|
|
1115
|
+
// Velocity/density advection use separate buffers — writeBuffer is a queue op;
|
|
1116
|
+
// a second write to the same buffer before queue.submit() aliases both passes.
|
|
1117
|
+
S(this, yg, null);
|
|
1118
|
+
// 16 bytes — velocity advection
|
|
1119
|
+
S(this, Vg, null);
|
|
1120
|
+
// 16 bytes — density advection
|
|
1121
|
+
S(this, Hg, null);
|
|
1122
|
+
// 16 bytes
|
|
1123
|
+
S(this, Xg, null);
|
|
1124
|
+
// 16 bytes
|
|
1125
|
+
S(this, Rg, null);
|
|
1126
|
+
// 16 bytes
|
|
1127
|
+
S(this, tg, null);
|
|
1128
|
+
// 48 bytes — velocity splat
|
|
1129
|
+
S(this, lg, null);
|
|
1130
|
+
// 48 bytes — density splat
|
|
1131
|
+
S(this, vg, null);
|
|
1132
|
+
// 16 bytes
|
|
1133
|
+
S(this, Wg, null);
|
|
1134
|
+
// 16 bytes
|
|
1135
|
+
S(this, Yg, null);
|
|
1136
|
+
// 64 bytes
|
|
1137
|
+
// ── Shared state ────────────────────────────────────────────────────────────
|
|
1138
|
+
S(this, V, 0);
|
|
1139
|
+
S(this, X, 0);
|
|
1140
|
+
S(this, O, 0);
|
|
1141
|
+
S(this, E, 0);
|
|
1142
|
+
S(this, D, 1);
|
|
1143
|
+
S(this, Jg, 1);
|
|
1144
|
+
S(this, og, 0.5);
|
|
1145
|
+
S(this, J, null);
|
|
1146
|
+
S(this, ng, "cover");
|
|
1147
|
+
S(this, Ag, void 0);
|
|
1148
|
+
S(this, L, { x: 0, y: 0, dx: 0, dy: 0, targetX: 0, targetY: 0, moved: !1 });
|
|
1149
|
+
S(this, k, null);
|
|
1150
|
+
S(this, ig, null);
|
|
1151
|
+
S(this, Ug, !1);
|
|
1152
|
+
S(this, gI, !1);
|
|
1153
|
+
if (l(this, bg, g), l(this, Jg, Math.max(0.1, Math.min(1, i.dpr ?? 1))), l(this, og, Math.max(0.1, Math.min(1, i.sim ?? 0.5))), l(this, Ag, qg(C)), s)
|
|
1154
|
+
l(this, F, s), h(this, cI, MI).call(this, s);
|
|
1155
|
+
else {
|
|
1156
|
+
const { gl: t, ext: o } = dC(g);
|
|
1157
|
+
l(this, Q, t), l(this, $g, o), l(this, hg, SC(t)), l(this, Fg, aC(t)), t.clearColor(0, 0, 0, 0);
|
|
1158
|
+
}
|
|
1159
|
+
}
|
|
1160
|
+
/**
|
|
1161
|
+
* WebGPU-first factory. Tries WebGPU, falls back to WebGL2 → WebGL1.
|
|
1162
|
+
* This is the recommended entry point when WebGPU support is desired.
|
|
1163
|
+
*/
|
|
1164
|
+
static async create(g, C = {}, i = {}, s = !0) {
|
|
1165
|
+
const t = s ? await cC(g) : null;
|
|
1166
|
+
return new yI(g, C, i, t ?? void 0);
|
|
549
1167
|
}
|
|
550
1168
|
// ---------------------------------------------------------------------------
|
|
551
1169
|
// Public API
|
|
552
1170
|
// ---------------------------------------------------------------------------
|
|
553
|
-
setTextSource(
|
|
554
|
-
l(this,
|
|
1171
|
+
setTextSource(g) {
|
|
1172
|
+
l(this, k, { type: "text", opts: g }), h(this, kg, Eg).call(this), h(this, Gg, fg).call(this), h(this, xg, jg).call(this);
|
|
555
1173
|
}
|
|
556
|
-
async setImageSource(
|
|
557
|
-
const s = await
|
|
558
|
-
if (
|
|
1174
|
+
async setImageSource(g, C = 0, i = "cover") {
|
|
1175
|
+
const s = await KI(g);
|
|
1176
|
+
if (I(this, gI)) {
|
|
559
1177
|
s.close();
|
|
560
1178
|
return;
|
|
561
1179
|
}
|
|
562
|
-
l(this,
|
|
1180
|
+
l(this, k, { type: "image", bitmap: s, effect: C, size: i }), h(this, kg, Eg).call(this), h(this, Gg, fg).call(this), h(this, xg, jg).call(this);
|
|
563
1181
|
}
|
|
564
|
-
setImageBitmap(
|
|
565
|
-
l(this,
|
|
1182
|
+
setImageBitmap(g, C = 0, i = "cover") {
|
|
1183
|
+
l(this, k, { type: "image", bitmap: g, effect: C, size: i }), h(this, kg, Eg).call(this), h(this, Gg, fg).call(this), h(this, xg, jg).call(this);
|
|
566
1184
|
}
|
|
567
|
-
setBackground(
|
|
568
|
-
|
|
1185
|
+
setBackground(g, C = "cover") {
|
|
1186
|
+
I(this, J) && I(this, J) !== g && I(this, J).close(), l(this, J, g), l(this, ng, C ?? "cover"), I(this, k) && I(this, V) > 0 && I(this, X) > 0 && h(this, Gg, fg).call(this);
|
|
569
1187
|
}
|
|
570
|
-
handleMove(
|
|
571
|
-
|
|
1188
|
+
handleMove(g, C, i = 1) {
|
|
1189
|
+
I(this, L).moved = !0, I(this, L).dx = (g - I(this, L).targetX) * i, I(this, L).dy = (C - I(this, L).targetY) * i, I(this, L).targetX = g, I(this, L).targetY = C;
|
|
572
1190
|
}
|
|
573
1191
|
/**
|
|
574
1192
|
* Immediately applies one fluid splat at (x, y) with explicit velocity (vx, vy).
|
|
@@ -577,157 +1195,413 @@ class EI {
|
|
|
577
1195
|
* where you want N independent injection points per frame without flooding the
|
|
578
1196
|
* mouse-state machine or the worker message queue.
|
|
579
1197
|
*/
|
|
580
|
-
splat(
|
|
581
|
-
|
|
582
|
-
return;
|
|
583
|
-
const t = g(this, X), u = g(this, tI), { splat: a } = g(this, oI), c = g(this, rI);
|
|
584
|
-
t.viewport(0, 0, g(this, CI), g(this, eI)), a.bind(), t.uniform1f(a.uniforms.aspectRatio, g(this, K) / g(this, R)), t.uniform2f(a.uniforms.point, I * g(this, k) / g(this, K), 1 - i * g(this, k) / g(this, R)), t.uniform1f(a.uniforms.radius, u.splatRadius), t.uniform1i(a.uniforms.uTarget, 0), t.activeTexture(t.TEXTURE0), t.bindTexture(t.TEXTURE_2D, g(this, B).read.tex), t.uniform3f(a.uniforms.color, e * u.splatForce * o, -s * u.splatForce * o, 0), c(g(this, B).write.fbo), g(this, B).swap(), t.activeTexture(t.TEXTURE0), t.bindTexture(t.TEXTURE_2D, g(this, H).read.tex), t.uniform3f(a.uniforms.color, o, o, o), c(g(this, H).write.fbo), g(this, H).swap();
|
|
1198
|
+
splat(g, C, i, s, t = 1) {
|
|
1199
|
+
!I(this, Ug) || I(this, V) === 0 || (I(this, F) ? h(this, pI, TI).call(this, g, C, i, s, t) : h(this, uI, JI).call(this, g, C, i, s, t));
|
|
585
1200
|
}
|
|
586
|
-
updateQuality(
|
|
587
|
-
|
|
1201
|
+
updateQuality(g) {
|
|
1202
|
+
g.dpr !== void 0 && l(this, Jg, Math.max(0.1, Math.min(1, g.dpr))), g.sim !== void 0 && l(this, og, Math.max(0.1, Math.min(1, g.sim)));
|
|
588
1203
|
}
|
|
589
|
-
resize(
|
|
590
|
-
if (
|
|
591
|
-
if (
|
|
1204
|
+
resize(g, C, i) {
|
|
1205
|
+
if (i !== void 0 ? l(this, D, i) : typeof window < "u" && window.devicePixelRatio && l(this, D, window.devicePixelRatio), g !== void 0 && g > 0) {
|
|
1206
|
+
if (C === void 0 || C <= 0)
|
|
592
1207
|
return;
|
|
593
|
-
l(this,
|
|
1208
|
+
l(this, V, I(this, bg).width = g), l(this, X, I(this, bg).height = C), l(this, O, Math.max(1, Math.round(g * I(this, og)))), l(this, E, Math.max(1, Math.round(C * I(this, og)))), h(this, II, LI).call(this);
|
|
594
1209
|
} else
|
|
595
|
-
|
|
596
|
-
|
|
1210
|
+
h(this, kg, Eg).call(this);
|
|
1211
|
+
I(this, k) && h(this, Gg, fg).call(this), h(this, xg, jg).call(this);
|
|
597
1212
|
}
|
|
598
|
-
updateConfig(
|
|
599
|
-
Object.assign(
|
|
1213
|
+
updateConfig(g) {
|
|
1214
|
+
Object.assign(I(this, Ag), g);
|
|
600
1215
|
}
|
|
601
1216
|
destroy() {
|
|
602
|
-
|
|
603
|
-
|
|
604
|
-
|
|
605
|
-
|
|
606
|
-
|
|
607
|
-
|
|
608
|
-
|
|
1217
|
+
var g, C, i, s, t, o, a, u, A, p, c;
|
|
1218
|
+
if (l(this, gI, !0), this.stop(), h(this, CI, wI).call(this), h(this, iI, bI).call(this), I(this, J) && (I(this, J).close(), l(this, J, null)), I(this, F))
|
|
1219
|
+
(g = I(this, yg)) == null || g.destroy(), (C = I(this, Vg)) == null || C.destroy(), (i = I(this, Hg)) == null || i.destroy(), (s = I(this, Xg)) == null || s.destroy(), (t = I(this, Rg)) == null || t.destroy(), (o = I(this, tg)) == null || o.destroy(), (a = I(this, lg)) == null || a.destroy(), (u = I(this, vg)) == null || u.destroy(), (A = I(this, Wg)) == null || A.destroy(), (p = I(this, Yg)) == null || p.destroy(), (c = I(this, Kg)) == null || c.destroy(), I(this, F).device.destroy();
|
|
1220
|
+
else {
|
|
1221
|
+
const d = I(this, Q);
|
|
1222
|
+
for (const m of Object.values(I(this, hg)))
|
|
1223
|
+
m.dispose();
|
|
1224
|
+
const Z = d.getExtension("WEBGL_lose_context");
|
|
1225
|
+
Z == null || Z.loseContext();
|
|
1226
|
+
}
|
|
609
1227
|
}
|
|
610
1228
|
// ---------------------------------------------------------------------------
|
|
611
1229
|
// Loop control
|
|
612
1230
|
// ---------------------------------------------------------------------------
|
|
613
1231
|
start() {
|
|
614
|
-
if (
|
|
1232
|
+
if (I(this, ig) !== null)
|
|
615
1233
|
return;
|
|
616
|
-
const
|
|
617
|
-
|
|
1234
|
+
const g = () => {
|
|
1235
|
+
h(this, SI, FI).call(this), l(this, ig, vI(g));
|
|
618
1236
|
};
|
|
619
|
-
l(this,
|
|
1237
|
+
l(this, ig, vI(g));
|
|
620
1238
|
}
|
|
621
1239
|
stop() {
|
|
622
|
-
|
|
1240
|
+
I(this, ig) !== null && (HC(I(this, ig)), l(this, ig, null));
|
|
623
1241
|
}
|
|
624
1242
|
get isRunning() {
|
|
625
|
-
return
|
|
1243
|
+
return I(this, ig) !== null;
|
|
626
1244
|
}
|
|
627
|
-
}
|
|
628
|
-
|
|
629
|
-
const
|
|
630
|
-
|
|
631
|
-
},
|
|
632
|
-
const
|
|
633
|
-
|
|
634
|
-
},
|
|
635
|
-
if (
|
|
636
|
-
|
|
637
|
-
|
|
638
|
-
|
|
639
|
-
|
|
640
|
-
|
|
641
|
-
|
|
642
|
-
|
|
643
|
-
|
|
644
|
-
|
|
645
|
-
|
|
1245
|
+
};
|
|
1246
|
+
bg = new WeakMap(), Q = new WeakMap(), $g = new WeakMap(), hg = new WeakMap(), Fg = new WeakMap(), x = new WeakMap(), H = new WeakMap(), q = new WeakMap(), $ = new WeakMap(), gg = new WeakMap(), sg = new WeakMap(), T = new WeakMap(), Ig = new WeakMap(), F = new WeakMap(), Dg = new WeakMap(), Kg = new WeakMap(), Tg = new WeakMap(), f = new WeakMap(), y = new WeakMap(), Zg = new WeakMap(), Cg = new WeakMap(), mg = new WeakMap(), z = new WeakMap(), yg = new WeakMap(), Vg = new WeakMap(), Hg = new WeakMap(), Xg = new WeakMap(), Rg = new WeakMap(), tg = new WeakMap(), lg = new WeakMap(), vg = new WeakMap(), Wg = new WeakMap(), Yg = new WeakMap(), V = new WeakMap(), X = new WeakMap(), O = new WeakMap(), E = new WeakMap(), D = new WeakMap(), Jg = new WeakMap(), og = new WeakMap(), J = new WeakMap(), ng = new WeakMap(), Ag = new WeakMap(), L = new WeakMap(), k = new WeakMap(), ig = new WeakMap(), Ug = new WeakMap(), gI = new WeakMap(), cI = new WeakSet(), MI = function(g) {
|
|
1247
|
+
const { device: C, format: i } = g;
|
|
1248
|
+
l(this, Dg, bC(C, i)), l(this, Kg, wC(C)), l(this, Tg, hC(C)), l(this, yg, _(C, 16)), l(this, Vg, _(C, 16)), l(this, Hg, _(C, 16)), l(this, Xg, _(C, 16)), l(this, Rg, _(C, 16)), l(this, tg, _(C, 48)), l(this, lg, _(C, 48)), l(this, vg, _(C, 16)), l(this, Wg, _(C, 16)), l(this, Yg, _(C, 64));
|
|
1249
|
+
}, kg = new WeakSet(), Eg = function() {
|
|
1250
|
+
const g = I(this, bg);
|
|
1251
|
+
"clientWidth" in g && g.clientWidth > 0 ? (l(this, D, (typeof window < "u" && window.devicePixelRatio || 1) * I(this, Jg)), l(this, V, g.width = Math.round(g.clientWidth * I(this, D))), l(this, X, g.height = Math.round(g.clientHeight * I(this, D)))) : (l(this, V, g.width), l(this, X, g.height)), !(I(this, V) === 0 || I(this, X) === 0) && (l(this, O, Math.max(1, Math.round(I(this, V) * I(this, og)))), l(this, E, Math.max(1, Math.round(I(this, X) * I(this, og)))), h(this, II, LI).call(this));
|
|
1252
|
+
}, II = new WeakSet(), LI = function() {
|
|
1253
|
+
if (h(this, CI, wI).call(this), I(this, F)) {
|
|
1254
|
+
const { device: g } = I(this, F), C = "rgba16float", i = I(this, O), s = I(this, E);
|
|
1255
|
+
l(this, f, rI(g, C, i, s)), l(this, y, rI(g, C, i, s)), l(this, Cg, rI(g, C, i, s)), l(this, Zg, AI(g, C, i, s)), l(this, mg, AI(g, C, i, s));
|
|
1256
|
+
} else {
|
|
1257
|
+
const g = I(this, Q), C = I(this, $g), i = I(this, O), s = I(this, E);
|
|
1258
|
+
l(this, x, GI(g, C, i, s)), l(this, H, GI(g, C, i, s)), l(this, $, GI(g, C, i, s)), l(this, q, oI(g, C, i, s)), l(this, gg, oI(g, C, i, s));
|
|
1259
|
+
}
|
|
1260
|
+
}, Gg = new WeakSet(), fg = function() {
|
|
1261
|
+
if (!(!I(this, k) || I(this, V) === 0 || I(this, X) === 0)) {
|
|
1262
|
+
if (h(this, iI, bI).call(this), I(this, F)) {
|
|
1263
|
+
const { device: g } = I(this, F);
|
|
1264
|
+
I(this, k).type === "text" ? l(this, z, gC(
|
|
1265
|
+
g,
|
|
1266
|
+
I(this, V),
|
|
1267
|
+
I(this, X),
|
|
1268
|
+
I(this, k).opts,
|
|
1269
|
+
I(this, J),
|
|
1270
|
+
I(this, ng)
|
|
1271
|
+
)) : l(this, z, IC(
|
|
1272
|
+
g,
|
|
1273
|
+
I(this, k).bitmap,
|
|
1274
|
+
I(this, V),
|
|
1275
|
+
I(this, X),
|
|
1276
|
+
I(this, k).effect,
|
|
1277
|
+
I(this, k).size,
|
|
1278
|
+
I(this, J),
|
|
1279
|
+
I(this, ng)
|
|
1280
|
+
));
|
|
646
1281
|
} else {
|
|
647
|
-
const
|
|
648
|
-
|
|
649
|
-
|
|
650
|
-
|
|
651
|
-
|
|
652
|
-
|
|
653
|
-
|
|
654
|
-
|
|
655
|
-
|
|
656
|
-
|
|
657
|
-
|
|
1282
|
+
const g = I(this, Q);
|
|
1283
|
+
if (I(this, k).type === "text") {
|
|
1284
|
+
const { backgroundTex: C, obstacleTex: i, coverageTex: s } = qI(
|
|
1285
|
+
g,
|
|
1286
|
+
I(this, V),
|
|
1287
|
+
I(this, X),
|
|
1288
|
+
I(this, k).opts,
|
|
1289
|
+
I(this, J),
|
|
1290
|
+
I(this, ng)
|
|
1291
|
+
);
|
|
1292
|
+
l(this, sg, C), l(this, T, i), l(this, Ig, s);
|
|
1293
|
+
} else {
|
|
1294
|
+
const { backgroundTex: C, obstacleTex: i, coverageTex: s } = $I(
|
|
1295
|
+
g,
|
|
1296
|
+
I(this, k).bitmap,
|
|
1297
|
+
I(this, V),
|
|
1298
|
+
I(this, X),
|
|
1299
|
+
I(this, k).effect,
|
|
1300
|
+
I(this, k).size,
|
|
1301
|
+
I(this, J),
|
|
1302
|
+
I(this, ng)
|
|
1303
|
+
);
|
|
1304
|
+
l(this, sg, C), l(this, T, i), l(this, Ig, s);
|
|
1305
|
+
}
|
|
1306
|
+
}
|
|
1307
|
+
l(this, Ug, !0);
|
|
1308
|
+
}
|
|
1309
|
+
}, xg = new WeakSet(), jg = function() {
|
|
1310
|
+
I(this, Ug) && !this.isRunning && this.start();
|
|
1311
|
+
}, CI = new WeakSet(), wI = function() {
|
|
1312
|
+
var g, C, i, s, t, o, a, u;
|
|
1313
|
+
if (I(this, F))
|
|
1314
|
+
(g = I(this, f)) == null || g.dispose(), (C = I(this, y)) == null || C.dispose(), (i = I(this, Cg)) == null || i.dispose(), (s = I(this, Zg)) == null || s.tex.destroy(), (t = I(this, mg)) == null || t.tex.destroy(), l(this, f, l(this, y, l(this, Cg, null))), l(this, Zg, l(this, mg, null));
|
|
1315
|
+
else {
|
|
1316
|
+
const A = I(this, Q);
|
|
1317
|
+
(o = I(this, x)) == null || o.dispose(), (a = I(this, H)) == null || a.dispose(), (u = I(this, $)) == null || u.dispose(), I(this, q) && (A.deleteTexture(I(this, q).tex), A.deleteFramebuffer(I(this, q).fbo)), I(this, gg) && (A.deleteTexture(I(this, gg).tex), A.deleteFramebuffer(I(this, gg).fbo)), l(this, x, l(this, H, l(this, $, l(this, q, l(this, gg, null)))));
|
|
1318
|
+
}
|
|
1319
|
+
}, iI = new WeakSet(), bI = function() {
|
|
1320
|
+
if (I(this, F))
|
|
1321
|
+
I(this, z) && (I(this, z).backgroundTex.destroy(), I(this, z).obstacleTex.destroy(), I(this, z).sharedCoverage || I(this, z).coverageTex.destroy(), l(this, z, null));
|
|
1322
|
+
else {
|
|
1323
|
+
const g = I(this, Q);
|
|
1324
|
+
I(this, sg) && g.deleteTexture(I(this, sg)), I(this, T) && g.deleteTexture(I(this, T)), I(this, Ig) && I(this, Ig) !== I(this, T) && g.deleteTexture(I(this, Ig)), l(this, sg, l(this, T, l(this, Ig, null)));
|
|
1325
|
+
}
|
|
1326
|
+
}, SI = new WeakSet(), FI = function() {
|
|
1327
|
+
!I(this, Ug) || I(this, V) === 0 || (I(this, F) ? h(this, aI, DI).call(this) : h(this, BI, zI).call(this));
|
|
1328
|
+
}, aI = new WeakSet(), DI = function() {
|
|
1329
|
+
const g = I(this, F), C = g.device, i = I(this, Dg), s = I(this, Kg), t = I(this, Tg), o = I(this, Ag), a = I(this, z);
|
|
1330
|
+
if (!I(this, f) || !I(this, y))
|
|
1331
|
+
return;
|
|
1332
|
+
I(this, L).x += (I(this, L).targetX - I(this, L).x) * 0.15, I(this, L).y += (I(this, L).targetY - I(this, L).y) * 0.15;
|
|
1333
|
+
const u = I(this, O), A = I(this, E), p = I(this, V), c = I(this, X), d = 1 / u, Z = 1 / A;
|
|
1334
|
+
RI(C, I(this, yg), d, Z, Og, o.velocityDissipation), tI(C, I(this, Hg), d, Z), tI(C, I(this, Xg), d, Z), tI(C, I(this, Rg), d, Z), tI(C, I(this, vg), d, Z), KC(C, I(this, Wg), d, Z, o.curl, Og), yC(
|
|
1335
|
+
C,
|
|
1336
|
+
I(this, Yg),
|
|
1337
|
+
1 / p,
|
|
1338
|
+
1 / c,
|
|
1339
|
+
o.refraction,
|
|
1340
|
+
o.specularExp,
|
|
1341
|
+
o.waterColor,
|
|
1342
|
+
o.glowColor,
|
|
1343
|
+
o.shine,
|
|
1344
|
+
o.warpStrength ?? 0.015,
|
|
1345
|
+
WI[o.algorithm] ?? 0
|
|
1346
|
+
);
|
|
1347
|
+
const m = C.createCommandEncoder(), w = (n, r) => C.createBindGroup({ layout: n.getBindGroupLayout(0), entries: r }), b = { binding: 1, resource: t };
|
|
1348
|
+
{
|
|
1349
|
+
const n = w(i.advection, [
|
|
1350
|
+
{ binding: 0, resource: { buffer: I(this, yg) } },
|
|
1351
|
+
b,
|
|
1352
|
+
{ binding: 2, resource: I(this, y).read.view },
|
|
1353
|
+
{ binding: 3, resource: I(this, y).read.view },
|
|
1354
|
+
{ binding: 4, resource: a.obstacleView }
|
|
1355
|
+
]);
|
|
1356
|
+
N(m, i.advection, n, s, I(this, y).write.view);
|
|
1357
|
+
}
|
|
1358
|
+
I(this, y).swap();
|
|
1359
|
+
{
|
|
1360
|
+
RI(C, I(this, Vg), d, Z, Og, o.densityDissipation);
|
|
1361
|
+
const n = w(i.advection, [
|
|
1362
|
+
{ binding: 0, resource: { buffer: I(this, Vg) } },
|
|
1363
|
+
b,
|
|
1364
|
+
{ binding: 2, resource: I(this, y).read.view },
|
|
1365
|
+
{ binding: 3, resource: I(this, f).read.view },
|
|
1366
|
+
{ binding: 4, resource: a.obstacleView }
|
|
1367
|
+
]);
|
|
1368
|
+
N(m, i.advection, n, s, I(this, f).write.view);
|
|
1369
|
+
}
|
|
1370
|
+
I(this, f).swap();
|
|
1371
|
+
{
|
|
1372
|
+
const n = w(i.curl, [
|
|
1373
|
+
{ binding: 0, resource: { buffer: I(this, vg) } },
|
|
1374
|
+
b,
|
|
1375
|
+
{ binding: 2, resource: I(this, y).read.view }
|
|
1376
|
+
]);
|
|
1377
|
+
N(m, i.curl, n, s, I(this, mg).view);
|
|
1378
|
+
}
|
|
1379
|
+
{
|
|
1380
|
+
const n = w(i.vorticity, [
|
|
1381
|
+
{ binding: 0, resource: { buffer: I(this, Wg) } },
|
|
1382
|
+
b,
|
|
1383
|
+
{ binding: 2, resource: I(this, y).read.view },
|
|
1384
|
+
{ binding: 3, resource: I(this, mg).view }
|
|
1385
|
+
]);
|
|
1386
|
+
N(m, i.vorticity, n, s, I(this, y).write.view);
|
|
1387
|
+
}
|
|
1388
|
+
if (I(this, y).swap(), I(this, L).moved) {
|
|
1389
|
+
const n = I(this, L).x * I(this, D) / p, r = I(this, L).y * I(this, D) / c;
|
|
1390
|
+
lI(
|
|
1391
|
+
C,
|
|
1392
|
+
I(this, tg),
|
|
1393
|
+
d,
|
|
1394
|
+
Z,
|
|
1395
|
+
p / c,
|
|
1396
|
+
o.splatRadius,
|
|
1397
|
+
I(this, L).dx * o.splatForce,
|
|
1398
|
+
I(this, L).dy * o.splatForce,
|
|
1399
|
+
0,
|
|
1400
|
+
n,
|
|
1401
|
+
r
|
|
1402
|
+
);
|
|
1403
|
+
{
|
|
1404
|
+
const R = w(i.splat, [
|
|
1405
|
+
{ binding: 0, resource: { buffer: I(this, tg) } },
|
|
1406
|
+
b,
|
|
1407
|
+
{ binding: 2, resource: I(this, y).read.view }
|
|
1408
|
+
]);
|
|
1409
|
+
N(m, i.splat, R, s, I(this, y).write.view);
|
|
1410
|
+
}
|
|
1411
|
+
I(this, y).swap(), lI(
|
|
1412
|
+
C,
|
|
1413
|
+
I(this, lg),
|
|
1414
|
+
d,
|
|
1415
|
+
Z,
|
|
1416
|
+
p / c,
|
|
1417
|
+
o.splatRadius,
|
|
1418
|
+
1,
|
|
1419
|
+
1,
|
|
1420
|
+
1,
|
|
1421
|
+
n,
|
|
1422
|
+
r
|
|
1423
|
+
);
|
|
1424
|
+
{
|
|
1425
|
+
const R = w(i.splat, [
|
|
1426
|
+
{ binding: 0, resource: { buffer: I(this, lg) } },
|
|
1427
|
+
b,
|
|
1428
|
+
{ binding: 2, resource: I(this, f).read.view }
|
|
1429
|
+
]);
|
|
1430
|
+
N(m, i.splat, R, s, I(this, f).write.view);
|
|
658
1431
|
}
|
|
659
|
-
|
|
1432
|
+
I(this, f).swap(), I(this, L).moved = !1;
|
|
1433
|
+
}
|
|
1434
|
+
{
|
|
1435
|
+
const n = w(i.divergence, [
|
|
1436
|
+
{ binding: 0, resource: { buffer: I(this, Hg) } },
|
|
1437
|
+
b,
|
|
1438
|
+
{ binding: 2, resource: I(this, y).read.view },
|
|
1439
|
+
{ binding: 3, resource: a.obstacleView }
|
|
1440
|
+
]);
|
|
1441
|
+
N(m, i.divergence, n, s, I(this, Zg).view);
|
|
1442
|
+
}
|
|
1443
|
+
for (let n = 0; n < o.pressureIterations; n++) {
|
|
1444
|
+
const r = w(i.pressure, [
|
|
1445
|
+
{ binding: 0, resource: { buffer: I(this, Xg) } },
|
|
1446
|
+
b,
|
|
1447
|
+
{ binding: 2, resource: I(this, Cg).read.view },
|
|
1448
|
+
{ binding: 3, resource: I(this, Zg).view },
|
|
1449
|
+
{ binding: 4, resource: a.obstacleView }
|
|
1450
|
+
]);
|
|
1451
|
+
N(m, i.pressure, r, s, I(this, Cg).write.view), I(this, Cg).swap();
|
|
1452
|
+
}
|
|
1453
|
+
{
|
|
1454
|
+
const n = w(i.gradientSubtract, [
|
|
1455
|
+
{ binding: 0, resource: { buffer: I(this, Rg) } },
|
|
1456
|
+
b,
|
|
1457
|
+
{ binding: 2, resource: I(this, Cg).read.view },
|
|
1458
|
+
{ binding: 3, resource: I(this, y).read.view },
|
|
1459
|
+
{ binding: 4, resource: a.obstacleView }
|
|
1460
|
+
]);
|
|
1461
|
+
N(m, i.gradientSubtract, n, s, I(this, y).write.view);
|
|
1462
|
+
}
|
|
1463
|
+
I(this, y).swap();
|
|
1464
|
+
{
|
|
1465
|
+
const n = g.context.getCurrentTexture().createView(), r = w(i.display, [
|
|
1466
|
+
{ binding: 0, resource: { buffer: I(this, Yg) } },
|
|
1467
|
+
b,
|
|
1468
|
+
{ binding: 2, resource: I(this, f).read.view },
|
|
1469
|
+
{ binding: 3, resource: a.obstacleView },
|
|
1470
|
+
{ binding: 4, resource: a.backgroundView },
|
|
1471
|
+
{ binding: 5, resource: a.coverageView },
|
|
1472
|
+
{ binding: 6, resource: I(this, y).read.view }
|
|
1473
|
+
]);
|
|
1474
|
+
VC(m, i.display, r, s, n);
|
|
1475
|
+
}
|
|
1476
|
+
C.queue.submit([m.finish()]);
|
|
1477
|
+
}, pI = new WeakSet(), TI = function(g, C, i, s, t) {
|
|
1478
|
+
const a = I(this, F).device, u = I(this, Dg).splat, A = I(this, Kg), p = I(this, Tg), c = I(this, Ag), d = I(this, O), Z = I(this, E), m = 1 / d, w = 1 / Z, b = a.createCommandEncoder(), n = { binding: 1, resource: p }, r = (W) => a.createBindGroup({ layout: u.getBindGroupLayout(0), entries: W }), R = g * I(this, D) / I(this, V), B = C * I(this, D) / I(this, X);
|
|
1479
|
+
lI(
|
|
1480
|
+
a,
|
|
1481
|
+
I(this, tg),
|
|
1482
|
+
m,
|
|
1483
|
+
w,
|
|
1484
|
+
I(this, V) / I(this, X),
|
|
1485
|
+
c.splatRadius,
|
|
1486
|
+
i * c.splatForce * t,
|
|
1487
|
+
s * c.splatForce * t,
|
|
1488
|
+
0,
|
|
1489
|
+
R,
|
|
1490
|
+
B
|
|
1491
|
+
);
|
|
1492
|
+
{
|
|
1493
|
+
const W = r([
|
|
1494
|
+
{ binding: 0, resource: { buffer: I(this, tg) } },
|
|
1495
|
+
n,
|
|
1496
|
+
{ binding: 2, resource: I(this, y).read.view }
|
|
1497
|
+
]);
|
|
1498
|
+
N(b, u, W, A, I(this, y).write.view);
|
|
660
1499
|
}
|
|
661
|
-
|
|
662
|
-
|
|
663
|
-
|
|
664
|
-
|
|
665
|
-
|
|
666
|
-
|
|
667
|
-
|
|
668
|
-
|
|
669
|
-
|
|
1500
|
+
I(this, y).swap(), lI(
|
|
1501
|
+
a,
|
|
1502
|
+
I(this, lg),
|
|
1503
|
+
m,
|
|
1504
|
+
w,
|
|
1505
|
+
I(this, V) / I(this, X),
|
|
1506
|
+
c.splatRadius,
|
|
1507
|
+
t,
|
|
1508
|
+
t,
|
|
1509
|
+
t,
|
|
1510
|
+
R,
|
|
1511
|
+
B
|
|
1512
|
+
);
|
|
1513
|
+
{
|
|
1514
|
+
const W = r([
|
|
1515
|
+
{ binding: 0, resource: { buffer: I(this, lg) } },
|
|
1516
|
+
n,
|
|
1517
|
+
{ binding: 2, resource: I(this, f).read.view }
|
|
1518
|
+
]);
|
|
1519
|
+
N(b, u, W, A, I(this, f).write.view);
|
|
1520
|
+
}
|
|
1521
|
+
I(this, f).swap(), a.queue.submit([b.finish()]);
|
|
1522
|
+
}, uI = new WeakSet(), JI = function(g, C, i, s, t) {
|
|
1523
|
+
const o = I(this, Q), a = I(this, Ag), u = I(this, hg).splat, A = I(this, Fg);
|
|
1524
|
+
o.viewport(0, 0, I(this, O), I(this, E)), u.bind(), o.uniform1f(u.uniforms.aspectRatio, I(this, V) / I(this, X)), o.uniform2f(u.uniforms.point, g * I(this, D) / I(this, V), 1 - C * I(this, D) / I(this, X)), o.uniform1f(u.uniforms.radius, a.splatRadius), o.uniform1i(u.uniforms.uTarget, 0), o.activeTexture(o.TEXTURE0), o.bindTexture(o.TEXTURE_2D, I(this, H).read.tex), o.uniform3f(u.uniforms.color, i * a.splatForce * t, -s * a.splatForce * t, 0), A(I(this, H).write.fbo), I(this, H).swap(), o.activeTexture(o.TEXTURE0), o.bindTexture(o.TEXTURE_2D, I(this, x).read.tex), o.uniform3f(u.uniforms.color, t, t, t), A(I(this, x).write.fbo), I(this, x).swap();
|
|
1525
|
+
}, BI = new WeakSet(), zI = function() {
|
|
1526
|
+
if (!I(this, x) || !I(this, H))
|
|
670
1527
|
return;
|
|
671
|
-
const
|
|
672
|
-
|
|
673
|
-
const
|
|
674
|
-
|
|
675
|
-
for (let
|
|
676
|
-
|
|
677
|
-
|
|
1528
|
+
const g = I(this, Q), C = I(this, Ag), { advection: i, divergence: s, pressure: t, gradientSubtract: o, splat: a, curl: u, vorticity: A, display: p } = I(this, hg);
|
|
1529
|
+
I(this, L).x += (I(this, L).targetX - I(this, L).x) * 0.15, I(this, L).y += (I(this, L).targetY - I(this, L).y) * 0.15;
|
|
1530
|
+
const c = I(this, O), d = I(this, E), Z = I(this, Fg);
|
|
1531
|
+
g.viewport(0, 0, c, d), i.bind(), g.uniform2f(i.uniforms.texelSize, 1 / c, 1 / d), g.uniform1f(i.uniforms.dt, Og), g.uniform1i(i.uniforms.uObstacle, 0), g.activeTexture(g.TEXTURE0), g.bindTexture(g.TEXTURE_2D, I(this, T)), g.uniform1f(i.uniforms.dissipation, C.velocityDissipation), g.uniform1i(i.uniforms.uVelocity, 1), g.activeTexture(g.TEXTURE1), g.bindTexture(g.TEXTURE_2D, I(this, H).read.tex), g.uniform1i(i.uniforms.uSource, 1), Z(I(this, H).write.fbo), I(this, H).swap(), g.uniform1f(i.uniforms.dissipation, C.densityDissipation), g.uniform1i(i.uniforms.uSource, 2), g.activeTexture(g.TEXTURE2), g.bindTexture(g.TEXTURE_2D, I(this, x).read.tex), Z(I(this, x).write.fbo), I(this, x).swap(), u.bind(), g.uniform2f(u.uniforms.texelSize, 1 / c, 1 / d), g.uniform1i(u.uniforms.uVelocity, 0), g.activeTexture(g.TEXTURE0), g.bindTexture(g.TEXTURE_2D, I(this, H).read.tex), Z(I(this, gg).fbo), A.bind(), g.uniform2f(A.uniforms.texelSize, 1 / c, 1 / d), g.uniform1f(A.uniforms.curl, C.curl), g.uniform1f(A.uniforms.dt, Og), g.uniform1i(A.uniforms.uVelocity, 0), g.activeTexture(g.TEXTURE0), g.bindTexture(g.TEXTURE_2D, I(this, H).read.tex), g.uniform1i(A.uniforms.uCurl, 1), g.activeTexture(g.TEXTURE1), g.bindTexture(g.TEXTURE_2D, I(this, gg).tex), Z(I(this, H).write.fbo), I(this, H).swap(), I(this, L).moved && (a.bind(), g.uniform1f(a.uniforms.aspectRatio, I(this, V) / I(this, X)), g.uniform2f(a.uniforms.point, I(this, L).x * I(this, D) / I(this, V), 1 - I(this, L).y * I(this, D) / I(this, X)), g.uniform1f(a.uniforms.radius, C.splatRadius), g.uniform1i(a.uniforms.uTarget, 0), g.activeTexture(g.TEXTURE0), g.bindTexture(g.TEXTURE_2D, I(this, H).read.tex), g.uniform3f(a.uniforms.color, I(this, L).dx * C.splatForce, -I(this, L).dy * C.splatForce, 0), Z(I(this, H).write.fbo), I(this, H).swap(), g.activeTexture(g.TEXTURE0), g.bindTexture(g.TEXTURE_2D, I(this, x).read.tex), g.uniform3f(a.uniforms.color, 1, 1, 1), Z(I(this, x).write.fbo), I(this, x).swap(), I(this, L).moved = !1), s.bind(), g.uniform2f(s.uniforms.texelSize, 1 / c, 1 / d), g.uniform1i(s.uniforms.uVelocity, 0), g.activeTexture(g.TEXTURE0), g.bindTexture(g.TEXTURE_2D, I(this, H).read.tex), g.uniform1i(s.uniforms.uObstacle, 1), g.activeTexture(g.TEXTURE1), g.bindTexture(g.TEXTURE_2D, I(this, T)), Z(I(this, q).fbo), t.bind(), g.uniform2f(t.uniforms.texelSize, 1 / c, 1 / d), g.uniform1i(t.uniforms.uDivergence, 0), g.activeTexture(g.TEXTURE0), g.bindTexture(g.TEXTURE_2D, I(this, q).tex), g.uniform1i(t.uniforms.uObstacle, 1), g.activeTexture(g.TEXTURE1), g.bindTexture(g.TEXTURE_2D, I(this, T));
|
|
1532
|
+
for (let m = 0; m < C.pressureIterations; m++)
|
|
1533
|
+
g.uniform1i(t.uniforms.uPressure, 2), g.activeTexture(g.TEXTURE2), g.bindTexture(g.TEXTURE_2D, I(this, $).read.tex), Z(I(this, $).write.fbo), I(this, $).swap();
|
|
1534
|
+
o.bind(), g.uniform2f(o.uniforms.texelSize, 1 / c, 1 / d), g.uniform1i(o.uniforms.uPressure, 0), g.activeTexture(g.TEXTURE0), g.bindTexture(g.TEXTURE_2D, I(this, $).read.tex), g.uniform1i(o.uniforms.uVelocity, 1), g.activeTexture(g.TEXTURE1), g.bindTexture(g.TEXTURE_2D, I(this, H).read.tex), g.uniform1i(o.uniforms.uObstacle, 2), g.activeTexture(g.TEXTURE2), g.bindTexture(g.TEXTURE_2D, I(this, T)), Z(I(this, H).write.fbo), I(this, H).swap(), g.viewport(0, 0, I(this, V), I(this, X)), g.bindFramebuffer(g.FRAMEBUFFER, null), g.clear(g.COLOR_BUFFER_BIT), p.bind(), g.uniform2f(p.uniforms.texelSize, 1 / I(this, V), 1 / I(this, X)), g.uniform3fv(p.uniforms.uWaterColor, C.waterColor), g.uniform3fv(p.uniforms.uGlowColor, C.glowColor), g.uniform1f(p.uniforms.uRefraction, C.refraction), g.uniform1f(p.uniforms.uSpecularExp, C.specularExp), g.uniform1f(p.uniforms.uShine, C.shine), g.uniform1f(p.uniforms.uWarpStrength, C.warpStrength ?? 0.015), g.uniform1i(p.uniforms.uAlgorithm, WI[C.algorithm] ?? 0), g.activeTexture(g.TEXTURE0), g.bindTexture(g.TEXTURE_2D, I(this, x).read.tex), g.activeTexture(g.TEXTURE1), g.bindTexture(g.TEXTURE_2D, I(this, T)), g.activeTexture(g.TEXTURE2), g.bindTexture(g.TEXTURE_2D, I(this, sg)), g.activeTexture(g.TEXTURE3), g.bindTexture(g.TEXTURE_2D, I(this, Ig)), g.activeTexture(g.TEXTURE4), g.bindTexture(g.TEXTURE_2D, I(this, H).read.tex), g.uniform1i(p.uniforms.uTexture, 0), g.uniform1i(p.uniforms.uObstacle, 1), g.uniform1i(p.uniforms.uBackground, 2), g.uniform1i(p.uniforms.uCoverage, 3), g.uniform1i(p.uniforms.uVelocity, 4), Z(null);
|
|
678
1535
|
};
|
|
679
|
-
const PI = "var Ue = Object.defineProperty;
var _e = (i, e, r) => e in i ? Ue(i, e, { enumerable: !0, configurable: !0, writable: !0, value: r }) : i[e] = r;
var ne = (i, e, r) => (_e(i, typeof e != "symbol" ? e + "" : e, r), r), xe = (i, e, r) => {
  if (!e.has(i))
    throw TypeError("Cannot " + r);
};
var t = (i, e, r) => (xe(i, e, "read from private field"), r ? r.call(i) : e.get(i)), l = (i, e, r) => {
  if (e.has(i))
    throw TypeError("Cannot add the same private member more than once");
  e instanceof WeakSet ? e.add(i) : e.set(i, r);
}, a = (i, e, r, o) => (xe(i, e, "write to private field"), o ? o.call(i, r) : e.set(i, r), r);
var T = (i, e, r) => (xe(i, e, "access private method"), r);
const ye = {
  densityDissipation: 0.992,
  velocityDissipation: 0.93,
  pressureIterations: 1,
  curl: 1e-4,
  splatRadius: 4e-3,
  splatForce: 0.91,
  refraction: 0.25,
  specularExp: 1.01,
  shine: 0.01,
  waterColor: [0, 0, 0],
  glowColor: [0.7, 0.85, 1],
  algorithm: "standard",
  warpStrength: 0.015
};
({
  ...ye
});
const we = {
  calm: {
    densityDissipation: 0.999,
    velocityDissipation: 0.98,
    curl: 1e-4,
    splatRadius: 3e-3,
    splatForce: 0.5,
    refraction: 0.15,
    shine: 5e-3,
    glowColor: [0.6, 0.85, 1],
    waterColor: [0, 0.02, 0.05]
  },
  sand: {
    densityDissipation: 0.997,
    velocityDissipation: 0.98,
    curl: 1,
    splatRadius: 0.01,
    splatForce: 0.9,
    refraction: 0.8,
    specularExp: 0.1,
    shine: 0.05,
    glowColor: [0.027, 0.027, 0.027],
    waterColor: [0.451, 0.329, 0.125]
  },
  wave: {
    densityDissipation: 0.994,
    velocityDissipation: 0.92,
    curl: 0.2,
    splatRadius: 5e-3,
    splatForce: 1.2,
    refraction: 0.35,
    shine: 0.03,
    glowColor: [0.5, 0.8, 1],
    waterColor: [0, 0.01, 0.03]
  },
  neon: {
    densityDissipation: 0.985,
    velocityDissipation: 0.93,
    curl: 0.05,
    splatRadius: 8e-3,
    splatForce: 1.5,
    refraction: 0.25,
    specularExp: 0.5,
    shine: 0.14,
    glowColor: [1, 0.2, 0.8],
    waterColor: [0.05, 0, 0.08]
  },
  smoke: {
    densityDissipation: 0.996,
    velocityDissipation: 0.97,
    curl: 0.04,
    splatRadius: 9e-3,
    splatForce: 0.8,
    refraction: 0.08,
    shine: 0,
    glowColor: [0.5, 0.5, 0.5],
    waterColor: [0.06, 0.06, 0.06]
  }
};
function Se(i = {}, e, r = ye) {
  return { ...e ? { ...r, ...we[e] } : r, ...i };
}
const P = (
  /* glsl */
  `
  precision highp float;
  attribute vec2 aPosition;
  varying vec2 vUv;
  varying vec2 vL; varying vec2 vR; varying vec2 vT; varying vec2 vB;
  uniform vec2 texelSize;
  void main () {
    vUv = aPosition * 0.5 + 0.5;
    vL = vUv - vec2(texelSize.x, 0.0);
    vR = vUv + vec2(texelSize.x, 0.0);
    vT = vUv + vec2(0.0, texelSize.y);
    vB = vUv - vec2(0.0, texelSize.y);
    gl_Position = vec4(aPosition, 0.0, 1.0);
  }
`
), Ce = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vUv;
  uniform sampler2D uVelocity;
  uniform sampler2D uSource;
  uniform sampler2D uObstacle;
  uniform vec2 texelSize;
  uniform float dt;
  uniform float dissipation;
  void main () {
    if (texture2D(uObstacle, vUv).r > 0.5) { gl_FragColor = vec4(0.0); return; }
    vec2 coord = vUv - dt * texture2D(uVelocity, vUv).xy * texelSize;
    gl_FragColor = dissipation * texture2D(uSource, coord);
  }
`
), Fe = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vL; varying vec2 vR; varying vec2 vT; varying vec2 vB;
  uniform sampler2D uVelocity;
  uniform sampler2D uObstacle;
  void main () {
    float L = texture2D(uObstacle, vL).r > 0.5 ? 0.0 : texture2D(uVelocity, vL).x;
    float R = texture2D(uObstacle, vR).r > 0.5 ? 0.0 : texture2D(uVelocity, vR).x;
    float T = texture2D(uObstacle, vT).r > 0.5 ? 0.0 : texture2D(uVelocity, vT).y;
    float B = texture2D(uObstacle, vB).r > 0.5 ? 0.0 : texture2D(uVelocity, vB).y;
    gl_FragColor = vec4(0.5 * (R - L + T - B), 0.0, 0.0, 1.0);
  }
`
), Xe = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vUv; varying vec2 vL; varying vec2 vR; varying vec2 vT; varying vec2 vB;
  uniform sampler2D uPressure;
  uniform sampler2D uDivergence;
  uniform sampler2D uObstacle;
  void main () {
    float C = texture2D(uPressure, vUv).x;
    float L = texture2D(uObstacle, vL).r > 0.5 ? C : texture2D(uPressure, vL).x;
    float R = texture2D(uObstacle, vR).r > 0.5 ? C : texture2D(uPressure, vR).x;
    float T = texture2D(uObstacle, vT).r > 0.5 ? C : texture2D(uPressure, vT).x;
    float B = texture2D(uObstacle, vB).r > 0.5 ? C : texture2D(uPressure, vB).x;
    float div = texture2D(uDivergence, vUv).x;
    gl_FragColor = vec4((L + R + B + T - div) * 0.25, 0.0, 0.0, 1.0);
  }
`
), Be = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vUv; varying vec2 vL; varying vec2 vR; varying vec2 vT; varying vec2 vB;
  uniform sampler2D uPressure;
  uniform sampler2D uVelocity;
  uniform sampler2D uObstacle;
  void main () {
    if (texture2D(uObstacle, vUv).r > 0.5) { gl_FragColor = vec4(0.0); return; }
    float L = texture2D(uPressure, vL).x;
    float R = texture2D(uPressure, vR).x;
    float T = texture2D(uPressure, vT).x;
    float B = texture2D(uPressure, vB).x;
    vec2 vel = texture2D(uVelocity, vUv).xy - vec2(R - L, T - B);
    gl_FragColor = vec4(vel, 0.0, 1.0);
  }
`
), Ae = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vUv;
  uniform sampler2D uTarget;
  uniform float aspectRatio;
  uniform vec3 color;
  uniform vec2 point;
  uniform float radius;
  void main () {
    vec2 p = vUv - point.xy;
    p.x *= aspectRatio;
    vec3 splat = exp(-dot(p, p) / radius) * color;
    gl_FragColor = vec4(texture2D(uTarget, vUv).xyz + splat, 1.0);
  }
`
), Le = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vL; varying vec2 vR; varying vec2 vT; varying vec2 vB;
  uniform sampler2D uVelocity;
  void main () {
    float L = texture2D(uVelocity, vL).y;
    float R = texture2D(uVelocity, vR).y;
    float T = texture2D(uVelocity, vT).x;
    float B = texture2D(uVelocity, vB).x;
    gl_FragColor = vec4(0.5 * (R - L - T + B), 0.0, 0.0, 1.0);
  }
`
), Me = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vUv; varying vec2 vL; varying vec2 vR; varying vec2 vT; varying vec2 vB;
  uniform sampler2D uVelocity;
  uniform sampler2D uCurl;
  uniform float curl;
  uniform float dt;
  void main () {
    float L = texture2D(uCurl, vL).x;
    float R = texture2D(uCurl, vR).x;
    float T = texture2D(uCurl, vT).x;
    float B = texture2D(uCurl, vB).x;
    float C = texture2D(uCurl, vUv).x;
    vec2 force = 0.5 * vec2(abs(T) - abs(B), abs(R) - abs(L));
    force /= length(force) + 0.0001;
    force *= curl * 30.0 * C;
    gl_FragColor = vec4(texture2D(uVelocity, vUv).xy + force * dt, 0.0, 1.0);
  }
`
), Pe = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vUv;

  uniform sampler2D uTexture;
  uniform sampler2D uObstacle;
  uniform sampler2D uBackground;
  uniform sampler2D uCoverage;
  uniform sampler2D uVelocity;

  uniform vec2  texelSize;
  uniform vec3  uWaterColor;
  uniform vec3  uGlowColor;
  uniform float uRefraction;
  uniform float uSpecularExp;
  uniform float uShine;
  uniform float uWarpStrength;
  uniform int   uAlgorithm;

  void main () {
    float obs      = texture2D(uObstacle,  vUv).r;
    // Mask density inside obstacles so splats don't flicker the text/image content.
    float density  = max(texture2D(uTexture, vUv).r, 0.0) * (1.0 - step(0.5, obs));
    float coverage = texture2D(uCoverage,  vUv).r;

    float dL = max(texture2D(uTexture, vUv - vec2(texelSize.x * 2.0, 0.0)).r, 0.0);
    float dR = max(texture2D(uTexture, vUv + vec2(texelSize.x * 2.0, 0.0)).r, 0.0);
    float dT = max(texture2D(uTexture, vUv + vec2(0.0, texelSize.y * 2.0)).r, 0.0);
    float dB = max(texture2D(uTexture, vUv - vec2(0.0, texelSize.y * 2.0)).r, 0.0);

    vec3  normal   = normalize(vec3(dL - dR, dB - dT, 0.2));
    vec3  lightDir = normalize(vec3(0.5, 1.0, 0.5));
    vec3  halfV    = normalize(lightDir + vec3(0.0, 0.0, 1.0));
    float spec     = pow(max(dot(normal, halfV), 0.0), uSpecularExp) * uShine * density;

    // In transparent (non-coverage) areas the background texture is empty black canvas.
    // Replace it with uWaterColor so fluid colour is not contaminated by that black,
    // allowing the CSS backgroundColor to show through correctly via premultiplied alpha.
    vec3 bgRaw = texture2D(uBackground, vUv).rgb;
    vec3 bg    = mix(uWaterColor, bgRaw, coverage);
    vec3 color = bg;

    if (uAlgorithm == 1) {
      // ── glass ──────────────────────────────────────────────────────────────
      // Strong UV distortion only. Image bends but no colour overlay.
      vec2 refrUv = clamp(vUv + normal.xy * uRefraction * density * 3.0, 0.0, 1.0);
      vec3 refrBg = mix(uWaterColor, texture2D(uBackground, mix(vUv, refrUv, 1.0 - obs)).rgb, coverage);
      color = refrBg + spec * uGlowColor * 2.5;
      color = mix(color, bg * 0.6, obs * 0.3);

    } else if (uAlgorithm == 2) {
      // ── ink ────────────────────────────────────────────────────────────────
      // Dense opaque pigment that stains. Subtle refraction underneath.
      float inkD  = min(density * 4.0, 1.0);
      vec2 refrUv = clamp(vUv + normal.xy * uRefraction * density * 0.4, 0.0, 1.0);
      vec3 refrBg = mix(uWaterColor, texture2D(uBackground, mix(vUv, refrUv, 1.0 - obs)).rgb, coverage);
      color = mix(refrBg, uWaterColor + spec * uGlowColor, inkD);
      color = mix(color, bg * 0.5, obs * 0.15);

    } else if (uAlgorithm == 3) {
      // ── aurora ─────────────────────────────────────────────────────────────
      // Velocity field warps background UVs — liquid metal / lava-lamp feel.
      vec2  vel    = texture2D(uVelocity, vUv).xy;
      float velMag = clamp(length(vel) * 20.0, 0.0, 1.0);
      vec2  warpUv = clamp(vUv + vel * uWarpStrength, 0.0, 1.0);
      vec3  warpBg = mix(uWaterColor, texture2D(uBackground, warpUv).rgb, coverage);
      color  = mix(bg, warpBg, velMag * (1.0 - obs));
      color += spec * uGlowColor * velMag * 1.5;
      color += uWaterColor * density * 0.3;
      color  = mix(color, bg * 0.5, obs * 0.2);

    } else if (uAlgorithm == 4) {
      // ── ripple ─────────────────────────────────────────────────────────────
      // Exaggerated normal perturbation + Fresnel rim — calm water surface.
      vec2  rippleUv = clamp(vUv + normal.xy * uRefraction * density * 6.0, 0.0, 1.0);
      vec3  refrBg   = mix(uWaterColor, texture2D(uBackground, mix(vUv, rippleUv, 1.0 - obs)).rgb, coverage);
      float fresnel  = pow(clamp(1.0 - dot(normal, vec3(0.0, 0.0, 1.0)), 0.0, 1.0), 3.0) * density;
      color  = refrBg;
      color += fresnel * uGlowColor * 2.0;
      color += spec * uGlowColor * density * 2.0;
      color  = mix(color, bg * 0.5, obs * 0.2);

    } else {
      // ── standard (0) ───────────────────────────────────────────────────────
      // Original: colour overlay blended over refracted background.
      vec2 refrUv = vUv + normal.xy * uRefraction * density;
      vec3 refrBg = mix(uWaterColor, texture2D(uBackground, mix(vUv, refrUv, 1.0 - obs)).rgb, coverage);
      color  = mix(refrBg, uWaterColor, min(density * 1.5, 0.8));
      color += spec * uGlowColor;
      color  = mix(color, bg * 0.5, obs * 0.2);
    }

    // Premultiplied alpha — transparent where there is neither content nor fluid,
    // letting the CSS backgroundColor on the container div show through.
    float alpha = clamp(max(density * 1.5, coverage), 0.0, 1.0);
    gl_FragColor = vec4(color * alpha, alpha);
  }
`
);
function ke(i) {
  const e = { alpha: !0, depth: !1, stencil: !1, antialias: !0, preserveDrawingBuffer: !1 };
  let r = i.getContext("webgl2", e);
  const o = !!r;
  o || (r = i.getContext("webgl", e), r.getExtension("EXT_color_buffer_half_float"));
  const s = o ? null : r.getExtension("OES_texture_half_float"), n = o ? r.HALF_FLOAT : s.HALF_FLOAT_OES;
  return r.getExtension("EXT_color_buffer_float"), r.getExtension("OES_texture_half_float_linear"), {
    gl: r,
    isWebGL2: o,
    ext: {
      internalFormat: o ? r.RGBA16F : r.RGBA,
      format: r.RGBA,
      type: n
    }
  };
}
class k {
  constructor(e, r, o) {
    ne(this, "program");
    ne(this, "uniforms", {});
    ne(this, "_gl");
    this._gl = e, this.program = e.createProgram(), e.attachShader(this.program, this._compile(e.VERTEX_SHADER, r)), e.attachShader(this.program, this._compile(e.FRAGMENT_SHADER, o)), e.linkProgram(this.program);
    const s = e.getProgramParameter(this.program, e.ACTIVE_UNIFORMS);
    for (let n = 0; n < s; n++) {
      const u = e.getActiveUniform(this.program, n).name;
      this.uniforms[u] = e.getUniformLocation(this.program, u);
    }
  }
  _compile(e, r) {
    const o = this._gl, s = o.createShader(e);
    return o.shaderSource(s, r), o.compileShader(s), s;
  }
  bind() {
    this._gl.useProgram(this.program);
  }
  dispose() {
    this._gl.deleteProgram(this.program);
  }
}
function Oe(i) {
  return {
    advection: new k(i, P, Ce),
    divergence: new k(i, P, Fe),
    pressure: new k(i, P, Xe),
    gradientSubtract: new k(i, P, Be),
    splat: new k(i, P, Ae),
    curl: new k(i, P, Le),
    vorticity: new k(i, P, Me),
    display: new k(i, P, Pe)
  };
}
function ue(i, e, r, o) {
  i.activeTexture(i.TEXTURE0);
  const s = i.createTexture();
  i.bindTexture(i.TEXTURE_2D, s), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_MIN_FILTER, i.LINEAR), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_MAG_FILTER, i.LINEAR), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_WRAP_S, i.CLAMP_TO_EDGE), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_WRAP_T, i.CLAMP_TO_EDGE), i.texImage2D(i.TEXTURE_2D, 0, e.internalFormat, r, o, 0, e.format, e.type, null);
  const n = i.createFramebuffer();
  return i.bindFramebuffer(i.FRAMEBUFFER, n), i.framebufferTexture2D(i.FRAMEBUFFER, i.COLOR_ATTACHMENT0, i.TEXTURE_2D, s, 0), { tex: s, fbo: n, width: r, height: o };
}
function de(i, e, r, o) {
  let s = ue(i, e, r, o), n = ue(i, e, r, o);
  return {
    get read() {
      return s;
    },
    get write() {
      return n;
    },
    swap() {
      [s, n] = [n, s];
    },
    dispose() {
      i.deleteTexture(s.tex), i.deleteFramebuffer(s.fbo), i.deleteTexture(n.tex), i.deleteFramebuffer(n.fbo);
    }
  };
}
function Ie(i) {
  const e = i.createBuffer();
  return i.bindBuffer(i.ARRAY_BUFFER, e), i.bufferData(i.ARRAY_BUFFER, new Float32Array([-1, -1, -1, 1, 1, 1, 1, -1]), i.STATIC_DRAW), i.vertexAttribPointer(0, 2, i.FLOAT, !1, 0, 0), i.enableVertexAttribArray(0), function(o) {
    i.bindFramebuffer(i.FRAMEBUFFER, o), i.drawArrays(i.TRIANGLE_FAN, 0, 4);
  };
}
function Te(i, e, r, o, s = "cover") {
  let n;
  s === "cover" ? n = Math.max(r / i, o / e) : s === "contain" ? n = Math.min(r / i, o / e) : typeof s == "string" && s.endsWith("%") ? n = Math.min(r / i, o / e) * (parseFloat(s) / 100) : typeof s == "string" && s.endsWith("px") ? n = parseFloat(s) / Math.max(i, e) : typeof s == "number" ? n = s : n = Math.max(r / i, o / e);
  const u = i * n, d = e * n;
  return { x: (r - u) / 2, y: (o - d) / 2, drawW: u, drawH: d };
}
function Ve(i, e, r, o, s = null, n = "cover") {
  const { text: u, fontSize: d, color: p, fontFamily: c = "sans-serif", fontWeight: m = 900 } = o, y = new OffscreenCanvas(e, r), f = y.getContext("2d");
  ((ce) => {
    if (s) {
      f.clearRect(0, 0, e, r), f.fillStyle = "black", f.fillRect(0, 0, e, r);
      const { x: fe, y: he, drawW: ve, drawH: me } = Te(
        s.width,
        s.height,
        e,
        r,
        n
      );
      f.drawImage(s, fe, he, ve, me);
    } else
      f.fillStyle = "black", f.fillRect(0, 0, e, r);
    f.fillStyle = ce, f.font = `${m} ${d}px ${c}`, f.textAlign = "center", f.textBaseline = "middle", f.fillText(u, e / 2, r / 2);
  })(p);
  const H = ee(i, y);
  f.fillStyle = "black", f.fillRect(0, 0, e, r), f.fillStyle = "white", f.font = `${m} ${d}px ${c}`, f.textAlign = "center", f.textBaseline = "middle", f.fillText(u, e / 2, r / 2);
  const ae = ee(i, y);
  return { backgroundTex: H, obstacleTex: ae, coverageTex: ae };
}
function We(i, e, r, o, s = 0, n = "cover", u = null, d = "cover") {
  const p = new OffscreenCanvas(r, o), c = p.getContext("2d"), { x: m, y, drawW: f, drawH: D } = Te(e.width, e.height, r, o, n);
  if (c.clearRect(0, 0, r, o), c.fillStyle = "black", c.fillRect(0, 0, r, o), u) {
    const {
      x: fe,
      y: he,
      drawW: ve,
      drawH: me
    } = Te(u.width, u.height, r, o, d);
    c.filter = `brightness(${s}) blur(8px)`, c.drawImage(u, fe, he, ve, me), c.filter = "none";
  }
  c.drawImage(e, m, y, f, D);
  const H = ee(i, p);
  c.clearRect(0, 0, r, o), c.fillStyle = "black", c.fillRect(0, 0, r, o), c.filter = `brightness(${s}) blur(8px)`, c.drawImage(e, m, y, f, D), c.filter = "none";
  const ae = ee(i, p);
  c.clearRect(0, 0, r, o), c.fillStyle = "black", c.fillRect(0, 0, r, o), c.fillStyle = "white", c.fillRect(
    Math.max(0, m),
    Math.max(0, y),
    Math.min(f, r - Math.max(0, m)),
    Math.min(D, o - Math.max(0, y))
  );
  const ce = ee(i, p);
  return { backgroundTex: H, obstacleTex: ae, coverageTex: ce };
}
function ee(i, e) {
  const r = i.createTexture();
  return i.bindTexture(i.TEXTURE_2D, r), i.pixelStorei(i.UNPACK_FLIP_Y_WEBGL, !0), i.texImage2D(i.TEXTURE_2D, 0, i.RGBA, i.RGBA, i.UNSIGNED_BYTE, e), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_MIN_FILTER, i.LINEAR), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_MAG_FILTER, i.LINEAR), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_WRAP_S, i.CLAMP_TO_EDGE), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_WRAP_T, i.CLAMP_TO_EDGE), r;
}
async function Ge(i) {
  const e = await fetch(i);
  if (!e.ok)
    throw new Error(`Failed to fetch image: ${i} (${e.status})`);
  const r = await e.blob();
  return createImageBitmap(r);
}
const ge = typeof requestAnimationFrame < "u" ? requestAnimationFrame.bind(globalThis) : (i) => setTimeout(i, 1e3 / 60), ze = typeof cancelAnimationFrame < "u" ? cancelAnimationFrame.bind(globalThis) : clearTimeout, Re = 0.016, Ne = { standard: 0, glass: 1, ink: 2, aurora: 3, ripple: 4 };
var W, g, te, G, q, E, b, O, I, w, Q, L, R, h, C, F, X, M, U, B, S, K, z, x, _, A, N, re, $, J, ie, pe, V, j, Y, Z, oe, Ee, se, be, le, De;
class $e {
  constructor(e, r = {}, o = {}) {
    // ---------------------------------------------------------------------------
    // Private helpers
    // ---------------------------------------------------------------------------
    l(this, $);
    l(this, ie);
    l(this, V);
    l(this, Y);
    l(this, oe);
    l(this, se);
    l(this, le);
    l(this, W, void 0);
    l(this, g, void 0);
    l(this, te, void 0);
    l(this, G, void 0);
    l(this, q, void 0);
    l(this, E, 0);
    l(this, b, 0);
    l(this, O, 0);
    l(this, I, 0);
    l(this, w, 1);
    l(this, Q, 1);
    l(this, L, 0.5);
    l(this, R, null);
    l(this, h, null);
    l(this, C, null);
    l(this, F, null);
    l(this, X, null);
    l(this, M, null);
    l(this, U, null);
    l(this, B, null);
    // binary content mask for transparent canvas support
    l(this, S, null);
    // optional background image (from backgroundSrc prop)
    l(this, K, "cover");
    l(this, z, void 0);
    l(this, x, { x: 0, y: 0, dx: 0, dy: 0, targetX: 0, targetY: 0, moved: !1 });
    // Stores source so textures can be rebuilt on resize
    l(this, _, null);
    l(this, A, null);
    l(this, N, !1);
    l(this, re, !1);
    a(this, W, e), a(this, Q, Math.max(0.1, Math.min(1, o.dpr ?? 1))), a(this, L, Math.max(0.1, Math.min(1, o.sim ?? 0.5))), a(this, z, Se(r));
    const { gl: s, ext: n } = ke(e);
    a(this, g, s), a(this, te, n), a(this, G, Oe(s)), a(this, q, Ie(s)), s.clearColor(0, 0, 0, 0);
  }
  // ---------------------------------------------------------------------------
  // Public API
  // ---------------------------------------------------------------------------
  setTextSource(e) {
    a(this, _, { type: "text", opts: e }), T(this, $, J).call(this), T(this, V, j).call(this), T(this, Y, Z).call(this);
  }
  async setImageSource(e, r = 0, o = "cover") {
    const s = await Ge(e);
    if (t(this, re)) {
      s.close();
      return;
    }
    a(this, _, { type: "image", bitmap: s, effect: r, size: o }), T(this, $, J).call(this), T(this, V, j).call(this), T(this, Y, Z).call(this);
  }
  setImageBitmap(e, r = 0, o = "cover") {
    a(this, _, { type: "image", bitmap: e, effect: r, size: o }), T(this, $, J).call(this), T(this, V, j).call(this), T(this, Y, Z).call(this);
  }
  setBackground(e, r = "cover") {
    t(this, S) && t(this, S) !== e && t(this, S).close(), a(this, S, e), a(this, K, r ?? "cover"), t(this, _) && t(this, E) > 0 && t(this, b) > 0 && T(this, V, j).call(this);
  }
  handleMove(e, r, o = 1) {
    t(this, x).moved = !0, t(this, x).dx = (e - t(this, x).targetX) * o, t(this, x).dy = (r - t(this, x).targetY) * o, t(this, x).targetX = e, t(this, x).targetY = r;
  }
  /**
   * Immediately applies one fluid splat at (x, y) with explicit velocity (vx, vy).
   * Safe to call multiple times per frame — each call writes directly to the FBOs.
   * Designed for programmatic use cases (e.g. particle systems, attractor paths)
   * where you want N independent injection points per frame without flooding the
   * mouse-state machine or the worker message queue.
   */
  splat(e, r, o, s, n = 1) {
    if (!t(this, N) || t(this, E) === 0)
      return;
    const u = t(this, g), d = t(this, z), { splat: p } = t(this, G), c = t(this, q);
    u.viewport(0, 0, t(this, O), t(this, I)), p.bind(), u.uniform1f(p.uniforms.aspectRatio, t(this, E) / t(this, b)), u.uniform2f(p.uniforms.point, e * t(this, w) / t(this, E), 1 - r * t(this, w) / t(this, b)), u.uniform1f(p.uniforms.radius, d.splatRadius), u.uniform1i(p.uniforms.uTarget, 0), u.activeTexture(u.TEXTURE0), u.bindTexture(u.TEXTURE_2D, t(this, h).read.tex), u.uniform3f(p.uniforms.color, o * d.splatForce * n, -s * d.splatForce * n, 0), c(t(this, h).write.fbo), t(this, h).swap(), u.activeTexture(u.TEXTURE0), u.bindTexture(u.TEXTURE_2D, t(this, R).read.tex), u.uniform3f(p.uniforms.color, n, n, n), c(t(this, R).write.fbo), t(this, R).swap();
  }
  updateQuality(e) {
    e.dpr !== void 0 && a(this, Q, Math.max(0.1, Math.min(1, e.dpr))), e.sim !== void 0 && a(this, L, Math.max(0.1, Math.min(1, e.sim)));
  }
  resize(e, r, o) {
    if (o !== void 0 ? a(this, w, o) : typeof window < "u" && window.devicePixelRatio && a(this, w, window.devicePixelRatio), e !== void 0 && e > 0) {
      if (r === void 0 || r <= 0)
        return;
      a(this, E, t(this, W).width = e), a(this, b, t(this, W).height = r), a(this, O, Math.max(1, Math.round(e * t(this, L)))), a(this, I, Math.max(1, Math.round(r * t(this, L)))), T(this, ie, pe).call(this);
    } else
      T(this, $, J).call(this);
    t(this, _) && T(this, V, j).call(this), T(this, Y, Z).call(this);
  }
  updateConfig(e) {
    Object.assign(t(this, z), e);
  }
  destroy() {
    a(this, re, !0), this.stop();
    const e = t(this, g);
    T(this, oe, Ee).call(this), T(this, se, be).call(this), t(this, S) && (t(this, S).close(), a(this, S, null));
    for (const o of Object.values(t(this, G)))
      o.dispose();
    const r = e.getExtension("WEBGL_lose_context");
    r == null || r.loseContext();
  }
  // ---------------------------------------------------------------------------
  // Loop control
  // ---------------------------------------------------------------------------
  start() {
    if (t(this, A) !== null)
      return;
    const e = () => {
      T(this, le, De).call(this), a(this, A, ge(e));
    };
    a(this, A, ge(e));
  }
  stop() {
    t(this, A) !== null && (ze(t(this, A)), a(this, A, null));
  }
  get isRunning() {
    return t(this, A) !== null;
  }
}
W = new WeakMap(), g = new WeakMap(), te = new WeakMap(), G = new WeakMap(), q = new WeakMap(), E = new WeakMap(), b = new WeakMap(), O = new WeakMap(), I = new WeakMap(), w = new WeakMap(), Q = new WeakMap(), L = new WeakMap(), R = new WeakMap(), h = new WeakMap(), C = new WeakMap(), F = new WeakMap(), X = new WeakMap(), M = new WeakMap(), U = new WeakMap(), B = new WeakMap(), S = new WeakMap(), K = new WeakMap(), z = new WeakMap(), x = new WeakMap(), _ = new WeakMap(), A = new WeakMap(), N = new WeakMap(), re = new WeakMap(), $ = new WeakSet(), J = function() {
  const e = t(this, W);
  "clientWidth" in e && e.clientWidth > 0 ? (a(this, w, (typeof window < "u" && window.devicePixelRatio || 1) * t(this, Q)), a(this, E, e.width = Math.round(e.clientWidth * t(this, w))), a(this, b, e.height = Math.round(e.clientHeight * t(this, w)))) : (a(this, E, e.width), a(this, b, e.height)), !(t(this, E) === 0 || t(this, b) === 0) && (a(this, O, Math.max(1, Math.round(t(this, E) * t(this, L)))), a(this, I, Math.max(1, Math.round(t(this, b) * t(this, L)))), T(this, ie, pe).call(this));
}, ie = new WeakSet(), pe = function() {
  const e = t(this, g), r = t(this, te), o = t(this, O), s = t(this, I);
  T(this, oe, Ee).call(this), a(this, R, de(e, r, o, s)), a(this, h, de(e, r, o, s)), a(this, F, de(e, r, o, s)), a(this, C, ue(e, r, o, s)), a(this, X, ue(e, r, o, s));
}, V = new WeakSet(), j = function() {
  if (!(!t(this, _) || t(this, E) === 0 || t(this, b) === 0)) {
    if (T(this, se, be).call(this), t(this, _).type === "text") {
      const { backgroundTex: e, obstacleTex: r, coverageTex: o } = Ve(
        t(this, g),
        t(this, E),
        t(this, b),
        t(this, _).opts,
        t(this, S),
        t(this, K)
      );
      a(this, M, e), a(this, U, r), a(this, B, o);
    } else {
      const { backgroundTex: e, obstacleTex: r, coverageTex: o } = We(
        t(this, g),
        t(this, _).bitmap,
        t(this, E),
        t(this, b),
        t(this, _).effect,
        t(this, _).size,
        t(this, S),
        t(this, K)
      );
      a(this, M, e), a(this, U, r), a(this, B, o);
    }
    a(this, N, !0);
  }
}, Y = new WeakSet(), Z = function() {
  t(this, N) && !this.isRunning && this.start();
}, oe = new WeakSet(), Ee = function() {
  var e, r, o;
  (e = t(this, R)) == null || e.dispose(), (r = t(this, h)) == null || r.dispose(), (o = t(this, F)) == null || o.dispose(), t(this, C) && (t(this, g).deleteTexture(t(this, C).tex), t(this, g).deleteFramebuffer(t(this, C).fbo)), t(this, X) && (t(this, g).deleteTexture(t(this, X).tex), t(this, g).deleteFramebuffer(t(this, X).fbo)), a(this, R, a(this, h, a(this, F, a(this, C, a(this, X, null)))));
}, se = new WeakSet(), be = function() {
  t(this, M) && t(this, g).deleteTexture(t(this, M)), t(this, U) && t(this, g).deleteTexture(t(this, U)), t(this, B) && t(this, B) !== t(this, U) && t(this, g).deleteTexture(t(this, B)), a(this, M, a(this, U, a(this, B, null)));
}, le = new WeakSet(), De = function() {
  if (!t(this, N) || t(this, E) === 0 || !t(this, R) || !t(this, h))
    return;
  const e = t(this, g), r = t(this, z), { advection: o, divergence: s, pressure: n, gradientSubtract: u, splat: d, curl: p, vorticity: c, display: m } = t(this, G);
  t(this, x).x += (t(this, x).targetX - t(this, x).x) * 0.15, t(this, x).y += (t(this, x).targetY - t(this, x).y) * 0.15;
  const y = t(this, O), f = t(this, I), D = t(this, q);
  e.viewport(0, 0, y, f), o.bind(), e.uniform2f(o.uniforms.texelSize, 1 / y, 1 / f), e.uniform1f(o.uniforms.dt, Re), e.uniform1i(o.uniforms.uObstacle, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, U)), e.uniform1f(o.uniforms.dissipation, r.velocityDissipation), e.uniform1i(o.uniforms.uVelocity, 1), e.activeTexture(e.TEXTURE1), e.bindTexture(e.TEXTURE_2D, t(this, h).read.tex), e.uniform1i(o.uniforms.uSource, 1), D(t(this, h).write.fbo), t(this, h).swap(), e.uniform1f(o.uniforms.dissipation, r.densityDissipation), e.uniform1i(o.uniforms.uSource, 2), e.activeTexture(e.TEXTURE2), e.bindTexture(e.TEXTURE_2D, t(this, R).read.tex), D(t(this, R).write.fbo), t(this, R).swap(), p.bind(), e.uniform2f(p.uniforms.texelSize, 1 / y, 1 / f), e.uniform1i(p.uniforms.uVelocity, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, h).read.tex), D(t(this, X).fbo), c.bind(), e.uniform2f(c.uniforms.texelSize, 1 / y, 1 / f), e.uniform1f(c.uniforms.curl, r.curl), e.uniform1f(c.uniforms.dt, Re), e.uniform1i(c.uniforms.uVelocity, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, h).read.tex), e.uniform1i(c.uniforms.uCurl, 1), e.activeTexture(e.TEXTURE1), e.bindTexture(e.TEXTURE_2D, t(this, X).tex), D(t(this, h).write.fbo), t(this, h).swap(), t(this, x).moved && (d.bind(), e.uniform1f(d.uniforms.aspectRatio, t(this, E) / t(this, b)), e.uniform2f(d.uniforms.point, t(this, x).x * t(this, w) / t(this, E), 1 - t(this, x).y * t(this, w) / t(this, b)), e.uniform1f(d.uniforms.radius, r.splatRadius), e.uniform1i(d.uniforms.uTarget, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, h).read.tex), e.uniform3f(d.uniforms.color, t(this, x).dx * r.splatForce, -t(this, x).dy * r.splatForce, 0), D(t(this, h).write.fbo), t(this, h).swap(), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, R).read.tex), e.uniform3f(d.uniforms.color, 1, 1, 1), D(t(this, R).write.fbo), t(this, R).swap(), t(this, x).moved = !1), s.bind(), e.uniform2f(s.uniforms.texelSize, 1 / y, 1 / f), e.uniform1i(s.uniforms.uVelocity, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, h).read.tex), e.uniform1i(s.uniforms.uObstacle, 1), e.activeTexture(e.TEXTURE1), e.bindTexture(e.TEXTURE_2D, t(this, U)), D(t(this, C).fbo), n.bind(), e.uniform2f(n.uniforms.texelSize, 1 / y, 1 / f), e.uniform1i(n.uniforms.uDivergence, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, C).tex), e.uniform1i(n.uniforms.uObstacle, 1), e.activeTexture(e.TEXTURE1), e.bindTexture(e.TEXTURE_2D, t(this, U));
  for (let H = 0; H < r.pressureIterations; H++)
    e.uniform1i(n.uniforms.uPressure, 2), e.activeTexture(e.TEXTURE2), e.bindTexture(e.TEXTURE_2D, t(this, F).read.tex), D(t(this, F).write.fbo), t(this, F).swap();
  u.bind(), e.uniform2f(u.uniforms.texelSize, 1 / y, 1 / f), e.uniform1i(u.uniforms.uPressure, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, F).read.tex), e.uniform1i(u.uniforms.uVelocity, 1), e.activeTexture(e.TEXTURE1), e.bindTexture(e.TEXTURE_2D, t(this, h).read.tex), e.uniform1i(u.uniforms.uObstacle, 2), e.activeTexture(e.TEXTURE2), e.bindTexture(e.TEXTURE_2D, t(this, U)), D(t(this, h).write.fbo), t(this, h).swap(), e.viewport(0, 0, t(this, E), t(this, b)), e.bindFramebuffer(e.FRAMEBUFFER, null), e.clear(e.COLOR_BUFFER_BIT), m.bind(), e.uniform2f(m.uniforms.texelSize, 1 / t(this, E), 1 / t(this, b)), e.uniform3fv(m.uniforms.uWaterColor, r.waterColor), e.uniform3fv(m.uniforms.uGlowColor, r.glowColor), e.uniform1f(m.uniforms.uRefraction, r.refraction), e.uniform1f(m.uniforms.uSpecularExp, r.specularExp), e.uniform1f(m.uniforms.uShine, r.shine), e.uniform1f(m.uniforms.uWarpStrength, r.warpStrength ?? 0.015), e.uniform1i(m.uniforms.uAlgorithm, Ne[r.algorithm] ?? 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, R).read.tex), e.activeTexture(e.TEXTURE1), e.bindTexture(e.TEXTURE_2D, t(this, U)), e.activeTexture(e.TEXTURE2), e.bindTexture(e.TEXTURE_2D, t(this, M)), e.activeTexture(e.TEXTURE3), e.bindTexture(e.TEXTURE_2D, t(this, B)), e.activeTexture(e.TEXTURE4), e.bindTexture(e.TEXTURE_2D, t(this, h).read.tex), e.uniform1i(m.uniforms.uTexture, 0), e.uniform1i(m.uniforms.uObstacle, 1), e.uniform1i(m.uniforms.uBackground, 2), e.uniform1i(m.uniforms.uCoverage, 3), e.uniform1i(m.uniforms.uVelocity, 4), D(null);
};
let v = null;
self.onmessage = async (i) => {
  const { type: e, ...r } = i.data;
  try {
    switch (e) {
      case "init": {
        const { canvas: o, width: s, height: n, config: u, dpr: d, quality: p } = r;
        o.width = s, o.height = n, v = new $e(o, u, p ?? {}), v.resize(s, n, d || 1), self.postMessage({ type: "ready" });
        break;
      }
      case "setTextSource": {
        if (!v)
          return;
        v.setTextSource(r.opts);
        break;
      }
      case "setImageSource": {
        if (!v)
          return;
        await v.setImageSource(
          r.src,
          r.effect,
          r.size
        );
        break;
      }
      case "setImageBitmap": {
        if (!v)
          return;
        v.setImageBitmap(
          r.bitmap,
          r.effect,
          r.size
        );
        break;
      }
      case "setBackground": {
        if (!v)
          return;
        v.setBackground(r.bitmap, r.size);
        break;
      }
      case "splat": {
        if (!v)
          return;
        v.splat(
          r.x,
          r.y,
          r.vx,
          r.vy,
          r.strength ?? 1
        );
        break;
      }
      case "move": {
        if (!v)
          return;
        v.handleMove(r.x, r.y, r.strength ?? 1);
        break;
      }
      case "resize": {
        if (!v)
          return;
        v.resize(r.width, r.height, r.dpr);
        break;
      }
      case "updateQuality": {
        if (!v)
          return;
        v.updateQuality(r.quality);
        break;
      }
      case "updateConfig": {
        if (!v)
          return;
        v.updateConfig(r.config);
        break;
      }
      case "destroy": {
        v == null || v.destroy(), v = null;
        break;
      }
      default:
        console.warn("[fluidity-js worker] Unknown message type:", e);
    }
  } catch (o) {
    self.postMessage({ type: "error", message: (o == null ? void 0 : o.message) ?? String(o) });
  }
};
", JI = typeof window < "u" && window.Blob && new Blob([atob(PI)], { type: "text/javascript;charset=utf-8" });
|
|
680
|
-
function Gg() {
|
|
681
|
-
|
|
1536
|
+
let dI = yI;
|
|
1537
|
+
const NI = "var gt = Object.defineProperty;
var bt = (i, e, r) => e in i ? gt(i, e, { enumerable: !0, configurable: !0, writable: !0, value: r }) : i[e] = r;
var ke = (i, e, r) => (bt(i, typeof e != "symbol" ? e + "" : e, r), r), Je = (i, e, r) => {
  if (!e.has(i))
    throw TypeError("Cannot " + r);
};
var t = (i, e, r) => (Je(i, e, "read from private field"), r ? r.call(i) : e.get(i)), v = (i, e, r) => {
  if (e.has(i))
    throw TypeError("Cannot add the same private member more than once");
  e instanceof WeakSet ? e.add(i) : e.set(i, r);
}, u = (i, e, r, s) => (Je(i, e, "write to private field"), s ? s.call(i, r) : e.set(i, r), r);
var S = (i, e, r) => (Je(i, e, "access private method"), r);
const ct = {
  densityDissipation: 0.992,
  velocityDissipation: 0.93,
  pressureIterations: 1,
  curl: 1e-4,
  splatRadius: 4e-3,
  splatForce: 0.91,
  refraction: 0.25,
  specularExp: 1.01,
  shine: 0.01,
  waterColor: [0, 0, 0],
  glowColor: [0.7, 0.85, 1],
  algorithm: "standard",
  warpStrength: 0.015
};
({
  ...ct
});
const Tt = {
  calm: {
    densityDissipation: 0.999,
    velocityDissipation: 0.98,
    curl: 1e-4,
    splatRadius: 3e-3,
    splatForce: 0.5,
    refraction: 0.15,
    shine: 5e-3,
    glowColor: [0.6, 0.85, 1],
    waterColor: [0, 0.02, 0.05]
  },
  sand: {
    densityDissipation: 0.997,
    velocityDissipation: 0.98,
    curl: 1,
    splatRadius: 0.01,
    splatForce: 0.9,
    refraction: 0.8,
    specularExp: 0.1,
    shine: 0.05,
    glowColor: [0.027, 0.027, 0.027],
    waterColor: [0.451, 0.329, 0.125]
  },
  wave: {
    densityDissipation: 0.994,
    velocityDissipation: 0.92,
    curl: 0.2,
    splatRadius: 5e-3,
    splatForce: 1.2,
    refraction: 0.35,
    shine: 0.03,
    glowColor: [0.5, 0.8, 1],
    waterColor: [0, 0.01, 0.03]
  },
  neon: {
    densityDissipation: 0.985,
    velocityDissipation: 0.93,
    curl: 0.05,
    splatRadius: 8e-3,
    splatForce: 1.5,
    refraction: 0.25,
    specularExp: 0.5,
    shine: 0.14,
    glowColor: [1, 0.2, 0.8],
    waterColor: [0.05, 0, 0.08]
  },
  smoke: {
    densityDissipation: 0.996,
    velocityDissipation: 0.97,
    curl: 0.04,
    splatRadius: 9e-3,
    splatForce: 0.8,
    refraction: 0.08,
    shine: 0,
    glowColor: [0.5, 0.5, 0.5],
    waterColor: [0.06, 0.06, 0.06]
  }
};
function yt(i = {}, e, r = ct) {
  return { ...e ? { ...r, ...Tt[e] } : r, ...i };
}
const re = (
  /* glsl */
  `
  precision highp float;
  attribute vec2 aPosition;
  varying vec2 vUv;
  varying vec2 vL; varying vec2 vR; varying vec2 vT; varying vec2 vB;
  uniform vec2 texelSize;
  void main () {
    vUv = aPosition * 0.5 + 0.5;
    vL = vUv - vec2(texelSize.x, 0.0);
    vR = vUv + vec2(texelSize.x, 0.0);
    vT = vUv + vec2(0.0, texelSize.y);
    vB = vUv - vec2(0.0, texelSize.y);
    gl_Position = vec4(aPosition, 0.0, 1.0);
  }
`
), St = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vUv;
  uniform sampler2D uVelocity;
  uniform sampler2D uSource;
  uniform sampler2D uObstacle;
  uniform vec2 texelSize;
  uniform float dt;
  uniform float dissipation;
  void main () {
    if (texture2D(uObstacle, vUv).r > 0.5) { gl_FragColor = vec4(0.0); return; }
    vec2 coord = vUv - dt * texture2D(uVelocity, vUv).xy * texelSize;
    gl_FragColor = dissipation * texture2D(uSource, coord);
  }
`
), wt = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vL; varying vec2 vR; varying vec2 vT; varying vec2 vB;
  uniform sampler2D uVelocity;
  uniform sampler2D uObstacle;
  void main () {
    float L = texture2D(uObstacle, vL).r > 0.5 ? 0.0 : texture2D(uVelocity, vL).x;
    float R = texture2D(uObstacle, vR).r > 0.5 ? 0.0 : texture2D(uVelocity, vR).x;
    float T = texture2D(uObstacle, vT).r > 0.5 ? 0.0 : texture2D(uVelocity, vT).y;
    float B = texture2D(uObstacle, vB).r > 0.5 ? 0.0 : texture2D(uVelocity, vB).y;
    gl_FragColor = vec4(0.5 * (R - L + T - B), 0.0, 0.0, 1.0);
  }
`
), Rt = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vUv; varying vec2 vL; varying vec2 vR; varying vec2 vT; varying vec2 vB;
  uniform sampler2D uPressure;
  uniform sampler2D uDivergence;
  uniform sampler2D uObstacle;
  void main () {
    float C = texture2D(uPressure, vUv).x;
    float L = texture2D(uObstacle, vL).r > 0.5 ? C : texture2D(uPressure, vL).x;
    float R = texture2D(uObstacle, vR).r > 0.5 ? C : texture2D(uPressure, vR).x;
    float T = texture2D(uObstacle, vT).r > 0.5 ? C : texture2D(uPressure, vT).x;
    float B = texture2D(uObstacle, vB).r > 0.5 ? C : texture2D(uPressure, vB).x;
    float div = texture2D(uDivergence, vUv).x;
    gl_FragColor = vec4((L + R + B + T - div) * 0.25, 0.0, 0.0, 1.0);
  }
`
), Et = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vUv; varying vec2 vL; varying vec2 vR; varying vec2 vT; varying vec2 vB;
  uniform sampler2D uPressure;
  uniform sampler2D uVelocity;
  uniform sampler2D uObstacle;
  void main () {
    if (texture2D(uObstacle, vUv).r > 0.5) { gl_FragColor = vec4(0.0); return; }
    float L = texture2D(uPressure, vL).x;
    float R = texture2D(uPressure, vR).x;
    float T = texture2D(uPressure, vT).x;
    float B = texture2D(uPressure, vB).x;
    vec2 vel = texture2D(uVelocity, vUv).xy - vec2(R - L, T - B);
    gl_FragColor = vec4(vel, 0.0, 1.0);
  }
`
), Ut = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vUv;
  uniform sampler2D uTarget;
  uniform float aspectRatio;
  uniform vec3 color;
  uniform vec2 point;
  uniform float radius;
  void main () {
    vec2 p = vUv - point.xy;
    p.x *= aspectRatio;
    vec3 splat = exp(-dot(p, p) / radius) * color;
    gl_FragColor = vec4(texture2D(uTarget, vUv).xyz + splat, 1.0);
  }
`
), Dt = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vL; varying vec2 vR; varying vec2 vT; varying vec2 vB;
  uniform sampler2D uVelocity;
  void main () {
    float L = texture2D(uVelocity, vL).y;
    float R = texture2D(uVelocity, vR).y;
    float T = texture2D(uVelocity, vT).x;
    float B = texture2D(uVelocity, vB).x;
    gl_FragColor = vec4(0.5 * (R - L - T + B), 0.0, 0.0, 1.0);
  }
`
), _t = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vUv; varying vec2 vL; varying vec2 vR; varying vec2 vT; varying vec2 vB;
  uniform sampler2D uVelocity;
  uniform sampler2D uCurl;
  uniform float curl;
  uniform float dt;
  void main () {
    float L = texture2D(uCurl, vL).x;
    float R = texture2D(uCurl, vR).x;
    float T = texture2D(uCurl, vT).x;
    float B = texture2D(uCurl, vB).x;
    float C = texture2D(uCurl, vUv).x;
    vec2 force = 0.5 * vec2(abs(T) - abs(B), abs(R) - abs(L));
    force /= length(force) + 0.0001;
    force *= curl * 30.0 * C;
    gl_FragColor = vec4(texture2D(uVelocity, vUv).xy + force * dt, 0.0, 1.0);
  }
`
), Ct = (
  /* glsl */
  `
  precision highp float;
  varying vec2 vUv;

  uniform sampler2D uTexture;
  uniform sampler2D uObstacle;
  uniform sampler2D uBackground;
  uniform sampler2D uCoverage;
  uniform sampler2D uVelocity;

  uniform vec2  texelSize;
  uniform vec3  uWaterColor;
  uniform vec3  uGlowColor;
  uniform float uRefraction;
  uniform float uSpecularExp;
  uniform float uShine;
  uniform float uWarpStrength;
  uniform int   uAlgorithm;

  void main () {
    float obs      = texture2D(uObstacle,  vUv).r;
    // Mask density inside obstacles so splats don't flicker the text/image content.
    float density  = max(texture2D(uTexture, vUv).r, 0.0) * (1.0 - step(0.5, obs));
    float coverage = texture2D(uCoverage,  vUv).r;

    float dL = max(texture2D(uTexture, vUv - vec2(texelSize.x * 2.0, 0.0)).r, 0.0);
    float dR = max(texture2D(uTexture, vUv + vec2(texelSize.x * 2.0, 0.0)).r, 0.0);
    float dT = max(texture2D(uTexture, vUv + vec2(0.0, texelSize.y * 2.0)).r, 0.0);
    float dB = max(texture2D(uTexture, vUv - vec2(0.0, texelSize.y * 2.0)).r, 0.0);

    vec3  normal   = normalize(vec3(dL - dR, dB - dT, 0.2));
    vec3  lightDir = normalize(vec3(0.5, 1.0, 0.5));
    vec3  halfV    = normalize(lightDir + vec3(0.0, 0.0, 1.0));
    float spec     = pow(max(dot(normal, halfV), 0.0), uSpecularExp) * uShine * density;

    // In transparent (non-coverage) areas the background texture is empty black canvas.
    // Replace it with uWaterColor so fluid colour is not contaminated by that black,
    // allowing the CSS backgroundColor to show through correctly via premultiplied alpha.
    vec3 bgRaw = texture2D(uBackground, vUv).rgb;
    vec3 bg    = mix(uWaterColor, bgRaw, coverage);
    vec3 color = bg;

    if (uAlgorithm == 1) {
      // ── glass ──────────────────────────────────────────────────────────────
      // Strong UV distortion only. Image bends but no colour overlay.
      vec2 refrUv = clamp(vUv + normal.xy * uRefraction * density * 3.0, 0.0, 1.0);
      vec3 refrBg = mix(uWaterColor, texture2D(uBackground, mix(vUv, refrUv, 1.0 - obs)).rgb, coverage);
      color = refrBg + spec * uGlowColor * 2.5;
      color = mix(color, bg * 0.6, obs * 0.3);

    } else if (uAlgorithm == 2) {
      // ── ink ────────────────────────────────────────────────────────────────
      // Dense opaque pigment that stains. Subtle refraction underneath.
      float inkD  = min(density * 4.0, 1.0);
      vec2 refrUv = clamp(vUv + normal.xy * uRefraction * density * 0.4, 0.0, 1.0);
      vec3 refrBg = mix(uWaterColor, texture2D(uBackground, mix(vUv, refrUv, 1.0 - obs)).rgb, coverage);
      color = mix(refrBg, uWaterColor + spec * uGlowColor, inkD);
      color = mix(color, bg * 0.5, obs * 0.15);

    } else if (uAlgorithm == 3) {
      // ── aurora ─────────────────────────────────────────────────────────────
      // Velocity field warps background UVs — liquid metal / lava-lamp feel.
      vec2  vel    = texture2D(uVelocity, vUv).xy;
      float velMag = clamp(length(vel) * 20.0, 0.0, 1.0);
      vec2  warpUv = clamp(vUv + vel * uWarpStrength, 0.0, 1.0);
      vec3  warpBg = mix(uWaterColor, texture2D(uBackground, warpUv).rgb, coverage);
      color  = mix(bg, warpBg, velMag * (1.0 - obs));
      color += spec * uGlowColor * velMag * 1.5;
      color += uWaterColor * density * 0.3;
      color  = mix(color, bg * 0.5, obs * 0.2);

    } else if (uAlgorithm == 4) {
      // ── ripple ─────────────────────────────────────────────────────────────
      // Exaggerated normal perturbation + Fresnel rim — calm water surface.
      vec2  rippleUv = clamp(vUv + normal.xy * uRefraction * density * 6.0, 0.0, 1.0);
      vec3  refrBg   = mix(uWaterColor, texture2D(uBackground, mix(vUv, rippleUv, 1.0 - obs)).rgb, coverage);
      float fresnel  = pow(clamp(1.0 - dot(normal, vec3(0.0, 0.0, 1.0)), 0.0, 1.0), 3.0) * density;
      color  = refrBg;
      color += fresnel * uGlowColor * 2.0;
      color += spec * uGlowColor * density * 2.0;
      color  = mix(color, bg * 0.5, obs * 0.2);

    } else {
      // ── standard (0) ───────────────────────────────────────────────────────
      // Original: colour overlay blended over refracted background.
      vec2 refrUv = vUv + normal.xy * uRefraction * density;
      vec3 refrBg = mix(uWaterColor, texture2D(uBackground, mix(vUv, refrUv, 1.0 - obs)).rgb, coverage);
      color  = mix(refrBg, uWaterColor, min(density * 1.5, 0.8));
      color += spec * uGlowColor;
      color  = mix(color, bg * 0.5, obs * 0.2);
    }

    // Premultiplied alpha — transparent where there is neither content nor fluid,
    // letting the CSS backgroundColor on the container div show through.
    float alpha = clamp(max(density * 1.5, coverage), 0.0, 1.0);
    gl_FragColor = vec4(color * alpha, alpha);
  }
`
);
function Bt(i) {
  const e = { alpha: !0, depth: !1, stencil: !1, antialias: !0, preserveDrawingBuffer: !1 };
  let r = i.getContext("webgl2", e);
  const s = !!r;
  s || (r = i.getContext("webgl", e), r.getExtension("EXT_color_buffer_half_float"));
  const o = s ? null : r.getExtension("OES_texture_half_float"), a = s ? r.HALF_FLOAT : o.HALF_FLOAT_OES;
  return r.getExtension("EXT_color_buffer_float"), r.getExtension("OES_texture_half_float_linear"), {
    type: s ? "webgl2" : "webgl1",
    gl: r,
    isWebGL2: s,
    ext: {
      internalFormat: s ? r.RGBA16F : r.RGBA,
      format: r.RGBA,
      type: a
    }
  };
}
async function Vt(i) {
  if (typeof navigator > "u" || !navigator.gpu)
    return null;
  try {
    const e = await navigator.gpu.requestAdapter();
    if (!e)
      return null;
    const r = await e.requestDevice(), s = i.getContext("webgpu");
    if (!s)
      return null;
    const o = navigator.gpu.getPreferredCanvasFormat();
    return s.configure({ device: r, format: o, alphaMode: "premultiplied" }), { type: "webgpu", adapter: e, device: r, context: s, format: o };
  } catch {
    return null;
  }
}
class ie {
  constructor(e, r, s) {
    ke(this, "program");
    ke(this, "uniforms", {});
    ke(this, "_gl");
    this._gl = e, this.program = e.createProgram(), e.attachShader(this.program, this._compile(e.VERTEX_SHADER, r)), e.attachShader(this.program, this._compile(e.FRAGMENT_SHADER, s)), e.linkProgram(this.program);
    const o = e.getProgramParameter(this.program, e.ACTIVE_UNIFORMS);
    for (let a = 0; a < o; a++) {
      const n = e.getActiveUniform(this.program, a).name;
      this.uniforms[n] = e.getUniformLocation(this.program, n);
    }
  }
  _compile(e, r) {
    const s = this._gl, o = s.createShader(e);
    return s.shaderSource(o, r), s.compileShader(o), o;
  }
  bind() {
    this._gl.useProgram(this.program);
  }
  dispose() {
    this._gl.deleteProgram(this.program);
  }
}
function Pt(i) {
  return {
    advection: new ie(i, re, St),
    divergence: new ie(i, re, wt),
    pressure: new ie(i, re, Rt),
    gradientSubtract: new ie(i, re, Et),
    splat: new ie(i, re, Ut),
    curl: new ie(i, re, Dt),
    vorticity: new ie(i, re, _t),
    display: new ie(i, re, Ct)
  };
}
function $e(i, e, r, s) {
  i.activeTexture(i.TEXTURE0);
  const o = i.createTexture();
  i.bindTexture(i.TEXTURE_2D, o), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_MIN_FILTER, i.LINEAR), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_MAG_FILTER, i.LINEAR), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_WRAP_S, i.CLAMP_TO_EDGE), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_WRAP_T, i.CLAMP_TO_EDGE), i.texImage2D(i.TEXTURE_2D, 0, e.internalFormat, r, s, 0, e.format, e.type, null);
  const a = i.createFramebuffer();
  return i.bindFramebuffer(i.FRAMEBUFFER, a), i.framebufferTexture2D(i.FRAMEBUFFER, i.COLOR_ATTACHMENT0, i.TEXTURE_2D, o, 0), { tex: o, fbo: a, width: r, height: s };
}
function Ze(i, e, r, s) {
  let o = $e(i, e, r, s), a = $e(i, e, r, s);
  return {
    get read() {
      return o;
    },
    get write() {
      return a;
    },
    swap() {
      [o, a] = [a, o];
    },
    dispose() {
      i.deleteTexture(o.tex), i.deleteFramebuffer(o.fbo), i.deleteTexture(a.tex), i.deleteFramebuffer(a.fbo);
    }
  };
}
function Ot(i) {
  const e = i.createBuffer();
  return i.bindBuffer(i.ARRAY_BUFFER, e), i.bufferData(i.ARRAY_BUFFER, new Float32Array([-1, -1, -1, 1, 1, 1, 1, -1]), i.STATIC_DRAW), i.vertexAttribPointer(0, 2, i.FLOAT, !1, 0, 0), i.enableVertexAttribArray(0), function(s) {
    i.bindFramebuffer(i.FRAMEBUFFER, s), i.drawArrays(i.TRIANGLE_FAN, 0, 4);
  };
}
const le = (
  /* wgsl */
  `
struct VSOut {
  @builtin(position) pos : vec4f,
  @location(0)       uv  : vec2f,
  @location(1)       vL  : vec2f,
  @location(2)       vR  : vec2f,
  @location(3)       vT  : vec2f,
  @location(4)       vB  : vec2f,
}`
), Ft = (
  /* wgsl */
  `
${le}

struct U {
  texelSize  : vec2f,
  dt         : f32,
  dissipation: f32,
}
@group(0) @binding(0) var<uniform> u     : U;
@group(0) @binding(1) var          samp  : sampler;
@group(0) @binding(2) var          uVel  : texture_2d<f32>;
@group(0) @binding(3) var          uSrc  : texture_2d<f32>;
@group(0) @binding(4) var          uObs  : texture_2d<f32>;

@vertex fn vs(@location(0) a: vec2f) -> VSOut {
  var o: VSOut;
  o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
  o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
  o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
  o.vT = o.uv + vec2f(0.0, u.texelSize.y);
  o.vB = o.uv - vec2f(0.0, u.texelSize.y);
  o.pos = vec4f(a, 0.0, 1.0);
  return o;
}

@fragment fn fs(i: VSOut) -> @location(0) vec4f {
  // Sample all textures before branching — textureSample requires uniform control flow
  let obs   = textureSample(uObs, samp, i.uv).r;
  let vel   = textureSample(uVel, samp, i.uv).xy;
  let coord = i.uv - u.dt * vel * u.texelSize;
  let src   = textureSample(uSrc, samp, coord);
  if (obs > 0.5) { return vec4f(0.0); }
  return u.dissipation * src;
}
`
), Lt = (
  /* wgsl */
  `
${le}

struct U { texelSize: vec2f, _pad: vec2f }
@group(0) @binding(0) var<uniform> u    : U;
@group(0) @binding(1) var          samp : sampler;
@group(0) @binding(2) var          uVel : texture_2d<f32>;
@group(0) @binding(3) var          uObs : texture_2d<f32>;

@vertex fn vs(@location(0) a: vec2f) -> VSOut {
  var o: VSOut;
  o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
  o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
  o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
  o.vT = o.uv + vec2f(0.0, u.texelSize.y);
  o.vB = o.uv - vec2f(0.0, u.texelSize.y);
  o.pos = vec4f(a, 0.0, 1.0);
  return o;
}

@fragment fn fs(i: VSOut) -> @location(0) vec4f {
  let L = select(textureSample(uVel, samp, i.vL).x, 0.0, textureSample(uObs, samp, i.vL).r > 0.5);
  let R = select(textureSample(uVel, samp, i.vR).x, 0.0, textureSample(uObs, samp, i.vR).r > 0.5);
  let T = select(textureSample(uVel, samp, i.vT).y, 0.0, textureSample(uObs, samp, i.vT).r > 0.5);
  let B = select(textureSample(uVel, samp, i.vB).y, 0.0, textureSample(uObs, samp, i.vB).r > 0.5);
  return vec4f(0.5 * (R - L + T - B), 0.0, 0.0, 1.0);
}
`
), At = (
  /* wgsl */
  `
${le}

struct U { texelSize: vec2f, _pad: vec2f }
@group(0) @binding(0) var<uniform> u    : U;
@group(0) @binding(1) var          samp : sampler;
@group(0) @binding(2) var          uPres: texture_2d<f32>;
@group(0) @binding(3) var          uDiv : texture_2d<f32>;
@group(0) @binding(4) var          uObs : texture_2d<f32>;

@vertex fn vs(@location(0) a: vec2f) -> VSOut {
  var o: VSOut;
  o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
  o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
  o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
  o.vT = o.uv + vec2f(0.0, u.texelSize.y);
  o.vB = o.uv - vec2f(0.0, u.texelSize.y);
  o.pos = vec4f(a, 0.0, 1.0);
  return o;
}

@fragment fn fs(i: VSOut) -> @location(0) vec4f {
  let C  = textureSample(uPres, samp, i.uv).x;
  let L  = select(textureSample(uPres, samp, i.vL).x, C, textureSample(uObs, samp, i.vL).r > 0.5);
  let R  = select(textureSample(uPres, samp, i.vR).x, C, textureSample(uObs, samp, i.vR).r > 0.5);
  let T  = select(textureSample(uPres, samp, i.vT).x, C, textureSample(uObs, samp, i.vT).r > 0.5);
  let B  = select(textureSample(uPres, samp, i.vB).x, C, textureSample(uObs, samp, i.vB).r > 0.5);
  let dv = textureSample(uDiv, samp, i.uv).x;
  return vec4f((L + R + B + T - dv) * 0.25, 0.0, 0.0, 1.0);
}
`
), Xt = (
  /* wgsl */
  `
${le}

struct U { texelSize: vec2f, _pad: vec2f }
@group(0) @binding(0) var<uniform> u    : U;
@group(0) @binding(1) var          samp : sampler;
@group(0) @binding(2) var          uPres: texture_2d<f32>;
@group(0) @binding(3) var          uVel : texture_2d<f32>;
@group(0) @binding(4) var          uObs : texture_2d<f32>;

@vertex fn vs(@location(0) a: vec2f) -> VSOut {
  var o: VSOut;
  o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
  o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
  o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
  o.vT = o.uv + vec2f(0.0, u.texelSize.y);
  o.vB = o.uv - vec2f(0.0, u.texelSize.y);
  o.pos = vec4f(a, 0.0, 1.0);
  return o;
}

@fragment fn fs(i: VSOut) -> @location(0) vec4f {
  // Sample all textures before branching — textureSample requires uniform control flow
  let obs = textureSample(uObs, samp, i.uv).r;
  let L   = textureSample(uPres, samp, i.vL).x;
  let R   = textureSample(uPres, samp, i.vR).x;
  let T   = textureSample(uPres, samp, i.vT).x;
  let B   = textureSample(uPres, samp, i.vB).x;
  let vel = textureSample(uVel, samp, i.uv).xy - vec2f(R - L, T - B);
  if (obs > 0.5) { return vec4f(0.0); }
  return vec4f(vel, 0.0, 1.0);
}
`
), zt = (
  /* wgsl */
  `
${le}

// texelSize occupies bytes 0-7 for the shared vertex stage; aspectRatio/radius fill the rest.
struct U {
  texelSize  : vec2f,
  aspectRatio: f32,
  radius     : f32,
  color      : vec4f,  // xyz = colour, w unused
  point      : vec2f,
  _pad       : vec2f,
}
@group(0) @binding(0) var<uniform> u    : U;
@group(0) @binding(1) var          samp : sampler;
@group(0) @binding(2) var          uTgt : texture_2d<f32>;

@vertex fn vs(@location(0) a: vec2f) -> VSOut {
  var o: VSOut;
  o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
  o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
  o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
  o.vT = o.uv + vec2f(0.0, u.texelSize.y);
  o.vB = o.uv - vec2f(0.0, u.texelSize.y);
  o.pos = vec4f(a, 0.0, 1.0);
  return o;
}

@fragment fn fs(i: VSOut) -> @location(0) vec4f {
  var p  = i.uv - u.point;
  p.x   *= u.aspectRatio;
  let sp = exp(-dot(p, p) / u.radius) * u.color.xyz;
  return vec4f(textureSample(uTgt, samp, i.uv).xyz + sp, 1.0);
}
`
), Mt = (
  /* wgsl */
  `
${le}

struct U { texelSize: vec2f, _pad: vec2f }
@group(0) @binding(0) var<uniform> u    : U;
@group(0) @binding(1) var          samp : sampler;
@group(0) @binding(2) var          uVel : texture_2d<f32>;

@vertex fn vs(@location(0) a: vec2f) -> VSOut {
  var o: VSOut;
  o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
  o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
  o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
  o.vT = o.uv + vec2f(0.0, u.texelSize.y);
  o.vB = o.uv - vec2f(0.0, u.texelSize.y);
  o.pos = vec4f(a, 0.0, 1.0);
  return o;
}

@fragment fn fs(i: VSOut) -> @location(0) vec4f {
  let L = textureSample(uVel, samp, i.vL).y;
  let R = textureSample(uVel, samp, i.vR).y;
  let T = textureSample(uVel, samp, i.vT).x;
  let B = textureSample(uVel, samp, i.vB).x;
  return vec4f(0.5 * (R - L - T + B), 0.0, 0.0, 1.0);
}
`
), Gt = (
  /* wgsl */
  `
${le}

struct U {
  texelSize: vec2f,
  curl     : f32,
  dt       : f32,
}
@group(0) @binding(0) var<uniform> u    : U;
@group(0) @binding(1) var          samp : sampler;
@group(0) @binding(2) var          uVel : texture_2d<f32>;
@group(0) @binding(3) var          uCrl : texture_2d<f32>;

@vertex fn vs(@location(0) a: vec2f) -> VSOut {
  var o: VSOut;
  o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
  o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
  o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
  o.vT = o.uv + vec2f(0.0, u.texelSize.y);
  o.vB = o.uv - vec2f(0.0, u.texelSize.y);
  o.pos = vec4f(a, 0.0, 1.0);
  return o;
}

@fragment fn fs(i: VSOut) -> @location(0) vec4f {
  let L     = textureSample(uCrl, samp, i.vL).x;
  let R     = textureSample(uCrl, samp, i.vR).x;
  let T     = textureSample(uCrl, samp, i.vT).x;
  let B     = textureSample(uCrl, samp, i.vB).x;
  let C     = textureSample(uCrl, samp, i.uv).x;
  var force = 0.5 * vec2f(abs(T) - abs(B), abs(R) - abs(L));
  force    /= length(force) + 0.0001;
  force    *= u.curl * 30.0 * C;
  let vel   = textureSample(uVel, samp, i.uv).xy + force * u.dt;
  return vec4f(vel, 0.0, 1.0);
}
`
), kt = (
  /* wgsl */
  `
${le}

struct U {
  texelSize   : vec2f,
  refraction  : f32,
  specularExp : f32,
  waterColor  : vec4f,
  glowColor   : vec4f,
  shine       : f32,
  warpStrength: f32,
  algorithm   : i32,
  _pad        : f32,
}
@group(0) @binding(0) var<uniform> u    : U;
@group(0) @binding(1) var          samp : sampler;
@group(0) @binding(2) var          uTex : texture_2d<f32>;
@group(0) @binding(3) var          uObs : texture_2d<f32>;
@group(0) @binding(4) var          uBg  : texture_2d<f32>;
@group(0) @binding(5) var          uCov : texture_2d<f32>;
@group(0) @binding(6) var          uVel : texture_2d<f32>;

@vertex fn vs(@location(0) a: vec2f) -> VSOut {
  var o: VSOut;
  o.uv = vec2f(a.x * 0.5 + 0.5, 0.5 - a.y * 0.5);
  o.vL = o.uv - vec2f(u.texelSize.x, 0.0);
  o.vR = o.uv + vec2f(u.texelSize.x, 0.0);
  o.vT = o.uv + vec2f(0.0, u.texelSize.y);
  o.vB = o.uv - vec2f(0.0, u.texelSize.y);
  o.pos = vec4f(a, 0.0, 1.0);
  return o;
}

@fragment fn fs(i: VSOut) -> @location(0) vec4f {
  let obs     = textureSample(uObs, samp, i.uv).r;
  let density = max(textureSample(uTex, samp, i.uv).r, 0.0) * (1.0 - step(0.5, obs));
  let cov     = textureSample(uCov, samp, i.uv).r;

  let ts2 = u.texelSize * 2.0;
  let dL  = max(textureSample(uTex, samp, i.uv - vec2f(ts2.x, 0.0)).r, 0.0);
  let dR  = max(textureSample(uTex, samp, i.uv + vec2f(ts2.x, 0.0)).r, 0.0);
  let dT  = max(textureSample(uTex, samp, i.uv + vec2f(0.0, ts2.y)).r, 0.0);
  let dB  = max(textureSample(uTex, samp, i.uv - vec2f(0.0, ts2.y)).r, 0.0);

  let norm  = normalize(vec3f(dL - dR, dB - dT, 0.2));
  let ldir  = normalize(vec3f(0.5, 1.0, 0.5));
  let halfV = normalize(ldir + vec3f(0.0, 0.0, 1.0));
  let spec  = pow(max(dot(norm, halfV), 0.0), u.specularExp) * u.shine * density;

  let bgRaw = textureSample(uBg, samp, i.uv).rgb;
  let wc    = u.waterColor.rgb;
  let gc    = u.glowColor.rgb;
  let bg    = mix(wc, bgRaw, cov);
  var color = bg;

  if (u.algorithm == 1) {
    let ruv = clamp(i.uv + norm.xy * u.refraction * density * 3.0, vec2f(0.0), vec2f(1.0));
    let rbg = mix(wc, textureSample(uBg, samp, mix(i.uv, ruv, 1.0 - obs)).rgb, cov);
    color   = rbg + spec * gc * 2.5;
    color   = mix(color, bg * 0.6, obs * 0.3);
  } else if (u.algorithm == 2) {
    let inkD = min(density * 4.0, 1.0);
    let ruv  = clamp(i.uv + norm.xy * u.refraction * density * 0.4, vec2f(0.0), vec2f(1.0));
    let rbg  = mix(wc, textureSample(uBg, samp, mix(i.uv, ruv, 1.0 - obs)).rgb, cov);
    color    = mix(rbg, wc + spec * gc, inkD);
    color    = mix(color, bg * 0.5, obs * 0.15);
  } else if (u.algorithm == 3) {
    let vel    = textureSample(uVel, samp, i.uv).xy;
    let velMag = clamp(length(vel) * 20.0, 0.0, 1.0);
    let wuv    = clamp(i.uv + vel * u.warpStrength, vec2f(0.0), vec2f(1.0));
    let wbg    = mix(wc, textureSample(uBg, samp, wuv).rgb, cov);
    color      = mix(bg, wbg, velMag * (1.0 - obs));
    color     += spec * gc * velMag * 1.5;
    color     += wc * density * 0.3;
    color      = mix(color, bg * 0.5, obs * 0.2);
  } else if (u.algorithm == 4) {
    let ruv   = clamp(i.uv + norm.xy * u.refraction * density * 6.0, vec2f(0.0), vec2f(1.0));
    let rbg   = mix(wc, textureSample(uBg, samp, mix(i.uv, ruv, 1.0 - obs)).rgb, cov);
    let fres  = pow(clamp(1.0 - dot(norm, vec3f(0.0, 0.0, 1.0)), 0.0, 1.0), 3.0) * density;
    color     = rbg;
    color    += fres * gc * 2.0;
    color    += spec * gc * density * 2.0;
    color     = mix(color, bg * 0.5, obs * 0.2);
  } else {
    let ruv = i.uv + norm.xy * u.refraction * density;
    let rbg = mix(wc, textureSample(uBg, samp, mix(i.uv, ruv, 1.0 - obs)).rgb, cov);
    color   = mix(rbg, wc, min(density * 1.5, 0.8));
    color  += spec * gc;
    color   = mix(color, bg * 0.5, obs * 0.2);
  }

  let alpha = clamp(max(density * 1.5, cov), 0.0, 1.0);
  return vec4f(color * alpha, alpha);
}
`
), It = {
  arrayStride: 8,
  attributes: [{ shaderLocation: 0, offset: 0, format: "float32x2" }]
}, at = new Float32Array([
  -1,
  -1,
  -1,
  1,
  1,
  -1,
  1,
  -1,
  -1,
  1,
  1,
  1
]);
function Wt(i) {
  const e = i.createBuffer({ size: at.byteLength, usage: GPUBufferUsage.VERTEX | GPUBufferUsage.COPY_DST });
  return i.queue.writeBuffer(e, 0, at), e;
}
function Ne(i, e, r, s) {
  const o = i.createTexture({
    size: [r, s],
    format: e,
    usage: GPUTextureUsage.TEXTURE_BINDING | GPUTextureUsage.RENDER_ATTACHMENT | GPUTextureUsage.COPY_SRC
  });
  return { tex: o, view: o.createView(), width: r, height: s };
}
function et(i, e, r, s) {
  let o = Ne(i, e, r, s), a = Ne(i, e, r, s);
  return {
    get read() {
      return o;
    },
    get write() {
      return a;
    },
    swap() {
      [o, a] = [a, o];
    },
    dispose() {
      o.tex.destroy(), a.tex.destroy();
    }
  };
}
function se(i, e, r) {
  const s = i.createShaderModule({ code: e });
  return i.createRenderPipeline({
    layout: "auto",
    vertex: { module: s, entryPoint: "vs", buffers: [It] },
    fragment: { module: s, entryPoint: "fs", targets: [{ format: r }] },
    primitive: { topology: "triangle-list" }
  });
}
function $t(i, e) {
  const r = "rgba16float";
  return {
    advection: se(i, Ft, r),
    divergence: se(i, Lt, r),
    pressure: se(i, At, r),
    gradientSubtract: se(i, Xt, r),
    splat: se(i, zt, r),
    curl: se(i, Mt, r),
    vorticity: se(i, Gt, r),
    display: se(i, kt, e)
  };
}
function Nt(i) {
  return i.createSampler({ magFilter: "linear", minFilter: "linear", addressModeU: "clamp-to-edge", addressModeV: "clamp-to-edge" });
}
function W(i, e) {
  return i.createBuffer({ size: e, usage: GPUBufferUsage.UNIFORM | GPUBufferUsage.COPY_DST });
}
function ut(i, e, r, s, o, a) {
  const n = new Float32Array([r, s, o, a]);
  i.queue.writeBuffer(e, 0, n);
}
function Ie(i, e, r, s) {
  const o = new Float32Array([r, s, 0, 0]);
  i.queue.writeBuffer(e, 0, o);
}
function qt(i, e, r, s, o, a) {
  const n = new Float32Array([r, s, o, a]);
  i.queue.writeBuffer(e, 0, n);
}
function We(i, e, r, s, o, a, n, x, m, l, p) {
  const c = new Float32Array(12);
  c[0] = r, c[1] = s, c[2] = o, c[3] = a, c[4] = n, c[5] = x, c[6] = m, c[7] = 0, c[8] = l, c[9] = p, c[10] = 0, c[11] = 0, i.queue.writeBuffer(e, 0, c);
}
function Ht(i, e, r, s, o, a, n, x, m, l, p) {
  const c = new Float32Array(16), f = new Int32Array(c.buffer);
  c[0] = r, c[1] = s, c[2] = o, c[3] = a, c[4] = n[0], c[5] = n[1], c[6] = n[2], c[7] = 0, c[8] = x[0], c[9] = x[1], c[10] = x[2], c[11] = 0, c[12] = m, c[13] = l, f[14] = p, i.queue.writeBuffer(e, 0, c);
}
function z(i, e, r, s, o) {
  const a = i.beginRenderPass({
    colorAttachments: [{
      view: o,
      clearValue: [0, 0, 0, 0],
      loadOp: "clear",
      storeOp: "store"
    }]
  });
  a.setPipeline(e), a.setBindGroup(0, r), a.setVertexBuffer(0, s), a.draw(6), a.end();
}
function Yt(i, e, r, s, o) {
  const a = i.beginRenderPass({
    colorAttachments: [{
      view: o,
      clearValue: [0, 0, 0, 0],
      loadOp: "clear",
      storeOp: "store"
    }]
  });
  a.setPipeline(e), a.setBindGroup(0, r), a.setVertexBuffer(0, s), a.draw(6), a.end();
}
function Ce(i, e, r, s, o = "cover") {
  let a;
  o === "cover" ? a = Math.max(r / i, s / e) : o === "contain" ? a = Math.min(r / i, s / e) : typeof o == "string" && o.endsWith("%") ? a = Math.min(r / i, s / e) * (parseFloat(o) / 100) : typeof o == "string" && o.endsWith("px") ? a = parseFloat(o) / Math.max(i, e) : typeof o == "number" ? a = o : a = Math.max(r / i, s / e);
  const n = i * a, x = e * a;
  return { x: (r - n) / 2, y: (s - x) / 2, drawW: n, drawH: x };
}
function Qt(i, e, r, s, o = null, a = "cover") {
  const { text: n, fontSize: x, color: m, fontFamily: l = "sans-serif", fontWeight: p = 900 } = s, c = new OffscreenCanvas(e, r), f = c.getContext("2d");
  ((D) => {
    if (o) {
      f.clearRect(0, 0, e, r), f.fillStyle = "black", f.fillRect(0, 0, e, r);
      const { x: g, y: _, drawW: F, drawH: I } = Ce(
        o.width,
        o.height,
        e,
        r,
        a
      );
      f.drawImage(o, g, _, F, I);
    } else
      f.fillStyle = "black", f.fillRect(0, 0, e, r);
    f.fillStyle = D, f.font = `${p} ${x}px ${l}`, f.textAlign = "center", f.textBaseline = "middle", f.fillText(n, e / 2, r / 2);
  })(m);
  const b = Fe(i, c);
  f.fillStyle = "black", f.fillRect(0, 0, e, r), f.fillStyle = "white", f.font = `${p} ${x}px ${l}`, f.textAlign = "center", f.textBaseline = "middle", f.fillText(n, e / 2, r / 2);
  const y = Fe(i, c);
  return { backgroundTex: b, obstacleTex: y, coverageTex: y };
}
function jt(i, e, r, s, o = 0, a = "cover", n = null, x = "cover") {
  const m = new OffscreenCanvas(r, s), l = m.getContext("2d"), { x: p, y: c, drawW: f, drawH: d } = Ce(e.width, e.height, r, s, a);
  if (l.clearRect(0, 0, r, s), l.fillStyle = "black", l.fillRect(0, 0, r, s), n) {
    const {
      x: g,
      y: _,
      drawW: F,
      drawH: I
    } = Ce(n.width, n.height, r, s, x);
    l.filter = `brightness(${o}) blur(8px)`, l.drawImage(n, g, _, F, I), l.filter = "none";
  }
  l.drawImage(e, p, c, f, d);
  const b = Fe(i, m);
  l.clearRect(0, 0, r, s), l.fillStyle = "black", l.fillRect(0, 0, r, s), l.filter = `brightness(${o}) blur(8px)`, l.drawImage(e, p, c, f, d), l.filter = "none";
  const y = Fe(i, m);
  l.clearRect(0, 0, r, s), l.fillStyle = "black", l.fillRect(0, 0, r, s), l.fillStyle = "white", l.fillRect(
    Math.max(0, p),
    Math.max(0, c),
    Math.min(f, r - Math.max(0, p)),
    Math.min(d, s - Math.max(0, c))
  );
  const D = Fe(i, m);
  return { backgroundTex: b, obstacleTex: y, coverageTex: D };
}
function Fe(i, e) {
  const r = i.createTexture();
  return i.bindTexture(i.TEXTURE_2D, r), i.pixelStorei(i.UNPACK_FLIP_Y_WEBGL, !0), i.texImage2D(i.TEXTURE_2D, 0, i.RGBA, i.RGBA, i.UNSIGNED_BYTE, e), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_MIN_FILTER, i.LINEAR), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_MAG_FILTER, i.LINEAR), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_WRAP_S, i.CLAMP_TO_EDGE), i.texParameteri(i.TEXTURE_2D, i.TEXTURE_WRAP_T, i.CLAMP_TO_EDGE), r;
}
function Kt(i, e, r, s, o = null, a = "cover") {
  const { text: n, fontSize: x, color: m, fontFamily: l = "sans-serif", fontWeight: p = 900 } = s, c = new OffscreenCanvas(e, r), f = c.getContext("2d");
  ((D) => {
    if (o) {
      f.clearRect(0, 0, e, r), f.fillStyle = "black", f.fillRect(0, 0, e, r);
      const { x: g, y: _, drawW: F, drawH: I } = Ce(
        o.width,
        o.height,
        e,
        r,
        a
      );
      f.drawImage(o, g, _, F, I);
    } else
      f.fillStyle = "black", f.fillRect(0, 0, e, r);
    f.fillStyle = D, f.font = `${p} ${x}px ${l}`, f.textAlign = "center", f.textBaseline = "middle", f.fillText(n, e / 2, r / 2);
  })(m);
  const b = Le(i, c, e, r);
  f.fillStyle = "black", f.fillRect(0, 0, e, r), f.fillStyle = "white", f.font = `${p} ${x}px ${l}`, f.textAlign = "center", f.textBaseline = "middle", f.fillText(n, e / 2, r / 2);
  const y = Le(i, c, e, r);
  return {
    backgroundTex: b,
    backgroundView: b.createView(),
    obstacleTex: y,
    obstacleView: y.createView(),
    coverageTex: y,
    coverageView: y.createView(),
    sharedCoverage: !0
  };
}
function Jt(i, e, r, s, o = 0, a = "cover", n = null, x = "cover") {
  const m = new OffscreenCanvas(r, s), l = m.getContext("2d"), { x: p, y: c, drawW: f, drawH: d } = Ce(e.width, e.height, r, s, a);
  if (l.clearRect(0, 0, r, s), l.fillStyle = "black", l.fillRect(0, 0, r, s), n) {
    const { x: g, y: _, drawW: F, drawH: I } = Ce(
      n.width,
      n.height,
      r,
      s,
      x
    );
    l.filter = `brightness(${o}) blur(8px)`, l.drawImage(n, g, _, F, I), l.filter = "none";
  }
  l.drawImage(e, p, c, f, d);
  const b = Le(i, m, r, s);
  l.clearRect(0, 0, r, s), l.fillStyle = "black", l.fillRect(0, 0, r, s), l.filter = `brightness(${o}) blur(8px)`, l.drawImage(e, p, c, f, d), l.filter = "none";
  const y = Le(i, m, r, s);
  l.clearRect(0, 0, r, s), l.fillStyle = "black", l.fillRect(0, 0, r, s), l.fillStyle = "white", l.fillRect(Math.max(0, p), Math.max(0, c), Math.min(f, r - Math.max(0, p)), Math.min(d, s - Math.max(0, c)));
  const D = Le(i, m, r, s);
  return {
    backgroundTex: b,
    backgroundView: b.createView(),
    obstacleTex: y,
    obstacleView: y.createView(),
    coverageTex: D,
    coverageView: D.createView(),
    sharedCoverage: !1
  };
}
function Le(i, e, r, s) {
  const o = i.createTexture({
    size: [r, s],
    format: "rgba8unorm",
    usage: GPUTextureUsage.TEXTURE_BINDING | GPUTextureUsage.COPY_DST | GPUTextureUsage.RENDER_ATTACHMENT
  });
  return i.queue.copyExternalImageToTexture(
    { source: e },
    { texture: o },
    [r, s]
  ), o;
}
async function Zt(i) {
  const e = await fetch(i);
  if (!e.ok)
    throw new Error(`Failed to fetch image: ${i} (${e.status})`);
  const r = await e.blob();
  return createImageBitmap(r);
}
const nt = typeof requestAnimationFrame < "u" ? requestAnimationFrame.bind(globalThis) : (i) => setTimeout(i, 1e3 / 60), er = typeof cancelAnimationFrame < "u" ? cancelAnimationFrame.bind(globalThis) : clearTimeout, Ve = 0.016, lt = { standard: 0, glass: 1, ink: 2, aurora: 3, ripple: 4 };
var ce, M, Ae, fe, Ee, B, R, N, q, H, K, L, Y, P, Ue, ve, De, V, T, oe, Q, ae, X, xe, pe, me, de, he, J, Z, ge, be, Te, w, U, G, k, O, _e, ee, A, ue, te, h, C, j, ye, Xe, qe, ft, Se, Pe, ze, rt, ne, Re, we, Oe, Me, it, Ge, st, He, vt, Ye, xt, Qe, pt, je, mt, Ke, dt;
const ot = class ot {
  // ── Constructor ─────────────────────────────────────────────────────────────
  constructor(e, r = {}, s = {}, o) {
    // ---------------------------------------------------------------------------
    // Private — GPU initialisation
    // ---------------------------------------------------------------------------
    v(this, qe);
    // ---------------------------------------------------------------------------
    // Private — shared helpers
    // ---------------------------------------------------------------------------
    v(this, Se);
    v(this, ze);
    v(this, ne);
    v(this, we);
    v(this, Me);
    v(this, Ge);
    // ---------------------------------------------------------------------------
    // Private — frame dispatch
    // ---------------------------------------------------------------------------
    v(this, He);
    // ---------------------------------------------------------------------------
    // Private — WebGPU simulation step
    // ---------------------------------------------------------------------------
    v(this, Ye);
    // ---------------------------------------------------------------------------
    // Private — WebGPU direct splat
    // ---------------------------------------------------------------------------
    v(this, Qe);
    // ---------------------------------------------------------------------------
    // Private — WebGL splat
    // ---------------------------------------------------------------------------
    v(this, je);
    // ---------------------------------------------------------------------------
    // Private — WebGL simulation step (unchanged from original)
    // ---------------------------------------------------------------------------
    v(this, Ke);
    v(this, ce, void 0);
    // ── WebGL path ──────────────────────────────────────────────────────────────
    v(this, M, null);
    v(this, Ae, null);
    v(this, fe, null);
    v(this, Ee, null);
    v(this, B, null);
    v(this, R, null);
    v(this, N, null);
    v(this, q, null);
    v(this, H, null);
    v(this, K, null);
    v(this, L, null);
    v(this, Y, null);
    // ── WebGPU path ─────────────────────────────────────────────────────────────
    v(this, P, null);
    v(this, Ue, null);
    v(this, ve, null);
    v(this, De, null);
    v(this, V, null);
    v(this, T, null);
    v(this, oe, null);
    v(this, Q, null);
    v(this, ae, null);
    v(this, X, null);
    // Pre-allocated uniform buffers (sizes: see gpu-utils writeXxx docs)
    // Velocity/density advection use separate buffers — writeBuffer is a queue op;
    // a second write to the same buffer before queue.submit() aliases both passes.
    v(this, xe, null);
    // 16 bytes — velocity advection
    v(this, pe, null);
    // 16 bytes — density advection
    v(this, me, null);
    // 16 bytes
    v(this, de, null);
    // 16 bytes
    v(this, he, null);
    // 16 bytes
    v(this, J, null);
    // 48 bytes — velocity splat
    v(this, Z, null);
    // 48 bytes — density splat
    v(this, ge, null);
    // 16 bytes
    v(this, be, null);
    // 16 bytes
    v(this, Te, null);
    // 64 bytes
    // ── Shared state ────────────────────────────────────────────────────────────
    v(this, w, 0);
    v(this, U, 0);
    v(this, G, 0);
    v(this, k, 0);
    v(this, O, 1);
    v(this, _e, 1);
    v(this, ee, 0.5);
    v(this, A, null);
    v(this, ue, "cover");
    v(this, te, void 0);
    v(this, h, { x: 0, y: 0, dx: 0, dy: 0, targetX: 0, targetY: 0, moved: !1 });
    v(this, C, null);
    v(this, j, null);
    v(this, ye, !1);
    v(this, Xe, !1);
    if (u(this, ce, e), u(this, _e, Math.max(0.1, Math.min(1, s.dpr ?? 1))), u(this, ee, Math.max(0.1, Math.min(1, s.sim ?? 0.5))), u(this, te, yt(r)), o)
      u(this, P, o), S(this, qe, ft).call(this, o);
    else {
      const { gl: a, ext: n } = Bt(e);
      u(this, M, a), u(this, Ae, n), u(this, fe, Pt(a)), u(this, Ee, Ot(a)), a.clearColor(0, 0, 0, 0);
    }
  }
  /**
   * WebGPU-first factory. Tries WebGPU, falls back to WebGL2 → WebGL1.
   * This is the recommended entry point when WebGPU support is desired.
   */
  static async create(e, r = {}, s = {}, o = !0) {
    const a = o ? await Vt(e) : null;
    return new ot(e, r, s, a ?? void 0);
  }
  // ---------------------------------------------------------------------------
  // Public API
  // ---------------------------------------------------------------------------
  setTextSource(e) {
    u(this, C, { type: "text", opts: e }), S(this, Se, Pe).call(this), S(this, ne, Re).call(this), S(this, we, Oe).call(this);
  }
  async setImageSource(e, r = 0, s = "cover") {
    const o = await Zt(e);
    if (t(this, Xe)) {
      o.close();
      return;
    }
    u(this, C, { type: "image", bitmap: o, effect: r, size: s }), S(this, Se, Pe).call(this), S(this, ne, Re).call(this), S(this, we, Oe).call(this);
  }
  setImageBitmap(e, r = 0, s = "cover") {
    u(this, C, { type: "image", bitmap: e, effect: r, size: s }), S(this, Se, Pe).call(this), S(this, ne, Re).call(this), S(this, we, Oe).call(this);
  }
  setBackground(e, r = "cover") {
    t(this, A) && t(this, A) !== e && t(this, A).close(), u(this, A, e), u(this, ue, r ?? "cover"), t(this, C) && t(this, w) > 0 && t(this, U) > 0 && S(this, ne, Re).call(this);
  }
  handleMove(e, r, s = 1) {
    t(this, h).moved = !0, t(this, h).dx = (e - t(this, h).targetX) * s, t(this, h).dy = (r - t(this, h).targetY) * s, t(this, h).targetX = e, t(this, h).targetY = r;
  }
  /**
   * Immediately applies one fluid splat at (x, y) with explicit velocity (vx, vy).
   * Safe to call multiple times per frame — each call writes directly to the FBOs.
   * Designed for programmatic use cases (e.g. particle systems, attractor paths)
   * where you want N independent injection points per frame without flooding the
   * mouse-state machine or the worker message queue.
   */
  splat(e, r, s, o, a = 1) {
    !t(this, ye) || t(this, w) === 0 || (t(this, P) ? S(this, Qe, pt).call(this, e, r, s, o, a) : S(this, je, mt).call(this, e, r, s, o, a));
  }
  updateQuality(e) {
    e.dpr !== void 0 && u(this, _e, Math.max(0.1, Math.min(1, e.dpr))), e.sim !== void 0 && u(this, ee, Math.max(0.1, Math.min(1, e.sim)));
  }
  resize(e, r, s) {
    if (s !== void 0 ? u(this, O, s) : typeof window < "u" && window.devicePixelRatio && u(this, O, window.devicePixelRatio), e !== void 0 && e > 0) {
      if (r === void 0 || r <= 0)
        return;
      u(this, w, t(this, ce).width = e), u(this, U, t(this, ce).height = r), u(this, G, Math.max(1, Math.round(e * t(this, ee)))), u(this, k, Math.max(1, Math.round(r * t(this, ee)))), S(this, ze, rt).call(this);
    } else
      S(this, Se, Pe).call(this);
    t(this, C) && S(this, ne, Re).call(this), S(this, we, Oe).call(this);
  }
  updateConfig(e) {
    Object.assign(t(this, te), e);
  }
  destroy() {
    var e, r, s, o, a, n, x, m, l, p, c;
    if (u(this, Xe, !0), this.stop(), S(this, Me, it).call(this), S(this, Ge, st).call(this), t(this, A) && (t(this, A).close(), u(this, A, null)), t(this, P))
      (e = t(this, xe)) == null || e.destroy(), (r = t(this, pe)) == null || r.destroy(), (s = t(this, me)) == null || s.destroy(), (o = t(this, de)) == null || o.destroy(), (a = t(this, he)) == null || a.destroy(), (n = t(this, J)) == null || n.destroy(), (x = t(this, Z)) == null || x.destroy(), (m = t(this, ge)) == null || m.destroy(), (l = t(this, be)) == null || l.destroy(), (p = t(this, Te)) == null || p.destroy(), (c = t(this, ve)) == null || c.destroy(), t(this, P).device.destroy();
    else {
      const f = t(this, M);
      for (const b of Object.values(t(this, fe)))
        b.dispose();
      const d = f.getExtension("WEBGL_lose_context");
      d == null || d.loseContext();
    }
  }
  // ---------------------------------------------------------------------------
  // Loop control
  // ---------------------------------------------------------------------------
  start() {
    if (t(this, j) !== null)
      return;
    const e = () => {
      S(this, He, vt).call(this), u(this, j, nt(e));
    };
    u(this, j, nt(e));
  }
  stop() {
    t(this, j) !== null && (er(t(this, j)), u(this, j, null));
  }
  get isRunning() {
    return t(this, j) !== null;
  }
};
ce = new WeakMap(), M = new WeakMap(), Ae = new WeakMap(), fe = new WeakMap(), Ee = new WeakMap(), B = new WeakMap(), R = new WeakMap(), N = new WeakMap(), q = new WeakMap(), H = new WeakMap(), K = new WeakMap(), L = new WeakMap(), Y = new WeakMap(), P = new WeakMap(), Ue = new WeakMap(), ve = new WeakMap(), De = new WeakMap(), V = new WeakMap(), T = new WeakMap(), oe = new WeakMap(), Q = new WeakMap(), ae = new WeakMap(), X = new WeakMap(), xe = new WeakMap(), pe = new WeakMap(), me = new WeakMap(), de = new WeakMap(), he = new WeakMap(), J = new WeakMap(), Z = new WeakMap(), ge = new WeakMap(), be = new WeakMap(), Te = new WeakMap(), w = new WeakMap(), U = new WeakMap(), G = new WeakMap(), k = new WeakMap(), O = new WeakMap(), _e = new WeakMap(), ee = new WeakMap(), A = new WeakMap(), ue = new WeakMap(), te = new WeakMap(), h = new WeakMap(), C = new WeakMap(), j = new WeakMap(), ye = new WeakMap(), Xe = new WeakMap(), qe = new WeakSet(), ft = function(e) {
  const { device: r, format: s } = e;
  u(this, Ue, $t(r, s)), u(this, ve, Wt(r)), u(this, De, Nt(r)), u(this, xe, W(r, 16)), u(this, pe, W(r, 16)), u(this, me, W(r, 16)), u(this, de, W(r, 16)), u(this, he, W(r, 16)), u(this, J, W(r, 48)), u(this, Z, W(r, 48)), u(this, ge, W(r, 16)), u(this, be, W(r, 16)), u(this, Te, W(r, 64));
}, Se = new WeakSet(), Pe = function() {
  const e = t(this, ce);
  "clientWidth" in e && e.clientWidth > 0 ? (u(this, O, (typeof window < "u" && window.devicePixelRatio || 1) * t(this, _e)), u(this, w, e.width = Math.round(e.clientWidth * t(this, O))), u(this, U, e.height = Math.round(e.clientHeight * t(this, O)))) : (u(this, w, e.width), u(this, U, e.height)), !(t(this, w) === 0 || t(this, U) === 0) && (u(this, G, Math.max(1, Math.round(t(this, w) * t(this, ee)))), u(this, k, Math.max(1, Math.round(t(this, U) * t(this, ee)))), S(this, ze, rt).call(this));
}, ze = new WeakSet(), rt = function() {
  if (S(this, Me, it).call(this), t(this, P)) {
    const { device: e } = t(this, P), r = "rgba16float", s = t(this, G), o = t(this, k);
    u(this, V, et(e, r, s, o)), u(this, T, et(e, r, s, o)), u(this, Q, et(e, r, s, o)), u(this, oe, Ne(e, r, s, o)), u(this, ae, Ne(e, r, s, o));
  } else {
    const e = t(this, M), r = t(this, Ae), s = t(this, G), o = t(this, k);
    u(this, B, Ze(e, r, s, o)), u(this, R, Ze(e, r, s, o)), u(this, q, Ze(e, r, s, o)), u(this, N, $e(e, r, s, o)), u(this, H, $e(e, r, s, o));
  }
}, ne = new WeakSet(), Re = function() {
  if (!(!t(this, C) || t(this, w) === 0 || t(this, U) === 0)) {
    if (S(this, Ge, st).call(this), t(this, P)) {
      const { device: e } = t(this, P);
      t(this, C).type === "text" ? u(this, X, Kt(
        e,
        t(this, w),
        t(this, U),
        t(this, C).opts,
        t(this, A),
        t(this, ue)
      )) : u(this, X, Jt(
        e,
        t(this, C).bitmap,
        t(this, w),
        t(this, U),
        t(this, C).effect,
        t(this, C).size,
        t(this, A),
        t(this, ue)
      ));
    } else {
      const e = t(this, M);
      if (t(this, C).type === "text") {
        const { backgroundTex: r, obstacleTex: s, coverageTex: o } = Qt(
          e,
          t(this, w),
          t(this, U),
          t(this, C).opts,
          t(this, A),
          t(this, ue)
        );
        u(this, K, r), u(this, L, s), u(this, Y, o);
      } else {
        const { backgroundTex: r, obstacleTex: s, coverageTex: o } = jt(
          e,
          t(this, C).bitmap,
          t(this, w),
          t(this, U),
          t(this, C).effect,
          t(this, C).size,
          t(this, A),
          t(this, ue)
        );
        u(this, K, r), u(this, L, s), u(this, Y, o);
      }
    }
    u(this, ye, !0);
  }
}, we = new WeakSet(), Oe = function() {
  t(this, ye) && !this.isRunning && this.start();
}, Me = new WeakSet(), it = function() {
  var e, r, s, o, a, n, x, m;
  if (t(this, P))
    (e = t(this, V)) == null || e.dispose(), (r = t(this, T)) == null || r.dispose(), (s = t(this, Q)) == null || s.dispose(), (o = t(this, oe)) == null || o.tex.destroy(), (a = t(this, ae)) == null || a.tex.destroy(), u(this, V, u(this, T, u(this, Q, null))), u(this, oe, u(this, ae, null));
  else {
    const l = t(this, M);
    (n = t(this, B)) == null || n.dispose(), (x = t(this, R)) == null || x.dispose(), (m = t(this, q)) == null || m.dispose(), t(this, N) && (l.deleteTexture(t(this, N).tex), l.deleteFramebuffer(t(this, N).fbo)), t(this, H) && (l.deleteTexture(t(this, H).tex), l.deleteFramebuffer(t(this, H).fbo)), u(this, B, u(this, R, u(this, q, u(this, N, u(this, H, null)))));
  }
}, Ge = new WeakSet(), st = function() {
  if (t(this, P))
    t(this, X) && (t(this, X).backgroundTex.destroy(), t(this, X).obstacleTex.destroy(), t(this, X).sharedCoverage || t(this, X).coverageTex.destroy(), u(this, X, null));
  else {
    const e = t(this, M);
    t(this, K) && e.deleteTexture(t(this, K)), t(this, L) && e.deleteTexture(t(this, L)), t(this, Y) && t(this, Y) !== t(this, L) && e.deleteTexture(t(this, Y)), u(this, K, u(this, L, u(this, Y, null)));
  }
}, He = new WeakSet(), vt = function() {
  !t(this, ye) || t(this, w) === 0 || (t(this, P) ? S(this, Ye, xt).call(this) : S(this, Ke, dt).call(this));
}, Ye = new WeakSet(), xt = function() {
  const e = t(this, P), r = e.device, s = t(this, Ue), o = t(this, ve), a = t(this, De), n = t(this, te), x = t(this, X);
  if (!t(this, V) || !t(this, T))
    return;
  t(this, h).x += (t(this, h).targetX - t(this, h).x) * 0.15, t(this, h).y += (t(this, h).targetY - t(this, h).y) * 0.15;
  const m = t(this, G), l = t(this, k), p = t(this, w), c = t(this, U), f = 1 / m, d = 1 / l;
  ut(r, t(this, xe), f, d, Ve, n.velocityDissipation), Ie(r, t(this, me), f, d), Ie(r, t(this, de), f, d), Ie(r, t(this, he), f, d), Ie(r, t(this, ge), f, d), qt(r, t(this, be), f, d, n.curl, Ve), Ht(
    r,
    t(this, Te),
    1 / p,
    1 / c,
    n.refraction,
    n.specularExp,
    n.waterColor,
    n.glowColor,
    n.shine,
    n.warpStrength ?? 0.015,
    lt[n.algorithm] ?? 0
  );
  const b = r.createCommandEncoder(), y = (g, _) => r.createBindGroup({ layout: g.getBindGroupLayout(0), entries: _ }), D = { binding: 1, resource: a };
  {
    const g = y(s.advection, [
      { binding: 0, resource: { buffer: t(this, xe) } },
      D,
      { binding: 2, resource: t(this, T).read.view },
      { binding: 3, resource: t(this, T).read.view },
      { binding: 4, resource: x.obstacleView }
    ]);
    z(b, s.advection, g, o, t(this, T).write.view);
  }
  t(this, T).swap();
  {
    ut(r, t(this, pe), f, d, Ve, n.densityDissipation);
    const g = y(s.advection, [
      { binding: 0, resource: { buffer: t(this, pe) } },
      D,
      { binding: 2, resource: t(this, T).read.view },
      { binding: 3, resource: t(this, V).read.view },
      { binding: 4, resource: x.obstacleView }
    ]);
    z(b, s.advection, g, o, t(this, V).write.view);
  }
  t(this, V).swap();
  {
    const g = y(s.curl, [
      { binding: 0, resource: { buffer: t(this, ge) } },
      D,
      { binding: 2, resource: t(this, T).read.view }
    ]);
    z(b, s.curl, g, o, t(this, ae).view);
  }
  {
    const g = y(s.vorticity, [
      { binding: 0, resource: { buffer: t(this, be) } },
      D,
      { binding: 2, resource: t(this, T).read.view },
      { binding: 3, resource: t(this, ae).view }
    ]);
    z(b, s.vorticity, g, o, t(this, T).write.view);
  }
  if (t(this, T).swap(), t(this, h).moved) {
    const g = t(this, h).x * t(this, O) / p, _ = t(this, h).y * t(this, O) / c;
    We(
      r,
      t(this, J),
      f,
      d,
      p / c,
      n.splatRadius,
      t(this, h).dx * n.splatForce,
      t(this, h).dy * n.splatForce,
      0,
      g,
      _
    );
    {
      const F = y(s.splat, [
        { binding: 0, resource: { buffer: t(this, J) } },
        D,
        { binding: 2, resource: t(this, T).read.view }
      ]);
      z(b, s.splat, F, o, t(this, T).write.view);
    }
    t(this, T).swap(), We(
      r,
      t(this, Z),
      f,
      d,
      p / c,
      n.splatRadius,
      1,
      1,
      1,
      g,
      _
    );
    {
      const F = y(s.splat, [
        { binding: 0, resource: { buffer: t(this, Z) } },
        D,
        { binding: 2, resource: t(this, V).read.view }
      ]);
      z(b, s.splat, F, o, t(this, V).write.view);
    }
    t(this, V).swap(), t(this, h).moved = !1;
  }
  {
    const g = y(s.divergence, [
      { binding: 0, resource: { buffer: t(this, me) } },
      D,
      { binding: 2, resource: t(this, T).read.view },
      { binding: 3, resource: x.obstacleView }
    ]);
    z(b, s.divergence, g, o, t(this, oe).view);
  }
  for (let g = 0; g < n.pressureIterations; g++) {
    const _ = y(s.pressure, [
      { binding: 0, resource: { buffer: t(this, de) } },
      D,
      { binding: 2, resource: t(this, Q).read.view },
      { binding: 3, resource: t(this, oe).view },
      { binding: 4, resource: x.obstacleView }
    ]);
    z(b, s.pressure, _, o, t(this, Q).write.view), t(this, Q).swap();
  }
  {
    const g = y(s.gradientSubtract, [
      { binding: 0, resource: { buffer: t(this, he) } },
      D,
      { binding: 2, resource: t(this, Q).read.view },
      { binding: 3, resource: t(this, T).read.view },
      { binding: 4, resource: x.obstacleView }
    ]);
    z(b, s.gradientSubtract, g, o, t(this, T).write.view);
  }
  t(this, T).swap();
  {
    const g = e.context.getCurrentTexture().createView(), _ = y(s.display, [
      { binding: 0, resource: { buffer: t(this, Te) } },
      D,
      { binding: 2, resource: t(this, V).read.view },
      { binding: 3, resource: x.obstacleView },
      { binding: 4, resource: x.backgroundView },
      { binding: 5, resource: x.coverageView },
      { binding: 6, resource: t(this, T).read.view }
    ]);
    Yt(b, s.display, _, o, g);
  }
  r.queue.submit([b.finish()]);
}, Qe = new WeakSet(), pt = function(e, r, s, o, a) {
  const x = t(this, P).device, m = t(this, Ue).splat, l = t(this, ve), p = t(this, De), c = t(this, te), f = t(this, G), d = t(this, k), b = 1 / f, y = 1 / d, D = x.createCommandEncoder(), g = { binding: 1, resource: p }, _ = (Be) => x.createBindGroup({ layout: m.getBindGroupLayout(0), entries: Be }), F = e * t(this, O) / t(this, w), I = r * t(this, O) / t(this, U);
  We(
    x,
    t(this, J),
    b,
    y,
    t(this, w) / t(this, U),
    c.splatRadius,
    s * c.splatForce * a,
    o * c.splatForce * a,
    0,
    F,
    I
  );
  {
    const Be = _([
      { binding: 0, resource: { buffer: t(this, J) } },
      g,
      { binding: 2, resource: t(this, T).read.view }
    ]);
    z(D, m, Be, l, t(this, T).write.view);
  }
  t(this, T).swap(), We(
    x,
    t(this, Z),
    b,
    y,
    t(this, w) / t(this, U),
    c.splatRadius,
    a,
    a,
    a,
    F,
    I
  );
  {
    const Be = _([
      { binding: 0, resource: { buffer: t(this, Z) } },
      g,
      { binding: 2, resource: t(this, V).read.view }
    ]);
    z(D, m, Be, l, t(this, V).write.view);
  }
  t(this, V).swap(), x.queue.submit([D.finish()]);
}, je = new WeakSet(), mt = function(e, r, s, o, a) {
  const n = t(this, M), x = t(this, te), m = t(this, fe).splat, l = t(this, Ee);
  n.viewport(0, 0, t(this, G), t(this, k)), m.bind(), n.uniform1f(m.uniforms.aspectRatio, t(this, w) / t(this, U)), n.uniform2f(m.uniforms.point, e * t(this, O) / t(this, w), 1 - r * t(this, O) / t(this, U)), n.uniform1f(m.uniforms.radius, x.splatRadius), n.uniform1i(m.uniforms.uTarget, 0), n.activeTexture(n.TEXTURE0), n.bindTexture(n.TEXTURE_2D, t(this, R).read.tex), n.uniform3f(m.uniforms.color, s * x.splatForce * a, -o * x.splatForce * a, 0), l(t(this, R).write.fbo), t(this, R).swap(), n.activeTexture(n.TEXTURE0), n.bindTexture(n.TEXTURE_2D, t(this, B).read.tex), n.uniform3f(m.uniforms.color, a, a, a), l(t(this, B).write.fbo), t(this, B).swap();
}, Ke = new WeakSet(), dt = function() {
  if (!t(this, B) || !t(this, R))
    return;
  const e = t(this, M), r = t(this, te), { advection: s, divergence: o, pressure: a, gradientSubtract: n, splat: x, curl: m, vorticity: l, display: p } = t(this, fe);
  t(this, h).x += (t(this, h).targetX - t(this, h).x) * 0.15, t(this, h).y += (t(this, h).targetY - t(this, h).y) * 0.15;
  const c = t(this, G), f = t(this, k), d = t(this, Ee);
  e.viewport(0, 0, c, f), s.bind(), e.uniform2f(s.uniforms.texelSize, 1 / c, 1 / f), e.uniform1f(s.uniforms.dt, Ve), e.uniform1i(s.uniforms.uObstacle, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, L)), e.uniform1f(s.uniforms.dissipation, r.velocityDissipation), e.uniform1i(s.uniforms.uVelocity, 1), e.activeTexture(e.TEXTURE1), e.bindTexture(e.TEXTURE_2D, t(this, R).read.tex), e.uniform1i(s.uniforms.uSource, 1), d(t(this, R).write.fbo), t(this, R).swap(), e.uniform1f(s.uniforms.dissipation, r.densityDissipation), e.uniform1i(s.uniforms.uSource, 2), e.activeTexture(e.TEXTURE2), e.bindTexture(e.TEXTURE_2D, t(this, B).read.tex), d(t(this, B).write.fbo), t(this, B).swap(), m.bind(), e.uniform2f(m.uniforms.texelSize, 1 / c, 1 / f), e.uniform1i(m.uniforms.uVelocity, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, R).read.tex), d(t(this, H).fbo), l.bind(), e.uniform2f(l.uniforms.texelSize, 1 / c, 1 / f), e.uniform1f(l.uniforms.curl, r.curl), e.uniform1f(l.uniforms.dt, Ve), e.uniform1i(l.uniforms.uVelocity, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, R).read.tex), e.uniform1i(l.uniforms.uCurl, 1), e.activeTexture(e.TEXTURE1), e.bindTexture(e.TEXTURE_2D, t(this, H).tex), d(t(this, R).write.fbo), t(this, R).swap(), t(this, h).moved && (x.bind(), e.uniform1f(x.uniforms.aspectRatio, t(this, w) / t(this, U)), e.uniform2f(x.uniforms.point, t(this, h).x * t(this, O) / t(this, w), 1 - t(this, h).y * t(this, O) / t(this, U)), e.uniform1f(x.uniforms.radius, r.splatRadius), e.uniform1i(x.uniforms.uTarget, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, R).read.tex), e.uniform3f(x.uniforms.color, t(this, h).dx * r.splatForce, -t(this, h).dy * r.splatForce, 0), d(t(this, R).write.fbo), t(this, R).swap(), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, B).read.tex), e.uniform3f(x.uniforms.color, 1, 1, 1), d(t(this, B).write.fbo), t(this, B).swap(), t(this, h).moved = !1), o.bind(), e.uniform2f(o.uniforms.texelSize, 1 / c, 1 / f), e.uniform1i(o.uniforms.uVelocity, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, R).read.tex), e.uniform1i(o.uniforms.uObstacle, 1), e.activeTexture(e.TEXTURE1), e.bindTexture(e.TEXTURE_2D, t(this, L)), d(t(this, N).fbo), a.bind(), e.uniform2f(a.uniforms.texelSize, 1 / c, 1 / f), e.uniform1i(a.uniforms.uDivergence, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, N).tex), e.uniform1i(a.uniforms.uObstacle, 1), e.activeTexture(e.TEXTURE1), e.bindTexture(e.TEXTURE_2D, t(this, L));
  for (let b = 0; b < r.pressureIterations; b++)
    e.uniform1i(a.uniforms.uPressure, 2), e.activeTexture(e.TEXTURE2), e.bindTexture(e.TEXTURE_2D, t(this, q).read.tex), d(t(this, q).write.fbo), t(this, q).swap();
  n.bind(), e.uniform2f(n.uniforms.texelSize, 1 / c, 1 / f), e.uniform1i(n.uniforms.uPressure, 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, q).read.tex), e.uniform1i(n.uniforms.uVelocity, 1), e.activeTexture(e.TEXTURE1), e.bindTexture(e.TEXTURE_2D, t(this, R).read.tex), e.uniform1i(n.uniforms.uObstacle, 2), e.activeTexture(e.TEXTURE2), e.bindTexture(e.TEXTURE_2D, t(this, L)), d(t(this, R).write.fbo), t(this, R).swap(), e.viewport(0, 0, t(this, w), t(this, U)), e.bindFramebuffer(e.FRAMEBUFFER, null), e.clear(e.COLOR_BUFFER_BIT), p.bind(), e.uniform2f(p.uniforms.texelSize, 1 / t(this, w), 1 / t(this, U)), e.uniform3fv(p.uniforms.uWaterColor, r.waterColor), e.uniform3fv(p.uniforms.uGlowColor, r.glowColor), e.uniform1f(p.uniforms.uRefraction, r.refraction), e.uniform1f(p.uniforms.uSpecularExp, r.specularExp), e.uniform1f(p.uniforms.uShine, r.shine), e.uniform1f(p.uniforms.uWarpStrength, r.warpStrength ?? 0.015), e.uniform1i(p.uniforms.uAlgorithm, lt[r.algorithm] ?? 0), e.activeTexture(e.TEXTURE0), e.bindTexture(e.TEXTURE_2D, t(this, B).read.tex), e.activeTexture(e.TEXTURE1), e.bindTexture(e.TEXTURE_2D, t(this, L)), e.activeTexture(e.TEXTURE2), e.bindTexture(e.TEXTURE_2D, t(this, K)), e.activeTexture(e.TEXTURE3), e.bindTexture(e.TEXTURE_2D, t(this, Y)), e.activeTexture(e.TEXTURE4), e.bindTexture(e.TEXTURE_2D, t(this, R).read.tex), e.uniform1i(p.uniforms.uTexture, 0), e.uniform1i(p.uniforms.uObstacle, 1), e.uniform1i(p.uniforms.uBackground, 2), e.uniform1i(p.uniforms.uCoverage, 3), e.uniform1i(p.uniforms.uVelocity, 4), d(null);
};
let tt = ot, E = null, ht;
const $ = new Promise((i) => {
  ht = i;
});
self.onmessage = async (i) => {
  const { type: e, ...r } = i.data;
  try {
    switch (e) {
      case "init": {
        const { canvas: s, width: o, height: a, config: n, dpr: x, quality: m, useWebGPU: l } = r;
        s.width = o, s.height = a, E = await tt.create(s, n, m ?? {}, l ?? !0), E.resize(o, a, x || 1), ht(), self.postMessage({ type: "ready" });
        break;
      }
      case "setTextSource": {
        if (await $, !E)
          return;
        E.setTextSource(r.opts);
        break;
      }
      case "setImageSource": {
        if (await $, !E)
          return;
        await E.setImageSource(
          r.src,
          r.effect,
          r.size
        );
        break;
      }
      case "setImageBitmap": {
        if (await $, !E)
          return;
        E.setImageBitmap(
          r.bitmap,
          r.effect,
          r.size
        );
        break;
      }
      case "setBackground": {
        if (await $, !E)
          return;
        E.setBackground(r.bitmap, r.size);
        break;
      }
      case "splat": {
        if (await $, !E)
          return;
        E.splat(
          r.x,
          r.y,
          r.vx,
          r.vy,
          r.strength ?? 1
        );
        break;
      }
      case "move": {
        if (await $, !E)
          return;
        E.handleMove(r.x, r.y, r.strength ?? 1);
        break;
      }
      case "resize": {
        if (await $, !E)
          return;
        E.resize(r.width, r.height, r.dpr);
        break;
      }
      case "updateQuality": {
        if (await $, !E)
          return;
        E.updateQuality(r.quality);
        break;
      }
      case "updateConfig": {
        if (await $, !E)
          return;
        E.updateConfig(r.config);
        break;
      }
      case "destroy": {
        await $, E == null || E.destroy(), E = null;
        break;
      }
      default:
        console.warn("[fluidity-js worker] Unknown message type:", e);
    }
  } catch (s) {
    self.postMessage({ type: "error", message: (s == null ? void 0 : s.message) ?? String(s) });
  }
};
", YI = typeof window < "u" && window.Blob && new Blob([atob(NI)], { type: "text/javascript;charset=utf-8" });
|
|
1538
|
+
function XC() {
|
|
1539
|
+
let e;
|
|
682
1540
|
try {
|
|
683
|
-
if (
|
|
1541
|
+
if (e = YI && (window.URL || window.webkitURL).createObjectURL(YI), !e)
|
|
684
1542
|
throw "";
|
|
685
|
-
return new Worker(
|
|
1543
|
+
return new Worker(e);
|
|
686
1544
|
} catch {
|
|
687
|
-
return new Worker("data:application/javascript;base64," +
|
|
1545
|
+
return new Worker("data:application/javascript;base64," + NI, { type: "module" });
|
|
688
1546
|
} finally {
|
|
689
|
-
|
|
1547
|
+
e && (window.URL || window.webkitURL).revokeObjectURL(e);
|
|
690
1548
|
}
|
|
691
1549
|
}
|
|
692
|
-
const
|
|
693
|
-
var
|
|
694
|
-
class
|
|
695
|
-
constructor(
|
|
696
|
-
isWorkerEnabled:
|
|
697
|
-
|
|
698
|
-
|
|
1550
|
+
const RC = typeof Worker < "u" && typeof OffscreenCanvas < "u";
|
|
1551
|
+
var v, M, zg, Ng, j, dg, rg, Lg, eI, hI, ZI, QI;
|
|
1552
|
+
class vC {
|
|
1553
|
+
constructor(g, {
|
|
1554
|
+
isWorkerEnabled: C = !0,
|
|
1555
|
+
useWebGPU: i = !0,
|
|
1556
|
+
quality: s = {},
|
|
1557
|
+
config: t = {}
|
|
699
1558
|
} = {}) {
|
|
700
1559
|
// ---------------------------------------------------------------------------
|
|
701
1560
|
// Private
|
|
702
1561
|
// ---------------------------------------------------------------------------
|
|
703
|
-
|
|
704
|
-
|
|
705
|
-
|
|
706
|
-
|
|
707
|
-
|
|
708
|
-
|
|
709
|
-
|
|
1562
|
+
/**
|
|
1563
|
+
* Main-thread renderer init.
|
|
1564
|
+
*
|
|
1565
|
+
* When `navigator.gpu` is present we attempt WebGPU asynchronously and
|
|
1566
|
+
* queue any source calls that arrive before the init resolves.
|
|
1567
|
+
*
|
|
1568
|
+
* When `navigator.gpu` is absent we fall back to WebGL synchronously —
|
|
1569
|
+
* this is the common path in jsdom/test environments and keeps the
|
|
1570
|
+
* constructor behaviour synchronous where possible.
|
|
1571
|
+
*/
|
|
1572
|
+
S(this, eI);
|
|
1573
|
+
S(this, ZI);
|
|
1574
|
+
S(this, v, null);
|
|
1575
|
+
S(this, M, null);
|
|
1576
|
+
S(this, zg, void 0);
|
|
1577
|
+
S(this, Ng, void 0);
|
|
1578
|
+
S(this, j, void 0);
|
|
1579
|
+
S(this, dg, void 0);
|
|
1580
|
+
// Pending source calls queued while WebGPU async init is in progress (main-thread only)
|
|
1581
|
+
S(this, rg, null);
|
|
1582
|
+
S(this, Lg, null);
|
|
1583
|
+
l(this, j, Math.max(0.1, Math.min(1, s.dpr ?? 1))), l(this, dg, Math.max(0.1, Math.min(1, s.sim ?? 0.5))), l(this, Ng, i), l(this, zg, C && RC), I(this, zg) ? h(this, ZI, QI).call(this, g, t) : h(this, eI, hI).call(this, g, t);
|
|
710
1584
|
}
|
|
711
1585
|
// ---------------------------------------------------------------------------
|
|
712
1586
|
// Source setters
|
|
713
1587
|
// ---------------------------------------------------------------------------
|
|
714
|
-
setTextSource(
|
|
715
|
-
|
|
1588
|
+
setTextSource(g) {
|
|
1589
|
+
I(this, v) ? I(this, v).postMessage({ type: "setTextSource", opts: g }) : I(this, M) ? I(this, M).setTextSource(g) : (l(this, rg, g), l(this, Lg, null));
|
|
716
1590
|
}
|
|
717
|
-
setImageSource(
|
|
718
|
-
if (
|
|
719
|
-
const s = new URL(
|
|
720
|
-
|
|
1591
|
+
setImageSource(g, C = Bg.effect, i = Bg.imageSize) {
|
|
1592
|
+
if (I(this, v)) {
|
|
1593
|
+
const s = new URL(g, location.href).href;
|
|
1594
|
+
I(this, v).postMessage({ type: "setImageSource", src: s, effect: C, size: i });
|
|
721
1595
|
} else
|
|
722
|
-
|
|
1596
|
+
I(this, M) ? I(this, M).setImageSource(g, C, i) : (l(this, Lg, { src: g, effect: C, size: i }), l(this, rg, null));
|
|
723
1597
|
}
|
|
724
|
-
setBackground(
|
|
725
|
-
var
|
|
726
|
-
if (
|
|
727
|
-
const s =
|
|
728
|
-
|
|
1598
|
+
setBackground(g, C = "cover") {
|
|
1599
|
+
var i;
|
|
1600
|
+
if (I(this, v)) {
|
|
1601
|
+
const s = g ? [g] : [];
|
|
1602
|
+
I(this, v).postMessage({ type: "setBackground", bitmap: g ?? null, size: C }, s);
|
|
729
1603
|
} else
|
|
730
|
-
(
|
|
1604
|
+
(i = I(this, M)) == null || i.setBackground(g ?? null, C);
|
|
731
1605
|
}
|
|
732
1606
|
// ---------------------------------------------------------------------------
|
|
733
1607
|
// Interaction
|
|
@@ -736,315 +1610,336 @@ class Bg {
|
|
|
736
1610
|
* Immediately injects one splat at (x, y) with explicit velocity (vx, vy).
|
|
737
1611
|
* Safe to call multiple times per frame. See FluidSimulation.splat for details.
|
|
738
1612
|
*/
|
|
739
|
-
splat(
|
|
740
|
-
|
|
1613
|
+
splat(g, C, i, s, t = 1) {
|
|
1614
|
+
I(this, v) ? I(this, v).postMessage({ type: "splat", x: g, y: C, vx: i, vy: s, strength: t }) : I(this, M).splat(g, C, i, s, t);
|
|
741
1615
|
}
|
|
742
|
-
handleMove(
|
|
743
|
-
|
|
1616
|
+
handleMove(g, C, i = 1) {
|
|
1617
|
+
I(this, v) ? I(this, v).postMessage({ type: "move", x: g, y: C, strength: i }) : I(this, M).handleMove(g, C, i);
|
|
744
1618
|
}
|
|
745
1619
|
// ---------------------------------------------------------------------------
|
|
746
1620
|
// Config + control
|
|
747
1621
|
// ---------------------------------------------------------------------------
|
|
748
|
-
updateQuality(
|
|
749
|
-
l(this,
|
|
1622
|
+
updateQuality(g) {
|
|
1623
|
+
l(this, j, Math.max(0.1, Math.min(1, g.dpr ?? I(this, j)))), l(this, dg, Math.max(0.1, Math.min(1, g.sim ?? I(this, dg)))), I(this, v) ? I(this, v).postMessage({ type: "updateQuality", quality: { dpr: I(this, j), sim: I(this, dg) } }) : I(this, M).updateQuality(g);
|
|
750
1624
|
}
|
|
751
|
-
updateConfig(
|
|
752
|
-
|
|
1625
|
+
updateConfig(g) {
|
|
1626
|
+
I(this, v) ? I(this, v).postMessage({ type: "updateConfig", config: g }) : I(this, M).updateConfig(g);
|
|
753
1627
|
}
|
|
754
|
-
resize(
|
|
755
|
-
const
|
|
756
|
-
|
|
1628
|
+
resize(g, C) {
|
|
1629
|
+
const i = (typeof window < "u" && window.devicePixelRatio || 1) * I(this, j);
|
|
1630
|
+
I(this, v) ? I(this, v).postMessage({ type: "resize", width: g, height: C, dpr: i }) : I(this, M).resize(g, C, i);
|
|
757
1631
|
}
|
|
758
1632
|
destroy() {
|
|
759
|
-
var
|
|
760
|
-
if (
|
|
761
|
-
const
|
|
762
|
-
l(this,
|
|
1633
|
+
var g;
|
|
1634
|
+
if (I(this, v)) {
|
|
1635
|
+
const C = I(this, v);
|
|
1636
|
+
l(this, v, null), C.postMessage({ type: "destroy" }), setTimeout(() => C.terminate(), 50);
|
|
763
1637
|
} else
|
|
764
|
-
(
|
|
1638
|
+
(g = I(this, M)) == null || g.destroy(), l(this, M, null);
|
|
765
1639
|
}
|
|
766
1640
|
}
|
|
767
|
-
|
|
768
|
-
const
|
|
769
|
-
I.
|
|
770
|
-
|
|
1641
|
+
v = new WeakMap(), M = new WeakMap(), zg = new WeakMap(), Ng = new WeakMap(), j = new WeakMap(), dg = new WeakMap(), rg = new WeakMap(), Lg = new WeakMap(), eI = new WeakSet(), hI = function(g, C) {
|
|
1642
|
+
const i = { dpr: I(this, j), sim: I(this, dg) };
|
|
1643
|
+
I(this, Ng) && typeof navigator < "u" && !!navigator.gpu ? dI.create(g, C, i, !0).then((t) => {
|
|
1644
|
+
if (l(this, M, t), I(this, rg))
|
|
1645
|
+
t.setTextSource(I(this, rg)), l(this, rg, null);
|
|
1646
|
+
else if (I(this, Lg)) {
|
|
1647
|
+
const { src: o, effect: a, size: u } = I(this, Lg);
|
|
1648
|
+
t.setImageSource(o, a, u), l(this, Lg, null);
|
|
1649
|
+
}
|
|
1650
|
+
}).catch((t) => {
|
|
1651
|
+
console.error("[fluidity-js] Renderer init failed:", t);
|
|
1652
|
+
}) : l(this, M, new dI(g, C, i));
|
|
1653
|
+
}, ZI = new WeakSet(), QI = function(g, C) {
|
|
1654
|
+
const i = (typeof window < "u" && window.devicePixelRatio || 1) * I(this, j), s = Math.round(g.clientWidth * i), t = Math.round(g.clientHeight * i);
|
|
1655
|
+
g.width = s, g.height = t;
|
|
1656
|
+
let o;
|
|
771
1657
|
try {
|
|
772
|
-
|
|
1658
|
+
o = g.transferControlToOffscreen();
|
|
773
1659
|
} catch {
|
|
774
1660
|
console.warn(
|
|
775
1661
|
"[fluidity-js] OffscreenCanvas transfer failed — falling back to main-thread mode. This is expected in React StrictMode development."
|
|
776
|
-
), l(this,
|
|
1662
|
+
), l(this, zg, !1), h(this, eI, hI).call(this, g, C);
|
|
777
1663
|
return;
|
|
778
1664
|
}
|
|
779
|
-
const
|
|
780
|
-
|
|
781
|
-
console.error("[fluidity-js] Worker error:",
|
|
782
|
-
},
|
|
783
|
-
|
|
784
|
-
},
|
|
785
|
-
{ type: "init", canvas:
|
|
786
|
-
[
|
|
1665
|
+
const a = l(this, v, new XC());
|
|
1666
|
+
a.onerror = (u) => {
|
|
1667
|
+
console.error("[fluidity-js] Worker error:", u.message);
|
|
1668
|
+
}, a.onmessage = (u) => {
|
|
1669
|
+
u.data.type === "error" && console.error("[fluidity-js] Simulation error:", u.data.message);
|
|
1670
|
+
}, a.postMessage(
|
|
1671
|
+
{ type: "init", canvas: o, width: s, height: t, config: C, dpr: i, quality: { dpr: I(this, j), sim: I(this, dg) }, useWebGPU: I(this, Ng) },
|
|
1672
|
+
[o]
|
|
787
1673
|
);
|
|
788
1674
|
};
|
|
789
|
-
function
|
|
790
|
-
|
|
1675
|
+
function OI(e, {
|
|
1676
|
+
isWorkerEnabled: g = !0,
|
|
1677
|
+
useWebGPU: C = !0,
|
|
1678
|
+
quality: i = {},
|
|
1679
|
+
config: s = {}
|
|
1680
|
+
} = {}) {
|
|
1681
|
+
const t = Mg(null), o = Mg({ isWorkerEnabled: g, quality: i, config: s }), a = Mg(Math.max(0.1, Math.min(1, i.dpr ?? 1))), u = Mg({
|
|
791
1682
|
dpr: i.dpr,
|
|
792
1683
|
sim: i.sim
|
|
793
1684
|
});
|
|
794
|
-
return
|
|
795
|
-
const
|
|
796
|
-
if (!
|
|
1685
|
+
return eg(() => {
|
|
1686
|
+
const A = e.current;
|
|
1687
|
+
if (!A)
|
|
797
1688
|
return;
|
|
798
|
-
const
|
|
799
|
-
|
|
800
|
-
const { isWorkerEnabled:
|
|
801
|
-
|
|
1689
|
+
const p = document.createElement("canvas");
|
|
1690
|
+
p.id = `fluid_canvas_${Date.now()}`, p.style.cssText = "position:absolute;inset:0;width:100%;height:100%;display:block;", A.appendChild(p);
|
|
1691
|
+
const { isWorkerEnabled: c, quality: d, config: Z } = o.current, m = (window.devicePixelRatio || 1) * a.current, w = A.getBoundingClientRect(), b = Math.round((w.width || A.clientWidth) * m) || 0, n = Math.round((w.height || A.clientHeight) * m) || 0;
|
|
1692
|
+
b > 0 && (p.width = b, p.height = n), n === 0 && console.warn(
|
|
802
1693
|
"[fluidity-js] Container has zero height — simulation will not render. Avoid height:auto or percentage heights without a sized ancestor. Use explicit pixel values instead."
|
|
803
1694
|
);
|
|
804
|
-
const
|
|
805
|
-
|
|
806
|
-
const
|
|
807
|
-
for (const
|
|
808
|
-
const
|
|
809
|
-
|
|
1695
|
+
const r = new vC(p, { isWorkerEnabled: c, useWebGPU: C, quality: d, config: Z });
|
|
1696
|
+
t.current = r;
|
|
1697
|
+
const R = new ResizeObserver((B) => {
|
|
1698
|
+
for (const W of B) {
|
|
1699
|
+
const G = (window.devicePixelRatio || 1) * a.current, { inlineSize: K, blockSize: U } = W.contentBoxSize[0];
|
|
1700
|
+
r.resize(Math.round(K * G), Math.round(U * G));
|
|
810
1701
|
}
|
|
811
1702
|
});
|
|
812
|
-
return
|
|
813
|
-
|
|
1703
|
+
return R.observe(A), () => {
|
|
1704
|
+
R.disconnect(), r.destroy(), p.remove(), t.current = null;
|
|
814
1705
|
};
|
|
815
|
-
}, []),
|
|
816
|
-
|
|
817
|
-
const
|
|
1706
|
+
}, [C]), eg(() => {
|
|
1707
|
+
a.current = Math.max(0.1, Math.min(1, i.dpr ?? 1));
|
|
1708
|
+
const A = u.current;
|
|
818
1709
|
u.current = { dpr: i.dpr, sim: i.sim };
|
|
819
|
-
const
|
|
820
|
-
if (!
|
|
1710
|
+
const p = t.current, c = e.current;
|
|
1711
|
+
if (!p || !c || A.dpr === i.dpr && A.sim === i.sim)
|
|
821
1712
|
return;
|
|
822
|
-
|
|
823
|
-
const
|
|
824
|
-
|
|
825
|
-
}, [i.dpr, i.sim]),
|
|
1713
|
+
p.updateQuality(i);
|
|
1714
|
+
const d = (window.devicePixelRatio || 1) * a.current, Z = c.clientWidth, m = c.clientHeight;
|
|
1715
|
+
Z > 0 && m > 0 && p.resize(Math.round(Z * d), Math.round(m * d));
|
|
1716
|
+
}, [i.dpr, i.sim, C]), t;
|
|
826
1717
|
}
|
|
827
|
-
const
|
|
828
|
-
text:
|
|
829
|
-
fontSize:
|
|
830
|
-
color:
|
|
831
|
-
fontFamily: s =
|
|
832
|
-
fontWeight:
|
|
833
|
-
className:
|
|
834
|
-
style:
|
|
835
|
-
config:
|
|
836
|
-
preset:
|
|
837
|
-
algorithm:
|
|
838
|
-
backgroundColor:
|
|
1718
|
+
const kC = kI(function({
|
|
1719
|
+
text: g,
|
|
1720
|
+
fontSize: C = Sg.fontSize,
|
|
1721
|
+
color: i = Sg.color,
|
|
1722
|
+
fontFamily: s = Sg.fontFamily,
|
|
1723
|
+
fontWeight: t = Sg.fontWeight,
|
|
1724
|
+
className: o,
|
|
1725
|
+
style: a,
|
|
1726
|
+
config: u,
|
|
1727
|
+
preset: A,
|
|
1728
|
+
algorithm: p,
|
|
1729
|
+
backgroundColor: c = Sg.backgroundColor,
|
|
839
1730
|
backgroundSrc: d,
|
|
840
|
-
backgroundSize:
|
|
841
|
-
isMouseEnabled:
|
|
842
|
-
isWorkerEnabled:
|
|
843
|
-
|
|
844
|
-
|
|
845
|
-
|
|
846
|
-
|
|
847
|
-
|
|
848
|
-
|
|
1731
|
+
backgroundSize: Z = Sg.backgroundSize,
|
|
1732
|
+
isMouseEnabled: m = Sg.isMouseEnabled,
|
|
1733
|
+
isWorkerEnabled: w = Sg.isWorkerEnabled,
|
|
1734
|
+
useWebGPU: b = !0,
|
|
1735
|
+
quality: n = mI.quality
|
|
1736
|
+
}, r) {
|
|
1737
|
+
const R = Mg(null), B = OI(R, {
|
|
1738
|
+
isWorkerEnabled: w,
|
|
1739
|
+
useWebGPU: b,
|
|
1740
|
+
quality: n,
|
|
1741
|
+
config: qg({ ...u, ...p ? { algorithm: p } : {} }, A, HI)
|
|
849
1742
|
});
|
|
850
|
-
|
|
851
|
-
|
|
1743
|
+
xI(
|
|
1744
|
+
r,
|
|
852
1745
|
() => ({
|
|
853
1746
|
reset() {
|
|
854
|
-
var
|
|
855
|
-
(
|
|
1747
|
+
var G;
|
|
1748
|
+
(G = B.current) == null || G.setTextSource({ text: g, fontSize: C, color: i, fontFamily: s, fontWeight: t });
|
|
856
1749
|
},
|
|
857
|
-
move(
|
|
858
|
-
var
|
|
859
|
-
(
|
|
1750
|
+
move(G, K, U = 1) {
|
|
1751
|
+
var Y;
|
|
1752
|
+
(Y = B.current) == null || Y.handleMove(G, K, U);
|
|
860
1753
|
},
|
|
861
|
-
splat(
|
|
862
|
-
var
|
|
863
|
-
(
|
|
1754
|
+
splat(G, K, U, Y, P = 1) {
|
|
1755
|
+
var cg;
|
|
1756
|
+
(cg = B.current) == null || cg.splat(G, K, U, Y, P);
|
|
864
1757
|
},
|
|
865
|
-
updateConfig(
|
|
866
|
-
var
|
|
867
|
-
(
|
|
1758
|
+
updateConfig(G) {
|
|
1759
|
+
var K;
|
|
1760
|
+
(K = B.current) == null || K.updateConfig(G);
|
|
868
1761
|
}
|
|
869
1762
|
}),
|
|
870
|
-
[
|
|
871
|
-
),
|
|
872
|
-
var
|
|
873
|
-
(
|
|
874
|
-
}, [
|
|
875
|
-
const
|
|
876
|
-
return
|
|
877
|
-
var
|
|
878
|
-
(
|
|
879
|
-
|
|
1763
|
+
[g, C, i, s, t]
|
|
1764
|
+
), eg(() => {
|
|
1765
|
+
var G;
|
|
1766
|
+
(G = B.current) == null || G.setTextSource({ text: g, fontSize: C, color: i, fontFamily: s, fontWeight: t });
|
|
1767
|
+
}, [g, C, i, s, t, b]);
|
|
1768
|
+
const W = JSON.stringify(u);
|
|
1769
|
+
return eg(() => {
|
|
1770
|
+
var G;
|
|
1771
|
+
(G = B.current) == null || G.updateConfig(
|
|
1772
|
+
qg({ ...u, ...p !== void 0 ? { algorithm: p } : {} }, A, HI)
|
|
880
1773
|
);
|
|
881
|
-
}, [
|
|
882
|
-
var
|
|
1774
|
+
}, [A, p, W, b]), eg(() => {
|
|
1775
|
+
var K;
|
|
883
1776
|
if (!d) {
|
|
884
|
-
(
|
|
1777
|
+
(K = B.current) == null || K.setBackground(null);
|
|
885
1778
|
return;
|
|
886
1779
|
}
|
|
887
|
-
let
|
|
888
|
-
return
|
|
889
|
-
var
|
|
890
|
-
if (
|
|
891
|
-
|
|
1780
|
+
let G = !1;
|
|
1781
|
+
return KI(d).then((U) => {
|
|
1782
|
+
var Y;
|
|
1783
|
+
if (G) {
|
|
1784
|
+
U.close();
|
|
892
1785
|
return;
|
|
893
1786
|
}
|
|
894
|
-
(
|
|
895
|
-
}).catch((
|
|
896
|
-
|
|
1787
|
+
(Y = B.current) == null || Y.setBackground(U, Z);
|
|
1788
|
+
}).catch((U) => console.error("[fluidity-js] backgroundSrc load failed:", U)), () => {
|
|
1789
|
+
G = !0;
|
|
897
1790
|
};
|
|
898
|
-
}, [d,
|
|
899
|
-
if (!
|
|
1791
|
+
}, [d, Z, b]), eg(() => {
|
|
1792
|
+
if (!m)
|
|
900
1793
|
return;
|
|
901
|
-
const
|
|
902
|
-
if (!
|
|
1794
|
+
const G = R.current;
|
|
1795
|
+
if (!G)
|
|
903
1796
|
return;
|
|
904
|
-
const
|
|
905
|
-
var
|
|
906
|
-
const
|
|
907
|
-
(
|
|
908
|
-
},
|
|
909
|
-
var
|
|
910
|
-
|
|
911
|
-
const
|
|
912
|
-
(
|
|
1797
|
+
const K = (Y) => {
|
|
1798
|
+
var cg;
|
|
1799
|
+
const P = G.getBoundingClientRect();
|
|
1800
|
+
(cg = B.current) == null || cg.handleMove(Y.clientX - P.left, Y.clientY - P.top, 2);
|
|
1801
|
+
}, U = (Y) => {
|
|
1802
|
+
var VI;
|
|
1803
|
+
Y.preventDefault();
|
|
1804
|
+
const P = G.getBoundingClientRect(), cg = Y.touches[0];
|
|
1805
|
+
(VI = B.current) == null || VI.handleMove(cg.clientX - P.left, cg.clientY - P.top, 1);
|
|
913
1806
|
};
|
|
914
|
-
return
|
|
915
|
-
|
|
1807
|
+
return G.addEventListener("mousemove", K), G.addEventListener("touchmove", U, { passive: !1 }), () => {
|
|
1808
|
+
G.removeEventListener("mousemove", K), G.removeEventListener("touchmove", U);
|
|
916
1809
|
};
|
|
917
|
-
}, [
|
|
1810
|
+
}, [m]), /* @__PURE__ */ UI(
|
|
918
1811
|
"div",
|
|
919
1812
|
{
|
|
920
|
-
ref:
|
|
921
|
-
className:
|
|
1813
|
+
ref: R,
|
|
1814
|
+
className: o,
|
|
922
1815
|
style: {
|
|
923
1816
|
position: "relative",
|
|
924
1817
|
display: "block",
|
|
925
1818
|
width: "100%",
|
|
926
1819
|
height: "100%",
|
|
927
|
-
background:
|
|
928
|
-
...
|
|
1820
|
+
background: c,
|
|
1821
|
+
...a
|
|
929
1822
|
}
|
|
930
1823
|
}
|
|
931
1824
|
);
|
|
932
|
-
}),
|
|
933
|
-
src:
|
|
934
|
-
effect:
|
|
935
|
-
imageSize:
|
|
1825
|
+
}), xC = kI(function({
|
|
1826
|
+
src: g,
|
|
1827
|
+
effect: C = Bg.effect,
|
|
1828
|
+
imageSize: i = Bg.imageSize,
|
|
936
1829
|
className: s,
|
|
937
|
-
style:
|
|
938
|
-
config:
|
|
939
|
-
preset:
|
|
940
|
-
algorithm:
|
|
941
|
-
backgroundColor:
|
|
942
|
-
backgroundSrc:
|
|
943
|
-
backgroundSize:
|
|
944
|
-
isMouseEnabled: d =
|
|
945
|
-
isWorkerEnabled:
|
|
946
|
-
|
|
947
|
-
|
|
948
|
-
|
|
949
|
-
|
|
950
|
-
|
|
951
|
-
|
|
1830
|
+
style: t,
|
|
1831
|
+
config: o,
|
|
1832
|
+
preset: a,
|
|
1833
|
+
algorithm: u,
|
|
1834
|
+
backgroundColor: A = Bg.backgroundColor,
|
|
1835
|
+
backgroundSrc: p,
|
|
1836
|
+
backgroundSize: c = Bg.backgroundSize,
|
|
1837
|
+
isMouseEnabled: d = Bg.isMouseEnabled,
|
|
1838
|
+
isWorkerEnabled: Z = Bg.isWorkerEnabled,
|
|
1839
|
+
useWebGPU: m = !0,
|
|
1840
|
+
quality: w = mI.quality
|
|
1841
|
+
}, b) {
|
|
1842
|
+
const n = Mg(null), r = OI(n, {
|
|
1843
|
+
isWorkerEnabled: Z,
|
|
1844
|
+
useWebGPU: m,
|
|
1845
|
+
quality: w,
|
|
1846
|
+
config: qg({ ...o, ...u ? { algorithm: u } : {} }, a)
|
|
952
1847
|
});
|
|
953
|
-
|
|
954
|
-
|
|
1848
|
+
xI(
|
|
1849
|
+
b,
|
|
955
1850
|
() => ({
|
|
956
1851
|
reset() {
|
|
957
|
-
var
|
|
958
|
-
|
|
1852
|
+
var B;
|
|
1853
|
+
g && ((B = r.current) == null || B.setImageSource(g, C, i));
|
|
959
1854
|
},
|
|
960
|
-
move(
|
|
961
|
-
var
|
|
962
|
-
(
|
|
1855
|
+
move(B, W, G = 1) {
|
|
1856
|
+
var K;
|
|
1857
|
+
(K = r.current) == null || K.handleMove(B, W, G);
|
|
963
1858
|
},
|
|
964
|
-
splat(
|
|
965
|
-
var
|
|
966
|
-
(
|
|
1859
|
+
splat(B, W, G, K, U = 1) {
|
|
1860
|
+
var Y;
|
|
1861
|
+
(Y = r.current) == null || Y.splat(B, W, G, K, U);
|
|
967
1862
|
},
|
|
968
|
-
updateConfig(
|
|
969
|
-
var
|
|
970
|
-
(
|
|
1863
|
+
updateConfig(B) {
|
|
1864
|
+
var W;
|
|
1865
|
+
(W = r.current) == null || W.updateConfig(B);
|
|
971
1866
|
}
|
|
972
1867
|
}),
|
|
973
|
-
[
|
|
974
|
-
),
|
|
975
|
-
var
|
|
976
|
-
|
|
977
|
-
}, [
|
|
978
|
-
const
|
|
979
|
-
return
|
|
980
|
-
var
|
|
981
|
-
(
|
|
982
|
-
|
|
1868
|
+
[g, C, i]
|
|
1869
|
+
), eg(() => {
|
|
1870
|
+
var B;
|
|
1871
|
+
g && ((B = r.current) == null || B.setImageSource(g, C, i));
|
|
1872
|
+
}, [g, C, i, m]);
|
|
1873
|
+
const R = JSON.stringify(o);
|
|
1874
|
+
return eg(() => {
|
|
1875
|
+
var B;
|
|
1876
|
+
(B = r.current) == null || B.updateConfig(
|
|
1877
|
+
qg({ ...o, ...u !== void 0 ? { algorithm: u } : {} }, a)
|
|
983
1878
|
);
|
|
984
|
-
}, [u,
|
|
985
|
-
var
|
|
986
|
-
if (!
|
|
987
|
-
(
|
|
1879
|
+
}, [a, u, R, m]), eg(() => {
|
|
1880
|
+
var W;
|
|
1881
|
+
if (!p) {
|
|
1882
|
+
(W = r.current) == null || W.setBackground(null);
|
|
988
1883
|
return;
|
|
989
1884
|
}
|
|
990
|
-
let
|
|
991
|
-
return
|
|
992
|
-
var
|
|
993
|
-
if (
|
|
994
|
-
|
|
1885
|
+
let B = !1;
|
|
1886
|
+
return KI(p).then((G) => {
|
|
1887
|
+
var K;
|
|
1888
|
+
if (B) {
|
|
1889
|
+
G.close();
|
|
995
1890
|
return;
|
|
996
1891
|
}
|
|
997
|
-
(
|
|
998
|
-
}).catch((
|
|
999
|
-
|
|
1892
|
+
(K = r.current) == null || K.setBackground(G, c);
|
|
1893
|
+
}).catch((G) => console.error("[fluidity-js] backgroundSrc load failed:", G)), () => {
|
|
1894
|
+
B = !0;
|
|
1000
1895
|
};
|
|
1001
|
-
}, [
|
|
1896
|
+
}, [p, c, m]), eg(() => {
|
|
1002
1897
|
if (!d)
|
|
1003
1898
|
return;
|
|
1004
|
-
const
|
|
1005
|
-
if (!
|
|
1899
|
+
const B = n.current;
|
|
1900
|
+
if (!B)
|
|
1006
1901
|
return;
|
|
1007
|
-
const
|
|
1008
|
-
var
|
|
1009
|
-
const
|
|
1010
|
-
(
|
|
1011
|
-
},
|
|
1012
|
-
var
|
|
1013
|
-
|
|
1014
|
-
const
|
|
1015
|
-
(
|
|
1902
|
+
const W = (K) => {
|
|
1903
|
+
var Y;
|
|
1904
|
+
const U = B.getBoundingClientRect();
|
|
1905
|
+
(Y = r.current) == null || Y.handleMove(K.clientX - U.left, K.clientY - U.top, 2);
|
|
1906
|
+
}, G = (K) => {
|
|
1907
|
+
var P;
|
|
1908
|
+
K.preventDefault();
|
|
1909
|
+
const U = B.getBoundingClientRect(), Y = K.touches[0];
|
|
1910
|
+
(P = r.current) == null || P.handleMove(Y.clientX - U.left, Y.clientY - U.top, 1);
|
|
1016
1911
|
};
|
|
1017
|
-
return
|
|
1018
|
-
|
|
1912
|
+
return B.addEventListener("mousemove", W), B.addEventListener("touchmove", G, { passive: !1 }), () => {
|
|
1913
|
+
B.removeEventListener("mousemove", W), B.removeEventListener("touchmove", G);
|
|
1019
1914
|
};
|
|
1020
|
-
}, [d]), /* @__PURE__ */
|
|
1915
|
+
}, [d]), /* @__PURE__ */ UI(
|
|
1021
1916
|
"div",
|
|
1022
1917
|
{
|
|
1023
|
-
ref:
|
|
1918
|
+
ref: n,
|
|
1024
1919
|
className: s,
|
|
1025
1920
|
style: {
|
|
1026
1921
|
position: "relative",
|
|
1027
1922
|
display: "block",
|
|
1028
1923
|
width: "100%",
|
|
1029
1924
|
height: "100%",
|
|
1030
|
-
background:
|
|
1031
|
-
...
|
|
1925
|
+
background: A,
|
|
1926
|
+
...t
|
|
1032
1927
|
}
|
|
1033
1928
|
}
|
|
1034
1929
|
);
|
|
1035
1930
|
});
|
|
1036
1931
|
export {
|
|
1037
|
-
|
|
1038
|
-
|
|
1039
|
-
|
|
1040
|
-
|
|
1041
|
-
|
|
1042
|
-
|
|
1043
|
-
|
|
1044
|
-
|
|
1045
|
-
|
|
1046
|
-
|
|
1047
|
-
|
|
1048
|
-
|
|
1049
|
-
|
|
1932
|
+
fI as DEFAULT_CONFIG,
|
|
1933
|
+
HI as DEFAULT_CONFIG_TEXT,
|
|
1934
|
+
Bg as DEFAULT_PROPS_IMAGE,
|
|
1935
|
+
mI as DEFAULT_PROPS_SHARED,
|
|
1936
|
+
Sg as DEFAULT_PROPS_TEXT,
|
|
1937
|
+
vC as FluidController,
|
|
1938
|
+
xC as FluidImage,
|
|
1939
|
+
dI as FluidSimulation,
|
|
1940
|
+
kC as FluidText,
|
|
1941
|
+
_I as PRESETS,
|
|
1942
|
+
KI as loadImageBitmap,
|
|
1943
|
+
qg as mergeConfig,
|
|
1944
|
+
OI as useFluid
|
|
1050
1945
|
};
|