@jayf0x/fluidity-js 0.1.5 → 0.1.7

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/dist/index.js CHANGED
@@ -1,18 +1,18 @@
1
- var jI = Object.defineProperty;
2
- var PI = (C, I, i) => I in C ? jI(C, I, { enumerable: !0, configurable: !0, writable: !0, value: i }) : C[I] = i;
3
- var BI = (C, I, i) => (PI(C, typeof I != "symbol" ? I + "" : I, i), i), LI = (C, I, i) => {
1
+ var $I = Object.defineProperty;
2
+ var Ig = (C, I, i) => I in C ? $I(C, I, { enumerable: !0, configurable: !0, writable: !0, value: i }) : C[I] = i;
3
+ var yI = (C, I, i) => (Ig(C, typeof I != "symbol" ? I + "" : I, i), i), UI = (C, I, i) => {
4
4
  if (!I.has(C))
5
5
  throw TypeError("Cannot " + i);
6
6
  };
7
- var g = (C, I, i) => (LI(C, I, "read from private field"), i ? i.call(C) : I.get(C)), n = (C, I, i) => {
7
+ var g = (C, I, i) => (UI(C, I, "read from private field"), i ? i.call(C) : I.get(C)), m = (C, I, i) => {
8
8
  if (I.has(C))
9
9
  throw TypeError("Cannot add the same private member more than once");
10
10
  I instanceof WeakSet ? I.add(C) : I.set(C, i);
11
- }, o = (C, I, i, e) => (LI(C, I, "write to private field"), e ? e.call(C, i) : I.set(C, i), i);
12
- var b = (C, I, i) => (LI(C, I, "access private method"), i);
13
- import { jsx as TI } from "react/jsx-runtime";
14
- import { useRef as bI, useEffect as Q, forwardRef as xI, useImperativeHandle as MI } from "react";
15
- const JI = {
11
+ }, l = (C, I, i, e) => (UI(C, I, "write to private field"), e ? e.call(C, i) : I.set(C, i), i);
12
+ var b = (C, I, i) => (UI(C, I, "access private method"), i);
13
+ import { jsx as NI } from "react/jsx-runtime";
14
+ import { useRef as mI, useEffect as j, forwardRef as zI, useImperativeHandle as QI } from "react";
15
+ const OI = {
16
16
  densityDissipation: 0.992,
17
17
  velocityDissipation: 0.93,
18
18
  pressureIterations: 1,
@@ -26,8 +26,8 @@ const JI = {
26
26
  glowColor: [0.7, 0.85, 1],
27
27
  algorithm: "standard",
28
28
  warpStrength: 0.015
29
- }, kI = {
30
- ...JI,
29
+ }, TI = {
30
+ ...OI,
31
31
  densityDissipation: 0.99,
32
32
  velocityDissipation: 0.98,
33
33
  pressureIterations: 3,
@@ -38,22 +38,23 @@ const JI = {
38
38
  specularExp: 1,
39
39
  shine: 0.1,
40
40
  glowColor: [0, 0.502, 1]
41
- }, EI = {
41
+ }, gg = { dpr: 1, sim: 0.5 }, HI = {
42
42
  backgroundColor: "#0a0a0a",
43
43
  backgroundSize: "cover",
44
44
  isMouseEnabled: !0,
45
- isWorkerEnabled: !0
46
- }, q = {
47
- ...EI,
45
+ isWorkerEnabled: !0,
46
+ quality: gg
47
+ }, iI = {
48
+ ...HI,
48
49
  effect: 0,
49
50
  imageSize: "cover"
50
- }, j = {
51
- ...EI,
51
+ }, $ = {
52
+ ...HI,
52
53
  fontSize: 100,
53
54
  color: "#ffffff",
54
55
  fontFamily: "sans-serif",
55
56
  fontWeight: 900
56
- }, _I = {
57
+ }, ig = {
57
58
  calm: {
58
59
  densityDissipation: 0.999,
59
60
  velocityDissipation: 0.98,
@@ -112,73 +113,73 @@ const JI = {
112
113
  waterColor: [0.06, 0.06, 0.06]
113
114
  }
114
115
  };
115
- function uI(C = {}, I, i = JI) {
116
- return { ...I ? { ...i, ..._I[I] } : i, ...C };
116
+ function pI(C = {}, I, i = OI) {
117
+ return { ...I ? { ...i, ...ig[I] } : i, ...C };
117
118
  }
118
- function RI(C, I, i, e, l = "cover") {
119
- let s;
120
- l === "cover" ? s = Math.max(i / C, e / I) : l === "contain" ? s = Math.min(i / C, e / I) : typeof l == "string" && l.endsWith("%") ? s = Math.min(i / C, e / I) * (parseFloat(l) / 100) : typeof l == "string" && l.endsWith("px") ? s = parseFloat(l) / Math.max(C, I) : typeof l == "number" ? s = l : s = Math.max(i / C, e / I);
121
- const t = C * s, m = I * s;
122
- return { x: (i - t) / 2, y: (e - m) / 2, drawW: t, drawH: m };
119
+ function vI(C, I, i, e, s = "cover") {
120
+ let o;
121
+ s === "cover" ? o = Math.max(i / C, e / I) : s === "contain" ? o = Math.min(i / C, e / I) : typeof s == "string" && s.endsWith("%") ? o = Math.min(i / C, e / I) * (parseFloat(s) / 100) : typeof s == "string" && s.endsWith("px") ? o = parseFloat(s) / Math.max(C, I) : typeof s == "number" ? o = s : o = Math.max(i / C, e / I);
122
+ const t = C * o, u = I * o;
123
+ return { x: (i - t) / 2, y: (e - u) / 2, drawW: t, drawH: u };
123
124
  }
124
- function qI(C, I, i, e, l = null, s = "cover") {
125
- const { text: t, fontSize: m, color: r, fontFamily: a = "sans-serif", fontWeight: u = 900 } = e, S = new OffscreenCanvas(I, i), A = S.getContext("2d");
126
- ((K) => {
127
- if (l) {
128
- A.clearRect(0, 0, I, i), A.fillStyle = "black", A.fillRect(0, 0, I, i);
129
- const { x: D, y: c, drawW: w, drawH: d } = RI(
130
- l.width,
131
- l.height,
125
+ function Cg(C, I, i, e, s = null, o = "cover") {
126
+ const { text: t, fontSize: u, color: a, fontFamily: c = "sans-serif", fontWeight: n = 900 } = e, G = new OffscreenCanvas(I, i), d = G.getContext("2d");
127
+ ((F) => {
128
+ if (s) {
129
+ d.clearRect(0, 0, I, i), d.fillStyle = "black", d.fillRect(0, 0, I, i);
130
+ const { x: w, y: D, drawW: A, drawH: y } = vI(
131
+ s.width,
132
+ s.height,
132
133
  I,
133
134
  i,
134
- s
135
+ o
135
136
  );
136
- A.drawImage(l, D, c, w, d);
137
+ d.drawImage(s, w, D, A, y);
137
138
  } else
138
- A.fillStyle = "black", A.fillRect(0, 0, I, i);
139
- A.fillStyle = K, A.font = `${u} ${m}px ${a}`, A.textAlign = "center", A.textBaseline = "middle", A.fillText(t, I / 2, i / 2);
140
- })(r);
141
- const v = rI(C, S);
142
- A.fillStyle = "black", A.fillRect(0, 0, I, i), A.fillStyle = "white", A.font = `${u} ${m}px ${a}`, A.textAlign = "center", A.textBaseline = "middle", A.fillText(t, I / 2, i / 2);
143
- const Y = rI(C, S);
144
- return { backgroundTex: v, obstacleTex: Y, coverageTex: Y };
139
+ d.fillStyle = "black", d.fillRect(0, 0, I, i);
140
+ d.fillStyle = F, d.font = `${n} ${u}px ${c}`, d.textAlign = "center", d.textBaseline = "middle", d.fillText(t, I / 2, i / 2);
141
+ })(a);
142
+ const W = BI(C, G);
143
+ d.fillStyle = "black", d.fillRect(0, 0, I, i), d.fillStyle = "white", d.font = `${n} ${u}px ${c}`, d.textAlign = "center", d.textBaseline = "middle", d.fillText(t, I / 2, i / 2);
144
+ const x = BI(C, G);
145
+ return { backgroundTex: W, obstacleTex: x, coverageTex: x };
145
146
  }
146
- function $I(C, I, i, e, l = 0, s = "cover", t = null, m = "cover") {
147
- const r = new OffscreenCanvas(i, e), a = r.getContext("2d"), { x: u, y: S, drawW: A, drawH: h } = RI(I.width, I.height, i, e, s);
148
- if (a.clearRect(0, 0, i, e), a.fillStyle = "black", a.fillRect(0, 0, i, e), t) {
147
+ function eg(C, I, i, e, s = 0, o = "cover", t = null, u = "cover") {
148
+ const a = new OffscreenCanvas(i, e), c = a.getContext("2d"), { x: n, y: G, drawW: d, drawH: h } = vI(I.width, I.height, i, e, o);
149
+ if (c.clearRect(0, 0, i, e), c.fillStyle = "black", c.fillRect(0, 0, i, e), t) {
149
150
  const {
150
- x: D,
151
- y: c,
152
- drawW: w,
153
- drawH: d
154
- } = RI(t.width, t.height, i, e, m);
155
- a.filter = `brightness(${l}) blur(8px)`, a.drawImage(t, D, c, w, d), a.filter = "none";
156
- }
157
- a.drawImage(I, u, S, A, h);
158
- const v = rI(C, r);
159
- a.clearRect(0, 0, i, e), a.fillStyle = "black", a.fillRect(0, 0, i, e), a.filter = `brightness(${l}) blur(8px)`, a.drawImage(I, u, S, A, h), a.filter = "none";
160
- const Y = rI(C, r);
161
- a.clearRect(0, 0, i, e), a.fillStyle = "black", a.fillRect(0, 0, i, e), a.fillStyle = "white", a.fillRect(
162
- Math.max(0, u),
163
- Math.max(0, S),
164
- Math.min(A, i - Math.max(0, u)),
165
- Math.min(h, e - Math.max(0, S))
151
+ x: w,
152
+ y: D,
153
+ drawW: A,
154
+ drawH: y
155
+ } = vI(t.width, t.height, i, e, u);
156
+ c.filter = `brightness(${s}) blur(8px)`, c.drawImage(t, w, D, A, y), c.filter = "none";
157
+ }
158
+ c.drawImage(I, n, G, d, h);
159
+ const W = BI(C, a);
160
+ c.clearRect(0, 0, i, e), c.fillStyle = "black", c.fillRect(0, 0, i, e), c.filter = `brightness(${s}) blur(8px)`, c.drawImage(I, n, G, d, h), c.filter = "none";
161
+ const x = BI(C, a);
162
+ c.clearRect(0, 0, i, e), c.fillStyle = "black", c.fillRect(0, 0, i, e), c.fillStyle = "white", c.fillRect(
163
+ Math.max(0, n),
164
+ Math.max(0, G),
165
+ Math.min(d, i - Math.max(0, n)),
166
+ Math.min(h, e - Math.max(0, G))
166
167
  );
167
- const K = rI(C, r);
168
- return { backgroundTex: v, obstacleTex: Y, coverageTex: K };
168
+ const F = BI(C, a);
169
+ return { backgroundTex: W, obstacleTex: x, coverageTex: F };
169
170
  }
170
- function rI(C, I) {
171
+ function BI(C, I) {
171
172
  const i = C.createTexture();
172
173
  return C.bindTexture(C.TEXTURE_2D, i), C.pixelStorei(C.UNPACK_FLIP_Y_WEBGL, !0), C.texImage2D(C.TEXTURE_2D, 0, C.RGBA, C.RGBA, C.UNSIGNED_BYTE, I), C.texParameteri(C.TEXTURE_2D, C.TEXTURE_MIN_FILTER, C.LINEAR), C.texParameteri(C.TEXTURE_2D, C.TEXTURE_MAG_FILTER, C.LINEAR), C.texParameteri(C.TEXTURE_2D, C.TEXTURE_WRAP_S, C.CLAMP_TO_EDGE), C.texParameteri(C.TEXTURE_2D, C.TEXTURE_WRAP_T, C.CLAMP_TO_EDGE), i;
173
174
  }
174
- async function XI(C) {
175
+ async function kI(C) {
175
176
  const I = await fetch(C);
176
177
  if (!I.ok)
177
178
  throw new Error(`Failed to fetch image: ${C} (${I.status})`);
178
179
  const i = await I.blob();
179
180
  return createImageBitmap(i);
180
181
  }
181
- const P = (
182
+ const II = (
182
183
  /* glsl */
183
184
  `
184
185
  precision highp float;
@@ -195,7 +196,7 @@ const P = (
195
196
  gl_Position = vec4(aPosition, 0.0, 1.0);
196
197
  }
197
198
  `
198
- ), Ig = (
199
+ ), sg = (
199
200
  /* glsl */
200
201
  `
201
202
  precision highp float;
@@ -212,7 +213,7 @@ const P = (
212
213
  gl_FragColor = dissipation * texture2D(uSource, coord);
213
214
  }
214
215
  `
215
- ), gg = (
216
+ ), lg = (
216
217
  /* glsl */
217
218
  `
218
219
  precision highp float;
@@ -227,7 +228,7 @@ const P = (
227
228
  gl_FragColor = vec4(0.5 * (R - L + T - B), 0.0, 0.0, 1.0);
228
229
  }
229
230
  `
230
- ), ig = (
231
+ ), og = (
231
232
  /* glsl */
232
233
  `
233
234
  precision highp float;
@@ -245,7 +246,7 @@ const P = (
245
246
  gl_FragColor = vec4((L + R + B + T - div) * 0.25, 0.0, 0.0, 1.0);
246
247
  }
247
248
  `
248
- ), Cg = (
249
+ ), tg = (
249
250
  /* glsl */
250
251
  `
251
252
  precision highp float;
@@ -263,7 +264,7 @@ const P = (
263
264
  gl_FragColor = vec4(vel, 0.0, 1.0);
264
265
  }
265
266
  `
266
- ), eg = (
267
+ ), cg = (
267
268
  /* glsl */
268
269
  `
269
270
  precision highp float;
@@ -280,7 +281,7 @@ const P = (
280
281
  gl_FragColor = vec4(texture2D(uTarget, vUv).xyz + splat, 1.0);
281
282
  }
282
283
  `
283
- ), lg = (
284
+ ), ag = (
284
285
  /* glsl */
285
286
  `
286
287
  precision highp float;
@@ -294,7 +295,7 @@ const P = (
294
295
  gl_FragColor = vec4(0.5 * (R - L - T + B), 0.0, 0.0, 1.0);
295
296
  }
296
297
  `
297
- ), sg = (
298
+ ), dg = (
298
299
  /* glsl */
299
300
  `
300
301
  precision highp float;
@@ -315,7 +316,7 @@ const P = (
315
316
  gl_FragColor = vec4(texture2D(uVelocity, vUv).xy + force * dt, 0.0, 1.0);
316
317
  }
317
318
  `
318
- ), og = (
319
+ ), Ag = (
319
320
  /* glsl */
320
321
  `
321
322
  precision highp float;
@@ -416,37 +417,37 @@ const P = (
416
417
  }
417
418
  `
418
419
  );
419
- function tg(C) {
420
+ function mg(C) {
420
421
  const I = { alpha: !0, depth: !1, stencil: !1, antialias: !0, preserveDrawingBuffer: !1 };
421
422
  let i = C.getContext("webgl2", I);
422
423
  const e = !!i;
423
424
  e || (i = C.getContext("webgl", I), i.getExtension("EXT_color_buffer_half_float"));
424
- const l = e ? null : i.getExtension("OES_texture_half_float"), s = e ? i.HALF_FLOAT : l.HALF_FLOAT_OES;
425
+ const s = e ? null : i.getExtension("OES_texture_half_float"), o = e ? i.HALF_FLOAT : s.HALF_FLOAT_OES;
425
426
  return i.getExtension("EXT_color_buffer_float"), i.getExtension("OES_texture_half_float_linear"), {
426
427
  gl: i,
427
428
  isWebGL2: e,
428
429
  ext: {
429
430
  internalFormat: e ? i.RGBA16F : i.RGBA,
430
431
  format: i.RGBA,
431
- type: s
432
+ type: o
432
433
  }
433
434
  };
434
435
  }
435
- class _ {
436
+ class gI {
436
437
  constructor(I, i, e) {
437
- BI(this, "program");
438
- BI(this, "uniforms", {});
439
- BI(this, "_gl");
438
+ yI(this, "program");
439
+ yI(this, "uniforms", {});
440
+ yI(this, "_gl");
440
441
  this._gl = I, this.program = I.createProgram(), I.attachShader(this.program, this._compile(I.VERTEX_SHADER, i)), I.attachShader(this.program, this._compile(I.FRAGMENT_SHADER, e)), I.linkProgram(this.program);
441
- const l = I.getProgramParameter(this.program, I.ACTIVE_UNIFORMS);
442
- for (let s = 0; s < l; s++) {
443
- const t = I.getActiveUniform(this.program, s).name;
442
+ const s = I.getProgramParameter(this.program, I.ACTIVE_UNIFORMS);
443
+ for (let o = 0; o < s; o++) {
444
+ const t = I.getActiveUniform(this.program, o).name;
444
445
  this.uniforms[t] = I.getUniformLocation(this.program, t);
445
446
  }
446
447
  }
447
448
  _compile(I, i) {
448
- const e = this._gl, l = e.createShader(I);
449
- return e.shaderSource(l, i), e.compileShader(l), l;
449
+ const e = this._gl, s = e.createShader(I);
450
+ return e.shaderSource(s, i), e.compileShader(s), s;
450
451
  }
451
452
  bind() {
452
453
  this._gl.useProgram(this.program);
@@ -455,117 +456,119 @@ class _ {
455
456
  this._gl.deleteProgram(this.program);
456
457
  }
457
458
  }
458
- function cg(C) {
459
+ function rg(C) {
459
460
  return {
460
- advection: new _(C, P, Ig),
461
- divergence: new _(C, P, gg),
462
- pressure: new _(C, P, ig),
463
- gradientSubtract: new _(C, P, Cg),
464
- splat: new _(C, P, eg),
465
- curl: new _(C, P, lg),
466
- vorticity: new _(C, P, sg),
467
- display: new _(C, P, og)
461
+ advection: new gI(C, II, sg),
462
+ divergence: new gI(C, II, lg),
463
+ pressure: new gI(C, II, og),
464
+ gradientSubtract: new gI(C, II, tg),
465
+ splat: new gI(C, II, cg),
466
+ curl: new gI(C, II, ag),
467
+ vorticity: new gI(C, II, dg),
468
+ display: new gI(C, II, Ag)
468
469
  };
469
470
  }
470
- function VI(C, I, i, e) {
471
+ function KI(C, I, i, e) {
471
472
  C.activeTexture(C.TEXTURE0);
472
- const l = C.createTexture();
473
- C.bindTexture(C.TEXTURE_2D, l), C.texParameteri(C.TEXTURE_2D, C.TEXTURE_MIN_FILTER, C.LINEAR), C.texParameteri(C.TEXTURE_2D, C.TEXTURE_MAG_FILTER, C.LINEAR), C.texParameteri(C.TEXTURE_2D, C.TEXTURE_WRAP_S, C.CLAMP_TO_EDGE), C.texParameteri(C.TEXTURE_2D, C.TEXTURE_WRAP_T, C.CLAMP_TO_EDGE), C.texImage2D(C.TEXTURE_2D, 0, I.internalFormat, i, e, 0, I.format, I.type, null);
474
- const s = C.createFramebuffer();
475
- return C.bindFramebuffer(C.FRAMEBUFFER, s), C.framebufferTexture2D(C.FRAMEBUFFER, C.COLOR_ATTACHMENT0, C.TEXTURE_2D, l, 0), { tex: l, fbo: s, width: i, height: e };
473
+ const s = C.createTexture();
474
+ C.bindTexture(C.TEXTURE_2D, s), C.texParameteri(C.TEXTURE_2D, C.TEXTURE_MIN_FILTER, C.LINEAR), C.texParameteri(C.TEXTURE_2D, C.TEXTURE_MAG_FILTER, C.LINEAR), C.texParameteri(C.TEXTURE_2D, C.TEXTURE_WRAP_S, C.CLAMP_TO_EDGE), C.texParameteri(C.TEXTURE_2D, C.TEXTURE_WRAP_T, C.CLAMP_TO_EDGE), C.texImage2D(C.TEXTURE_2D, 0, I.internalFormat, i, e, 0, I.format, I.type, null);
475
+ const o = C.createFramebuffer();
476
+ return C.bindFramebuffer(C.FRAMEBUFFER, o), C.framebufferTexture2D(C.FRAMEBUFFER, C.COLOR_ATTACHMENT0, C.TEXTURE_2D, s, 0), { tex: s, fbo: o, width: i, height: e };
476
477
  }
477
- function KI(C, I, i, e) {
478
- let l = VI(C, I, i, e), s = VI(C, I, i, e);
478
+ function WI(C, I, i, e) {
479
+ let s = KI(C, I, i, e), o = KI(C, I, i, e);
479
480
  return {
480
481
  get read() {
481
- return l;
482
+ return s;
482
483
  },
483
484
  get write() {
484
- return s;
485
+ return o;
485
486
  },
486
487
  swap() {
487
- [l, s] = [s, l];
488
+ [s, o] = [o, s];
488
489
  },
489
490
  dispose() {
490
- C.deleteTexture(l.tex), C.deleteFramebuffer(l.fbo), C.deleteTexture(s.tex), C.deleteFramebuffer(s.fbo);
491
+ C.deleteTexture(s.tex), C.deleteFramebuffer(s.fbo), C.deleteTexture(o.tex), C.deleteFramebuffer(o.fbo);
491
492
  }
492
493
  };
493
494
  }
494
- function dg(C) {
495
+ function ng(C) {
495
496
  const I = C.createBuffer();
496
497
  return C.bindBuffer(C.ARRAY_BUFFER, I), C.bufferData(C.ARRAY_BUFFER, new Float32Array([-1, -1, -1, 1, 1, 1, 1, -1]), C.STATIC_DRAW), C.vertexAttribPointer(0, 2, C.FLOAT, !1, 0, 0), C.enableVertexAttribArray(0), function(e) {
497
498
  C.bindFramebuffer(C.FRAMEBUFFER, e), C.drawArrays(C.TRIANGLE_FAN, 0, 4);
498
499
  };
499
500
  }
500
- const YI = typeof requestAnimationFrame < "u" ? requestAnimationFrame.bind(globalThis) : (C) => setTimeout(C, 1e3 / 60), ag = typeof cancelAnimationFrame < "u" ? cancelAnimationFrame.bind(globalThis) : clearTimeout, DI = 0.016, Ag = { standard: 0, glass: 1, ink: 2, aurora: 3, ripple: 4 };
501
- var iI, R, mI, CI, cI, y, W, $, II, f, H, Z, x, M, J, z, X, E, F, dI, eI, V, U, N, lI, GI, sI, AI, ZI, HI, gI, tI, oI, nI, hI, WI, SI, vI, pI, NI;
502
- class fI {
503
- constructor(I, i = {}) {
501
+ const xI = typeof requestAnimationFrame < "u" ? requestAnimationFrame.bind(globalThis) : (C) => setTimeout(C, 1e3 / 60), ug = typeof cancelAnimationFrame < "u" ? cancelAnimationFrame.bind(globalThis) : clearTimeout, MI = 0.016, Zg = { standard: 0, glass: 1, ink: 2, aurora: 3, ripple: 4 };
502
+ var lI, X, SI, oI, rI, K, R, CI, eI, k, nI, P, H, B, E, J, N, _, v, z, f, uI, tI, L, Y, Q, cI, bI, aI, GI, LI, YI, sI, AI, dI, hI, VI, FI, wI, DI, RI, jI;
503
+ class EI {
504
+ constructor(I, i = {}, e = {}) {
504
505
  // ---------------------------------------------------------------------------
505
506
  // Private helpers
506
507
  // ---------------------------------------------------------------------------
507
- n(this, sI);
508
- n(this, ZI);
509
- n(this, gI);
510
- n(this, oI);
511
- n(this, hI);
512
- n(this, SI);
513
- n(this, pI);
514
- n(this, iI, void 0);
515
- n(this, R, void 0);
516
- n(this, mI, void 0);
517
- n(this, CI, void 0);
518
- n(this, cI, void 0);
519
- n(this, y, 0);
520
- n(this, W, 0);
521
- n(this, $, 0);
522
- n(this, II, 0);
523
- n(this, f, 1);
524
- n(this, H, null);
525
- n(this, Z, null);
526
- n(this, x, null);
527
- n(this, M, null);
528
- n(this, J, null);
529
- n(this, z, null);
530
- n(this, X, null);
531
- n(this, E, null);
508
+ m(this, aI);
509
+ m(this, LI);
510
+ m(this, sI);
511
+ m(this, dI);
512
+ m(this, VI);
513
+ m(this, wI);
514
+ m(this, RI);
515
+ m(this, lI, void 0);
516
+ m(this, X, void 0);
517
+ m(this, SI, void 0);
518
+ m(this, oI, void 0);
519
+ m(this, rI, void 0);
520
+ m(this, K, 0);
521
+ m(this, R, 0);
522
+ m(this, CI, 0);
523
+ m(this, eI, 0);
524
+ m(this, k, 1);
525
+ m(this, nI, 1);
526
+ m(this, P, 0.5);
527
+ m(this, H, null);
528
+ m(this, B, null);
529
+ m(this, E, null);
530
+ m(this, J, null);
531
+ m(this, N, null);
532
+ m(this, _, null);
533
+ m(this, v, null);
534
+ m(this, z, null);
532
535
  // binary content mask for transparent canvas support
533
- n(this, F, null);
536
+ m(this, f, null);
534
537
  // optional background image (from backgroundSrc prop)
535
- n(this, dI, "cover");
536
- n(this, eI, void 0);
537
- n(this, V, { x: 0, y: 0, dx: 0, dy: 0, targetX: 0, targetY: 0, moved: !1 });
538
+ m(this, uI, "cover");
539
+ m(this, tI, void 0);
540
+ m(this, L, { x: 0, y: 0, dx: 0, dy: 0, targetX: 0, targetY: 0, moved: !1 });
538
541
  // Stores source so textures can be rebuilt on resize
539
- n(this, U, null);
540
- n(this, N, null);
541
- n(this, lI, !1);
542
- n(this, GI, !1);
543
- o(this, iI, I), o(this, eI, uI(i));
544
- const { gl: e, ext: l } = tg(I);
545
- o(this, R, e), o(this, mI, l), o(this, CI, cg(e)), o(this, cI, dg(e)), e.clearColor(0, 0, 0, 0);
542
+ m(this, Y, null);
543
+ m(this, Q, null);
544
+ m(this, cI, !1);
545
+ m(this, bI, !1);
546
+ l(this, lI, I), l(this, nI, Math.max(0.1, Math.min(1, e.dpr ?? 1))), l(this, P, Math.max(0.1, Math.min(1, e.sim ?? 0.5))), l(this, tI, pI(i));
547
+ const { gl: s, ext: o } = mg(I);
548
+ l(this, X, s), l(this, SI, o), l(this, oI, rg(s)), l(this, rI, ng(s)), s.clearColor(0, 0, 0, 0);
546
549
  }
547
550
  // ---------------------------------------------------------------------------
548
551
  // Public API
549
552
  // ---------------------------------------------------------------------------
550
553
  setTextSource(I) {
551
- o(this, U, { type: "text", opts: I }), b(this, sI, AI).call(this), b(this, gI, tI).call(this), b(this, oI, nI).call(this);
554
+ l(this, Y, { type: "text", opts: I }), b(this, aI, GI).call(this), b(this, sI, AI).call(this), b(this, dI, hI).call(this);
552
555
  }
553
556
  async setImageSource(I, i = 0, e = "cover") {
554
- const l = await XI(I);
555
- if (g(this, GI)) {
556
- l.close();
557
+ const s = await kI(I);
558
+ if (g(this, bI)) {
559
+ s.close();
557
560
  return;
558
561
  }
559
- o(this, U, { type: "image", bitmap: l, effect: i, size: e }), b(this, sI, AI).call(this), b(this, gI, tI).call(this), b(this, oI, nI).call(this);
562
+ l(this, Y, { type: "image", bitmap: s, effect: i, size: e }), b(this, aI, GI).call(this), b(this, sI, AI).call(this), b(this, dI, hI).call(this);
560
563
  }
561
564
  setImageBitmap(I, i = 0, e = "cover") {
562
- o(this, U, { type: "image", bitmap: I, effect: i, size: e }), b(this, sI, AI).call(this), b(this, gI, tI).call(this), b(this, oI, nI).call(this);
565
+ l(this, Y, { type: "image", bitmap: I, effect: i, size: e }), b(this, aI, GI).call(this), b(this, sI, AI).call(this), b(this, dI, hI).call(this);
563
566
  }
564
567
  setBackground(I, i = "cover") {
565
- g(this, F) && g(this, F) !== I && g(this, F).close(), o(this, F, I), o(this, dI, i ?? "cover"), g(this, U) && g(this, y) > 0 && b(this, gI, tI).call(this);
568
+ g(this, f) && g(this, f) !== I && g(this, f).close(), l(this, f, I), l(this, uI, i ?? "cover"), g(this, Y) && g(this, K) > 0 && g(this, R) > 0 && b(this, sI, AI).call(this);
566
569
  }
567
570
  handleMove(I, i, e = 1) {
568
- g(this, V).moved = !0, g(this, V).dx = (I - g(this, V).targetX) * e, g(this, V).dy = (i - g(this, V).targetY) * e, g(this, V).targetX = I, g(this, V).targetY = i;
571
+ g(this, L).moved = !0, g(this, L).dx = (I - g(this, L).targetX) * e, g(this, L).dy = (i - g(this, L).targetY) * e, g(this, L).targetX = I, g(this, L).targetY = i;
569
572
  }
570
573
  /**
571
574
  * Immediately applies one fluid splat at (x, y) with explicit velocity (vx, vy).
@@ -574,29 +577,32 @@ class fI {
574
577
  * where you want N independent injection points per frame without flooding the
575
578
  * mouse-state machine or the worker message queue.
576
579
  */
577
- splat(I, i, e, l, s = 1) {
578
- if (!g(this, lI) || g(this, y) === 0)
580
+ splat(I, i, e, s, o = 1) {
581
+ if (!g(this, cI) || g(this, K) === 0)
579
582
  return;
580
- const t = g(this, R), m = g(this, eI), { splat: r } = g(this, CI), a = g(this, cI);
581
- t.viewport(0, 0, g(this, $), g(this, II)), r.bind(), t.uniform1f(r.uniforms.aspectRatio, g(this, y) / g(this, W)), t.uniform2f(r.uniforms.point, I * g(this, f) / g(this, y), 1 - i * g(this, f) / g(this, W)), t.uniform1f(r.uniforms.radius, m.splatRadius), t.uniform1i(r.uniforms.uTarget, 0), t.activeTexture(t.TEXTURE0), t.bindTexture(t.TEXTURE_2D, g(this, Z).read.tex), t.uniform3f(r.uniforms.color, e * m.splatForce * s, -l * m.splatForce * s, 0), a(g(this, Z).write.fbo), g(this, Z).swap(), t.activeTexture(t.TEXTURE0), t.bindTexture(t.TEXTURE_2D, g(this, H).read.tex), t.uniform3f(r.uniforms.color, s, s, s), a(g(this, H).write.fbo), g(this, H).swap();
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();
585
+ }
586
+ updateQuality(I) {
587
+ I.dpr !== void 0 && l(this, nI, Math.max(0.1, Math.min(1, I.dpr))), I.sim !== void 0 && l(this, P, Math.max(0.1, Math.min(1, I.sim)));
582
588
  }
583
589
  resize(I, i, e) {
584
- if (e !== void 0 ? o(this, f, e) : typeof window < "u" && window.devicePixelRatio && o(this, f, window.devicePixelRatio), I !== void 0 && I > 0) {
590
+ if (e !== void 0 ? l(this, k, e) : typeof window < "u" && window.devicePixelRatio && l(this, k, window.devicePixelRatio), I !== void 0 && I > 0) {
585
591
  if (i === void 0 || i <= 0)
586
592
  return;
587
- o(this, y, g(this, iI).width = I), o(this, W, g(this, iI).height = i), o(this, $, I >> 1), o(this, II, i >> 1), b(this, ZI, HI).call(this);
593
+ l(this, K, g(this, lI).width = I), l(this, R, g(this, lI).height = i), l(this, CI, Math.max(1, Math.round(I * g(this, P)))), l(this, eI, Math.max(1, Math.round(i * g(this, P)))), b(this, LI, YI).call(this);
588
594
  } else
589
- b(this, sI, AI).call(this);
590
- g(this, U) && b(this, gI, tI).call(this), b(this, oI, nI).call(this);
595
+ b(this, aI, GI).call(this);
596
+ g(this, Y) && b(this, sI, AI).call(this), b(this, dI, hI).call(this);
591
597
  }
592
598
  updateConfig(I) {
593
- Object.assign(g(this, eI), I);
599
+ Object.assign(g(this, tI), I);
594
600
  }
595
601
  destroy() {
596
- o(this, GI, !0), this.stop();
597
- const I = g(this, R);
598
- b(this, hI, WI).call(this), b(this, SI, vI).call(this), g(this, F) && (g(this, F).close(), o(this, F, null));
599
- for (const e of Object.values(g(this, CI)))
602
+ l(this, bI, !0), this.stop();
603
+ const I = g(this, X);
604
+ b(this, VI, FI).call(this), b(this, wI, DI).call(this), g(this, f) && (g(this, f).close(), l(this, f, null));
605
+ for (const e of Object.values(g(this, oI)))
600
606
  e.dispose();
601
607
  const i = I.getExtension("WEBGL_lose_context");
602
608
  i == null || i.loseContext();
@@ -605,118 +611,123 @@ class fI {
605
611
  // Loop control
606
612
  // ---------------------------------------------------------------------------
607
613
  start() {
608
- if (g(this, N) !== null)
614
+ if (g(this, Q) !== null)
609
615
  return;
610
616
  const I = () => {
611
- b(this, pI, NI).call(this), o(this, N, YI(I));
617
+ b(this, RI, jI).call(this), l(this, Q, xI(I));
612
618
  };
613
- o(this, N, YI(I));
619
+ l(this, Q, xI(I));
614
620
  }
615
621
  stop() {
616
- g(this, N) !== null && (ag(g(this, N)), o(this, N, null));
622
+ g(this, Q) !== null && (ug(g(this, Q)), l(this, Q, null));
617
623
  }
618
624
  get isRunning() {
619
- return g(this, N) !== null;
625
+ return g(this, Q) !== null;
620
626
  }
621
627
  }
622
- iI = new WeakMap(), R = new WeakMap(), mI = new WeakMap(), CI = new WeakMap(), cI = new WeakMap(), y = new WeakMap(), W = new WeakMap(), $ = new WeakMap(), II = new WeakMap(), f = new WeakMap(), H = new WeakMap(), Z = new WeakMap(), x = new WeakMap(), M = new WeakMap(), J = new WeakMap(), z = new WeakMap(), X = new WeakMap(), E = new WeakMap(), F = new WeakMap(), dI = new WeakMap(), eI = new WeakMap(), V = new WeakMap(), U = new WeakMap(), N = new WeakMap(), lI = new WeakMap(), GI = new WeakMap(), sI = new WeakSet(), AI = function() {
623
- const I = g(this, iI);
624
- "clientWidth" in I && I.clientWidth > 0 ? (o(this, f, typeof window < "u" && window.devicePixelRatio || 1), o(this, y, I.width = Math.round(I.clientWidth * g(this, f))), o(this, W, I.height = Math.round(I.clientHeight * g(this, f)))) : (o(this, y, I.width), o(this, W, I.height)), !(g(this, y) === 0 || g(this, W) === 0) && (o(this, $, g(this, y) >> 1), o(this, II, g(this, W) >> 1), b(this, ZI, HI).call(this));
625
- }, ZI = new WeakSet(), HI = function() {
626
- const I = g(this, R), i = g(this, mI), e = g(this, $), l = g(this, II);
627
- b(this, hI, WI).call(this), o(this, H, KI(I, i, e, l)), o(this, Z, KI(I, i, e, l)), o(this, M, KI(I, i, e, l)), o(this, x, VI(I, i, e, l)), o(this, J, VI(I, i, e, l));
628
- }, gI = new WeakSet(), tI = function() {
629
- if (!(!g(this, U) || g(this, y) === 0)) {
630
- if (b(this, SI, vI).call(this), g(this, U).type === "text") {
631
- const { backgroundTex: I, obstacleTex: i, coverageTex: e } = qI(
628
+ lI = new WeakMap(), X = new WeakMap(), SI = new WeakMap(), oI = new WeakMap(), rI = new WeakMap(), K = new WeakMap(), R = new WeakMap(), CI = new WeakMap(), eI = new WeakMap(), k = new WeakMap(), nI = new WeakMap(), P = new WeakMap(), H = new WeakMap(), B = new WeakMap(), E = new WeakMap(), J = new WeakMap(), N = new WeakMap(), _ = new WeakMap(), v = new WeakMap(), z = new WeakMap(), f = new WeakMap(), uI = new WeakMap(), tI = new WeakMap(), L = new WeakMap(), Y = new WeakMap(), Q = new WeakMap(), cI = new WeakMap(), bI = new WeakMap(), aI = new WeakSet(), GI = function() {
629
+ const I = g(this, lI);
630
+ "clientWidth" in I && I.clientWidth > 0 ? (l(this, k, (typeof window < "u" && window.devicePixelRatio || 1) * g(this, nI)), l(this, K, I.width = Math.round(I.clientWidth * g(this, k))), l(this, R, I.height = Math.round(I.clientHeight * g(this, k)))) : (l(this, K, I.width), l(this, R, I.height)), !(g(this, K) === 0 || g(this, R) === 0) && (l(this, CI, Math.max(1, Math.round(g(this, K) * g(this, P)))), l(this, eI, Math.max(1, Math.round(g(this, R) * g(this, P)))), b(this, LI, YI).call(this));
631
+ }, LI = new WeakSet(), YI = function() {
632
+ const I = g(this, X), i = g(this, SI), e = g(this, CI), s = g(this, eI);
633
+ b(this, VI, FI).call(this), l(this, H, WI(I, i, e, s)), l(this, B, WI(I, i, e, s)), l(this, J, WI(I, i, e, s)), l(this, E, KI(I, i, e, s)), l(this, N, KI(I, i, e, s));
634
+ }, sI = new WeakSet(), AI = function() {
635
+ if (!(!g(this, Y) || g(this, K) === 0 || g(this, R) === 0)) {
636
+ if (b(this, wI, DI).call(this), g(this, Y).type === "text") {
637
+ const { backgroundTex: I, obstacleTex: i, coverageTex: e } = Cg(
638
+ g(this, X),
639
+ g(this, K),
632
640
  g(this, R),
633
- g(this, y),
634
- g(this, W),
635
- g(this, U).opts,
636
- g(this, F),
637
- g(this, dI)
641
+ g(this, Y).opts,
642
+ g(this, f),
643
+ g(this, uI)
638
644
  );
639
- o(this, z, I), o(this, X, i), o(this, E, e);
645
+ l(this, _, I), l(this, v, i), l(this, z, e);
640
646
  } else {
641
- const { backgroundTex: I, obstacleTex: i, coverageTex: e } = $I(
647
+ const { backgroundTex: I, obstacleTex: i, coverageTex: e } = eg(
648
+ g(this, X),
649
+ g(this, Y).bitmap,
650
+ g(this, K),
642
651
  g(this, R),
643
- g(this, U).bitmap,
644
- g(this, y),
645
- g(this, W),
646
- g(this, U).effect,
647
- g(this, U).size,
648
- g(this, F),
649
- g(this, dI)
652
+ g(this, Y).effect,
653
+ g(this, Y).size,
654
+ g(this, f),
655
+ g(this, uI)
650
656
  );
651
- o(this, z, I), o(this, X, i), o(this, E, e);
657
+ l(this, _, I), l(this, v, i), l(this, z, e);
652
658
  }
653
- o(this, lI, !0);
659
+ l(this, cI, !0);
654
660
  }
655
- }, oI = new WeakSet(), nI = function() {
656
- g(this, lI) && !this.isRunning && this.start();
657
- }, hI = new WeakSet(), WI = function() {
661
+ }, dI = new WeakSet(), hI = function() {
662
+ g(this, cI) && !this.isRunning && this.start();
663
+ }, VI = new WeakSet(), FI = function() {
658
664
  var I, i, e;
659
- (I = g(this, H)) == null || I.dispose(), (i = g(this, Z)) == null || i.dispose(), (e = g(this, M)) == null || e.dispose(), g(this, x) && (g(this, R).deleteTexture(g(this, x).tex), g(this, R).deleteFramebuffer(g(this, x).fbo)), g(this, J) && (g(this, R).deleteTexture(g(this, J).tex), g(this, R).deleteFramebuffer(g(this, J).fbo)), o(this, H, o(this, Z, o(this, M, o(this, x, o(this, J, null)))));
660
- }, SI = new WeakSet(), vI = function() {
661
- g(this, z) && g(this, R).deleteTexture(g(this, z)), g(this, X) && g(this, R).deleteTexture(g(this, X)), g(this, E) && g(this, E) !== g(this, X) && g(this, R).deleteTexture(g(this, E)), o(this, z, o(this, X, o(this, E, null)));
662
- }, pI = new WeakSet(), NI = function() {
663
- if (!g(this, lI) || g(this, y) === 0 || !g(this, H) || !g(this, Z))
665
+ (I = g(this, H)) == null || I.dispose(), (i = g(this, B)) == null || i.dispose(), (e = g(this, J)) == null || e.dispose(), g(this, E) && (g(this, X).deleteTexture(g(this, E).tex), g(this, X).deleteFramebuffer(g(this, E).fbo)), g(this, N) && (g(this, X).deleteTexture(g(this, N).tex), g(this, X).deleteFramebuffer(g(this, N).fbo)), l(this, H, l(this, B, l(this, J, l(this, E, l(this, N, null)))));
666
+ }, wI = new WeakSet(), DI = function() {
667
+ g(this, _) && g(this, X).deleteTexture(g(this, _)), g(this, v) && g(this, X).deleteTexture(g(this, v)), g(this, z) && g(this, z) !== g(this, v) && g(this, X).deleteTexture(g(this, z)), l(this, _, l(this, v, l(this, z, null)));
668
+ }, RI = new WeakSet(), jI = function() {
669
+ if (!g(this, cI) || g(this, K) === 0 || !g(this, H) || !g(this, B))
664
670
  return;
665
- const I = g(this, R), i = g(this, eI), { advection: e, divergence: l, pressure: s, gradientSubtract: t, splat: m, curl: r, vorticity: a, display: u } = g(this, CI);
666
- g(this, V).x += (g(this, V).targetX - g(this, V).x) * 0.15, g(this, V).y += (g(this, V).targetY - g(this, V).y) * 0.15;
667
- const S = g(this, $), A = g(this, II), h = g(this, cI);
668
- I.viewport(0, 0, S, A), e.bind(), I.uniform2f(e.uniforms.texelSize, 1 / S, 1 / A), I.uniform1f(e.uniforms.dt, DI), I.uniform1i(e.uniforms.uObstacle, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, X)), I.uniform1f(e.uniforms.dissipation, i.velocityDissipation), I.uniform1i(e.uniforms.uVelocity, 1), I.activeTexture(I.TEXTURE1), I.bindTexture(I.TEXTURE_2D, g(this, Z).read.tex), I.uniform1i(e.uniforms.uSource, 1), h(g(this, Z).write.fbo), g(this, Z).swap(), I.uniform1f(e.uniforms.dissipation, i.densityDissipation), I.uniform1i(e.uniforms.uSource, 2), I.activeTexture(I.TEXTURE2), I.bindTexture(I.TEXTURE_2D, g(this, H).read.tex), h(g(this, H).write.fbo), g(this, H).swap(), r.bind(), I.uniform2f(r.uniforms.texelSize, 1 / S, 1 / A), I.uniform1i(r.uniforms.uVelocity, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, Z).read.tex), h(g(this, J).fbo), a.bind(), I.uniform2f(a.uniforms.texelSize, 1 / S, 1 / A), I.uniform1f(a.uniforms.curl, i.curl), I.uniform1f(a.uniforms.dt, DI), I.uniform1i(a.uniforms.uVelocity, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, Z).read.tex), I.uniform1i(a.uniforms.uCurl, 1), I.activeTexture(I.TEXTURE1), I.bindTexture(I.TEXTURE_2D, g(this, J).tex), h(g(this, Z).write.fbo), g(this, Z).swap(), g(this, V).moved && (m.bind(), I.uniform1f(m.uniforms.aspectRatio, g(this, y) / g(this, W)), I.uniform2f(m.uniforms.point, g(this, V).x * g(this, f) / g(this, y), 1 - g(this, V).y * g(this, f) / g(this, W)), I.uniform1f(m.uniforms.radius, i.splatRadius), I.uniform1i(m.uniforms.uTarget, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, Z).read.tex), I.uniform3f(m.uniforms.color, g(this, V).dx * i.splatForce, -g(this, V).dy * i.splatForce, 0), h(g(this, Z).write.fbo), g(this, Z).swap(), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, H).read.tex), I.uniform3f(m.uniforms.color, 1, 1, 1), h(g(this, H).write.fbo), g(this, H).swap(), g(this, V).moved = !1), l.bind(), I.uniform2f(l.uniforms.texelSize, 1 / S, 1 / A), I.uniform1i(l.uniforms.uVelocity, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, Z).read.tex), I.uniform1i(l.uniforms.uObstacle, 1), I.activeTexture(I.TEXTURE1), I.bindTexture(I.TEXTURE_2D, g(this, X)), h(g(this, x).fbo), s.bind(), I.uniform2f(s.uniforms.texelSize, 1 / S, 1 / A), I.uniform1i(s.uniforms.uDivergence, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, x).tex), I.uniform1i(s.uniforms.uObstacle, 1), I.activeTexture(I.TEXTURE1), I.bindTexture(I.TEXTURE_2D, g(this, X));
669
- for (let v = 0; v < i.pressureIterations; v++)
670
- I.uniform1i(s.uniforms.uPressure, 2), I.activeTexture(I.TEXTURE2), I.bindTexture(I.TEXTURE_2D, g(this, M).read.tex), h(g(this, M).write.fbo), g(this, M).swap();
671
- t.bind(), I.uniform2f(t.uniforms.texelSize, 1 / S, 1 / A), I.uniform1i(t.uniforms.uPressure, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, M).read.tex), I.uniform1i(t.uniforms.uVelocity, 1), I.activeTexture(I.TEXTURE1), I.bindTexture(I.TEXTURE_2D, g(this, Z).read.tex), I.uniform1i(t.uniforms.uObstacle, 2), I.activeTexture(I.TEXTURE2), I.bindTexture(I.TEXTURE_2D, g(this, X)), h(g(this, Z).write.fbo), g(this, Z).swap(), I.viewport(0, 0, g(this, y), g(this, W)), I.bindFramebuffer(I.FRAMEBUFFER, null), I.clear(I.COLOR_BUFFER_BIT), u.bind(), I.uniform2f(u.uniforms.texelSize, 1 / g(this, y), 1 / g(this, W)), I.uniform3fv(u.uniforms.uWaterColor, i.waterColor), I.uniform3fv(u.uniforms.uGlowColor, i.glowColor), I.uniform1f(u.uniforms.uRefraction, i.refraction), I.uniform1f(u.uniforms.uSpecularExp, i.specularExp), I.uniform1f(u.uniforms.uShine, i.shine), I.uniform1f(u.uniforms.uWarpStrength, i.warpStrength ?? 0.015), I.uniform1i(u.uniforms.uAlgorithm, Ag[i.algorithm] ?? 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, H).read.tex), I.activeTexture(I.TEXTURE1), I.bindTexture(I.TEXTURE_2D, g(this, X)), I.activeTexture(I.TEXTURE2), I.bindTexture(I.TEXTURE_2D, g(this, z)), I.activeTexture(I.TEXTURE3), I.bindTexture(I.TEXTURE_2D, g(this, E)), I.activeTexture(I.TEXTURE4), I.bindTexture(I.TEXTURE_2D, g(this, Z).read.tex), I.uniform1i(u.uniforms.uTexture, 0), I.uniform1i(u.uniforms.uObstacle, 1), I.uniform1i(u.uniforms.uBackground, 2), I.uniform1i(u.uniforms.uCoverage, 3), I.uniform1i(u.uniforms.uVelocity, 4), h(null);
671
+ const I = g(this, X), i = g(this, tI), { advection: e, divergence: s, pressure: o, gradientSubtract: t, splat: u, curl: a, vorticity: c, display: n } = g(this, oI);
672
+ g(this, L).x += (g(this, L).targetX - g(this, L).x) * 0.15, g(this, L).y += (g(this, L).targetY - g(this, L).y) * 0.15;
673
+ const G = g(this, CI), d = g(this, eI), h = g(this, rI);
674
+ I.viewport(0, 0, G, d), e.bind(), I.uniform2f(e.uniforms.texelSize, 1 / G, 1 / d), I.uniform1f(e.uniforms.dt, MI), I.uniform1i(e.uniforms.uObstacle, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, v)), I.uniform1f(e.uniforms.dissipation, i.velocityDissipation), I.uniform1i(e.uniforms.uVelocity, 1), I.activeTexture(I.TEXTURE1), I.bindTexture(I.TEXTURE_2D, g(this, B).read.tex), I.uniform1i(e.uniforms.uSource, 1), h(g(this, B).write.fbo), g(this, B).swap(), I.uniform1f(e.uniforms.dissipation, i.densityDissipation), I.uniform1i(e.uniforms.uSource, 2), I.activeTexture(I.TEXTURE2), I.bindTexture(I.TEXTURE_2D, g(this, H).read.tex), h(g(this, H).write.fbo), g(this, H).swap(), a.bind(), I.uniform2f(a.uniforms.texelSize, 1 / G, 1 / d), I.uniform1i(a.uniforms.uVelocity, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, B).read.tex), h(g(this, N).fbo), c.bind(), I.uniform2f(c.uniforms.texelSize, 1 / G, 1 / d), I.uniform1f(c.uniforms.curl, i.curl), I.uniform1f(c.uniforms.dt, MI), I.uniform1i(c.uniforms.uVelocity, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, B).read.tex), I.uniform1i(c.uniforms.uCurl, 1), I.activeTexture(I.TEXTURE1), I.bindTexture(I.TEXTURE_2D, g(this, N).tex), h(g(this, B).write.fbo), g(this, B).swap(), g(this, L).moved && (u.bind(), I.uniform1f(u.uniforms.aspectRatio, g(this, K) / g(this, R)), I.uniform2f(u.uniforms.point, g(this, L).x * g(this, k) / g(this, K), 1 - g(this, L).y * g(this, k) / g(this, R)), I.uniform1f(u.uniforms.radius, i.splatRadius), I.uniform1i(u.uniforms.uTarget, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, B).read.tex), I.uniform3f(u.uniforms.color, g(this, L).dx * i.splatForce, -g(this, L).dy * i.splatForce, 0), h(g(this, B).write.fbo), g(this, B).swap(), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, H).read.tex), I.uniform3f(u.uniforms.color, 1, 1, 1), h(g(this, H).write.fbo), g(this, H).swap(), g(this, L).moved = !1), s.bind(), I.uniform2f(s.uniforms.texelSize, 1 / G, 1 / d), I.uniform1i(s.uniforms.uVelocity, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, B).read.tex), I.uniform1i(s.uniforms.uObstacle, 1), I.activeTexture(I.TEXTURE1), I.bindTexture(I.TEXTURE_2D, g(this, v)), h(g(this, E).fbo), o.bind(), I.uniform2f(o.uniforms.texelSize, 1 / G, 1 / d), I.uniform1i(o.uniforms.uDivergence, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, E).tex), I.uniform1i(o.uniforms.uObstacle, 1), I.activeTexture(I.TEXTURE1), I.bindTexture(I.TEXTURE_2D, g(this, v));
675
+ for (let W = 0; W < i.pressureIterations; W++)
676
+ I.uniform1i(o.uniforms.uPressure, 2), I.activeTexture(I.TEXTURE2), I.bindTexture(I.TEXTURE_2D, g(this, J).read.tex), h(g(this, J).write.fbo), g(this, J).swap();
677
+ t.bind(), I.uniform2f(t.uniforms.texelSize, 1 / G, 1 / d), I.uniform1i(t.uniforms.uPressure, 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, J).read.tex), I.uniform1i(t.uniforms.uVelocity, 1), I.activeTexture(I.TEXTURE1), I.bindTexture(I.TEXTURE_2D, g(this, B).read.tex), I.uniform1i(t.uniforms.uObstacle, 2), I.activeTexture(I.TEXTURE2), I.bindTexture(I.TEXTURE_2D, g(this, v)), h(g(this, B).write.fbo), g(this, B).swap(), I.viewport(0, 0, g(this, K), g(this, R)), I.bindFramebuffer(I.FRAMEBUFFER, null), I.clear(I.COLOR_BUFFER_BIT), n.bind(), I.uniform2f(n.uniforms.texelSize, 1 / g(this, K), 1 / g(this, R)), I.uniform3fv(n.uniforms.uWaterColor, i.waterColor), I.uniform3fv(n.uniforms.uGlowColor, i.glowColor), I.uniform1f(n.uniforms.uRefraction, i.refraction), I.uniform1f(n.uniforms.uSpecularExp, i.specularExp), I.uniform1f(n.uniforms.uShine, i.shine), I.uniform1f(n.uniforms.uWarpStrength, i.warpStrength ?? 0.015), I.uniform1i(n.uniforms.uAlgorithm, Zg[i.algorithm] ?? 0), I.activeTexture(I.TEXTURE0), I.bindTexture(I.TEXTURE_2D, g(this, H).read.tex), I.activeTexture(I.TEXTURE1), I.bindTexture(I.TEXTURE_2D, g(this, v)), I.activeTexture(I.TEXTURE2), I.bindTexture(I.TEXTURE_2D, g(this, _)), I.activeTexture(I.TEXTURE3), I.bindTexture(I.TEXTURE_2D, g(this, z)), I.activeTexture(I.TEXTURE4), I.bindTexture(I.TEXTURE_2D, g(this, B).read.tex), I.uniform1i(n.uniforms.uTexture, 0), I.uniform1i(n.uniforms.uObstacle, 1), I.uniform1i(n.uniforms.uBackground, 2), I.uniform1i(n.uniforms.uCoverage, 3), I.uniform1i(n.uniforms.uVelocity, 4), h(null);
672
678
  };
673
- const zI = "dmFyIHllID0gT2JqZWN0LmRlZmluZVByb3BlcnR5Owp2YXIgRGUgPSAoaSwgZSwgcikgPT4gZSBpbiBpID8geWUoaSwgZSwgeyBlbnVtZXJhYmxlOiAhMCwgY29uZmlndXJhYmxlOiAhMCwgd3JpdGFibGU6ICEwLCB2YWx1ZTogciB9KSA6IGlbZV0gPSByOwp2YXIgc2UgPSAoaSwgZSwgcikgPT4gKERlKGksIHR5cGVvZiBlICE9ICJzeW1ib2wiID8gZSArICIiIDogZSwgciksIHIpLCBoZSA9IChpLCBlLCByKSA9PiB7CiAgaWYgKCFlLmhhcyhpKSkKICAgIHRocm93IFR5cGVFcnJvcigiQ2Fubm90ICIgKyByKTsKfTsKdmFyIHQgPSAoaSwgZSwgcikgPT4gKGhlKGksIGUsICJyZWFkIGZyb20gcHJpdmF0ZSBmaWVsZCIpLCByID8gci5jYWxsKGkpIDogZS5nZXQoaSkpLCBsID0gKGksIGUsIHIpID0+IHsKICBpZiAoZS5oYXMoaSkpCiAgICB0aHJvdyBUeXBlRXJyb3IoIkNhbm5vdCBhZGQgdGhlIHNhbWUgcHJpdmF0ZSBtZW1iZXIgbW9yZSB0aGFuIG9uY2UiKTsKICBlIGluc3RhbmNlb2YgV2Vha1NldCA/IGUuYWRkKGkpIDogZS5zZXQoaSwgcik7Cn0sIGEgPSAoaSwgZSwgciwgbykgPT4gKGhlKGksIGUsICJ3cml0ZSB0byBwcml2YXRlIGZpZWxkIiksIG8gPyBvLmNhbGwoaSwgcikgOiBlLnNldChpLCByKSwgcik7CnZhciBkID0gKGksIGUsIHIpID0+IChoZShpLCBlLCAiYWNjZXNzIHByaXZhdGUgbWV0aG9kIiksIHIpOwpjb25zdCBnZSA9IHsKICBkZW5zaXR5RGlzc2lwYXRpb246IDAuOTkyLAogIHZlbG9jaXR5RGlzc2lwYXRpb246IDAuOTMsCiAgcHJlc3N1cmVJdGVyYXRpb25zOiAxLAogIGN1cmw6IDFlLTQsCiAgc3BsYXRSYWRpdXM6IDRlLTMsCiAgc3BsYXRGb3JjZTogMC45MSwKICByZWZyYWN0aW9uOiAwLjI1LAogIHNwZWN1bGFyRXhwOiAxLjAxLAogIHNoaW5lOiAwLjAxLAogIHdhdGVyQ29sb3I6IFswLCAwLCAwXSwKICBnbG93Q29sb3I6IFswLjcsIDAuODUsIDFdLAogIGFsZ29yaXRobTogInN0YW5kYXJkIiwKICB3YXJwU3RyZW5ndGg6IDAuMDE1Cn07Cih7CiAgLi4uZ2UKfSk7CmNvbnN0IFVlID0gewogIGNhbG06IHsKICAgIGRlbnNpdHlEaXNzaXBhdGlvbjogMC45OTksCiAgICB2ZWxvY2l0eURpc3NpcGF0aW9uOiAwLjk4LAogICAgY3VybDogMWUtNCwKICAgIHNwbGF0UmFkaXVzOiAzZS0zLAogICAgc3BsYXRGb3JjZTogMC41LAogICAgcmVmcmFjdGlvbjogMC4xNSwKICAgIHNoaW5lOiA1ZS0zLAogICAgZ2xvd0NvbG9yOiBbMC42LCAwLjg1LCAxXSwKICAgIHdhdGVyQ29sb3I6IFswLCAwLjAyLCAwLjA1XQogIH0sCiAgc2FuZDogewogICAgZGVuc2l0eURpc3NpcGF0aW9uOiAwLjk5NywKICAgIHZlbG9jaXR5RGlzc2lwYXRpb246IDAuOTgsCiAgICBjdXJsOiAxLAogICAgc3BsYXRSYWRpdXM6IDAuMDEsCiAgICBzcGxhdEZvcmNlOiAwLjksCiAgICByZWZyYWN0aW9uOiAwLjgsCiAgICBzcGVjdWxhckV4cDogMC4xLAogICAgc2hpbmU6IDAuMDUsCiAgICBnbG93Q29sb3I6IFswLjAyNywgMC4wMjcsIDAuMDI3XSwKICAgIHdhdGVyQ29sb3I6IFswLjQ1MSwgMC4zMjksIDAuMTI1XQogIH0sCiAgd2F2ZTogewogICAgZGVuc2l0eURpc3NpcGF0aW9uOiAwLjk5NCwKICAgIHZlbG9jaXR5RGlzc2lwYXRpb246IDAuOTIsCiAgICBjdXJsOiAwLjIsCiAgICBzcGxhdFJhZGl1czogNWUtMywKICAgIHNwbGF0Rm9yY2U6IDEuMiwKICAgIHJlZnJhY3Rpb246IDAuMzUsCiAgICBzaGluZTogMC4wMywKICAgIGdsb3dDb2xvcjogWzAuNSwgMC44LCAxXSwKICAgIHdhdGVyQ29sb3I6IFswLCAwLjAxLCAwLjAzXQogIH0sCiAgbmVvbjogewogICAgZGVuc2l0eURpc3NpcGF0aW9uOiAwLjk4NSwKICAgIHZlbG9jaXR5RGlzc2lwYXRpb246IDAuOTMsCiAgICBjdXJsOiAwLjA1LAogICAgc3BsYXRSYWRpdXM6IDhlLTMsCiAgICBzcGxhdEZvcmNlOiAxLjUsCiAgICByZWZyYWN0aW9uOiAwLjI1LAogICAgc3BlY3VsYXJFeHA6IDAuNSwKICAgIHNoaW5lOiAwLjE0LAogICAgZ2xvd0NvbG9yOiBbMSwgMC4yLCAwLjhdLAogICAgd2F0ZXJDb2xvcjogWzAuMDUsIDAsIDAuMDhdCiAgfSwKICBzbW9rZTogewogICAgZGVuc2l0eURpc3NpcGF0aW9uOiAwLjk5NiwKICAgIHZlbG9jaXR5RGlzc2lwYXRpb246IDAuOTcsCiAgICBjdXJsOiAwLjA0LAogICAgc3BsYXRSYWRpdXM6IDllLTMsCiAgICBzcGxhdEZvcmNlOiAwLjgsCiAgICByZWZyYWN0aW9uOiAwLjA4LAogICAgc2hpbmU6IDAsCiAgICBnbG93Q29sb3I6IFswLjUsIDAuNSwgMC41XSwKICAgIHdhdGVyQ29sb3I6IFswLjA2LCAwLjA2LCAwLjA2XQogIH0KfTsKZnVuY3Rpb24gX2UoaSA9IHt9LCBlLCByID0gZ2UpIHsKICByZXR1cm4geyAuLi5lID8geyAuLi5yLCAuLi5VZVtlXSB9IDogciwgLi4uaSB9Owp9CmNvbnN0IFAgPSAoCiAgLyogZ2xzbCAqLwogIGAKICBwcmVjaXNpb24gaGlnaHAgZmxvYXQ7CiAgYXR0cmlidXRlIHZlYzIgYVBvc2l0aW9uOwogIHZhcnlpbmcgdmVjMiB2VXY7CiAgdmFyeWluZyB2ZWMyIHZMOyB2YXJ5aW5nIHZlYzIgdlI7IHZhcnlpbmcgdmVjMiB2VDsgdmFyeWluZyB2ZWMyIHZCOwogIHVuaWZvcm0gdmVjMiB0ZXhlbFNpemU7CiAgdm9pZCBtYWluICgpIHsKICAgIHZVdiA9IGFQb3NpdGlvbiAqIDAuNSArIDAuNTsKICAgIHZMID0gdlV2IC0gdmVjMih0ZXhlbFNpemUueCwgMC4wKTsKICAgIHZSID0gdlV2ICsgdmVjMih0ZXhlbFNpemUueCwgMC4wKTsKICAgIHZUID0gdlV2ICsgdmVjMigwLjAsIHRleGVsU2l6ZS55KTsKICAgIHZCID0gdlV2IC0gdmVjMigwLjAsIHRleGVsU2l6ZS55KTsKICAgIGdsX1Bvc2l0aW9uID0gdmVjNChhUG9zaXRpb24sIDAuMCwgMS4wKTsKICB9CmAKKSwgd2UgPSAoCiAgLyogZ2xzbCAqLwogIGAKICBwcmVjaXNpb24gaGlnaHAgZmxvYXQ7CiAgdmFyeWluZyB2ZWMyIHZVdjsKICB1bmlmb3JtIHNhbXBsZXIyRCB1VmVsb2NpdHk7CiAgdW5pZm9ybSBzYW1wbGVyMkQgdVNvdXJjZTsKICB1bmlmb3JtIHNhbXBsZXIyRCB1T2JzdGFjbGU7CiAgdW5pZm9ybSB2ZWMyIHRleGVsU2l6ZTsKICB1bmlmb3JtIGZsb2F0IGR0OwogIHVuaWZvcm0gZmxvYXQgZGlzc2lwYXRpb247CiAgdm9pZCBtYWluICgpIHsKICAgIGlmICh0ZXh0dXJlMkQodU9ic3RhY2xlLCB2VXYpLnIgPiAwLjUpIHsgZ2xfRnJhZ0NvbG9yID0gdmVjNCgwLjApOyByZXR1cm47IH0KICAgIHZlYzIgY29vcmQgPSB2VXYgLSBkdCAqIHRleHR1cmUyRCh1VmVsb2NpdHksIHZVdikueHkgKiB0ZXhlbFNpemU7CiAgICBnbF9GcmFnQ29sb3IgPSBkaXNzaXBhdGlvbiAqIHRleHR1cmUyRCh1U291cmNlLCBjb29yZCk7CiAgfQpgCiksIENlID0gKAogIC8qIGdsc2wgKi8KICBgCiAgcHJlY2lzaW9uIGhpZ2hwIGZsb2F0OwogIHZhcnlpbmcgdmVjMiB2TDsgdmFyeWluZyB2ZWMyIHZSOyB2YXJ5aW5nIHZlYzIgdlQ7IHZhcnlpbmcgdmVjMiB2QjsKICB1bmlmb3JtIHNhbXBsZXIyRCB1VmVsb2NpdHk7CiAgdW5pZm9ybSBzYW1wbGVyMkQgdU9ic3RhY2xlOwogIHZvaWQgbWFpbiAoKSB7CiAgICBmbG9hdCBMID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgdkwpLnIgPiAwLjUgPyAwLjAgOiB0ZXh0dXJlMkQodVZlbG9jaXR5LCB2TCkueDsKICAgIGZsb2F0IFIgPSB0ZXh0dXJlMkQodU9ic3RhY2xlLCB2UikuciA+IDAuNSA/IDAuMCA6IHRleHR1cmUyRCh1VmVsb2NpdHksIHZSKS54OwogICAgZmxvYXQgVCA9IHRleHR1cmUyRCh1T2JzdGFjbGUsIHZUKS5yID4gMC41ID8gMC4wIDogdGV4dHVyZTJEKHVWZWxvY2l0eSwgdlQpLnk7CiAgICBmbG9hdCBCID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgdkIpLnIgPiAwLjUgPyAwLjAgOiB0ZXh0dXJlMkQodVZlbG9jaXR5LCB2QikueTsKICAgIGdsX0ZyYWdDb2xvciA9IHZlYzQoMC41ICogKFIgLSBMICsgVCAtIEIpLCAwLjAsIDAuMCwgMS4wKTsKICB9CmAKKSwgU2UgPSAoCiAgLyogZ2xzbCAqLwogIGAKICBwcmVjaXNpb24gaGlnaHAgZmxvYXQ7CiAgdmFyeWluZyB2ZWMyIHZVdjsgdmFyeWluZyB2ZWMyIHZMOyB2YXJ5aW5nIHZlYzIgdlI7IHZhcnlpbmcgdmVjMiB2VDsgdmFyeWluZyB2ZWMyIHZCOwogIHVuaWZvcm0gc2FtcGxlcjJEIHVQcmVzc3VyZTsKICB1bmlmb3JtIHNhbXBsZXIyRCB1RGl2ZXJnZW5jZTsKICB1bmlmb3JtIHNhbXBsZXIyRCB1T2JzdGFjbGU7CiAgdm9pZCBtYWluICgpIHsKICAgIGZsb2F0IEMgPSB0ZXh0dXJlMkQodVByZXNzdXJlLCB2VXYpLng7CiAgICBmbG9hdCBMID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgdkwpLnIgPiAwLjUgPyBDIDogdGV4dHVyZTJEKHVQcmVzc3VyZSwgdkwpLng7CiAgICBmbG9hdCBSID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgdlIpLnIgPiAwLjUgPyBDIDogdGV4dHVyZTJEKHVQcmVzc3VyZSwgdlIpLng7CiAgICBmbG9hdCBUID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgdlQpLnIgPiAwLjUgPyBDIDogdGV4dHVyZTJEKHVQcmVzc3VyZSwgdlQpLng7CiAgICBmbG9hdCBCID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgdkIpLnIgPiAwLjUgPyBDIDogdGV4dHVyZTJEKHVQcmVzc3VyZSwgdkIpLng7CiAgICBmbG9hdCBkaXYgPSB0ZXh0dXJlMkQodURpdmVyZ2VuY2UsIHZVdikueDsKICAgIGdsX0ZyYWdDb2xvciA9IHZlYzQoKEwgKyBSICsgQiArIFQgLSBkaXYpICogMC4yNSwgMC4wLCAwLjAsIDEuMCk7CiAgfQpgCiksIEZlID0gKAogIC8qIGdsc2wgKi8KICBgCiAgcHJlY2lzaW9uIGhpZ2hwIGZsb2F0OwogIHZhcnlpbmcgdmVjMiB2VXY7IHZhcnlpbmcgdmVjMiB2TDsgdmFyeWluZyB2ZWMyIHZSOyB2YXJ5aW5nIHZlYzIgdlQ7IHZhcnlpbmcgdmVjMiB2QjsKICB1bmlmb3JtIHNhbXBsZXIyRCB1UHJlc3N1cmU7CiAgdW5pZm9ybSBzYW1wbGVyMkQgdVZlbG9jaXR5OwogIHVuaWZvcm0gc2FtcGxlcjJEIHVPYnN0YWNsZTsKICB2b2lkIG1haW4gKCkgewogICAgaWYgKHRleHR1cmUyRCh1T2JzdGFjbGUsIHZVdikuciA+IDAuNSkgeyBnbF9GcmFnQ29sb3IgPSB2ZWM0KDAuMCk7IHJldHVybjsgfQogICAgZmxvYXQgTCA9IHRleHR1cmUyRCh1UHJlc3N1cmUsIHZMKS54OwogICAgZmxvYXQgUiA9IHRleHR1cmUyRCh1UHJlc3N1cmUsIHZSKS54OwogICAgZmxvYXQgVCA9IHRleHR1cmUyRCh1UHJlc3N1cmUsIHZUKS54OwogICAgZmxvYXQgQiA9IHRleHR1cmUyRCh1UHJlc3N1cmUsIHZCKS54OwogICAgdmVjMiB2ZWwgPSB0ZXh0dXJlMkQodVZlbG9jaXR5LCB2VXYpLnh5IC0gdmVjMihSIC0gTCwgVCAtIEIpOwogICAgZ2xfRnJhZ0NvbG9yID0gdmVjNCh2ZWwsIDAuMCwgMS4wKTsKICB9CmAKKSwgWGUgPSAoCiAgLyogZ2xzbCAqLwogIGAKICBwcmVjaXNpb24gaGlnaHAgZmxvYXQ7CiAgdmFyeWluZyB2ZWMyIHZVdjsKICB1bmlmb3JtIHNhbXBsZXIyRCB1VGFyZ2V0OwogIHVuaWZvcm0gZmxvYXQgYXNwZWN0UmF0aW87CiAgdW5pZm9ybSB2ZWMzIGNvbG9yOwogIHVuaWZvcm0gdmVjMiBwb2ludDsKICB1bmlmb3JtIGZsb2F0IHJhZGl1czsKICB2b2lkIG1haW4gKCkgewogICAgdmVjMiBwID0gdlV2IC0gcG9pbnQueHk7CiAgICBwLnggKj0gYXNwZWN0UmF0aW87CiAgICB2ZWMzIHNwbGF0ID0gZXhwKC1kb3QocCwgcCkgLyByYWRpdXMpICogY29sb3I7CiAgICBnbF9GcmFnQ29sb3IgPSB2ZWM0KHRleHR1cmUyRCh1VGFyZ2V0LCB2VXYpLnh5eiArIHNwbGF0LCAxLjApOwogIH0KYAopLCBCZSA9ICgKICAvKiBnbHNsICovCiAgYAogIHByZWNpc2lvbiBoaWdocCBmbG9hdDsKICB2YXJ5aW5nIHZlYzIgdkw7IHZhcnlpbmcgdmVjMiB2UjsgdmFyeWluZyB2ZWMyIHZUOyB2YXJ5aW5nIHZlYzIgdkI7CiAgdW5pZm9ybSBzYW1wbGVyMkQgdVZlbG9jaXR5OwogIHZvaWQgbWFpbiAoKSB7CiAgICBmbG9hdCBMID0gdGV4dHVyZTJEKHVWZWxvY2l0eSwgdkwpLnk7CiAgICBmbG9hdCBSID0gdGV4dHVyZTJEKHVWZWxvY2l0eSwgdlIpLnk7CiAgICBmbG9hdCBUID0gdGV4dHVyZTJEKHVWZWxvY2l0eSwgdlQpLng7CiAgICBmbG9hdCBCID0gdGV4dHVyZTJEKHVWZWxvY2l0eSwgdkIpLng7CiAgICBnbF9GcmFnQ29sb3IgPSB2ZWM0KDAuNSAqIChSIC0gTCAtIFQgKyBCKSwgMC4wLCAwLjAsIDEuMCk7CiAgfQpgCiksIEFlID0gKAogIC8qIGdsc2wgKi8KICBgCiAgcHJlY2lzaW9uIGhpZ2hwIGZsb2F0OwogIHZhcnlpbmcgdmVjMiB2VXY7IHZhcnlpbmcgdmVjMiB2TDsgdmFyeWluZyB2ZWMyIHZSOyB2YXJ5aW5nIHZlYzIgdlQ7IHZhcnlpbmcgdmVjMiB2QjsKICB1bmlmb3JtIHNhbXBsZXIyRCB1VmVsb2NpdHk7CiAgdW5pZm9ybSBzYW1wbGVyMkQgdUN1cmw7CiAgdW5pZm9ybSBmbG9hdCBjdXJsOwogIHVuaWZvcm0gZmxvYXQgZHQ7CiAgdm9pZCBtYWluICgpIHsKICAgIGZsb2F0IEwgPSB0ZXh0dXJlMkQodUN1cmwsIHZMKS54OwogICAgZmxvYXQgUiA9IHRleHR1cmUyRCh1Q3VybCwgdlIpLng7CiAgICBmbG9hdCBUID0gdGV4dHVyZTJEKHVDdXJsLCB2VCkueDsKICAgIGZsb2F0IEIgPSB0ZXh0dXJlMkQodUN1cmwsIHZCKS54OwogICAgZmxvYXQgQyA9IHRleHR1cmUyRCh1Q3VybCwgdlV2KS54OwogICAgdmVjMiBmb3JjZSA9IDAuNSAqIHZlYzIoYWJzKFQpIC0gYWJzKEIpLCBhYnMoUikgLSBhYnMoTCkpOwogICAgZm9yY2UgLz0gbGVuZ3RoKGZvcmNlKSArIDAuMDAwMTsKICAgIGZvcmNlICo9IGN1cmwgKiAzMC4wICogQzsKICAgIGdsX0ZyYWdDb2xvciA9IHZlYzQodGV4dHVyZTJEKHVWZWxvY2l0eSwgdlV2KS54eSArIGZvcmNlICogZHQsIDAuMCwgMS4wKTsKICB9CmAKKSwgTGUgPSAoCiAgLyogZ2xzbCAqLwogIGAKICBwcmVjaXNpb24gaGlnaHAgZmxvYXQ7CiAgdmFyeWluZyB2ZWMyIHZVdjsKCiAgdW5pZm9ybSBzYW1wbGVyMkQgdVRleHR1cmU7CiAgdW5pZm9ybSBzYW1wbGVyMkQgdU9ic3RhY2xlOwogIHVuaWZvcm0gc2FtcGxlcjJEIHVCYWNrZ3JvdW5kOwogIHVuaWZvcm0gc2FtcGxlcjJEIHVDb3ZlcmFnZTsKICB1bmlmb3JtIHNhbXBsZXIyRCB1VmVsb2NpdHk7CgogIHVuaWZvcm0gdmVjMiAgdGV4ZWxTaXplOwogIHVuaWZvcm0gdmVjMyAgdVdhdGVyQ29sb3I7CiAgdW5pZm9ybSB2ZWMzICB1R2xvd0NvbG9yOwogIHVuaWZvcm0gZmxvYXQgdVJlZnJhY3Rpb247CiAgdW5pZm9ybSBmbG9hdCB1U3BlY3VsYXJFeHA7CiAgdW5pZm9ybSBmbG9hdCB1U2hpbmU7CiAgdW5pZm9ybSBmbG9hdCB1V2FycFN0cmVuZ3RoOwogIHVuaWZvcm0gaW50ICAgdUFsZ29yaXRobTsKCiAgdm9pZCBtYWluICgpIHsKICAgIGZsb2F0IG9icyAgICAgID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgIHZVdikucjsKICAgIC8vIE1hc2sgZGVuc2l0eSBpbnNpZGUgb2JzdGFjbGVzIHNvIHNwbGF0cyBkb24ndCBmbGlja2VyIHRoZSB0ZXh0L2ltYWdlIGNvbnRlbnQuCiAgICBmbG9hdCBkZW5zaXR5ICA9IG1heCh0ZXh0dXJlMkQodVRleHR1cmUsIHZVdikuciwgMC4wKSAqICgxLjAgLSBzdGVwKDAuNSwgb2JzKSk7CiAgICBmbG9hdCBjb3ZlcmFnZSA9IHRleHR1cmUyRCh1Q292ZXJhZ2UsICB2VXYpLnI7CgogICAgZmxvYXQgZEwgPSBtYXgodGV4dHVyZTJEKHVUZXh0dXJlLCB2VXYgLSB2ZWMyKHRleGVsU2l6ZS54ICogMi4wLCAwLjApKS5yLCAwLjApOwogICAgZmxvYXQgZFIgPSBtYXgodGV4dHVyZTJEKHVUZXh0dXJlLCB2VXYgKyB2ZWMyKHRleGVsU2l6ZS54ICogMi4wLCAwLjApKS5yLCAwLjApOwogICAgZmxvYXQgZFQgPSBtYXgodGV4dHVyZTJEKHVUZXh0dXJlLCB2VXYgKyB2ZWMyKDAuMCwgdGV4ZWxTaXplLnkgKiAyLjApKS5yLCAwLjApOwogICAgZmxvYXQgZEIgPSBtYXgodGV4dHVyZTJEKHVUZXh0dXJlLCB2VXYgLSB2ZWMyKDAuMCwgdGV4ZWxTaXplLnkgKiAyLjApKS5yLCAwLjApOwoKICAgIHZlYzMgIG5vcm1hbCAgID0gbm9ybWFsaXplKHZlYzMoZEwgLSBkUiwgZEIgLSBkVCwgMC4yKSk7CiAgICB2ZWMzICBsaWdodERpciA9IG5vcm1hbGl6ZSh2ZWMzKDAuNSwgMS4wLCAwLjUpKTsKICAgIHZlYzMgIGhhbGZWICAgID0gbm9ybWFsaXplKGxpZ2h0RGlyICsgdmVjMygwLjAsIDAuMCwgMS4wKSk7CiAgICBmbG9hdCBzcGVjICAgICA9IHBvdyhtYXgoZG90KG5vcm1hbCwgaGFsZlYpLCAwLjApLCB1U3BlY3VsYXJFeHApICogdVNoaW5lICogZGVuc2l0eTsKCiAgICAvLyBJbiB0cmFuc3BhcmVudCAobm9uLWNvdmVyYWdlKSBhcmVhcyB0aGUgYmFja2dyb3VuZCB0ZXh0dXJlIGlzIGVtcHR5IGJsYWNrIGNhbnZhcy4KICAgIC8vIFJlcGxhY2UgaXQgd2l0aCB1V2F0ZXJDb2xvciBzbyBmbHVpZCBjb2xvdXIgaXMgbm90IGNvbnRhbWluYXRlZCBieSB0aGF0IGJsYWNrLAogICAgLy8gYWxsb3dpbmcgdGhlIENTUyBiYWNrZ3JvdW5kQ29sb3IgdG8gc2hvdyB0aHJvdWdoIGNvcnJlY3RseSB2aWEgcHJlbXVsdGlwbGllZCBhbHBoYS4KICAgIHZlYzMgYmdSYXcgPSB0ZXh0dXJlMkQodUJhY2tncm91bmQsIHZVdikucmdiOwogICAgdmVjMyBiZyAgICA9IG1peCh1V2F0ZXJDb2xvciwgYmdSYXcsIGNvdmVyYWdlKTsKICAgIHZlYzMgY29sb3IgPSBiZzsKCiAgICBpZiAodUFsZ29yaXRobSA9PSAxKSB7CiAgICAgIC8vIOKUgOKUgCBnbGFzcyDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIAKICAgICAgLy8gU3Ryb25nIFVWIGRpc3RvcnRpb24gb25seS4gSW1hZ2UgYmVuZHMgYnV0IG5vIGNvbG91ciBvdmVybGF5LgogICAgICB2ZWMyIHJlZnJVdiA9IGNsYW1wKHZVdiArIG5vcm1hbC54eSAqIHVSZWZyYWN0aW9uICogZGVuc2l0eSAqIDMuMCwgMC4wLCAxLjApOwogICAgICB2ZWMzIHJlZnJCZyA9IG1peCh1V2F0ZXJDb2xvciwgdGV4dHVyZTJEKHVCYWNrZ3JvdW5kLCBtaXgodlV2LCByZWZyVXYsIDEuMCAtIG9icykpLnJnYiwgY292ZXJhZ2UpOwogICAgICBjb2xvciA9IHJlZnJCZyArIHNwZWMgKiB1R2xvd0NvbG9yICogMi41OwogICAgICBjb2xvciA9IG1peChjb2xvciwgYmcgKiAwLjYsIG9icyAqIDAuMyk7CgogICAgfSBlbHNlIGlmICh1QWxnb3JpdGhtID09IDIpIHsKICAgICAgLy8g4pSA4pSAIGluayDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIAKICAgICAgLy8gRGVuc2Ugb3BhcXVlIHBpZ21lbnQgdGhhdCBzdGFpbnMuIFN1YnRsZSByZWZyYWN0aW9uIHVuZGVybmVhdGguCiAgICAgIGZsb2F0IGlua0QgID0gbWluKGRlbnNpdHkgKiA0LjAsIDEuMCk7CiAgICAgIHZlYzIgcmVmclV2ID0gY2xhbXAodlV2ICsgbm9ybWFsLnh5ICogdVJlZnJhY3Rpb24gKiBkZW5zaXR5ICogMC40LCAwLjAsIDEuMCk7CiAgICAgIHZlYzMgcmVmckJnID0gbWl4KHVXYXRlckNvbG9yLCB0ZXh0dXJlMkQodUJhY2tncm91bmQsIG1peCh2VXYsIHJlZnJVdiwgMS4wIC0gb2JzKSkucmdiLCBjb3ZlcmFnZSk7CiAgICAgIGNvbG9yID0gbWl4KHJlZnJCZywgdVdhdGVyQ29sb3IgKyBzcGVjICogdUdsb3dDb2xvciwgaW5rRCk7CiAgICAgIGNvbG9yID0gbWl4KGNvbG9yLCBiZyAqIDAuNSwgb2JzICogMC4xNSk7CgogICAgfSBlbHNlIGlmICh1QWxnb3JpdGhtID09IDMpIHsKICAgICAgLy8g4pSA4pSAIGF1cm9yYSDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIAKICAgICAgLy8gVmVsb2NpdHkgZmllbGQgd2FycHMgYmFja2dyb3VuZCBVVnMg4oCUIGxpcXVpZCBtZXRhbCAvIGxhdmEtbGFtcCBmZWVsLgogICAgICB2ZWMyICB2ZWwgICAgPSB0ZXh0dXJlMkQodVZlbG9jaXR5LCB2VXYpLnh5OwogICAgICBmbG9hdCB2ZWxNYWcgPSBjbGFtcChsZW5ndGgodmVsKSAqIDIwLjAsIDAuMCwgMS4wKTsKICAgICAgdmVjMiAgd2FycFV2ID0gY2xhbXAodlV2ICsgdmVsICogdVdhcnBTdHJlbmd0aCwgMC4wLCAxLjApOwogICAgICB2ZWMzICB3YXJwQmcgPSBtaXgodVdhdGVyQ29sb3IsIHRleHR1cmUyRCh1QmFja2dyb3VuZCwgd2FycFV2KS5yZ2IsIGNvdmVyYWdlKTsKICAgICAgY29sb3IgID0gbWl4KGJnLCB3YXJwQmcsIHZlbE1hZyAqICgxLjAgLSBvYnMpKTsKICAgICAgY29sb3IgKz0gc3BlYyAqIHVHbG93Q29sb3IgKiB2ZWxNYWcgKiAxLjU7CiAgICAgIGNvbG9yICs9IHVXYXRlckNvbG9yICogZGVuc2l0eSAqIDAuMzsKICAgICAgY29sb3IgID0gbWl4KGNvbG9yLCBiZyAqIDAuNSwgb2JzICogMC4yKTsKCiAgICB9IGVsc2UgaWYgKHVBbGdvcml0aG0gPT0gNCkgewogICAgICAvLyDilIDilIAgcmlwcGxlIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgAogICAgICAvLyBFeGFnZ2VyYXRlZCBub3JtYWwgcGVydHVyYmF0aW9uICsgRnJlc25lbCByaW0g4oCUIGNhbG0gd2F0ZXIgc3VyZmFjZS4KICAgICAgdmVjMiAgcmlwcGxlVXYgPSBjbGFtcCh2VXYgKyBub3JtYWwueHkgKiB1UmVmcmFjdGlvbiAqIGRlbnNpdHkgKiA2LjAsIDAuMCwgMS4wKTsKICAgICAgdmVjMyAgcmVmckJnICAgPSBtaXgodVdhdGVyQ29sb3IsIHRleHR1cmUyRCh1QmFja2dyb3VuZCwgbWl4KHZVdiwgcmlwcGxlVXYsIDEuMCAtIG9icykpLnJnYiwgY292ZXJhZ2UpOwogICAgICBmbG9hdCBmcmVzbmVsICA9IHBvdyhjbGFtcCgxLjAgLSBkb3Qobm9ybWFsLCB2ZWMzKDAuMCwgMC4wLCAxLjApKSwgMC4wLCAxLjApLCAzLjApICogZGVuc2l0eTsKICAgICAgY29sb3IgID0gcmVmckJnOwogICAgICBjb2xvciArPSBmcmVzbmVsICogdUdsb3dDb2xvciAqIDIuMDsKICAgICAgY29sb3IgKz0gc3BlYyAqIHVHbG93Q29sb3IgKiBkZW5zaXR5ICogMi4wOwogICAgICBjb2xvciAgPSBtaXgoY29sb3IsIGJnICogMC41LCBvYnMgKiAwLjIpOwoKICAgIH0gZWxzZSB7CiAgICAgIC8vIOKUgOKUgCBzdGFuZGFyZCAoMCkg4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSACiAgICAgIC8vIE9yaWdpbmFsOiBjb2xvdXIgb3ZlcmxheSBibGVuZGVkIG92ZXIgcmVmcmFjdGVkIGJhY2tncm91bmQuCiAgICAgIHZlYzIgcmVmclV2ID0gdlV2ICsgbm9ybWFsLnh5ICogdVJlZnJhY3Rpb24gKiBkZW5zaXR5OwogICAgICB2ZWMzIHJlZnJCZyA9IG1peCh1V2F0ZXJDb2xvciwgdGV4dHVyZTJEKHVCYWNrZ3JvdW5kLCBtaXgodlV2LCByZWZyVXYsIDEuMCAtIG9icykpLnJnYiwgY292ZXJhZ2UpOwogICAgICBjb2xvciAgPSBtaXgocmVmckJnLCB1V2F0ZXJDb2xvciwgbWluKGRlbnNpdHkgKiAxLjUsIDAuOCkpOwogICAgICBjb2xvciArPSBzcGVjICogdUdsb3dDb2xvcjsKICAgICAgY29sb3IgID0gbWl4KGNvbG9yLCBiZyAqIDAuNSwgb2JzICogMC4yKTsKICAgIH0KCiAgICAvLyBQcmVtdWx0aXBsaWVkIGFscGhhIOKAlCB0cmFuc3BhcmVudCB3aGVyZSB0aGVyZSBpcyBuZWl0aGVyIGNvbnRlbnQgbm9yIGZsdWlkLAogICAgLy8gbGV0dGluZyB0aGUgQ1NTIGJhY2tncm91bmRDb2xvciBvbiB0aGUgY29udGFpbmVyIGRpdiBzaG93IHRocm91Z2guCiAgICBmbG9hdCBhbHBoYSA9IGNsYW1wKG1heChkZW5zaXR5ICogMS41LCBjb3ZlcmFnZSksIDAuMCwgMS4wKTsKICAgIGdsX0ZyYWdDb2xvciA9IHZlYzQoY29sb3IgKiBhbHBoYSwgYWxwaGEpOwogIH0KYAopOwpmdW5jdGlvbiBQZShpKSB7CiAgY29uc3QgZSA9IHsgYWxwaGE6ICEwLCBkZXB0aDogITEsIHN0ZW5jaWw6ICExLCBhbnRpYWxpYXM6ICEwLCBwcmVzZXJ2ZURyYXdpbmdCdWZmZXI6ICExIH07CiAgbGV0IHIgPSBpLmdldENvbnRleHQoIndlYmdsMiIsIGUpOwogIGNvbnN0IG8gPSAhIXI7CiAgbyB8fCAociA9IGkuZ2V0Q29udGV4dCgid2ViZ2wiLCBlKSwgci5nZXRFeHRlbnNpb24oIkVYVF9jb2xvcl9idWZmZXJfaGFsZl9mbG9hdCIpKTsKICBjb25zdCBzID0gbyA/IG51bGwgOiByLmdldEV4dGVuc2lvbigiT0VTX3RleHR1cmVfaGFsZl9mbG9hdCIpLCBuID0gbyA/IHIuSEFMRl9GTE9BVCA6IHMuSEFMRl9GTE9BVF9PRVM7CiAgcmV0dXJuIHIuZ2V0RXh0ZW5zaW9uKCJFWFRfY29sb3JfYnVmZmVyX2Zsb2F0IiksIHIuZ2V0RXh0ZW5zaW9uKCJPRVNfdGV4dHVyZV9oYWxmX2Zsb2F0X2xpbmVhciIpLCB7CiAgICBnbDogciwKICAgIGlzV2ViR0wyOiBvLAogICAgZXh0OiB7CiAgICAgIGludGVybmFsRm9ybWF0OiBvID8gci5SR0JBMTZGIDogci5SR0JBLAogICAgICBmb3JtYXQ6IHIuUkdCQSwKICAgICAgdHlwZTogbgogICAgfQogIH07Cn0KY2xhc3MgTyB7CiAgY29uc3RydWN0b3IoZSwgciwgbykgewogICAgc2UodGhpcywgInByb2dyYW0iKTsKICAgIHNlKHRoaXMsICJ1bmlmb3JtcyIsIHt9KTsKICAgIHNlKHRoaXMsICJfZ2wiKTsKICAgIHRoaXMuX2dsID0gZSwgdGhpcy5wcm9ncmFtID0gZS5jcmVhdGVQcm9ncmFtKCksIGUuYXR0YWNoU2hhZGVyKHRoaXMucHJvZ3JhbSwgdGhpcy5fY29tcGlsZShlLlZFUlRFWF9TSEFERVIsIHIpKSwgZS5hdHRhY2hTaGFkZXIodGhpcy5wcm9ncmFtLCB0aGlzLl9jb21waWxlKGUuRlJBR01FTlRfU0hBREVSLCBvKSksIGUubGlua1Byb2dyYW0odGhpcy5wcm9ncmFtKTsKICAgIGNvbnN0IHMgPSBlLmdldFByb2dyYW1QYXJhbWV0ZXIodGhpcy5wcm9ncmFtLCBlLkFDVElWRV9VTklGT1JNUyk7CiAgICBmb3IgKGxldCBuID0gMDsgbiA8IHM7IG4rKykgewogICAgICBjb25zdCB1ID0gZS5nZXRBY3RpdmVVbmlmb3JtKHRoaXMucHJvZ3JhbSwgbikubmFtZTsKICAgICAgdGhpcy51bmlmb3Jtc1t1XSA9IGUuZ2V0VW5pZm9ybUxvY2F0aW9uKHRoaXMucHJvZ3JhbSwgdSk7CiAgICB9CiAgfQogIF9jb21waWxlKGUsIHIpIHsKICAgIGNvbnN0IG8gPSB0aGlzLl9nbCwgcyA9IG8uY3JlYXRlU2hhZGVyKGUpOwogICAgcmV0dXJuIG8uc2hhZGVyU291cmNlKHMsIHIpLCBvLmNvbXBpbGVTaGFkZXIocyksIHM7CiAgfQogIGJpbmQoKSB7CiAgICB0aGlzLl9nbC51c2VQcm9ncmFtKHRoaXMucHJvZ3JhbSk7CiAgfQogIGRpc3Bvc2UoKSB7CiAgICB0aGlzLl9nbC5kZWxldGVQcm9ncmFtKHRoaXMucHJvZ3JhbSk7CiAgfQp9CmZ1bmN0aW9uIE9lKGkpIHsKICByZXR1cm4gewogICAgYWR2ZWN0aW9uOiBuZXcgTyhpLCBQLCB3ZSksCiAgICBkaXZlcmdlbmNlOiBuZXcgTyhpLCBQLCBDZSksCiAgICBwcmVzc3VyZTogbmV3IE8oaSwgUCwgU2UpLAogICAgZ3JhZGllbnRTdWJ0cmFjdDogbmV3IE8oaSwgUCwgRmUpLAogICAgc3BsYXQ6IG5ldyBPKGksIFAsIFhlKSwKICAgIGN1cmw6IG5ldyBPKGksIFAsIEJlKSwKICAgIHZvcnRpY2l0eTogbmV3IE8oaSwgUCwgQWUpLAogICAgZGlzcGxheTogbmV3IE8oaSwgUCwgTGUpCiAgfTsKfQpmdW5jdGlvbiBhZShpLCBlLCByLCBvKSB7CiAgaS5hY3RpdmVUZXh0dXJlKGkuVEVYVFVSRTApOwogIGNvbnN0IHMgPSBpLmNyZWF0ZVRleHR1cmUoKTsKICBpLmJpbmRUZXh0dXJlKGkuVEVYVFVSRV8yRCwgcyksIGkudGV4UGFyYW1ldGVyaShpLlRFWFRVUkVfMkQsIGkuVEVYVFVSRV9NSU5fRklMVEVSLCBpLkxJTkVBUiksIGkudGV4UGFyYW1ldGVyaShpLlRFWFRVUkVfMkQsIGkuVEVYVFVSRV9NQUdfRklMVEVSLCBpLkxJTkVBUiksIGkudGV4UGFyYW1ldGVyaShpLlRFWFRVUkVfMkQsIGkuVEVYVFVSRV9XUkFQX1MsIGkuQ0xBTVBfVE9fRURHRSksIGkudGV4UGFyYW1ldGVyaShpLlRFWFRVUkVfMkQsIGkuVEVYVFVSRV9XUkFQX1QsIGkuQ0xBTVBfVE9fRURHRSksIGkudGV4SW1hZ2UyRChpLlRFWFRVUkVfMkQsIDAsIGUuaW50ZXJuYWxGb3JtYXQsIHIsIG8sIDAsIGUuZm9ybWF0LCBlLnR5cGUsIG51bGwpOwogIGNvbnN0IG4gPSBpLmNyZWF0ZUZyYW1lYnVmZmVyKCk7CiAgcmV0dXJuIGkuYmluZEZyYW1lYnVmZmVyKGkuRlJBTUVCVUZGRVIsIG4pLCBpLmZyYW1lYnVmZmVyVGV4dHVyZTJEKGkuRlJBTUVCVUZGRVIsIGkuQ09MT1JfQVRUQUNITUVOVDAsIGkuVEVYVFVSRV8yRCwgcywgMCksIHsgdGV4OiBzLCBmYm86IG4sIHdpZHRoOiByLCBoZWlnaHQ6IG8gfTsKfQpmdW5jdGlvbiBtZShpLCBlLCByLCBvKSB7CiAgbGV0IHMgPSBhZShpLCBlLCByLCBvKSwgbiA9IGFlKGksIGUsIHIsIG8pOwogIHJldHVybiB7CiAgICBnZXQgcmVhZCgpIHsKICAgICAgcmV0dXJuIHM7CiAgICB9LAogICAgZ2V0IHdyaXRlKCkgewogICAgICByZXR1cm4gbjsKICAgIH0sCiAgICBzd2FwKCkgewogICAgICBbcywgbl0gPSBbbiwgc107CiAgICB9LAogICAgZGlzcG9zZSgpIHsKICAgICAgaS5kZWxldGVUZXh0dXJlKHMudGV4KSwgaS5kZWxldGVGcmFtZWJ1ZmZlcihzLmZibyksIGkuZGVsZXRlVGV4dHVyZShuLnRleCksIGkuZGVsZXRlRnJhbWVidWZmZXIobi5mYm8pOwogICAgfQogIH07Cn0KZnVuY3Rpb24ga2UoaSkgewogIGNvbnN0IGUgPSBpLmNyZWF0ZUJ1ZmZlcigpOwogIHJldHVybiBpLmJpbmRCdWZmZXIoaS5BUlJBWV9CVUZGRVIsIGUpLCBpLmJ1ZmZlckRhdGEoaS5BUlJBWV9CVUZGRVIsIG5ldyBGbG9hdDMyQXJyYXkoWy0xLCAtMSwgLTEsIDEsIDEsIDEsIDEsIC0xXSksIGkuU1RBVElDX0RSQVcpLCBpLnZlcnRleEF0dHJpYlBvaW50ZXIoMCwgMiwgaS5GTE9BVCwgITEsIDAsIDApLCBpLmVuYWJsZVZlcnRleEF0dHJpYkFycmF5KDApLCBmdW5jdGlvbihvKSB7CiAgICBpLmJpbmRGcmFtZWJ1ZmZlcihpLkZSQU1FQlVGRkVSLCBvKSwgaS5kcmF3QXJyYXlzKGkuVFJJQU5HTEVfRkFOLCAwLCA0KTsKICB9Owp9CmZ1bmN0aW9uIHhlKGksIGUsIHIsIG8sIHMgPSAiY292ZXIiKSB7CiAgbGV0IG47CiAgcyA9PT0gImNvdmVyIiA/IG4gPSBNYXRoLm1heChyIC8gaSwgbyAvIGUpIDogcyA9PT0gImNvbnRhaW4iID8gbiA9IE1hdGgubWluKHIgLyBpLCBvIC8gZSkgOiB0eXBlb2YgcyA9PSAic3RyaW5nIiAmJiBzLmVuZHNXaXRoKCIlIikgPyBuID0gTWF0aC5taW4ociAvIGksIG8gLyBlKSAqIChwYXJzZUZsb2F0KHMpIC8gMTAwKSA6IHR5cGVvZiBzID09ICJzdHJpbmciICYmIHMuZW5kc1dpdGgoInB4IikgPyBuID0gcGFyc2VGbG9hdChzKSAvIE1hdGgubWF4KGksIGUpIDogdHlwZW9mIHMgPT0gIm51bWJlciIgPyBuID0gcyA6IG4gPSBNYXRoLm1heChyIC8gaSwgbyAvIGUpOwogIGNvbnN0IHUgPSBpICogbiwgVCA9IGUgKiBuOwogIHJldHVybiB7IHg6IChyIC0gdSkgLyAyLCB5OiAobyAtIFQpIC8gMiwgZHJhd1c6IHUsIGRyYXdIOiBUIH07Cn0KZnVuY3Rpb24gSWUoaSwgZSwgciwgbywgcyA9IG51bGwsIG4gPSAiY292ZXIiKSB7CiAgY29uc3QgeyB0ZXh0OiB1LCBmb250U2l6ZTogVCwgY29sb3I6IEUsIGZvbnRGYW1pbHk6IGMgPSAic2Fucy1zZXJpZiIsIGZvbnRXZWlnaHQ6IGggPSA5MDAgfSA9IG8sIFIgPSBuZXcgT2Zmc2NyZWVuQ2FudmFzKGUsIHIpLCBmID0gUi5nZXRDb250ZXh0KCIyZCIpOwogICgodWUpID0+IHsKICAgIGlmIChzKSB7CiAgICAgIGYuY2xlYXJSZWN0KDAsIDAsIGUsIHIpLCBmLmZpbGxTdHlsZSA9ICJibGFjayIsIGYuZmlsbFJlY3QoMCwgMCwgZSwgcik7CiAgICAgIGNvbnN0IHsgeDogbGUsIHk6IGNlLCBkcmF3VzogZmUsIGRyYXdIOiB2ZSB9ID0geGUoCiAgICAgICAgcy53aWR0aCwKICAgICAgICBzLmhlaWdodCwKICAgICAgICBlLAogICAgICAgIHIsCiAgICAgICAgbgogICAgICApOwogICAgICBmLmRyYXdJbWFnZShzLCBsZSwgY2UsIGZlLCB2ZSk7CiAgICB9IGVsc2UKICAgICAgZi5maWxsU3R5bGUgPSAiYmxhY2siLCBmLmZpbGxSZWN0KDAsIDAsIGUsIHIpOwogICAgZi5maWxsU3R5bGUgPSB1ZSwgZi5mb250ID0gYCR7aH0gJHtUfXB4ICR7Y31gLCBmLnRleHRBbGlnbiA9ICJjZW50ZXIiLCBmLnRleHRCYXNlbGluZSA9ICJtaWRkbGUiLCBmLmZpbGxUZXh0KHUsIGUgLyAyLCByIC8gMik7CiAgfSkoRSk7CiAgY29uc3QgWSA9IFEoaSwgUik7CiAgZi5maWxsU3R5bGUgPSAiYmxhY2siLCBmLmZpbGxSZWN0KDAsIDAsIGUsIHIpLCBmLmZpbGxTdHlsZSA9ICJ3aGl0ZSIsIGYuZm9udCA9IGAke2h9ICR7VH1weCAke2N9YCwgZi50ZXh0QWxpZ24gPSAiY2VudGVyIiwgZi50ZXh0QmFzZWxpbmUgPSAibWlkZGxlIiwgZi5maWxsVGV4dCh1LCBlIC8gMiwgciAvIDIpOwogIGNvbnN0IG9lID0gUShpLCBSKTsKICByZXR1cm4geyBiYWNrZ3JvdW5kVGV4OiBZLCBvYnN0YWNsZVRleDogb2UsIGNvdmVyYWdlVGV4OiBvZSB9Owp9CmZ1bmN0aW9uIE1lKGksIGUsIHIsIG8sIHMgPSAwLCBuID0gImNvdmVyIiwgdSA9IG51bGwsIFQgPSAiY292ZXIiKSB7CiAgY29uc3QgRSA9IG5ldyBPZmZzY3JlZW5DYW52YXMociwgbyksIGMgPSBFLmdldENvbnRleHQoIjJkIiksIHsgeDogaCwgeTogUiwgZHJhd1c6IGYsIGRyYXdIOiBEIH0gPSB4ZShlLndpZHRoLCBlLmhlaWdodCwgciwgbywgbik7CiAgaWYgKGMuY2xlYXJSZWN0KDAsIDAsIHIsIG8pLCBjLmZpbGxTdHlsZSA9ICJibGFjayIsIGMuZmlsbFJlY3QoMCwgMCwgciwgbyksIHUpIHsKICAgIGNvbnN0IHsKICAgICAgeDogbGUsCiAgICAgIHk6IGNlLAogICAgICBkcmF3VzogZmUsCiAgICAgIGRyYXdIOiB2ZQogICAgfSA9IHhlKHUud2lkdGgsIHUuaGVpZ2h0LCByLCBvLCBUKTsKICAgIGMuZmlsdGVyID0gYGJyaWdodG5lc3MoJHtzfSkgYmx1cig4cHgpYCwgYy5kcmF3SW1hZ2UodSwgbGUsIGNlLCBmZSwgdmUpLCBjLmZpbHRlciA9ICJub25lIjsKICB9CiAgYy5kcmF3SW1hZ2UoZSwgaCwgUiwgZiwgRCk7CiAgY29uc3QgWSA9IFEoaSwgRSk7CiAgYy5jbGVhclJlY3QoMCwgMCwgciwgbyksIGMuZmlsbFN0eWxlID0gImJsYWNrIiwgYy5maWxsUmVjdCgwLCAwLCByLCBvKSwgYy5maWx0ZXIgPSBgYnJpZ2h0bmVzcygke3N9KSBibHVyKDhweClgLCBjLmRyYXdJbWFnZShlLCBoLCBSLCBmLCBEKSwgYy5maWx0ZXIgPSAibm9uZSI7CiAgY29uc3Qgb2UgPSBRKGksIEUpOwogIGMuY2xlYXJSZWN0KDAsIDAsIHIsIG8pLCBjLmZpbGxTdHlsZSA9ICJibGFjayIsIGMuZmlsbFJlY3QoMCwgMCwgciwgbyksIGMuZmlsbFN0eWxlID0gIndoaXRlIiwgYy5maWxsUmVjdCgKICAgIE1hdGgubWF4KDAsIGgpLAogICAgTWF0aC5tYXgoMCwgUiksCiAgICBNYXRoLm1pbihmLCByIC0gTWF0aC5tYXgoMCwgaCkpLAogICAgTWF0aC5taW4oRCwgbyAtIE1hdGgubWF4KDAsIFIpKQogICk7CiAgY29uc3QgdWUgPSBRKGksIEUpOwogIHJldHVybiB7IGJhY2tncm91bmRUZXg6IFksIG9ic3RhY2xlVGV4OiBvZSwgY292ZXJhZ2VUZXg6IHVlIH07Cn0KZnVuY3Rpb24gUShpLCBlKSB7CiAgY29uc3QgciA9IGkuY3JlYXRlVGV4dHVyZSgpOwogIHJldHVybiBpLmJpbmRUZXh0dXJlKGkuVEVYVFVSRV8yRCwgciksIGkucGl4ZWxTdG9yZWkoaS5VTlBBQ0tfRkxJUF9ZX1dFQkdMLCAhMCksIGkudGV4SW1hZ2UyRChpLlRFWFRVUkVfMkQsIDAsIGkuUkdCQSwgaS5SR0JBLCBpLlVOU0lHTkVEX0JZVEUsIGUpLCBpLnRleFBhcmFtZXRlcmkoaS5URVhUVVJFXzJELCBpLlRFWFRVUkVfTUlOX0ZJTFRFUiwgaS5MSU5FQVIpLCBpLnRleFBhcmFtZXRlcmkoaS5URVhUVVJFXzJELCBpLlRFWFRVUkVfTUFHX0ZJTFRFUiwgaS5MSU5FQVIpLCBpLnRleFBhcmFtZXRlcmkoaS5URVhUVVJFXzJELCBpLlRFWFRVUkVfV1JBUF9TLCBpLkNMQU1QX1RPX0VER0UpLCBpLnRleFBhcmFtZXRlcmkoaS5URVhUVVJFXzJELCBpLlRFWFRVUkVfV1JBUF9ULCBpLkNMQU1QX1RPX0VER0UpLCByOwp9CmFzeW5jIGZ1bmN0aW9uIFZlKGkpIHsKICBjb25zdCBlID0gYXdhaXQgZmV0Y2goaSk7CiAgaWYgKCFlLm9rKQogICAgdGhyb3cgbmV3IEVycm9yKGBGYWlsZWQgdG8gZmV0Y2ggaW1hZ2U6ICR7aX0gKCR7ZS5zdGF0dXN9KWApOwogIGNvbnN0IHIgPSBhd2FpdCBlLmJsb2IoKTsKICByZXR1cm4gY3JlYXRlSW1hZ2VCaXRtYXAocik7Cn0KY29uc3QgRWUgPSB0eXBlb2YgcmVxdWVzdEFuaW1hdGlvbkZyYW1lIDwgInUiID8gcmVxdWVzdEFuaW1hdGlvbkZyYW1lLmJpbmQoZ2xvYmFsVGhpcykgOiAoaSkgPT4gc2V0VGltZW91dChpLCAxZTMgLyA2MCksIFdlID0gdHlwZW9mIGNhbmNlbEFuaW1hdGlvbkZyYW1lIDwgInUiID8gY2FuY2VsQW5pbWF0aW9uRnJhbWUuYmluZChnbG9iYWxUaGlzKSA6IGNsZWFyVGltZW91dCwgYmUgPSAwLjAxNiwgR2UgPSB7IHN0YW5kYXJkOiAwLCBnbGFzczogMSwgaW5rOiAyLCBhdXJvcmE6IDMsIHJpcHBsZTogNCB9Owp2YXIgViwgYiwgWiwgVywgSCwgcCwgeSwgaywgSSwgdywgZywgdiwgUywgRiwgWCwgTCwgVSwgQiwgQywgaiwgRywgeCwgXywgQSwgeiwgZWUsIE4sIEssIHRlLCBUZSwgTSwgcSwgJCwgSiwgcmUsIGRlLCBpZSwgcGUsIG5lLCBSZTsKY2xhc3MgemUgewogIGNvbnN0cnVjdG9yKGUsIHIgPSB7fSkgewogICAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAgICAvLyBQcml2YXRlIGhlbHBlcnMKICAgIC8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQogICAgbCh0aGlzLCBOKTsKICAgIGwodGhpcywgdGUpOwogICAgbCh0aGlzLCBNKTsKICAgIGwodGhpcywgJCk7CiAgICBsKHRoaXMsIHJlKTsKICAgIGwodGhpcywgaWUpOwogICAgbCh0aGlzLCBuZSk7CiAgICBsKHRoaXMsIFYsIHZvaWQgMCk7CiAgICBsKHRoaXMsIGIsIHZvaWQgMCk7CiAgICBsKHRoaXMsIFosIHZvaWQgMCk7CiAgICBsKHRoaXMsIFcsIHZvaWQgMCk7CiAgICBsKHRoaXMsIEgsIHZvaWQgMCk7CiAgICBsKHRoaXMsIHAsIDApOwogICAgbCh0aGlzLCB5LCAwKTsKICAgIGwodGhpcywgaywgMCk7CiAgICBsKHRoaXMsIEksIDApOwogICAgbCh0aGlzLCB3LCAxKTsKICAgIGwodGhpcywgZywgbnVsbCk7CiAgICBsKHRoaXMsIHYsIG51bGwpOwogICAgbCh0aGlzLCBTLCBudWxsKTsKICAgIGwodGhpcywgRiwgbnVsbCk7CiAgICBsKHRoaXMsIFgsIG51bGwpOwogICAgbCh0aGlzLCBMLCBudWxsKTsKICAgIGwodGhpcywgVSwgbnVsbCk7CiAgICBsKHRoaXMsIEIsIG51bGwpOwogICAgLy8gYmluYXJ5IGNvbnRlbnQgbWFzayBmb3IgdHJhbnNwYXJlbnQgY2FudmFzIHN1cHBvcnQKICAgIGwodGhpcywgQywgbnVsbCk7CiAgICAvLyBvcHRpb25hbCBiYWNrZ3JvdW5kIGltYWdlIChmcm9tIGJhY2tncm91bmRTcmMgcHJvcCkKICAgIGwodGhpcywgaiwgImNvdmVyIik7CiAgICBsKHRoaXMsIEcsIHZvaWQgMCk7CiAgICBsKHRoaXMsIHgsIHsgeDogMCwgeTogMCwgZHg6IDAsIGR5OiAwLCB0YXJnZXRYOiAwLCB0YXJnZXRZOiAwLCBtb3ZlZDogITEgfSk7CiAgICAvLyBTdG9yZXMgc291cmNlIHNvIHRleHR1cmVzIGNhbiBiZSByZWJ1aWx0IG9uIHJlc2l6ZQogICAgbCh0aGlzLCBfLCBudWxsKTsKICAgIGwodGhpcywgQSwgbnVsbCk7CiAgICBsKHRoaXMsIHosICExKTsKICAgIGwodGhpcywgZWUsICExKTsKICAgIGEodGhpcywgViwgZSksIGEodGhpcywgRywgX2UocikpOwogICAgY29uc3QgeyBnbDogbywgZXh0OiBzIH0gPSBQZShlKTsKICAgIGEodGhpcywgYiwgbyksIGEodGhpcywgWiwgcyksIGEodGhpcywgVywgT2UobykpLCBhKHRoaXMsIEgsIGtlKG8pKSwgby5jbGVhckNvbG9yKDAsIDAsIDAsIDApOwogIH0KICAvLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KICAvLyBQdWJsaWMgQVBJCiAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAgc2V0VGV4dFNvdXJjZShlKSB7CiAgICBhKHRoaXMsIF8sIHsgdHlwZTogInRleHQiLCBvcHRzOiBlIH0pLCBkKHRoaXMsIE4sIEspLmNhbGwodGhpcyksIGQodGhpcywgTSwgcSkuY2FsbCh0aGlzKSwgZCh0aGlzLCAkLCBKKS5jYWxsKHRoaXMpOwogIH0KICBhc3luYyBzZXRJbWFnZVNvdXJjZShlLCByID0gMCwgbyA9ICJjb3ZlciIpIHsKICAgIGNvbnN0IHMgPSBhd2FpdCBWZShlKTsKICAgIGlmICh0KHRoaXMsIGVlKSkgewogICAgICBzLmNsb3NlKCk7CiAgICAgIHJldHVybjsKICAgIH0KICAgIGEodGhpcywgXywgeyB0eXBlOiAiaW1hZ2UiLCBiaXRtYXA6IHMsIGVmZmVjdDogciwgc2l6ZTogbyB9KSwgZCh0aGlzLCBOLCBLKS5jYWxsKHRoaXMpLCBkKHRoaXMsIE0sIHEpLmNhbGwodGhpcyksIGQodGhpcywgJCwgSikuY2FsbCh0aGlzKTsKICB9CiAgc2V0SW1hZ2VCaXRtYXAoZSwgciA9IDAsIG8gPSAiY292ZXIiKSB7CiAgICBhKHRoaXMsIF8sIHsgdHlwZTogImltYWdlIiwgYml0bWFwOiBlLCBlZmZlY3Q6IHIsIHNpemU6IG8gfSksIGQodGhpcywgTiwgSykuY2FsbCh0aGlzKSwgZCh0aGlzLCBNLCBxKS5jYWxsKHRoaXMpLCBkKHRoaXMsICQsIEopLmNhbGwodGhpcyk7CiAgfQogIHNldEJhY2tncm91bmQoZSwgciA9ICJjb3ZlciIpIHsKICAgIHQodGhpcywgQykgJiYgdCh0aGlzLCBDKSAhPT0gZSAmJiB0KHRoaXMsIEMpLmNsb3NlKCksIGEodGhpcywgQywgZSksIGEodGhpcywgaiwgciA/PyAiY292ZXIiKSwgdCh0aGlzLCBfKSAmJiB0KHRoaXMsIHApID4gMCAmJiBkKHRoaXMsIE0sIHEpLmNhbGwodGhpcyk7CiAgfQogIGhhbmRsZU1vdmUoZSwgciwgbyA9IDEpIHsKICAgIHQodGhpcywgeCkubW92ZWQgPSAhMCwgdCh0aGlzLCB4KS5keCA9IChlIC0gdCh0aGlzLCB4KS50YXJnZXRYKSAqIG8sIHQodGhpcywgeCkuZHkgPSAociAtIHQodGhpcywgeCkudGFyZ2V0WSkgKiBvLCB0KHRoaXMsIHgpLnRhcmdldFggPSBlLCB0KHRoaXMsIHgpLnRhcmdldFkgPSByOwogIH0KICAvKioKICAgKiBJbW1lZGlhdGVseSBhcHBsaWVzIG9uZSBmbHVpZCBzcGxhdCBhdCAoeCwgeSkgd2l0aCBleHBsaWNpdCB2ZWxvY2l0eSAodngsIHZ5KS4KICAgKiBTYWZlIHRvIGNhbGwgbXVsdGlwbGUgdGltZXMgcGVyIGZyYW1lIOKAlCBlYWNoIGNhbGwgd3JpdGVzIGRpcmVjdGx5IHRvIHRoZSBGQk9zLgogICAqIERlc2lnbmVkIGZvciBwcm9ncmFtbWF0aWMgdXNlIGNhc2VzIChlLmcuIHBhcnRpY2xlIHN5c3RlbXMsIGF0dHJhY3RvciBwYXRocykKICAgKiB3aGVyZSB5b3Ugd2FudCBOIGluZGVwZW5kZW50IGluamVjdGlvbiBwb2ludHMgcGVyIGZyYW1lIHdpdGhvdXQgZmxvb2RpbmcgdGhlCiAgICogbW91c2Utc3RhdGUgbWFjaGluZSBvciB0aGUgd29ya2VyIG1lc3NhZ2UgcXVldWUuCiAgICovCiAgc3BsYXQoZSwgciwgbywgcywgbiA9IDEpIHsKICAgIGlmICghdCh0aGlzLCB6KSB8fCB0KHRoaXMsIHApID09PSAwKQogICAgICByZXR1cm47CiAgICBjb25zdCB1ID0gdCh0aGlzLCBiKSwgVCA9IHQodGhpcywgRyksIHsgc3BsYXQ6IEUgfSA9IHQodGhpcywgVyksIGMgPSB0KHRoaXMsIEgpOwogICAgdS52aWV3cG9ydCgwLCAwLCB0KHRoaXMsIGspLCB0KHRoaXMsIEkpKSwgRS5iaW5kKCksIHUudW5pZm9ybTFmKEUudW5pZm9ybXMuYXNwZWN0UmF0aW8sIHQodGhpcywgcCkgLyB0KHRoaXMsIHkpKSwgdS51bmlmb3JtMmYoRS51bmlmb3Jtcy5wb2ludCwgZSAqIHQodGhpcywgdykgLyB0KHRoaXMsIHApLCAxIC0gciAqIHQodGhpcywgdykgLyB0KHRoaXMsIHkpKSwgdS51bmlmb3JtMWYoRS51bmlmb3Jtcy5yYWRpdXMsIFQuc3BsYXRSYWRpdXMpLCB1LnVuaWZvcm0xaShFLnVuaWZvcm1zLnVUYXJnZXQsIDApLCB1LmFjdGl2ZVRleHR1cmUodS5URVhUVVJFMCksIHUuYmluZFRleHR1cmUodS5URVhUVVJFXzJELCB0KHRoaXMsIHYpLnJlYWQudGV4KSwgdS51bmlmb3JtM2YoRS51bmlmb3Jtcy5jb2xvciwgbyAqIFQuc3BsYXRGb3JjZSAqIG4sIC1zICogVC5zcGxhdEZvcmNlICogbiwgMCksIGModCh0aGlzLCB2KS53cml0ZS5mYm8pLCB0KHRoaXMsIHYpLnN3YXAoKSwgdS5hY3RpdmVUZXh0dXJlKHUuVEVYVFVSRTApLCB1LmJpbmRUZXh0dXJlKHUuVEVYVFVSRV8yRCwgdCh0aGlzLCBnKS5yZWFkLnRleCksIHUudW5pZm9ybTNmKEUudW5pZm9ybXMuY29sb3IsIG4sIG4sIG4pLCBjKHQodGhpcywgZykud3JpdGUuZmJvKSwgdCh0aGlzLCBnKS5zd2FwKCk7CiAgfQogIHJlc2l6ZShlLCByLCBvKSB7CiAgICBpZiAobyAhPT0gdm9pZCAwID8gYSh0aGlzLCB3LCBvKSA6IHR5cGVvZiB3aW5kb3cgPCAidSIgJiYgd2luZG93LmRldmljZVBpeGVsUmF0aW8gJiYgYSh0aGlzLCB3LCB3aW5kb3cuZGV2aWNlUGl4ZWxSYXRpbyksIGUgIT09IHZvaWQgMCAmJiBlID4gMCkgewogICAgICBpZiAociA9PT0gdm9pZCAwIHx8IHIgPD0gMCkKICAgICAgICByZXR1cm47CiAgICAgIGEodGhpcywgcCwgdCh0aGlzLCBWKS53aWR0aCA9IGUpLCBhKHRoaXMsIHksIHQodGhpcywgVikuaGVpZ2h0ID0gciksIGEodGhpcywgaywgZSA+PiAxKSwgYSh0aGlzLCBJLCByID4+IDEpLCBkKHRoaXMsIHRlLCBUZSkuY2FsbCh0aGlzKTsKICAgIH0gZWxzZQogICAgICBkKHRoaXMsIE4sIEspLmNhbGwodGhpcyk7CiAgICB0KHRoaXMsIF8pICYmIGQodGhpcywgTSwgcSkuY2FsbCh0aGlzKSwgZCh0aGlzLCAkLCBKKS5jYWxsKHRoaXMpOwogIH0KICB1cGRhdGVDb25maWcoZSkgewogICAgT2JqZWN0LmFzc2lnbih0KHRoaXMsIEcpLCBlKTsKICB9CiAgZGVzdHJveSgpIHsKICAgIGEodGhpcywgZWUsICEwKSwgdGhpcy5zdG9wKCk7CiAgICBjb25zdCBlID0gdCh0aGlzLCBiKTsKICAgIGQodGhpcywgcmUsIGRlKS5jYWxsKHRoaXMpLCBkKHRoaXMsIGllLCBwZSkuY2FsbCh0aGlzKSwgdCh0aGlzLCBDKSAmJiAodCh0aGlzLCBDKS5jbG9zZSgpLCBhKHRoaXMsIEMsIG51bGwpKTsKICAgIGZvciAoY29uc3QgbyBvZiBPYmplY3QudmFsdWVzKHQodGhpcywgVykpKQogICAgICBvLmRpc3Bvc2UoKTsKICAgIGNvbnN0IHIgPSBlLmdldEV4dGVuc2lvbigiV0VCR0xfbG9zZV9jb250ZXh0Iik7CiAgICByID09IG51bGwgfHwgci5sb3NlQ29udGV4dCgpOwogIH0KICAvLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KICAvLyBMb29wIGNvbnRyb2wKICAvLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KICBzdGFydCgpIHsKICAgIGlmICh0KHRoaXMsIEEpICE9PSBudWxsKQogICAgICByZXR1cm47CiAgICBjb25zdCBlID0gKCkgPT4gewogICAgICBkKHRoaXMsIG5lLCBSZSkuY2FsbCh0aGlzKSwgYSh0aGlzLCBBLCBFZShlKSk7CiAgICB9OwogICAgYSh0aGlzLCBBLCBFZShlKSk7CiAgfQogIHN0b3AoKSB7CiAgICB0KHRoaXMsIEEpICE9PSBudWxsICYmIChXZSh0KHRoaXMsIEEpKSwgYSh0aGlzLCBBLCBudWxsKSk7CiAgfQogIGdldCBpc1J1bm5pbmcoKSB7CiAgICByZXR1cm4gdCh0aGlzLCBBKSAhPT0gbnVsbDsKICB9Cn0KViA9IG5ldyBXZWFrTWFwKCksIGIgPSBuZXcgV2Vha01hcCgpLCBaID0gbmV3IFdlYWtNYXAoKSwgVyA9IG5ldyBXZWFrTWFwKCksIEggPSBuZXcgV2Vha01hcCgpLCBwID0gbmV3IFdlYWtNYXAoKSwgeSA9IG5ldyBXZWFrTWFwKCksIGsgPSBuZXcgV2Vha01hcCgpLCBJID0gbmV3IFdlYWtNYXAoKSwgdyA9IG5ldyBXZWFrTWFwKCksIGcgPSBuZXcgV2Vha01hcCgpLCB2ID0gbmV3IFdlYWtNYXAoKSwgUyA9IG5ldyBXZWFrTWFwKCksIEYgPSBuZXcgV2Vha01hcCgpLCBYID0gbmV3IFdlYWtNYXAoKSwgTCA9IG5ldyBXZWFrTWFwKCksIFUgPSBuZXcgV2Vha01hcCgpLCBCID0gbmV3IFdlYWtNYXAoKSwgQyA9IG5ldyBXZWFrTWFwKCksIGogPSBuZXcgV2Vha01hcCgpLCBHID0gbmV3IFdlYWtNYXAoKSwgeCA9IG5ldyBXZWFrTWFwKCksIF8gPSBuZXcgV2Vha01hcCgpLCBBID0gbmV3IFdlYWtNYXAoKSwgeiA9IG5ldyBXZWFrTWFwKCksIGVlID0gbmV3IFdlYWtNYXAoKSwgTiA9IG5ldyBXZWFrU2V0KCksIEsgPSBmdW5jdGlvbigpIHsKICBjb25zdCBlID0gdCh0aGlzLCBWKTsKICAiY2xpZW50V2lkdGgiIGluIGUgJiYgZS5jbGllbnRXaWR0aCA+IDAgPyAoYSh0aGlzLCB3LCB0eXBlb2Ygd2luZG93IDwgInUiICYmIHdpbmRvdy5kZXZpY2VQaXhlbFJhdGlvIHx8IDEpLCBhKHRoaXMsIHAsIGUud2lkdGggPSBNYXRoLnJvdW5kKGUuY2xpZW50V2lkdGggKiB0KHRoaXMsIHcpKSksIGEodGhpcywgeSwgZS5oZWlnaHQgPSBNYXRoLnJvdW5kKGUuY2xpZW50SGVpZ2h0ICogdCh0aGlzLCB3KSkpKSA6IChhKHRoaXMsIHAsIGUud2lkdGgpLCBhKHRoaXMsIHksIGUuaGVpZ2h0KSksICEodCh0aGlzLCBwKSA9PT0gMCB8fCB0KHRoaXMsIHkpID09PSAwKSAmJiAoYSh0aGlzLCBrLCB0KHRoaXMsIHApID4+IDEpLCBhKHRoaXMsIEksIHQodGhpcywgeSkgPj4gMSksIGQodGhpcywgdGUsIFRlKS5jYWxsKHRoaXMpKTsKfSwgdGUgPSBuZXcgV2Vha1NldCgpLCBUZSA9IGZ1bmN0aW9uKCkgewogIGNvbnN0IGUgPSB0KHRoaXMsIGIpLCByID0gdCh0aGlzLCBaKSwgbyA9IHQodGhpcywgayksIHMgPSB0KHRoaXMsIEkpOwogIGQodGhpcywgcmUsIGRlKS5jYWxsKHRoaXMpLCBhKHRoaXMsIGcsIG1lKGUsIHIsIG8sIHMpKSwgYSh0aGlzLCB2LCBtZShlLCByLCBvLCBzKSksIGEodGhpcywgRiwgbWUoZSwgciwgbywgcykpLCBhKHRoaXMsIFMsIGFlKGUsIHIsIG8sIHMpKSwgYSh0aGlzLCBYLCBhZShlLCByLCBvLCBzKSk7Cn0sIE0gPSBuZXcgV2Vha1NldCgpLCBxID0gZnVuY3Rpb24oKSB7CiAgaWYgKCEoIXQodGhpcywgXykgfHwgdCh0aGlzLCBwKSA9PT0gMCkpIHsKICAgIGlmIChkKHRoaXMsIGllLCBwZSkuY2FsbCh0aGlzKSwgdCh0aGlzLCBfKS50eXBlID09PSAidGV4dCIpIHsKICAgICAgY29uc3QgeyBiYWNrZ3JvdW5kVGV4OiBlLCBvYnN0YWNsZVRleDogciwgY292ZXJhZ2VUZXg6IG8gfSA9IEllKAogICAgICAgIHQodGhpcywgYiksCiAgICAgICAgdCh0aGlzLCBwKSwKICAgICAgICB0KHRoaXMsIHkpLAogICAgICAgIHQodGhpcywgXykub3B0cywKICAgICAgICB0KHRoaXMsIEMpLAogICAgICAgIHQodGhpcywgaikKICAgICAgKTsKICAgICAgYSh0aGlzLCBMLCBlKSwgYSh0aGlzLCBVLCByKSwgYSh0aGlzLCBCLCBvKTsKICAgIH0gZWxzZSB7CiAgICAgIGNvbnN0IHsgYmFja2dyb3VuZFRleDogZSwgb2JzdGFjbGVUZXg6IHIsIGNvdmVyYWdlVGV4OiBvIH0gPSBNZSgKICAgICAgICB0KHRoaXMsIGIpLAogICAgICAgIHQodGhpcywgXykuYml0bWFwLAogICAgICAgIHQodGhpcywgcCksCiAgICAgICAgdCh0aGlzLCB5KSwKICAgICAgICB0KHRoaXMsIF8pLmVmZmVjdCwKICAgICAgICB0KHRoaXMsIF8pLnNpemUsCiAgICAgICAgdCh0aGlzLCBDKSwKICAgICAgICB0KHRoaXMsIGopCiAgICAgICk7CiAgICAgIGEodGhpcywgTCwgZSksIGEodGhpcywgVSwgciksIGEodGhpcywgQiwgbyk7CiAgICB9CiAgICBhKHRoaXMsIHosICEwKTsKICB9Cn0sICQgPSBuZXcgV2Vha1NldCgpLCBKID0gZnVuY3Rpb24oKSB7CiAgdCh0aGlzLCB6KSAmJiAhdGhpcy5pc1J1bm5pbmcgJiYgdGhpcy5zdGFydCgpOwp9LCByZSA9IG5ldyBXZWFrU2V0KCksIGRlID0gZnVuY3Rpb24oKSB7CiAgdmFyIGUsIHIsIG87CiAgKGUgPSB0KHRoaXMsIGcpKSA9PSBudWxsIHx8IGUuZGlzcG9zZSgpLCAociA9IHQodGhpcywgdikpID09IG51bGwgfHwgci5kaXNwb3NlKCksIChvID0gdCh0aGlzLCBGKSkgPT0gbnVsbCB8fCBvLmRpc3Bvc2UoKSwgdCh0aGlzLCBTKSAmJiAodCh0aGlzLCBiKS5kZWxldGVUZXh0dXJlKHQodGhpcywgUykudGV4KSwgdCh0aGlzLCBiKS5kZWxldGVGcmFtZWJ1ZmZlcih0KHRoaXMsIFMpLmZibykpLCB0KHRoaXMsIFgpICYmICh0KHRoaXMsIGIpLmRlbGV0ZVRleHR1cmUodCh0aGlzLCBYKS50ZXgpLCB0KHRoaXMsIGIpLmRlbGV0ZUZyYW1lYnVmZmVyKHQodGhpcywgWCkuZmJvKSksIGEodGhpcywgZywgYSh0aGlzLCB2LCBhKHRoaXMsIEYsIGEodGhpcywgUywgYSh0aGlzLCBYLCBudWxsKSkpKSk7Cn0sIGllID0gbmV3IFdlYWtTZXQoKSwgcGUgPSBmdW5jdGlvbigpIHsKICB0KHRoaXMsIEwpICYmIHQodGhpcywgYikuZGVsZXRlVGV4dHVyZSh0KHRoaXMsIEwpKSwgdCh0aGlzLCBVKSAmJiB0KHRoaXMsIGIpLmRlbGV0ZVRleHR1cmUodCh0aGlzLCBVKSksIHQodGhpcywgQikgJiYgdCh0aGlzLCBCKSAhPT0gdCh0aGlzLCBVKSAmJiB0KHRoaXMsIGIpLmRlbGV0ZVRleHR1cmUodCh0aGlzLCBCKSksIGEodGhpcywgTCwgYSh0aGlzLCBVLCBhKHRoaXMsIEIsIG51bGwpKSk7Cn0sIG5lID0gbmV3IFdlYWtTZXQoKSwgUmUgPSBmdW5jdGlvbigpIHsKICBpZiAoIXQodGhpcywgeikgfHwgdCh0aGlzLCBwKSA9PT0gMCB8fCAhdCh0aGlzLCBnKSB8fCAhdCh0aGlzLCB2KSkKICAgIHJldHVybjsKICBjb25zdCBlID0gdCh0aGlzLCBiKSwgciA9IHQodGhpcywgRyksIHsgYWR2ZWN0aW9uOiBvLCBkaXZlcmdlbmNlOiBzLCBwcmVzc3VyZTogbiwgZ3JhZGllbnRTdWJ0cmFjdDogdSwgc3BsYXQ6IFQsIGN1cmw6IEUsIHZvcnRpY2l0eTogYywgZGlzcGxheTogaCB9ID0gdCh0aGlzLCBXKTsKICB0KHRoaXMsIHgpLnggKz0gKHQodGhpcywgeCkudGFyZ2V0WCAtIHQodGhpcywgeCkueCkgKiAwLjE1LCB0KHRoaXMsIHgpLnkgKz0gKHQodGhpcywgeCkudGFyZ2V0WSAtIHQodGhpcywgeCkueSkgKiAwLjE1OwogIGNvbnN0IFIgPSB0KHRoaXMsIGspLCBmID0gdCh0aGlzLCBJKSwgRCA9IHQodGhpcywgSCk7CiAgZS52aWV3cG9ydCgwLCAwLCBSLCBmKSwgby5iaW5kKCksIGUudW5pZm9ybTJmKG8udW5pZm9ybXMudGV4ZWxTaXplLCAxIC8gUiwgMSAvIGYpLCBlLnVuaWZvcm0xZihvLnVuaWZvcm1zLmR0LCBiZSksIGUudW5pZm9ybTFpKG8udW5pZm9ybXMudU9ic3RhY2xlLCAwKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTApLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBVKSksIGUudW5pZm9ybTFmKG8udW5pZm9ybXMuZGlzc2lwYXRpb24sIHIudmVsb2NpdHlEaXNzaXBhdGlvbiksIGUudW5pZm9ybTFpKG8udW5pZm9ybXMudVZlbG9jaXR5LCAxKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTEpLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCB2KS5yZWFkLnRleCksIGUudW5pZm9ybTFpKG8udW5pZm9ybXMudVNvdXJjZSwgMSksIEQodCh0aGlzLCB2KS53cml0ZS5mYm8pLCB0KHRoaXMsIHYpLnN3YXAoKSwgZS51bmlmb3JtMWYoby51bmlmb3Jtcy5kaXNzaXBhdGlvbiwgci5kZW5zaXR5RGlzc2lwYXRpb24pLCBlLnVuaWZvcm0xaShvLnVuaWZvcm1zLnVTb3VyY2UsIDIpLCBlLmFjdGl2ZVRleHR1cmUoZS5URVhUVVJFMiksIGUuYmluZFRleHR1cmUoZS5URVhUVVJFXzJELCB0KHRoaXMsIGcpLnJlYWQudGV4KSwgRCh0KHRoaXMsIGcpLndyaXRlLmZibyksIHQodGhpcywgZykuc3dhcCgpLCBFLmJpbmQoKSwgZS51bmlmb3JtMmYoRS51bmlmb3Jtcy50ZXhlbFNpemUsIDEgLyBSLCAxIC8gZiksIGUudW5pZm9ybTFpKEUudW5pZm9ybXMudVZlbG9jaXR5LCAwKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTApLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCB2KS5yZWFkLnRleCksIEQodCh0aGlzLCBYKS5mYm8pLCBjLmJpbmQoKSwgZS51bmlmb3JtMmYoYy51bmlmb3Jtcy50ZXhlbFNpemUsIDEgLyBSLCAxIC8gZiksIGUudW5pZm9ybTFmKGMudW5pZm9ybXMuY3VybCwgci5jdXJsKSwgZS51bmlmb3JtMWYoYy51bmlmb3Jtcy5kdCwgYmUpLCBlLnVuaWZvcm0xaShjLnVuaWZvcm1zLnVWZWxvY2l0eSwgMCksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUwKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgdikucmVhZC50ZXgpLCBlLnVuaWZvcm0xaShjLnVuaWZvcm1zLnVDdXJsLCAxKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTEpLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBYKS50ZXgpLCBEKHQodGhpcywgdikud3JpdGUuZmJvKSwgdCh0aGlzLCB2KS5zd2FwKCksIHQodGhpcywgeCkubW92ZWQgJiYgKFQuYmluZCgpLCBlLnVuaWZvcm0xZihULnVuaWZvcm1zLmFzcGVjdFJhdGlvLCB0KHRoaXMsIHApIC8gdCh0aGlzLCB5KSksIGUudW5pZm9ybTJmKFQudW5pZm9ybXMucG9pbnQsIHQodGhpcywgeCkueCAqIHQodGhpcywgdykgLyB0KHRoaXMsIHApLCAxIC0gdCh0aGlzLCB4KS55ICogdCh0aGlzLCB3KSAvIHQodGhpcywgeSkpLCBlLnVuaWZvcm0xZihULnVuaWZvcm1zLnJhZGl1cywgci5zcGxhdFJhZGl1cyksIGUudW5pZm9ybTFpKFQudW5pZm9ybXMudVRhcmdldCwgMCksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUwKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgdikucmVhZC50ZXgpLCBlLnVuaWZvcm0zZihULnVuaWZvcm1zLmNvbG9yLCB0KHRoaXMsIHgpLmR4ICogci5zcGxhdEZvcmNlLCAtdCh0aGlzLCB4KS5keSAqIHIuc3BsYXRGb3JjZSwgMCksIEQodCh0aGlzLCB2KS53cml0ZS5mYm8pLCB0KHRoaXMsIHYpLnN3YXAoKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTApLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBnKS5yZWFkLnRleCksIGUudW5pZm9ybTNmKFQudW5pZm9ybXMuY29sb3IsIDEsIDEsIDEpLCBEKHQodGhpcywgZykud3JpdGUuZmJvKSwgdCh0aGlzLCBnKS5zd2FwKCksIHQodGhpcywgeCkubW92ZWQgPSAhMSksIHMuYmluZCgpLCBlLnVuaWZvcm0yZihzLnVuaWZvcm1zLnRleGVsU2l6ZSwgMSAvIFIsIDEgLyBmKSwgZS51bmlmb3JtMWkocy51bmlmb3Jtcy51VmVsb2NpdHksIDApLCBlLmFjdGl2ZVRleHR1cmUoZS5URVhUVVJFMCksIGUuYmluZFRleHR1cmUoZS5URVhUVVJFXzJELCB0KHRoaXMsIHYpLnJlYWQudGV4KSwgZS51bmlmb3JtMWkocy51bmlmb3Jtcy51T2JzdGFjbGUsIDEpLCBlLmFjdGl2ZVRleHR1cmUoZS5URVhUVVJFMSksIGUuYmluZFRleHR1cmUoZS5URVhUVVJFXzJELCB0KHRoaXMsIFUpKSwgRCh0KHRoaXMsIFMpLmZibyksIG4uYmluZCgpLCBlLnVuaWZvcm0yZihuLnVuaWZvcm1zLnRleGVsU2l6ZSwgMSAvIFIsIDEgLyBmKSwgZS51bmlmb3JtMWkobi51bmlmb3Jtcy51RGl2ZXJnZW5jZSwgMCksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUwKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgUykudGV4KSwgZS51bmlmb3JtMWkobi51bmlmb3Jtcy51T2JzdGFjbGUsIDEpLCBlLmFjdGl2ZVRleHR1cmUoZS5URVhUVVJFMSksIGUuYmluZFRleHR1cmUoZS5URVhUVVJFXzJELCB0KHRoaXMsIFUpKTsKICBmb3IgKGxldCBZID0gMDsgWSA8IHIucHJlc3N1cmVJdGVyYXRpb25zOyBZKyspCiAgICBlLnVuaWZvcm0xaShuLnVuaWZvcm1zLnVQcmVzc3VyZSwgMiksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUyKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgRikucmVhZC50ZXgpLCBEKHQodGhpcywgRikud3JpdGUuZmJvKSwgdCh0aGlzLCBGKS5zd2FwKCk7CiAgdS5iaW5kKCksIGUudW5pZm9ybTJmKHUudW5pZm9ybXMudGV4ZWxTaXplLCAxIC8gUiwgMSAvIGYpLCBlLnVuaWZvcm0xaSh1LnVuaWZvcm1zLnVQcmVzc3VyZSwgMCksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUwKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgRikucmVhZC50ZXgpLCBlLnVuaWZvcm0xaSh1LnVuaWZvcm1zLnVWZWxvY2l0eSwgMSksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUxKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgdikucmVhZC50ZXgpLCBlLnVuaWZvcm0xaSh1LnVuaWZvcm1zLnVPYnN0YWNsZSwgMiksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUyKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgVSkpLCBEKHQodGhpcywgdikud3JpdGUuZmJvKSwgdCh0aGlzLCB2KS5zd2FwKCksIGUudmlld3BvcnQoMCwgMCwgdCh0aGlzLCBwKSwgdCh0aGlzLCB5KSksIGUuYmluZEZyYW1lYnVmZmVyKGUuRlJBTUVCVUZGRVIsIG51bGwpLCBlLmNsZWFyKGUuQ09MT1JfQlVGRkVSX0JJVCksIGguYmluZCgpLCBlLnVuaWZvcm0yZihoLnVuaWZvcm1zLnRleGVsU2l6ZSwgMSAvIHQodGhpcywgcCksIDEgLyB0KHRoaXMsIHkpKSwgZS51bmlmb3JtM2Z2KGgudW5pZm9ybXMudVdhdGVyQ29sb3IsIHIud2F0ZXJDb2xvciksIGUudW5pZm9ybTNmdihoLnVuaWZvcm1zLnVHbG93Q29sb3IsIHIuZ2xvd0NvbG9yKSwgZS51bmlmb3JtMWYoaC51bmlmb3Jtcy51UmVmcmFjdGlvbiwgci5yZWZyYWN0aW9uKSwgZS51bmlmb3JtMWYoaC51bmlmb3Jtcy51U3BlY3VsYXJFeHAsIHIuc3BlY3VsYXJFeHApLCBlLnVuaWZvcm0xZihoLnVuaWZvcm1zLnVTaGluZSwgci5zaGluZSksIGUudW5pZm9ybTFmKGgudW5pZm9ybXMudVdhcnBTdHJlbmd0aCwgci53YXJwU3RyZW5ndGggPz8gMC4wMTUpLCBlLnVuaWZvcm0xaShoLnVuaWZvcm1zLnVBbGdvcml0aG0sIEdlW3IuYWxnb3JpdGhtXSA/PyAwKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTApLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBnKS5yZWFkLnRleCksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUxKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgVSkpLCBlLmFjdGl2ZVRleHR1cmUoZS5URVhUVVJFMiksIGUuYmluZFRleHR1cmUoZS5URVhUVVJFXzJELCB0KHRoaXMsIEwpKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTMpLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBCKSksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkU0KSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgdikucmVhZC50ZXgpLCBlLnVuaWZvcm0xaShoLnVuaWZvcm1zLnVUZXh0dXJlLCAwKSwgZS51bmlmb3JtMWkoaC51bmlmb3Jtcy51T2JzdGFjbGUsIDEpLCBlLnVuaWZvcm0xaShoLnVuaWZvcm1zLnVCYWNrZ3JvdW5kLCAyKSwgZS51bmlmb3JtMWkoaC51bmlmb3Jtcy51Q292ZXJhZ2UsIDMpLCBlLnVuaWZvcm0xaShoLnVuaWZvcm1zLnVWZWxvY2l0eSwgNCksIEQobnVsbCk7Cn07CmxldCBtID0gbnVsbDsKc2VsZi5vbm1lc3NhZ2UgPSBhc3luYyAoaSkgPT4gewogIGNvbnN0IHsgdHlwZTogZSwgLi4uciB9ID0gaS5kYXRhOwogIHRyeSB7CiAgICBzd2l0Y2ggKGUpIHsKICAgICAgY2FzZSAiaW5pdCI6IHsKICAgICAgICBjb25zdCB7IGNhbnZhczogbywgd2lkdGg6IHMsIGhlaWdodDogbiwgY29uZmlnOiB1LCBkcHI6IFQgfSA9IHI7CiAgICAgICAgby53aWR0aCA9IHMsIG8uaGVpZ2h0ID0gbiwgbSA9IG5ldyB6ZShvLCB1KSwgbS5yZXNpemUocywgbiwgVCB8fCAxKSwgc2VsZi5wb3N0TWVzc2FnZSh7IHR5cGU6ICJyZWFkeSIgfSk7CiAgICAgICAgYnJlYWs7CiAgICAgIH0KICAgICAgY2FzZSAic2V0VGV4dFNvdXJjZSI6IHsKICAgICAgICBpZiAoIW0pCiAgICAgICAgICByZXR1cm47CiAgICAgICAgbS5zZXRUZXh0U291cmNlKHIub3B0cyk7CiAgICAgICAgYnJlYWs7CiAgICAgIH0KICAgICAgY2FzZSAic2V0SW1hZ2VTb3VyY2UiOiB7CiAgICAgICAgaWYgKCFtKQogICAgICAgICAgcmV0dXJuOwogICAgICAgIGF3YWl0IG0uc2V0SW1hZ2VTb3VyY2UoCiAgICAgICAgICByLnNyYywKICAgICAgICAgIHIuZWZmZWN0LAogICAgICAgICAgci5zaXplCiAgICAgICAgKTsKICAgICAgICBicmVhazsKICAgICAgfQogICAgICBjYXNlICJzZXRJbWFnZUJpdG1hcCI6IHsKICAgICAgICBpZiAoIW0pCiAgICAgICAgICByZXR1cm47CiAgICAgICAgbS5zZXRJbWFnZUJpdG1hcCgKICAgICAgICAgIHIuYml0bWFwLAogICAgICAgICAgci5lZmZlY3QsCiAgICAgICAgICByLnNpemUKICAgICAgICApOwogICAgICAgIGJyZWFrOwogICAgICB9CiAgICAgIGNhc2UgInNldEJhY2tncm91bmQiOiB7CiAgICAgICAgaWYgKCFtKQogICAgICAgICAgcmV0dXJuOwogICAgICAgIG0uc2V0QmFja2dyb3VuZChyLmJpdG1hcCwgci5zaXplKTsKICAgICAgICBicmVhazsKICAgICAgfQogICAgICBjYXNlICJzcGxhdCI6IHsKICAgICAgICBpZiAoIW0pCiAgICAgICAgICByZXR1cm47CiAgICAgICAgbS5zcGxhdCgKICAgICAgICAgIHIueCwKICAgICAgICAgIHIueSwKICAgICAgICAgIHIudngsCiAgICAgICAgICByLnZ5LAogICAgICAgICAgci5zdHJlbmd0aCA/PyAxCiAgICAgICAgKTsKICAgICAgICBicmVhazsKICAgICAgfQogICAgICBjYXNlICJtb3ZlIjogewogICAgICAgIGlmICghbSkKICAgICAgICAgIHJldHVybjsKICAgICAgICBtLmhhbmRsZU1vdmUoci54LCByLnksIHIuc3RyZW5ndGggPz8gMSk7CiAgICAgICAgYnJlYWs7CiAgICAgIH0KICAgICAgY2FzZSAicmVzaXplIjogewogICAgICAgIGlmICghbSkKICAgICAgICAgIHJldHVybjsKICAgICAgICBtLnJlc2l6ZShyLndpZHRoLCByLmhlaWdodCwgci5kcHIpOwogICAgICAgIGJyZWFrOwogICAgICB9CiAgICAgIGNhc2UgInVwZGF0ZUNvbmZpZyI6IHsKICAgICAgICBpZiAoIW0pCiAgICAgICAgICByZXR1cm47CiAgICAgICAgbS51cGRhdGVDb25maWcoci5jb25maWcpOwogICAgICAgIGJyZWFrOwogICAgICB9CiAgICAgIGNhc2UgImRlc3Ryb3kiOiB7CiAgICAgICAgbSA9PSBudWxsIHx8IG0uZGVzdHJveSgpLCBtID0gbnVsbDsKICAgICAgICBicmVhazsKICAgICAgfQogICAgICBkZWZhdWx0OgogICAgICAgIGNvbnNvbGUud2FybigiW2ZsdWlkaXR5LWpzIHdvcmtlcl0gVW5rbm93biBtZXNzYWdlIHR5cGU6IiwgZSk7CiAgICB9CiAgfSBjYXRjaCAobykgewogICAgc2VsZi5wb3N0TWVzc2FnZSh7IHR5cGU6ICJlcnJvciIsIG1lc3NhZ2U6IChvID09IG51bGwgPyB2b2lkIDAgOiBvLm1lc3NhZ2UpID8/IFN0cmluZyhvKSB9KTsKICB9Cn07Cg==", FI = typeof window < "u" && window.Blob && new Blob([atob(zI)], { type: "text/javascript;charset=utf-8" });
674
- function ng() {
679
+ const PI = "dmFyIFVlID0gT2JqZWN0LmRlZmluZVByb3BlcnR5Owp2YXIgX2UgPSAoaSwgZSwgcikgPT4gZSBpbiBpID8gVWUoaSwgZSwgeyBlbnVtZXJhYmxlOiAhMCwgY29uZmlndXJhYmxlOiAhMCwgd3JpdGFibGU6ICEwLCB2YWx1ZTogciB9KSA6IGlbZV0gPSByOwp2YXIgbmUgPSAoaSwgZSwgcikgPT4gKF9lKGksIHR5cGVvZiBlICE9ICJzeW1ib2wiID8gZSArICIiIDogZSwgciksIHIpLCB4ZSA9IChpLCBlLCByKSA9PiB7CiAgaWYgKCFlLmhhcyhpKSkKICAgIHRocm93IFR5cGVFcnJvcigiQ2Fubm90ICIgKyByKTsKfTsKdmFyIHQgPSAoaSwgZSwgcikgPT4gKHhlKGksIGUsICJyZWFkIGZyb20gcHJpdmF0ZSBmaWVsZCIpLCByID8gci5jYWxsKGkpIDogZS5nZXQoaSkpLCBsID0gKGksIGUsIHIpID0+IHsKICBpZiAoZS5oYXMoaSkpCiAgICB0aHJvdyBUeXBlRXJyb3IoIkNhbm5vdCBhZGQgdGhlIHNhbWUgcHJpdmF0ZSBtZW1iZXIgbW9yZSB0aGFuIG9uY2UiKTsKICBlIGluc3RhbmNlb2YgV2Vha1NldCA/IGUuYWRkKGkpIDogZS5zZXQoaSwgcik7Cn0sIGEgPSAoaSwgZSwgciwgbykgPT4gKHhlKGksIGUsICJ3cml0ZSB0byBwcml2YXRlIGZpZWxkIiksIG8gPyBvLmNhbGwoaSwgcikgOiBlLnNldChpLCByKSwgcik7CnZhciBUID0gKGksIGUsIHIpID0+ICh4ZShpLCBlLCAiYWNjZXNzIHByaXZhdGUgbWV0aG9kIiksIHIpOwpjb25zdCB5ZSA9IHsKICBkZW5zaXR5RGlzc2lwYXRpb246IDAuOTkyLAogIHZlbG9jaXR5RGlzc2lwYXRpb246IDAuOTMsCiAgcHJlc3N1cmVJdGVyYXRpb25zOiAxLAogIGN1cmw6IDFlLTQsCiAgc3BsYXRSYWRpdXM6IDRlLTMsCiAgc3BsYXRGb3JjZTogMC45MSwKICByZWZyYWN0aW9uOiAwLjI1LAogIHNwZWN1bGFyRXhwOiAxLjAxLAogIHNoaW5lOiAwLjAxLAogIHdhdGVyQ29sb3I6IFswLCAwLCAwXSwKICBnbG93Q29sb3I6IFswLjcsIDAuODUsIDFdLAogIGFsZ29yaXRobTogInN0YW5kYXJkIiwKICB3YXJwU3RyZW5ndGg6IDAuMDE1Cn07Cih7CiAgLi4ueWUKfSk7CmNvbnN0IHdlID0gewogIGNhbG06IHsKICAgIGRlbnNpdHlEaXNzaXBhdGlvbjogMC45OTksCiAgICB2ZWxvY2l0eURpc3NpcGF0aW9uOiAwLjk4LAogICAgY3VybDogMWUtNCwKICAgIHNwbGF0UmFkaXVzOiAzZS0zLAogICAgc3BsYXRGb3JjZTogMC41LAogICAgcmVmcmFjdGlvbjogMC4xNSwKICAgIHNoaW5lOiA1ZS0zLAogICAgZ2xvd0NvbG9yOiBbMC42LCAwLjg1LCAxXSwKICAgIHdhdGVyQ29sb3I6IFswLCAwLjAyLCAwLjA1XQogIH0sCiAgc2FuZDogewogICAgZGVuc2l0eURpc3NpcGF0aW9uOiAwLjk5NywKICAgIHZlbG9jaXR5RGlzc2lwYXRpb246IDAuOTgsCiAgICBjdXJsOiAxLAogICAgc3BsYXRSYWRpdXM6IDAuMDEsCiAgICBzcGxhdEZvcmNlOiAwLjksCiAgICByZWZyYWN0aW9uOiAwLjgsCiAgICBzcGVjdWxhckV4cDogMC4xLAogICAgc2hpbmU6IDAuMDUsCiAgICBnbG93Q29sb3I6IFswLjAyNywgMC4wMjcsIDAuMDI3XSwKICAgIHdhdGVyQ29sb3I6IFswLjQ1MSwgMC4zMjksIDAuMTI1XQogIH0sCiAgd2F2ZTogewogICAgZGVuc2l0eURpc3NpcGF0aW9uOiAwLjk5NCwKICAgIHZlbG9jaXR5RGlzc2lwYXRpb246IDAuOTIsCiAgICBjdXJsOiAwLjIsCiAgICBzcGxhdFJhZGl1czogNWUtMywKICAgIHNwbGF0Rm9yY2U6IDEuMiwKICAgIHJlZnJhY3Rpb246IDAuMzUsCiAgICBzaGluZTogMC4wMywKICAgIGdsb3dDb2xvcjogWzAuNSwgMC44LCAxXSwKICAgIHdhdGVyQ29sb3I6IFswLCAwLjAxLCAwLjAzXQogIH0sCiAgbmVvbjogewogICAgZGVuc2l0eURpc3NpcGF0aW9uOiAwLjk4NSwKICAgIHZlbG9jaXR5RGlzc2lwYXRpb246IDAuOTMsCiAgICBjdXJsOiAwLjA1LAogICAgc3BsYXRSYWRpdXM6IDhlLTMsCiAgICBzcGxhdEZvcmNlOiAxLjUsCiAgICByZWZyYWN0aW9uOiAwLjI1LAogICAgc3BlY3VsYXJFeHA6IDAuNSwKICAgIHNoaW5lOiAwLjE0LAogICAgZ2xvd0NvbG9yOiBbMSwgMC4yLCAwLjhdLAogICAgd2F0ZXJDb2xvcjogWzAuMDUsIDAsIDAuMDhdCiAgfSwKICBzbW9rZTogewogICAgZGVuc2l0eURpc3NpcGF0aW9uOiAwLjk5NiwKICAgIHZlbG9jaXR5RGlzc2lwYXRpb246IDAuOTcsCiAgICBjdXJsOiAwLjA0LAogICAgc3BsYXRSYWRpdXM6IDllLTMsCiAgICBzcGxhdEZvcmNlOiAwLjgsCiAgICByZWZyYWN0aW9uOiAwLjA4LAogICAgc2hpbmU6IDAsCiAgICBnbG93Q29sb3I6IFswLjUsIDAuNSwgMC41XSwKICAgIHdhdGVyQ29sb3I6IFswLjA2LCAwLjA2LCAwLjA2XQogIH0KfTsKZnVuY3Rpb24gU2UoaSA9IHt9LCBlLCByID0geWUpIHsKICByZXR1cm4geyAuLi5lID8geyAuLi5yLCAuLi53ZVtlXSB9IDogciwgLi4uaSB9Owp9CmNvbnN0IFAgPSAoCiAgLyogZ2xzbCAqLwogIGAKICBwcmVjaXNpb24gaGlnaHAgZmxvYXQ7CiAgYXR0cmlidXRlIHZlYzIgYVBvc2l0aW9uOwogIHZhcnlpbmcgdmVjMiB2VXY7CiAgdmFyeWluZyB2ZWMyIHZMOyB2YXJ5aW5nIHZlYzIgdlI7IHZhcnlpbmcgdmVjMiB2VDsgdmFyeWluZyB2ZWMyIHZCOwogIHVuaWZvcm0gdmVjMiB0ZXhlbFNpemU7CiAgdm9pZCBtYWluICgpIHsKICAgIHZVdiA9IGFQb3NpdGlvbiAqIDAuNSArIDAuNTsKICAgIHZMID0gdlV2IC0gdmVjMih0ZXhlbFNpemUueCwgMC4wKTsKICAgIHZSID0gdlV2ICsgdmVjMih0ZXhlbFNpemUueCwgMC4wKTsKICAgIHZUID0gdlV2ICsgdmVjMigwLjAsIHRleGVsU2l6ZS55KTsKICAgIHZCID0gdlV2IC0gdmVjMigwLjAsIHRleGVsU2l6ZS55KTsKICAgIGdsX1Bvc2l0aW9uID0gdmVjNChhUG9zaXRpb24sIDAuMCwgMS4wKTsKICB9CmAKKSwgQ2UgPSAoCiAgLyogZ2xzbCAqLwogIGAKICBwcmVjaXNpb24gaGlnaHAgZmxvYXQ7CiAgdmFyeWluZyB2ZWMyIHZVdjsKICB1bmlmb3JtIHNhbXBsZXIyRCB1VmVsb2NpdHk7CiAgdW5pZm9ybSBzYW1wbGVyMkQgdVNvdXJjZTsKICB1bmlmb3JtIHNhbXBsZXIyRCB1T2JzdGFjbGU7CiAgdW5pZm9ybSB2ZWMyIHRleGVsU2l6ZTsKICB1bmlmb3JtIGZsb2F0IGR0OwogIHVuaWZvcm0gZmxvYXQgZGlzc2lwYXRpb247CiAgdm9pZCBtYWluICgpIHsKICAgIGlmICh0ZXh0dXJlMkQodU9ic3RhY2xlLCB2VXYpLnIgPiAwLjUpIHsgZ2xfRnJhZ0NvbG9yID0gdmVjNCgwLjApOyByZXR1cm47IH0KICAgIHZlYzIgY29vcmQgPSB2VXYgLSBkdCAqIHRleHR1cmUyRCh1VmVsb2NpdHksIHZVdikueHkgKiB0ZXhlbFNpemU7CiAgICBnbF9GcmFnQ29sb3IgPSBkaXNzaXBhdGlvbiAqIHRleHR1cmUyRCh1U291cmNlLCBjb29yZCk7CiAgfQpgCiksIEZlID0gKAogIC8qIGdsc2wgKi8KICBgCiAgcHJlY2lzaW9uIGhpZ2hwIGZsb2F0OwogIHZhcnlpbmcgdmVjMiB2TDsgdmFyeWluZyB2ZWMyIHZSOyB2YXJ5aW5nIHZlYzIgdlQ7IHZhcnlpbmcgdmVjMiB2QjsKICB1bmlmb3JtIHNhbXBsZXIyRCB1VmVsb2NpdHk7CiAgdW5pZm9ybSBzYW1wbGVyMkQgdU9ic3RhY2xlOwogIHZvaWQgbWFpbiAoKSB7CiAgICBmbG9hdCBMID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgdkwpLnIgPiAwLjUgPyAwLjAgOiB0ZXh0dXJlMkQodVZlbG9jaXR5LCB2TCkueDsKICAgIGZsb2F0IFIgPSB0ZXh0dXJlMkQodU9ic3RhY2xlLCB2UikuciA+IDAuNSA/IDAuMCA6IHRleHR1cmUyRCh1VmVsb2NpdHksIHZSKS54OwogICAgZmxvYXQgVCA9IHRleHR1cmUyRCh1T2JzdGFjbGUsIHZUKS5yID4gMC41ID8gMC4wIDogdGV4dHVyZTJEKHVWZWxvY2l0eSwgdlQpLnk7CiAgICBmbG9hdCBCID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgdkIpLnIgPiAwLjUgPyAwLjAgOiB0ZXh0dXJlMkQodVZlbG9jaXR5LCB2QikueTsKICAgIGdsX0ZyYWdDb2xvciA9IHZlYzQoMC41ICogKFIgLSBMICsgVCAtIEIpLCAwLjAsIDAuMCwgMS4wKTsKICB9CmAKKSwgWGUgPSAoCiAgLyogZ2xzbCAqLwogIGAKICBwcmVjaXNpb24gaGlnaHAgZmxvYXQ7CiAgdmFyeWluZyB2ZWMyIHZVdjsgdmFyeWluZyB2ZWMyIHZMOyB2YXJ5aW5nIHZlYzIgdlI7IHZhcnlpbmcgdmVjMiB2VDsgdmFyeWluZyB2ZWMyIHZCOwogIHVuaWZvcm0gc2FtcGxlcjJEIHVQcmVzc3VyZTsKICB1bmlmb3JtIHNhbXBsZXIyRCB1RGl2ZXJnZW5jZTsKICB1bmlmb3JtIHNhbXBsZXIyRCB1T2JzdGFjbGU7CiAgdm9pZCBtYWluICgpIHsKICAgIGZsb2F0IEMgPSB0ZXh0dXJlMkQodVByZXNzdXJlLCB2VXYpLng7CiAgICBmbG9hdCBMID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgdkwpLnIgPiAwLjUgPyBDIDogdGV4dHVyZTJEKHVQcmVzc3VyZSwgdkwpLng7CiAgICBmbG9hdCBSID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgdlIpLnIgPiAwLjUgPyBDIDogdGV4dHVyZTJEKHVQcmVzc3VyZSwgdlIpLng7CiAgICBmbG9hdCBUID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgdlQpLnIgPiAwLjUgPyBDIDogdGV4dHVyZTJEKHVQcmVzc3VyZSwgdlQpLng7CiAgICBmbG9hdCBCID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgdkIpLnIgPiAwLjUgPyBDIDogdGV4dHVyZTJEKHVQcmVzc3VyZSwgdkIpLng7CiAgICBmbG9hdCBkaXYgPSB0ZXh0dXJlMkQodURpdmVyZ2VuY2UsIHZVdikueDsKICAgIGdsX0ZyYWdDb2xvciA9IHZlYzQoKEwgKyBSICsgQiArIFQgLSBkaXYpICogMC4yNSwgMC4wLCAwLjAsIDEuMCk7CiAgfQpgCiksIEJlID0gKAogIC8qIGdsc2wgKi8KICBgCiAgcHJlY2lzaW9uIGhpZ2hwIGZsb2F0OwogIHZhcnlpbmcgdmVjMiB2VXY7IHZhcnlpbmcgdmVjMiB2TDsgdmFyeWluZyB2ZWMyIHZSOyB2YXJ5aW5nIHZlYzIgdlQ7IHZhcnlpbmcgdmVjMiB2QjsKICB1bmlmb3JtIHNhbXBsZXIyRCB1UHJlc3N1cmU7CiAgdW5pZm9ybSBzYW1wbGVyMkQgdVZlbG9jaXR5OwogIHVuaWZvcm0gc2FtcGxlcjJEIHVPYnN0YWNsZTsKICB2b2lkIG1haW4gKCkgewogICAgaWYgKHRleHR1cmUyRCh1T2JzdGFjbGUsIHZVdikuciA+IDAuNSkgeyBnbF9GcmFnQ29sb3IgPSB2ZWM0KDAuMCk7IHJldHVybjsgfQogICAgZmxvYXQgTCA9IHRleHR1cmUyRCh1UHJlc3N1cmUsIHZMKS54OwogICAgZmxvYXQgUiA9IHRleHR1cmUyRCh1UHJlc3N1cmUsIHZSKS54OwogICAgZmxvYXQgVCA9IHRleHR1cmUyRCh1UHJlc3N1cmUsIHZUKS54OwogICAgZmxvYXQgQiA9IHRleHR1cmUyRCh1UHJlc3N1cmUsIHZCKS54OwogICAgdmVjMiB2ZWwgPSB0ZXh0dXJlMkQodVZlbG9jaXR5LCB2VXYpLnh5IC0gdmVjMihSIC0gTCwgVCAtIEIpOwogICAgZ2xfRnJhZ0NvbG9yID0gdmVjNCh2ZWwsIDAuMCwgMS4wKTsKICB9CmAKKSwgQWUgPSAoCiAgLyogZ2xzbCAqLwogIGAKICBwcmVjaXNpb24gaGlnaHAgZmxvYXQ7CiAgdmFyeWluZyB2ZWMyIHZVdjsKICB1bmlmb3JtIHNhbXBsZXIyRCB1VGFyZ2V0OwogIHVuaWZvcm0gZmxvYXQgYXNwZWN0UmF0aW87CiAgdW5pZm9ybSB2ZWMzIGNvbG9yOwogIHVuaWZvcm0gdmVjMiBwb2ludDsKICB1bmlmb3JtIGZsb2F0IHJhZGl1czsKICB2b2lkIG1haW4gKCkgewogICAgdmVjMiBwID0gdlV2IC0gcG9pbnQueHk7CiAgICBwLnggKj0gYXNwZWN0UmF0aW87CiAgICB2ZWMzIHNwbGF0ID0gZXhwKC1kb3QocCwgcCkgLyByYWRpdXMpICogY29sb3I7CiAgICBnbF9GcmFnQ29sb3IgPSB2ZWM0KHRleHR1cmUyRCh1VGFyZ2V0LCB2VXYpLnh5eiArIHNwbGF0LCAxLjApOwogIH0KYAopLCBMZSA9ICgKICAvKiBnbHNsICovCiAgYAogIHByZWNpc2lvbiBoaWdocCBmbG9hdDsKICB2YXJ5aW5nIHZlYzIgdkw7IHZhcnlpbmcgdmVjMiB2UjsgdmFyeWluZyB2ZWMyIHZUOyB2YXJ5aW5nIHZlYzIgdkI7CiAgdW5pZm9ybSBzYW1wbGVyMkQgdVZlbG9jaXR5OwogIHZvaWQgbWFpbiAoKSB7CiAgICBmbG9hdCBMID0gdGV4dHVyZTJEKHVWZWxvY2l0eSwgdkwpLnk7CiAgICBmbG9hdCBSID0gdGV4dHVyZTJEKHVWZWxvY2l0eSwgdlIpLnk7CiAgICBmbG9hdCBUID0gdGV4dHVyZTJEKHVWZWxvY2l0eSwgdlQpLng7CiAgICBmbG9hdCBCID0gdGV4dHVyZTJEKHVWZWxvY2l0eSwgdkIpLng7CiAgICBnbF9GcmFnQ29sb3IgPSB2ZWM0KDAuNSAqIChSIC0gTCAtIFQgKyBCKSwgMC4wLCAwLjAsIDEuMCk7CiAgfQpgCiksIE1lID0gKAogIC8qIGdsc2wgKi8KICBgCiAgcHJlY2lzaW9uIGhpZ2hwIGZsb2F0OwogIHZhcnlpbmcgdmVjMiB2VXY7IHZhcnlpbmcgdmVjMiB2TDsgdmFyeWluZyB2ZWMyIHZSOyB2YXJ5aW5nIHZlYzIgdlQ7IHZhcnlpbmcgdmVjMiB2QjsKICB1bmlmb3JtIHNhbXBsZXIyRCB1VmVsb2NpdHk7CiAgdW5pZm9ybSBzYW1wbGVyMkQgdUN1cmw7CiAgdW5pZm9ybSBmbG9hdCBjdXJsOwogIHVuaWZvcm0gZmxvYXQgZHQ7CiAgdm9pZCBtYWluICgpIHsKICAgIGZsb2F0IEwgPSB0ZXh0dXJlMkQodUN1cmwsIHZMKS54OwogICAgZmxvYXQgUiA9IHRleHR1cmUyRCh1Q3VybCwgdlIpLng7CiAgICBmbG9hdCBUID0gdGV4dHVyZTJEKHVDdXJsLCB2VCkueDsKICAgIGZsb2F0IEIgPSB0ZXh0dXJlMkQodUN1cmwsIHZCKS54OwogICAgZmxvYXQgQyA9IHRleHR1cmUyRCh1Q3VybCwgdlV2KS54OwogICAgdmVjMiBmb3JjZSA9IDAuNSAqIHZlYzIoYWJzKFQpIC0gYWJzKEIpLCBhYnMoUikgLSBhYnMoTCkpOwogICAgZm9yY2UgLz0gbGVuZ3RoKGZvcmNlKSArIDAuMDAwMTsKICAgIGZvcmNlICo9IGN1cmwgKiAzMC4wICogQzsKICAgIGdsX0ZyYWdDb2xvciA9IHZlYzQodGV4dHVyZTJEKHVWZWxvY2l0eSwgdlV2KS54eSArIGZvcmNlICogZHQsIDAuMCwgMS4wKTsKICB9CmAKKSwgUGUgPSAoCiAgLyogZ2xzbCAqLwogIGAKICBwcmVjaXNpb24gaGlnaHAgZmxvYXQ7CiAgdmFyeWluZyB2ZWMyIHZVdjsKCiAgdW5pZm9ybSBzYW1wbGVyMkQgdVRleHR1cmU7CiAgdW5pZm9ybSBzYW1wbGVyMkQgdU9ic3RhY2xlOwogIHVuaWZvcm0gc2FtcGxlcjJEIHVCYWNrZ3JvdW5kOwogIHVuaWZvcm0gc2FtcGxlcjJEIHVDb3ZlcmFnZTsKICB1bmlmb3JtIHNhbXBsZXIyRCB1VmVsb2NpdHk7CgogIHVuaWZvcm0gdmVjMiAgdGV4ZWxTaXplOwogIHVuaWZvcm0gdmVjMyAgdVdhdGVyQ29sb3I7CiAgdW5pZm9ybSB2ZWMzICB1R2xvd0NvbG9yOwogIHVuaWZvcm0gZmxvYXQgdVJlZnJhY3Rpb247CiAgdW5pZm9ybSBmbG9hdCB1U3BlY3VsYXJFeHA7CiAgdW5pZm9ybSBmbG9hdCB1U2hpbmU7CiAgdW5pZm9ybSBmbG9hdCB1V2FycFN0cmVuZ3RoOwogIHVuaWZvcm0gaW50ICAgdUFsZ29yaXRobTsKCiAgdm9pZCBtYWluICgpIHsKICAgIGZsb2F0IG9icyAgICAgID0gdGV4dHVyZTJEKHVPYnN0YWNsZSwgIHZVdikucjsKICAgIC8vIE1hc2sgZGVuc2l0eSBpbnNpZGUgb2JzdGFjbGVzIHNvIHNwbGF0cyBkb24ndCBmbGlja2VyIHRoZSB0ZXh0L2ltYWdlIGNvbnRlbnQuCiAgICBmbG9hdCBkZW5zaXR5ICA9IG1heCh0ZXh0dXJlMkQodVRleHR1cmUsIHZVdikuciwgMC4wKSAqICgxLjAgLSBzdGVwKDAuNSwgb2JzKSk7CiAgICBmbG9hdCBjb3ZlcmFnZSA9IHRleHR1cmUyRCh1Q292ZXJhZ2UsICB2VXYpLnI7CgogICAgZmxvYXQgZEwgPSBtYXgodGV4dHVyZTJEKHVUZXh0dXJlLCB2VXYgLSB2ZWMyKHRleGVsU2l6ZS54ICogMi4wLCAwLjApKS5yLCAwLjApOwogICAgZmxvYXQgZFIgPSBtYXgodGV4dHVyZTJEKHVUZXh0dXJlLCB2VXYgKyB2ZWMyKHRleGVsU2l6ZS54ICogMi4wLCAwLjApKS5yLCAwLjApOwogICAgZmxvYXQgZFQgPSBtYXgodGV4dHVyZTJEKHVUZXh0dXJlLCB2VXYgKyB2ZWMyKDAuMCwgdGV4ZWxTaXplLnkgKiAyLjApKS5yLCAwLjApOwogICAgZmxvYXQgZEIgPSBtYXgodGV4dHVyZTJEKHVUZXh0dXJlLCB2VXYgLSB2ZWMyKDAuMCwgdGV4ZWxTaXplLnkgKiAyLjApKS5yLCAwLjApOwoKICAgIHZlYzMgIG5vcm1hbCAgID0gbm9ybWFsaXplKHZlYzMoZEwgLSBkUiwgZEIgLSBkVCwgMC4yKSk7CiAgICB2ZWMzICBsaWdodERpciA9IG5vcm1hbGl6ZSh2ZWMzKDAuNSwgMS4wLCAwLjUpKTsKICAgIHZlYzMgIGhhbGZWICAgID0gbm9ybWFsaXplKGxpZ2h0RGlyICsgdmVjMygwLjAsIDAuMCwgMS4wKSk7CiAgICBmbG9hdCBzcGVjICAgICA9IHBvdyhtYXgoZG90KG5vcm1hbCwgaGFsZlYpLCAwLjApLCB1U3BlY3VsYXJFeHApICogdVNoaW5lICogZGVuc2l0eTsKCiAgICAvLyBJbiB0cmFuc3BhcmVudCAobm9uLWNvdmVyYWdlKSBhcmVhcyB0aGUgYmFja2dyb3VuZCB0ZXh0dXJlIGlzIGVtcHR5IGJsYWNrIGNhbnZhcy4KICAgIC8vIFJlcGxhY2UgaXQgd2l0aCB1V2F0ZXJDb2xvciBzbyBmbHVpZCBjb2xvdXIgaXMgbm90IGNvbnRhbWluYXRlZCBieSB0aGF0IGJsYWNrLAogICAgLy8gYWxsb3dpbmcgdGhlIENTUyBiYWNrZ3JvdW5kQ29sb3IgdG8gc2hvdyB0aHJvdWdoIGNvcnJlY3RseSB2aWEgcHJlbXVsdGlwbGllZCBhbHBoYS4KICAgIHZlYzMgYmdSYXcgPSB0ZXh0dXJlMkQodUJhY2tncm91bmQsIHZVdikucmdiOwogICAgdmVjMyBiZyAgICA9IG1peCh1V2F0ZXJDb2xvciwgYmdSYXcsIGNvdmVyYWdlKTsKICAgIHZlYzMgY29sb3IgPSBiZzsKCiAgICBpZiAodUFsZ29yaXRobSA9PSAxKSB7CiAgICAgIC8vIOKUgOKUgCBnbGFzcyDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIAKICAgICAgLy8gU3Ryb25nIFVWIGRpc3RvcnRpb24gb25seS4gSW1hZ2UgYmVuZHMgYnV0IG5vIGNvbG91ciBvdmVybGF5LgogICAgICB2ZWMyIHJlZnJVdiA9IGNsYW1wKHZVdiArIG5vcm1hbC54eSAqIHVSZWZyYWN0aW9uICogZGVuc2l0eSAqIDMuMCwgMC4wLCAxLjApOwogICAgICB2ZWMzIHJlZnJCZyA9IG1peCh1V2F0ZXJDb2xvciwgdGV4dHVyZTJEKHVCYWNrZ3JvdW5kLCBtaXgodlV2LCByZWZyVXYsIDEuMCAtIG9icykpLnJnYiwgY292ZXJhZ2UpOwogICAgICBjb2xvciA9IHJlZnJCZyArIHNwZWMgKiB1R2xvd0NvbG9yICogMi41OwogICAgICBjb2xvciA9IG1peChjb2xvciwgYmcgKiAwLjYsIG9icyAqIDAuMyk7CgogICAgfSBlbHNlIGlmICh1QWxnb3JpdGhtID09IDIpIHsKICAgICAgLy8g4pSA4pSAIGluayDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIAKICAgICAgLy8gRGVuc2Ugb3BhcXVlIHBpZ21lbnQgdGhhdCBzdGFpbnMuIFN1YnRsZSByZWZyYWN0aW9uIHVuZGVybmVhdGguCiAgICAgIGZsb2F0IGlua0QgID0gbWluKGRlbnNpdHkgKiA0LjAsIDEuMCk7CiAgICAgIHZlYzIgcmVmclV2ID0gY2xhbXAodlV2ICsgbm9ybWFsLnh5ICogdVJlZnJhY3Rpb24gKiBkZW5zaXR5ICogMC40LCAwLjAsIDEuMCk7CiAgICAgIHZlYzMgcmVmckJnID0gbWl4KHVXYXRlckNvbG9yLCB0ZXh0dXJlMkQodUJhY2tncm91bmQsIG1peCh2VXYsIHJlZnJVdiwgMS4wIC0gb2JzKSkucmdiLCBjb3ZlcmFnZSk7CiAgICAgIGNvbG9yID0gbWl4KHJlZnJCZywgdVdhdGVyQ29sb3IgKyBzcGVjICogdUdsb3dDb2xvciwgaW5rRCk7CiAgICAgIGNvbG9yID0gbWl4KGNvbG9yLCBiZyAqIDAuNSwgb2JzICogMC4xNSk7CgogICAgfSBlbHNlIGlmICh1QWxnb3JpdGhtID09IDMpIHsKICAgICAgLy8g4pSA4pSAIGF1cm9yYSDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIAKICAgICAgLy8gVmVsb2NpdHkgZmllbGQgd2FycHMgYmFja2dyb3VuZCBVVnMg4oCUIGxpcXVpZCBtZXRhbCAvIGxhdmEtbGFtcCBmZWVsLgogICAgICB2ZWMyICB2ZWwgICAgPSB0ZXh0dXJlMkQodVZlbG9jaXR5LCB2VXYpLnh5OwogICAgICBmbG9hdCB2ZWxNYWcgPSBjbGFtcChsZW5ndGgodmVsKSAqIDIwLjAsIDAuMCwgMS4wKTsKICAgICAgdmVjMiAgd2FycFV2ID0gY2xhbXAodlV2ICsgdmVsICogdVdhcnBTdHJlbmd0aCwgMC4wLCAxLjApOwogICAgICB2ZWMzICB3YXJwQmcgPSBtaXgodVdhdGVyQ29sb3IsIHRleHR1cmUyRCh1QmFja2dyb3VuZCwgd2FycFV2KS5yZ2IsIGNvdmVyYWdlKTsKICAgICAgY29sb3IgID0gbWl4KGJnLCB3YXJwQmcsIHZlbE1hZyAqICgxLjAgLSBvYnMpKTsKICAgICAgY29sb3IgKz0gc3BlYyAqIHVHbG93Q29sb3IgKiB2ZWxNYWcgKiAxLjU7CiAgICAgIGNvbG9yICs9IHVXYXRlckNvbG9yICogZGVuc2l0eSAqIDAuMzsKICAgICAgY29sb3IgID0gbWl4KGNvbG9yLCBiZyAqIDAuNSwgb2JzICogMC4yKTsKCiAgICB9IGVsc2UgaWYgKHVBbGdvcml0aG0gPT0gNCkgewogICAgICAvLyDilIDilIAgcmlwcGxlIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgAogICAgICAvLyBFeGFnZ2VyYXRlZCBub3JtYWwgcGVydHVyYmF0aW9uICsgRnJlc25lbCByaW0g4oCUIGNhbG0gd2F0ZXIgc3VyZmFjZS4KICAgICAgdmVjMiAgcmlwcGxlVXYgPSBjbGFtcCh2VXYgKyBub3JtYWwueHkgKiB1UmVmcmFjdGlvbiAqIGRlbnNpdHkgKiA2LjAsIDAuMCwgMS4wKTsKICAgICAgdmVjMyAgcmVmckJnICAgPSBtaXgodVdhdGVyQ29sb3IsIHRleHR1cmUyRCh1QmFja2dyb3VuZCwgbWl4KHZVdiwgcmlwcGxlVXYsIDEuMCAtIG9icykpLnJnYiwgY292ZXJhZ2UpOwogICAgICBmbG9hdCBmcmVzbmVsICA9IHBvdyhjbGFtcCgxLjAgLSBkb3Qobm9ybWFsLCB2ZWMzKDAuMCwgMC4wLCAxLjApKSwgMC4wLCAxLjApLCAzLjApICogZGVuc2l0eTsKICAgICAgY29sb3IgID0gcmVmckJnOwogICAgICBjb2xvciArPSBmcmVzbmVsICogdUdsb3dDb2xvciAqIDIuMDsKICAgICAgY29sb3IgKz0gc3BlYyAqIHVHbG93Q29sb3IgKiBkZW5zaXR5ICogMi4wOwogICAgICBjb2xvciAgPSBtaXgoY29sb3IsIGJnICogMC41LCBvYnMgKiAwLjIpOwoKICAgIH0gZWxzZSB7CiAgICAgIC8vIOKUgOKUgCBzdGFuZGFyZCAoMCkg4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSACiAgICAgIC8vIE9yaWdpbmFsOiBjb2xvdXIgb3ZlcmxheSBibGVuZGVkIG92ZXIgcmVmcmFjdGVkIGJhY2tncm91bmQuCiAgICAgIHZlYzIgcmVmclV2ID0gdlV2ICsgbm9ybWFsLnh5ICogdVJlZnJhY3Rpb24gKiBkZW5zaXR5OwogICAgICB2ZWMzIHJlZnJCZyA9IG1peCh1V2F0ZXJDb2xvciwgdGV4dHVyZTJEKHVCYWNrZ3JvdW5kLCBtaXgodlV2LCByZWZyVXYsIDEuMCAtIG9icykpLnJnYiwgY292ZXJhZ2UpOwogICAgICBjb2xvciAgPSBtaXgocmVmckJnLCB1V2F0ZXJDb2xvciwgbWluKGRlbnNpdHkgKiAxLjUsIDAuOCkpOwogICAgICBjb2xvciArPSBzcGVjICogdUdsb3dDb2xvcjsKICAgICAgY29sb3IgID0gbWl4KGNvbG9yLCBiZyAqIDAuNSwgb2JzICogMC4yKTsKICAgIH0KCiAgICAvLyBQcmVtdWx0aXBsaWVkIGFscGhhIOKAlCB0cmFuc3BhcmVudCB3aGVyZSB0aGVyZSBpcyBuZWl0aGVyIGNvbnRlbnQgbm9yIGZsdWlkLAogICAgLy8gbGV0dGluZyB0aGUgQ1NTIGJhY2tncm91bmRDb2xvciBvbiB0aGUgY29udGFpbmVyIGRpdiBzaG93IHRocm91Z2guCiAgICBmbG9hdCBhbHBoYSA9IGNsYW1wKG1heChkZW5zaXR5ICogMS41LCBjb3ZlcmFnZSksIDAuMCwgMS4wKTsKICAgIGdsX0ZyYWdDb2xvciA9IHZlYzQoY29sb3IgKiBhbHBoYSwgYWxwaGEpOwogIH0KYAopOwpmdW5jdGlvbiBrZShpKSB7CiAgY29uc3QgZSA9IHsgYWxwaGE6ICEwLCBkZXB0aDogITEsIHN0ZW5jaWw6ICExLCBhbnRpYWxpYXM6ICEwLCBwcmVzZXJ2ZURyYXdpbmdCdWZmZXI6ICExIH07CiAgbGV0IHIgPSBpLmdldENvbnRleHQoIndlYmdsMiIsIGUpOwogIGNvbnN0IG8gPSAhIXI7CiAgbyB8fCAociA9IGkuZ2V0Q29udGV4dCgid2ViZ2wiLCBlKSwgci5nZXRFeHRlbnNpb24oIkVYVF9jb2xvcl9idWZmZXJfaGFsZl9mbG9hdCIpKTsKICBjb25zdCBzID0gbyA/IG51bGwgOiByLmdldEV4dGVuc2lvbigiT0VTX3RleHR1cmVfaGFsZl9mbG9hdCIpLCBuID0gbyA/IHIuSEFMRl9GTE9BVCA6IHMuSEFMRl9GTE9BVF9PRVM7CiAgcmV0dXJuIHIuZ2V0RXh0ZW5zaW9uKCJFWFRfY29sb3JfYnVmZmVyX2Zsb2F0IiksIHIuZ2V0RXh0ZW5zaW9uKCJPRVNfdGV4dHVyZV9oYWxmX2Zsb2F0X2xpbmVhciIpLCB7CiAgICBnbDogciwKICAgIGlzV2ViR0wyOiBvLAogICAgZXh0OiB7CiAgICAgIGludGVybmFsRm9ybWF0OiBvID8gci5SR0JBMTZGIDogci5SR0JBLAogICAgICBmb3JtYXQ6IHIuUkdCQSwKICAgICAgdHlwZTogbgogICAgfQogIH07Cn0KY2xhc3MgayB7CiAgY29uc3RydWN0b3IoZSwgciwgbykgewogICAgbmUodGhpcywgInByb2dyYW0iKTsKICAgIG5lKHRoaXMsICJ1bmlmb3JtcyIsIHt9KTsKICAgIG5lKHRoaXMsICJfZ2wiKTsKICAgIHRoaXMuX2dsID0gZSwgdGhpcy5wcm9ncmFtID0gZS5jcmVhdGVQcm9ncmFtKCksIGUuYXR0YWNoU2hhZGVyKHRoaXMucHJvZ3JhbSwgdGhpcy5fY29tcGlsZShlLlZFUlRFWF9TSEFERVIsIHIpKSwgZS5hdHRhY2hTaGFkZXIodGhpcy5wcm9ncmFtLCB0aGlzLl9jb21waWxlKGUuRlJBR01FTlRfU0hBREVSLCBvKSksIGUubGlua1Byb2dyYW0odGhpcy5wcm9ncmFtKTsKICAgIGNvbnN0IHMgPSBlLmdldFByb2dyYW1QYXJhbWV0ZXIodGhpcy5wcm9ncmFtLCBlLkFDVElWRV9VTklGT1JNUyk7CiAgICBmb3IgKGxldCBuID0gMDsgbiA8IHM7IG4rKykgewogICAgICBjb25zdCB1ID0gZS5nZXRBY3RpdmVVbmlmb3JtKHRoaXMucHJvZ3JhbSwgbikubmFtZTsKICAgICAgdGhpcy51bmlmb3Jtc1t1XSA9IGUuZ2V0VW5pZm9ybUxvY2F0aW9uKHRoaXMucHJvZ3JhbSwgdSk7CiAgICB9CiAgfQogIF9jb21waWxlKGUsIHIpIHsKICAgIGNvbnN0IG8gPSB0aGlzLl9nbCwgcyA9IG8uY3JlYXRlU2hhZGVyKGUpOwogICAgcmV0dXJuIG8uc2hhZGVyU291cmNlKHMsIHIpLCBvLmNvbXBpbGVTaGFkZXIocyksIHM7CiAgfQogIGJpbmQoKSB7CiAgICB0aGlzLl9nbC51c2VQcm9ncmFtKHRoaXMucHJvZ3JhbSk7CiAgfQogIGRpc3Bvc2UoKSB7CiAgICB0aGlzLl9nbC5kZWxldGVQcm9ncmFtKHRoaXMucHJvZ3JhbSk7CiAgfQp9CmZ1bmN0aW9uIE9lKGkpIHsKICByZXR1cm4gewogICAgYWR2ZWN0aW9uOiBuZXcgayhpLCBQLCBDZSksCiAgICBkaXZlcmdlbmNlOiBuZXcgayhpLCBQLCBGZSksCiAgICBwcmVzc3VyZTogbmV3IGsoaSwgUCwgWGUpLAogICAgZ3JhZGllbnRTdWJ0cmFjdDogbmV3IGsoaSwgUCwgQmUpLAogICAgc3BsYXQ6IG5ldyBrKGksIFAsIEFlKSwKICAgIGN1cmw6IG5ldyBrKGksIFAsIExlKSwKICAgIHZvcnRpY2l0eTogbmV3IGsoaSwgUCwgTWUpLAogICAgZGlzcGxheTogbmV3IGsoaSwgUCwgUGUpCiAgfTsKfQpmdW5jdGlvbiB1ZShpLCBlLCByLCBvKSB7CiAgaS5hY3RpdmVUZXh0dXJlKGkuVEVYVFVSRTApOwogIGNvbnN0IHMgPSBpLmNyZWF0ZVRleHR1cmUoKTsKICBpLmJpbmRUZXh0dXJlKGkuVEVYVFVSRV8yRCwgcyksIGkudGV4UGFyYW1ldGVyaShpLlRFWFRVUkVfMkQsIGkuVEVYVFVSRV9NSU5fRklMVEVSLCBpLkxJTkVBUiksIGkudGV4UGFyYW1ldGVyaShpLlRFWFRVUkVfMkQsIGkuVEVYVFVSRV9NQUdfRklMVEVSLCBpLkxJTkVBUiksIGkudGV4UGFyYW1ldGVyaShpLlRFWFRVUkVfMkQsIGkuVEVYVFVSRV9XUkFQX1MsIGkuQ0xBTVBfVE9fRURHRSksIGkudGV4UGFyYW1ldGVyaShpLlRFWFRVUkVfMkQsIGkuVEVYVFVSRV9XUkFQX1QsIGkuQ0xBTVBfVE9fRURHRSksIGkudGV4SW1hZ2UyRChpLlRFWFRVUkVfMkQsIDAsIGUuaW50ZXJuYWxGb3JtYXQsIHIsIG8sIDAsIGUuZm9ybWF0LCBlLnR5cGUsIG51bGwpOwogIGNvbnN0IG4gPSBpLmNyZWF0ZUZyYW1lYnVmZmVyKCk7CiAgcmV0dXJuIGkuYmluZEZyYW1lYnVmZmVyKGkuRlJBTUVCVUZGRVIsIG4pLCBpLmZyYW1lYnVmZmVyVGV4dHVyZTJEKGkuRlJBTUVCVUZGRVIsIGkuQ09MT1JfQVRUQUNITUVOVDAsIGkuVEVYVFVSRV8yRCwgcywgMCksIHsgdGV4OiBzLCBmYm86IG4sIHdpZHRoOiByLCBoZWlnaHQ6IG8gfTsKfQpmdW5jdGlvbiBkZShpLCBlLCByLCBvKSB7CiAgbGV0IHMgPSB1ZShpLCBlLCByLCBvKSwgbiA9IHVlKGksIGUsIHIsIG8pOwogIHJldHVybiB7CiAgICBnZXQgcmVhZCgpIHsKICAgICAgcmV0dXJuIHM7CiAgICB9LAogICAgZ2V0IHdyaXRlKCkgewogICAgICByZXR1cm4gbjsKICAgIH0sCiAgICBzd2FwKCkgewogICAgICBbcywgbl0gPSBbbiwgc107CiAgICB9LAogICAgZGlzcG9zZSgpIHsKICAgICAgaS5kZWxldGVUZXh0dXJlKHMudGV4KSwgaS5kZWxldGVGcmFtZWJ1ZmZlcihzLmZibyksIGkuZGVsZXRlVGV4dHVyZShuLnRleCksIGkuZGVsZXRlRnJhbWVidWZmZXIobi5mYm8pOwogICAgfQogIH07Cn0KZnVuY3Rpb24gSWUoaSkgewogIGNvbnN0IGUgPSBpLmNyZWF0ZUJ1ZmZlcigpOwogIHJldHVybiBpLmJpbmRCdWZmZXIoaS5BUlJBWV9CVUZGRVIsIGUpLCBpLmJ1ZmZlckRhdGEoaS5BUlJBWV9CVUZGRVIsIG5ldyBGbG9hdDMyQXJyYXkoWy0xLCAtMSwgLTEsIDEsIDEsIDEsIDEsIC0xXSksIGkuU1RBVElDX0RSQVcpLCBpLnZlcnRleEF0dHJpYlBvaW50ZXIoMCwgMiwgaS5GTE9BVCwgITEsIDAsIDApLCBpLmVuYWJsZVZlcnRleEF0dHJpYkFycmF5KDApLCBmdW5jdGlvbihvKSB7CiAgICBpLmJpbmRGcmFtZWJ1ZmZlcihpLkZSQU1FQlVGRkVSLCBvKSwgaS5kcmF3QXJyYXlzKGkuVFJJQU5HTEVfRkFOLCAwLCA0KTsKICB9Owp9CmZ1bmN0aW9uIFRlKGksIGUsIHIsIG8sIHMgPSAiY292ZXIiKSB7CiAgbGV0IG47CiAgcyA9PT0gImNvdmVyIiA/IG4gPSBNYXRoLm1heChyIC8gaSwgbyAvIGUpIDogcyA9PT0gImNvbnRhaW4iID8gbiA9IE1hdGgubWluKHIgLyBpLCBvIC8gZSkgOiB0eXBlb2YgcyA9PSAic3RyaW5nIiAmJiBzLmVuZHNXaXRoKCIlIikgPyBuID0gTWF0aC5taW4ociAvIGksIG8gLyBlKSAqIChwYXJzZUZsb2F0KHMpIC8gMTAwKSA6IHR5cGVvZiBzID09ICJzdHJpbmciICYmIHMuZW5kc1dpdGgoInB4IikgPyBuID0gcGFyc2VGbG9hdChzKSAvIE1hdGgubWF4KGksIGUpIDogdHlwZW9mIHMgPT0gIm51bWJlciIgPyBuID0gcyA6IG4gPSBNYXRoLm1heChyIC8gaSwgbyAvIGUpOwogIGNvbnN0IHUgPSBpICogbiwgZCA9IGUgKiBuOwogIHJldHVybiB7IHg6IChyIC0gdSkgLyAyLCB5OiAobyAtIGQpIC8gMiwgZHJhd1c6IHUsIGRyYXdIOiBkIH07Cn0KZnVuY3Rpb24gVmUoaSwgZSwgciwgbywgcyA9IG51bGwsIG4gPSAiY292ZXIiKSB7CiAgY29uc3QgeyB0ZXh0OiB1LCBmb250U2l6ZTogZCwgY29sb3I6IHAsIGZvbnRGYW1pbHk6IGMgPSAic2Fucy1zZXJpZiIsIGZvbnRXZWlnaHQ6IG0gPSA5MDAgfSA9IG8sIHkgPSBuZXcgT2Zmc2NyZWVuQ2FudmFzKGUsIHIpLCBmID0geS5nZXRDb250ZXh0KCIyZCIpOwogICgoY2UpID0+IHsKICAgIGlmIChzKSB7CiAgICAgIGYuY2xlYXJSZWN0KDAsIDAsIGUsIHIpLCBmLmZpbGxTdHlsZSA9ICJibGFjayIsIGYuZmlsbFJlY3QoMCwgMCwgZSwgcik7CiAgICAgIGNvbnN0IHsgeDogZmUsIHk6IGhlLCBkcmF3VzogdmUsIGRyYXdIOiBtZSB9ID0gVGUoCiAgICAgICAgcy53aWR0aCwKICAgICAgICBzLmhlaWdodCwKICAgICAgICBlLAogICAgICAgIHIsCiAgICAgICAgbgogICAgICApOwogICAgICBmLmRyYXdJbWFnZShzLCBmZSwgaGUsIHZlLCBtZSk7CiAgICB9IGVsc2UKICAgICAgZi5maWxsU3R5bGUgPSAiYmxhY2siLCBmLmZpbGxSZWN0KDAsIDAsIGUsIHIpOwogICAgZi5maWxsU3R5bGUgPSBjZSwgZi5mb250ID0gYCR7bX0gJHtkfXB4ICR7Y31gLCBmLnRleHRBbGlnbiA9ICJjZW50ZXIiLCBmLnRleHRCYXNlbGluZSA9ICJtaWRkbGUiLCBmLmZpbGxUZXh0KHUsIGUgLyAyLCByIC8gMik7CiAgfSkocCk7CiAgY29uc3QgSCA9IGVlKGksIHkpOwogIGYuZmlsbFN0eWxlID0gImJsYWNrIiwgZi5maWxsUmVjdCgwLCAwLCBlLCByKSwgZi5maWxsU3R5bGUgPSAid2hpdGUiLCBmLmZvbnQgPSBgJHttfSAke2R9cHggJHtjfWAsIGYudGV4dEFsaWduID0gImNlbnRlciIsIGYudGV4dEJhc2VsaW5lID0gIm1pZGRsZSIsIGYuZmlsbFRleHQodSwgZSAvIDIsIHIgLyAyKTsKICBjb25zdCBhZSA9IGVlKGksIHkpOwogIHJldHVybiB7IGJhY2tncm91bmRUZXg6IEgsIG9ic3RhY2xlVGV4OiBhZSwgY292ZXJhZ2VUZXg6IGFlIH07Cn0KZnVuY3Rpb24gV2UoaSwgZSwgciwgbywgcyA9IDAsIG4gPSAiY292ZXIiLCB1ID0gbnVsbCwgZCA9ICJjb3ZlciIpIHsKICBjb25zdCBwID0gbmV3IE9mZnNjcmVlbkNhbnZhcyhyLCBvKSwgYyA9IHAuZ2V0Q29udGV4dCgiMmQiKSwgeyB4OiBtLCB5LCBkcmF3VzogZiwgZHJhd0g6IEQgfSA9IFRlKGUud2lkdGgsIGUuaGVpZ2h0LCByLCBvLCBuKTsKICBpZiAoYy5jbGVhclJlY3QoMCwgMCwgciwgbyksIGMuZmlsbFN0eWxlID0gImJsYWNrIiwgYy5maWxsUmVjdCgwLCAwLCByLCBvKSwgdSkgewogICAgY29uc3QgewogICAgICB4OiBmZSwKICAgICAgeTogaGUsCiAgICAgIGRyYXdXOiB2ZSwKICAgICAgZHJhd0g6IG1lCiAgICB9ID0gVGUodS53aWR0aCwgdS5oZWlnaHQsIHIsIG8sIGQpOwogICAgYy5maWx0ZXIgPSBgYnJpZ2h0bmVzcygke3N9KSBibHVyKDhweClgLCBjLmRyYXdJbWFnZSh1LCBmZSwgaGUsIHZlLCBtZSksIGMuZmlsdGVyID0gIm5vbmUiOwogIH0KICBjLmRyYXdJbWFnZShlLCBtLCB5LCBmLCBEKTsKICBjb25zdCBIID0gZWUoaSwgcCk7CiAgYy5jbGVhclJlY3QoMCwgMCwgciwgbyksIGMuZmlsbFN0eWxlID0gImJsYWNrIiwgYy5maWxsUmVjdCgwLCAwLCByLCBvKSwgYy5maWx0ZXIgPSBgYnJpZ2h0bmVzcygke3N9KSBibHVyKDhweClgLCBjLmRyYXdJbWFnZShlLCBtLCB5LCBmLCBEKSwgYy5maWx0ZXIgPSAibm9uZSI7CiAgY29uc3QgYWUgPSBlZShpLCBwKTsKICBjLmNsZWFyUmVjdCgwLCAwLCByLCBvKSwgYy5maWxsU3R5bGUgPSAiYmxhY2siLCBjLmZpbGxSZWN0KDAsIDAsIHIsIG8pLCBjLmZpbGxTdHlsZSA9ICJ3aGl0ZSIsIGMuZmlsbFJlY3QoCiAgICBNYXRoLm1heCgwLCBtKSwKICAgIE1hdGgubWF4KDAsIHkpLAogICAgTWF0aC5taW4oZiwgciAtIE1hdGgubWF4KDAsIG0pKSwKICAgIE1hdGgubWluKEQsIG8gLSBNYXRoLm1heCgwLCB5KSkKICApOwogIGNvbnN0IGNlID0gZWUoaSwgcCk7CiAgcmV0dXJuIHsgYmFja2dyb3VuZFRleDogSCwgb2JzdGFjbGVUZXg6IGFlLCBjb3ZlcmFnZVRleDogY2UgfTsKfQpmdW5jdGlvbiBlZShpLCBlKSB7CiAgY29uc3QgciA9IGkuY3JlYXRlVGV4dHVyZSgpOwogIHJldHVybiBpLmJpbmRUZXh0dXJlKGkuVEVYVFVSRV8yRCwgciksIGkucGl4ZWxTdG9yZWkoaS5VTlBBQ0tfRkxJUF9ZX1dFQkdMLCAhMCksIGkudGV4SW1hZ2UyRChpLlRFWFRVUkVfMkQsIDAsIGkuUkdCQSwgaS5SR0JBLCBpLlVOU0lHTkVEX0JZVEUsIGUpLCBpLnRleFBhcmFtZXRlcmkoaS5URVhUVVJFXzJELCBpLlRFWFRVUkVfTUlOX0ZJTFRFUiwgaS5MSU5FQVIpLCBpLnRleFBhcmFtZXRlcmkoaS5URVhUVVJFXzJELCBpLlRFWFRVUkVfTUFHX0ZJTFRFUiwgaS5MSU5FQVIpLCBpLnRleFBhcmFtZXRlcmkoaS5URVhUVVJFXzJELCBpLlRFWFRVUkVfV1JBUF9TLCBpLkNMQU1QX1RPX0VER0UpLCBpLnRleFBhcmFtZXRlcmkoaS5URVhUVVJFXzJELCBpLlRFWFRVUkVfV1JBUF9ULCBpLkNMQU1QX1RPX0VER0UpLCByOwp9CmFzeW5jIGZ1bmN0aW9uIEdlKGkpIHsKICBjb25zdCBlID0gYXdhaXQgZmV0Y2goaSk7CiAgaWYgKCFlLm9rKQogICAgdGhyb3cgbmV3IEVycm9yKGBGYWlsZWQgdG8gZmV0Y2ggaW1hZ2U6ICR7aX0gKCR7ZS5zdGF0dXN9KWApOwogIGNvbnN0IHIgPSBhd2FpdCBlLmJsb2IoKTsKICByZXR1cm4gY3JlYXRlSW1hZ2VCaXRtYXAocik7Cn0KY29uc3QgZ2UgPSB0eXBlb2YgcmVxdWVzdEFuaW1hdGlvbkZyYW1lIDwgInUiID8gcmVxdWVzdEFuaW1hdGlvbkZyYW1lLmJpbmQoZ2xvYmFsVGhpcykgOiAoaSkgPT4gc2V0VGltZW91dChpLCAxZTMgLyA2MCksIHplID0gdHlwZW9mIGNhbmNlbEFuaW1hdGlvbkZyYW1lIDwgInUiID8gY2FuY2VsQW5pbWF0aW9uRnJhbWUuYmluZChnbG9iYWxUaGlzKSA6IGNsZWFyVGltZW91dCwgUmUgPSAwLjAxNiwgTmUgPSB7IHN0YW5kYXJkOiAwLCBnbGFzczogMSwgaW5rOiAyLCBhdXJvcmE6IDMsIHJpcHBsZTogNCB9Owp2YXIgVywgZywgdGUsIEcsIHEsIEUsIGIsIE8sIEksIHcsIFEsIEwsIFIsIGgsIEMsIEYsIFgsIE0sIFUsIEIsIFMsIEssIHosIHgsIF8sIEEsIE4sIHJlLCAkLCBKLCBpZSwgcGUsIFYsIGosIFksIFosIG9lLCBFZSwgc2UsIGJlLCBsZSwgRGU7CmNsYXNzICRlIHsKICBjb25zdHJ1Y3RvcihlLCByID0ge30sIG8gPSB7fSkgewogICAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAgICAvLyBQcml2YXRlIGhlbHBlcnMKICAgIC8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQogICAgbCh0aGlzLCAkKTsKICAgIGwodGhpcywgaWUpOwogICAgbCh0aGlzLCBWKTsKICAgIGwodGhpcywgWSk7CiAgICBsKHRoaXMsIG9lKTsKICAgIGwodGhpcywgc2UpOwogICAgbCh0aGlzLCBsZSk7CiAgICBsKHRoaXMsIFcsIHZvaWQgMCk7CiAgICBsKHRoaXMsIGcsIHZvaWQgMCk7CiAgICBsKHRoaXMsIHRlLCB2b2lkIDApOwogICAgbCh0aGlzLCBHLCB2b2lkIDApOwogICAgbCh0aGlzLCBxLCB2b2lkIDApOwogICAgbCh0aGlzLCBFLCAwKTsKICAgIGwodGhpcywgYiwgMCk7CiAgICBsKHRoaXMsIE8sIDApOwogICAgbCh0aGlzLCBJLCAwKTsKICAgIGwodGhpcywgdywgMSk7CiAgICBsKHRoaXMsIFEsIDEpOwogICAgbCh0aGlzLCBMLCAwLjUpOwogICAgbCh0aGlzLCBSLCBudWxsKTsKICAgIGwodGhpcywgaCwgbnVsbCk7CiAgICBsKHRoaXMsIEMsIG51bGwpOwogICAgbCh0aGlzLCBGLCBudWxsKTsKICAgIGwodGhpcywgWCwgbnVsbCk7CiAgICBsKHRoaXMsIE0sIG51bGwpOwogICAgbCh0aGlzLCBVLCBudWxsKTsKICAgIGwodGhpcywgQiwgbnVsbCk7CiAgICAvLyBiaW5hcnkgY29udGVudCBtYXNrIGZvciB0cmFuc3BhcmVudCBjYW52YXMgc3VwcG9ydAogICAgbCh0aGlzLCBTLCBudWxsKTsKICAgIC8vIG9wdGlvbmFsIGJhY2tncm91bmQgaW1hZ2UgKGZyb20gYmFja2dyb3VuZFNyYyBwcm9wKQogICAgbCh0aGlzLCBLLCAiY292ZXIiKTsKICAgIGwodGhpcywgeiwgdm9pZCAwKTsKICAgIGwodGhpcywgeCwgeyB4OiAwLCB5OiAwLCBkeDogMCwgZHk6IDAsIHRhcmdldFg6IDAsIHRhcmdldFk6IDAsIG1vdmVkOiAhMSB9KTsKICAgIC8vIFN0b3JlcyBzb3VyY2Ugc28gdGV4dHVyZXMgY2FuIGJlIHJlYnVpbHQgb24gcmVzaXplCiAgICBsKHRoaXMsIF8sIG51bGwpOwogICAgbCh0aGlzLCBBLCBudWxsKTsKICAgIGwodGhpcywgTiwgITEpOwogICAgbCh0aGlzLCByZSwgITEpOwogICAgYSh0aGlzLCBXLCBlKSwgYSh0aGlzLCBRLCBNYXRoLm1heCgwLjEsIE1hdGgubWluKDEsIG8uZHByID8/IDEpKSksIGEodGhpcywgTCwgTWF0aC5tYXgoMC4xLCBNYXRoLm1pbigxLCBvLnNpbSA/PyAwLjUpKSksIGEodGhpcywgeiwgU2UocikpOwogICAgY29uc3QgeyBnbDogcywgZXh0OiBuIH0gPSBrZShlKTsKICAgIGEodGhpcywgZywgcyksIGEodGhpcywgdGUsIG4pLCBhKHRoaXMsIEcsIE9lKHMpKSwgYSh0aGlzLCBxLCBJZShzKSksIHMuY2xlYXJDb2xvcigwLCAwLCAwLCAwKTsKICB9CiAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAgLy8gUHVibGljIEFQSQogIC8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQogIHNldFRleHRTb3VyY2UoZSkgewogICAgYSh0aGlzLCBfLCB7IHR5cGU6ICJ0ZXh0Iiwgb3B0czogZSB9KSwgVCh0aGlzLCAkLCBKKS5jYWxsKHRoaXMpLCBUKHRoaXMsIFYsIGopLmNhbGwodGhpcyksIFQodGhpcywgWSwgWikuY2FsbCh0aGlzKTsKICB9CiAgYXN5bmMgc2V0SW1hZ2VTb3VyY2UoZSwgciA9IDAsIG8gPSAiY292ZXIiKSB7CiAgICBjb25zdCBzID0gYXdhaXQgR2UoZSk7CiAgICBpZiAodCh0aGlzLCByZSkpIHsKICAgICAgcy5jbG9zZSgpOwogICAgICByZXR1cm47CiAgICB9CiAgICBhKHRoaXMsIF8sIHsgdHlwZTogImltYWdlIiwgYml0bWFwOiBzLCBlZmZlY3Q6IHIsIHNpemU6IG8gfSksIFQodGhpcywgJCwgSikuY2FsbCh0aGlzKSwgVCh0aGlzLCBWLCBqKS5jYWxsKHRoaXMpLCBUKHRoaXMsIFksIFopLmNhbGwodGhpcyk7CiAgfQogIHNldEltYWdlQml0bWFwKGUsIHIgPSAwLCBvID0gImNvdmVyIikgewogICAgYSh0aGlzLCBfLCB7IHR5cGU6ICJpbWFnZSIsIGJpdG1hcDogZSwgZWZmZWN0OiByLCBzaXplOiBvIH0pLCBUKHRoaXMsICQsIEopLmNhbGwodGhpcyksIFQodGhpcywgViwgaikuY2FsbCh0aGlzKSwgVCh0aGlzLCBZLCBaKS5jYWxsKHRoaXMpOwogIH0KICBzZXRCYWNrZ3JvdW5kKGUsIHIgPSAiY292ZXIiKSB7CiAgICB0KHRoaXMsIFMpICYmIHQodGhpcywgUykgIT09IGUgJiYgdCh0aGlzLCBTKS5jbG9zZSgpLCBhKHRoaXMsIFMsIGUpLCBhKHRoaXMsIEssIHIgPz8gImNvdmVyIiksIHQodGhpcywgXykgJiYgdCh0aGlzLCBFKSA+IDAgJiYgdCh0aGlzLCBiKSA+IDAgJiYgVCh0aGlzLCBWLCBqKS5jYWxsKHRoaXMpOwogIH0KICBoYW5kbGVNb3ZlKGUsIHIsIG8gPSAxKSB7CiAgICB0KHRoaXMsIHgpLm1vdmVkID0gITAsIHQodGhpcywgeCkuZHggPSAoZSAtIHQodGhpcywgeCkudGFyZ2V0WCkgKiBvLCB0KHRoaXMsIHgpLmR5ID0gKHIgLSB0KHRoaXMsIHgpLnRhcmdldFkpICogbywgdCh0aGlzLCB4KS50YXJnZXRYID0gZSwgdCh0aGlzLCB4KS50YXJnZXRZID0gcjsKICB9CiAgLyoqCiAgICogSW1tZWRpYXRlbHkgYXBwbGllcyBvbmUgZmx1aWQgc3BsYXQgYXQgKHgsIHkpIHdpdGggZXhwbGljaXQgdmVsb2NpdHkgKHZ4LCB2eSkuCiAgICogU2FmZSB0byBjYWxsIG11bHRpcGxlIHRpbWVzIHBlciBmcmFtZSDigJQgZWFjaCBjYWxsIHdyaXRlcyBkaXJlY3RseSB0byB0aGUgRkJPcy4KICAgKiBEZXNpZ25lZCBmb3IgcHJvZ3JhbW1hdGljIHVzZSBjYXNlcyAoZS5nLiBwYXJ0aWNsZSBzeXN0ZW1zLCBhdHRyYWN0b3IgcGF0aHMpCiAgICogd2hlcmUgeW91IHdhbnQgTiBpbmRlcGVuZGVudCBpbmplY3Rpb24gcG9pbnRzIHBlciBmcmFtZSB3aXRob3V0IGZsb29kaW5nIHRoZQogICAqIG1vdXNlLXN0YXRlIG1hY2hpbmUgb3IgdGhlIHdvcmtlciBtZXNzYWdlIHF1ZXVlLgogICAqLwogIHNwbGF0KGUsIHIsIG8sIHMsIG4gPSAxKSB7CiAgICBpZiAoIXQodGhpcywgTikgfHwgdCh0aGlzLCBFKSA9PT0gMCkKICAgICAgcmV0dXJuOwogICAgY29uc3QgdSA9IHQodGhpcywgZyksIGQgPSB0KHRoaXMsIHopLCB7IHNwbGF0OiBwIH0gPSB0KHRoaXMsIEcpLCBjID0gdCh0aGlzLCBxKTsKICAgIHUudmlld3BvcnQoMCwgMCwgdCh0aGlzLCBPKSwgdCh0aGlzLCBJKSksIHAuYmluZCgpLCB1LnVuaWZvcm0xZihwLnVuaWZvcm1zLmFzcGVjdFJhdGlvLCB0KHRoaXMsIEUpIC8gdCh0aGlzLCBiKSksIHUudW5pZm9ybTJmKHAudW5pZm9ybXMucG9pbnQsIGUgKiB0KHRoaXMsIHcpIC8gdCh0aGlzLCBFKSwgMSAtIHIgKiB0KHRoaXMsIHcpIC8gdCh0aGlzLCBiKSksIHUudW5pZm9ybTFmKHAudW5pZm9ybXMucmFkaXVzLCBkLnNwbGF0UmFkaXVzKSwgdS51bmlmb3JtMWkocC51bmlmb3Jtcy51VGFyZ2V0LCAwKSwgdS5hY3RpdmVUZXh0dXJlKHUuVEVYVFVSRTApLCB1LmJpbmRUZXh0dXJlKHUuVEVYVFVSRV8yRCwgdCh0aGlzLCBoKS5yZWFkLnRleCksIHUudW5pZm9ybTNmKHAudW5pZm9ybXMuY29sb3IsIG8gKiBkLnNwbGF0Rm9yY2UgKiBuLCAtcyAqIGQuc3BsYXRGb3JjZSAqIG4sIDApLCBjKHQodGhpcywgaCkud3JpdGUuZmJvKSwgdCh0aGlzLCBoKS5zd2FwKCksIHUuYWN0aXZlVGV4dHVyZSh1LlRFWFRVUkUwKSwgdS5iaW5kVGV4dHVyZSh1LlRFWFRVUkVfMkQsIHQodGhpcywgUikucmVhZC50ZXgpLCB1LnVuaWZvcm0zZihwLnVuaWZvcm1zLmNvbG9yLCBuLCBuLCBuKSwgYyh0KHRoaXMsIFIpLndyaXRlLmZibyksIHQodGhpcywgUikuc3dhcCgpOwogIH0KICB1cGRhdGVRdWFsaXR5KGUpIHsKICAgIGUuZHByICE9PSB2b2lkIDAgJiYgYSh0aGlzLCBRLCBNYXRoLm1heCgwLjEsIE1hdGgubWluKDEsIGUuZHByKSkpLCBlLnNpbSAhPT0gdm9pZCAwICYmIGEodGhpcywgTCwgTWF0aC5tYXgoMC4xLCBNYXRoLm1pbigxLCBlLnNpbSkpKTsKICB9CiAgcmVzaXplKGUsIHIsIG8pIHsKICAgIGlmIChvICE9PSB2b2lkIDAgPyBhKHRoaXMsIHcsIG8pIDogdHlwZW9mIHdpbmRvdyA8ICJ1IiAmJiB3aW5kb3cuZGV2aWNlUGl4ZWxSYXRpbyAmJiBhKHRoaXMsIHcsIHdpbmRvdy5kZXZpY2VQaXhlbFJhdGlvKSwgZSAhPT0gdm9pZCAwICYmIGUgPiAwKSB7CiAgICAgIGlmIChyID09PSB2b2lkIDAgfHwgciA8PSAwKQogICAgICAgIHJldHVybjsKICAgICAgYSh0aGlzLCBFLCB0KHRoaXMsIFcpLndpZHRoID0gZSksIGEodGhpcywgYiwgdCh0aGlzLCBXKS5oZWlnaHQgPSByKSwgYSh0aGlzLCBPLCBNYXRoLm1heCgxLCBNYXRoLnJvdW5kKGUgKiB0KHRoaXMsIEwpKSkpLCBhKHRoaXMsIEksIE1hdGgubWF4KDEsIE1hdGgucm91bmQociAqIHQodGhpcywgTCkpKSksIFQodGhpcywgaWUsIHBlKS5jYWxsKHRoaXMpOwogICAgfSBlbHNlCiAgICAgIFQodGhpcywgJCwgSikuY2FsbCh0aGlzKTsKICAgIHQodGhpcywgXykgJiYgVCh0aGlzLCBWLCBqKS5jYWxsKHRoaXMpLCBUKHRoaXMsIFksIFopLmNhbGwodGhpcyk7CiAgfQogIHVwZGF0ZUNvbmZpZyhlKSB7CiAgICBPYmplY3QuYXNzaWduKHQodGhpcywgeiksIGUpOwogIH0KICBkZXN0cm95KCkgewogICAgYSh0aGlzLCByZSwgITApLCB0aGlzLnN0b3AoKTsKICAgIGNvbnN0IGUgPSB0KHRoaXMsIGcpOwogICAgVCh0aGlzLCBvZSwgRWUpLmNhbGwodGhpcyksIFQodGhpcywgc2UsIGJlKS5jYWxsKHRoaXMpLCB0KHRoaXMsIFMpICYmICh0KHRoaXMsIFMpLmNsb3NlKCksIGEodGhpcywgUywgbnVsbCkpOwogICAgZm9yIChjb25zdCBvIG9mIE9iamVjdC52YWx1ZXModCh0aGlzLCBHKSkpCiAgICAgIG8uZGlzcG9zZSgpOwogICAgY29uc3QgciA9IGUuZ2V0RXh0ZW5zaW9uKCJXRUJHTF9sb3NlX2NvbnRleHQiKTsKICAgIHIgPT0gbnVsbCB8fCByLmxvc2VDb250ZXh0KCk7CiAgfQogIC8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQogIC8vIExvb3AgY29udHJvbAogIC8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQogIHN0YXJ0KCkgewogICAgaWYgKHQodGhpcywgQSkgIT09IG51bGwpCiAgICAgIHJldHVybjsKICAgIGNvbnN0IGUgPSAoKSA9PiB7CiAgICAgIFQodGhpcywgbGUsIERlKS5jYWxsKHRoaXMpLCBhKHRoaXMsIEEsIGdlKGUpKTsKICAgIH07CiAgICBhKHRoaXMsIEEsIGdlKGUpKTsKICB9CiAgc3RvcCgpIHsKICAgIHQodGhpcywgQSkgIT09IG51bGwgJiYgKHplKHQodGhpcywgQSkpLCBhKHRoaXMsIEEsIG51bGwpKTsKICB9CiAgZ2V0IGlzUnVubmluZygpIHsKICAgIHJldHVybiB0KHRoaXMsIEEpICE9PSBudWxsOwogIH0KfQpXID0gbmV3IFdlYWtNYXAoKSwgZyA9IG5ldyBXZWFrTWFwKCksIHRlID0gbmV3IFdlYWtNYXAoKSwgRyA9IG5ldyBXZWFrTWFwKCksIHEgPSBuZXcgV2Vha01hcCgpLCBFID0gbmV3IFdlYWtNYXAoKSwgYiA9IG5ldyBXZWFrTWFwKCksIE8gPSBuZXcgV2Vha01hcCgpLCBJID0gbmV3IFdlYWtNYXAoKSwgdyA9IG5ldyBXZWFrTWFwKCksIFEgPSBuZXcgV2Vha01hcCgpLCBMID0gbmV3IFdlYWtNYXAoKSwgUiA9IG5ldyBXZWFrTWFwKCksIGggPSBuZXcgV2Vha01hcCgpLCBDID0gbmV3IFdlYWtNYXAoKSwgRiA9IG5ldyBXZWFrTWFwKCksIFggPSBuZXcgV2Vha01hcCgpLCBNID0gbmV3IFdlYWtNYXAoKSwgVSA9IG5ldyBXZWFrTWFwKCksIEIgPSBuZXcgV2Vha01hcCgpLCBTID0gbmV3IFdlYWtNYXAoKSwgSyA9IG5ldyBXZWFrTWFwKCksIHogPSBuZXcgV2Vha01hcCgpLCB4ID0gbmV3IFdlYWtNYXAoKSwgXyA9IG5ldyBXZWFrTWFwKCksIEEgPSBuZXcgV2Vha01hcCgpLCBOID0gbmV3IFdlYWtNYXAoKSwgcmUgPSBuZXcgV2Vha01hcCgpLCAkID0gbmV3IFdlYWtTZXQoKSwgSiA9IGZ1bmN0aW9uKCkgewogIGNvbnN0IGUgPSB0KHRoaXMsIFcpOwogICJjbGllbnRXaWR0aCIgaW4gZSAmJiBlLmNsaWVudFdpZHRoID4gMCA/IChhKHRoaXMsIHcsICh0eXBlb2Ygd2luZG93IDwgInUiICYmIHdpbmRvdy5kZXZpY2VQaXhlbFJhdGlvIHx8IDEpICogdCh0aGlzLCBRKSksIGEodGhpcywgRSwgZS53aWR0aCA9IE1hdGgucm91bmQoZS5jbGllbnRXaWR0aCAqIHQodGhpcywgdykpKSwgYSh0aGlzLCBiLCBlLmhlaWdodCA9IE1hdGgucm91bmQoZS5jbGllbnRIZWlnaHQgKiB0KHRoaXMsIHcpKSkpIDogKGEodGhpcywgRSwgZS53aWR0aCksIGEodGhpcywgYiwgZS5oZWlnaHQpKSwgISh0KHRoaXMsIEUpID09PSAwIHx8IHQodGhpcywgYikgPT09IDApICYmIChhKHRoaXMsIE8sIE1hdGgubWF4KDEsIE1hdGgucm91bmQodCh0aGlzLCBFKSAqIHQodGhpcywgTCkpKSksIGEodGhpcywgSSwgTWF0aC5tYXgoMSwgTWF0aC5yb3VuZCh0KHRoaXMsIGIpICogdCh0aGlzLCBMKSkpKSwgVCh0aGlzLCBpZSwgcGUpLmNhbGwodGhpcykpOwp9LCBpZSA9IG5ldyBXZWFrU2V0KCksIHBlID0gZnVuY3Rpb24oKSB7CiAgY29uc3QgZSA9IHQodGhpcywgZyksIHIgPSB0KHRoaXMsIHRlKSwgbyA9IHQodGhpcywgTyksIHMgPSB0KHRoaXMsIEkpOwogIFQodGhpcywgb2UsIEVlKS5jYWxsKHRoaXMpLCBhKHRoaXMsIFIsIGRlKGUsIHIsIG8sIHMpKSwgYSh0aGlzLCBoLCBkZShlLCByLCBvLCBzKSksIGEodGhpcywgRiwgZGUoZSwgciwgbywgcykpLCBhKHRoaXMsIEMsIHVlKGUsIHIsIG8sIHMpKSwgYSh0aGlzLCBYLCB1ZShlLCByLCBvLCBzKSk7Cn0sIFYgPSBuZXcgV2Vha1NldCgpLCBqID0gZnVuY3Rpb24oKSB7CiAgaWYgKCEoIXQodGhpcywgXykgfHwgdCh0aGlzLCBFKSA9PT0gMCB8fCB0KHRoaXMsIGIpID09PSAwKSkgewogICAgaWYgKFQodGhpcywgc2UsIGJlKS5jYWxsKHRoaXMpLCB0KHRoaXMsIF8pLnR5cGUgPT09ICJ0ZXh0IikgewogICAgICBjb25zdCB7IGJhY2tncm91bmRUZXg6IGUsIG9ic3RhY2xlVGV4OiByLCBjb3ZlcmFnZVRleDogbyB9ID0gVmUoCiAgICAgICAgdCh0aGlzLCBnKSwKICAgICAgICB0KHRoaXMsIEUpLAogICAgICAgIHQodGhpcywgYiksCiAgICAgICAgdCh0aGlzLCBfKS5vcHRzLAogICAgICAgIHQodGhpcywgUyksCiAgICAgICAgdCh0aGlzLCBLKQogICAgICApOwogICAgICBhKHRoaXMsIE0sIGUpLCBhKHRoaXMsIFUsIHIpLCBhKHRoaXMsIEIsIG8pOwogICAgfSBlbHNlIHsKICAgICAgY29uc3QgeyBiYWNrZ3JvdW5kVGV4OiBlLCBvYnN0YWNsZVRleDogciwgY292ZXJhZ2VUZXg6IG8gfSA9IFdlKAogICAgICAgIHQodGhpcywgZyksCiAgICAgICAgdCh0aGlzLCBfKS5iaXRtYXAsCiAgICAgICAgdCh0aGlzLCBFKSwKICAgICAgICB0KHRoaXMsIGIpLAogICAgICAgIHQodGhpcywgXykuZWZmZWN0LAogICAgICAgIHQodGhpcywgXykuc2l6ZSwKICAgICAgICB0KHRoaXMsIFMpLAogICAgICAgIHQodGhpcywgSykKICAgICAgKTsKICAgICAgYSh0aGlzLCBNLCBlKSwgYSh0aGlzLCBVLCByKSwgYSh0aGlzLCBCLCBvKTsKICAgIH0KICAgIGEodGhpcywgTiwgITApOwogIH0KfSwgWSA9IG5ldyBXZWFrU2V0KCksIFogPSBmdW5jdGlvbigpIHsKICB0KHRoaXMsIE4pICYmICF0aGlzLmlzUnVubmluZyAmJiB0aGlzLnN0YXJ0KCk7Cn0sIG9lID0gbmV3IFdlYWtTZXQoKSwgRWUgPSBmdW5jdGlvbigpIHsKICB2YXIgZSwgciwgbzsKICAoZSA9IHQodGhpcywgUikpID09IG51bGwgfHwgZS5kaXNwb3NlKCksIChyID0gdCh0aGlzLCBoKSkgPT0gbnVsbCB8fCByLmRpc3Bvc2UoKSwgKG8gPSB0KHRoaXMsIEYpKSA9PSBudWxsIHx8IG8uZGlzcG9zZSgpLCB0KHRoaXMsIEMpICYmICh0KHRoaXMsIGcpLmRlbGV0ZVRleHR1cmUodCh0aGlzLCBDKS50ZXgpLCB0KHRoaXMsIGcpLmRlbGV0ZUZyYW1lYnVmZmVyKHQodGhpcywgQykuZmJvKSksIHQodGhpcywgWCkgJiYgKHQodGhpcywgZykuZGVsZXRlVGV4dHVyZSh0KHRoaXMsIFgpLnRleCksIHQodGhpcywgZykuZGVsZXRlRnJhbWVidWZmZXIodCh0aGlzLCBYKS5mYm8pKSwgYSh0aGlzLCBSLCBhKHRoaXMsIGgsIGEodGhpcywgRiwgYSh0aGlzLCBDLCBhKHRoaXMsIFgsIG51bGwpKSkpKTsKfSwgc2UgPSBuZXcgV2Vha1NldCgpLCBiZSA9IGZ1bmN0aW9uKCkgewogIHQodGhpcywgTSkgJiYgdCh0aGlzLCBnKS5kZWxldGVUZXh0dXJlKHQodGhpcywgTSkpLCB0KHRoaXMsIFUpICYmIHQodGhpcywgZykuZGVsZXRlVGV4dHVyZSh0KHRoaXMsIFUpKSwgdCh0aGlzLCBCKSAmJiB0KHRoaXMsIEIpICE9PSB0KHRoaXMsIFUpICYmIHQodGhpcywgZykuZGVsZXRlVGV4dHVyZSh0KHRoaXMsIEIpKSwgYSh0aGlzLCBNLCBhKHRoaXMsIFUsIGEodGhpcywgQiwgbnVsbCkpKTsKfSwgbGUgPSBuZXcgV2Vha1NldCgpLCBEZSA9IGZ1bmN0aW9uKCkgewogIGlmICghdCh0aGlzLCBOKSB8fCB0KHRoaXMsIEUpID09PSAwIHx8ICF0KHRoaXMsIFIpIHx8ICF0KHRoaXMsIGgpKQogICAgcmV0dXJuOwogIGNvbnN0IGUgPSB0KHRoaXMsIGcpLCByID0gdCh0aGlzLCB6KSwgeyBhZHZlY3Rpb246IG8sIGRpdmVyZ2VuY2U6IHMsIHByZXNzdXJlOiBuLCBncmFkaWVudFN1YnRyYWN0OiB1LCBzcGxhdDogZCwgY3VybDogcCwgdm9ydGljaXR5OiBjLCBkaXNwbGF5OiBtIH0gPSB0KHRoaXMsIEcpOwogIHQodGhpcywgeCkueCArPSAodCh0aGlzLCB4KS50YXJnZXRYIC0gdCh0aGlzLCB4KS54KSAqIDAuMTUsIHQodGhpcywgeCkueSArPSAodCh0aGlzLCB4KS50YXJnZXRZIC0gdCh0aGlzLCB4KS55KSAqIDAuMTU7CiAgY29uc3QgeSA9IHQodGhpcywgTyksIGYgPSB0KHRoaXMsIEkpLCBEID0gdCh0aGlzLCBxKTsKICBlLnZpZXdwb3J0KDAsIDAsIHksIGYpLCBvLmJpbmQoKSwgZS51bmlmb3JtMmYoby51bmlmb3Jtcy50ZXhlbFNpemUsIDEgLyB5LCAxIC8gZiksIGUudW5pZm9ybTFmKG8udW5pZm9ybXMuZHQsIFJlKSwgZS51bmlmb3JtMWkoby51bmlmb3Jtcy51T2JzdGFjbGUsIDApLCBlLmFjdGl2ZVRleHR1cmUoZS5URVhUVVJFMCksIGUuYmluZFRleHR1cmUoZS5URVhUVVJFXzJELCB0KHRoaXMsIFUpKSwgZS51bmlmb3JtMWYoby51bmlmb3Jtcy5kaXNzaXBhdGlvbiwgci52ZWxvY2l0eURpc3NpcGF0aW9uKSwgZS51bmlmb3JtMWkoby51bmlmb3Jtcy51VmVsb2NpdHksIDEpLCBlLmFjdGl2ZVRleHR1cmUoZS5URVhUVVJFMSksIGUuYmluZFRleHR1cmUoZS5URVhUVVJFXzJELCB0KHRoaXMsIGgpLnJlYWQudGV4KSwgZS51bmlmb3JtMWkoby51bmlmb3Jtcy51U291cmNlLCAxKSwgRCh0KHRoaXMsIGgpLndyaXRlLmZibyksIHQodGhpcywgaCkuc3dhcCgpLCBlLnVuaWZvcm0xZihvLnVuaWZvcm1zLmRpc3NpcGF0aW9uLCByLmRlbnNpdHlEaXNzaXBhdGlvbiksIGUudW5pZm9ybTFpKG8udW5pZm9ybXMudVNvdXJjZSwgMiksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUyKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgUikucmVhZC50ZXgpLCBEKHQodGhpcywgUikud3JpdGUuZmJvKSwgdCh0aGlzLCBSKS5zd2FwKCksIHAuYmluZCgpLCBlLnVuaWZvcm0yZihwLnVuaWZvcm1zLnRleGVsU2l6ZSwgMSAvIHksIDEgLyBmKSwgZS51bmlmb3JtMWkocC51bmlmb3Jtcy51VmVsb2NpdHksIDApLCBlLmFjdGl2ZVRleHR1cmUoZS5URVhUVVJFMCksIGUuYmluZFRleHR1cmUoZS5URVhUVVJFXzJELCB0KHRoaXMsIGgpLnJlYWQudGV4KSwgRCh0KHRoaXMsIFgpLmZibyksIGMuYmluZCgpLCBlLnVuaWZvcm0yZihjLnVuaWZvcm1zLnRleGVsU2l6ZSwgMSAvIHksIDEgLyBmKSwgZS51bmlmb3JtMWYoYy51bmlmb3Jtcy5jdXJsLCByLmN1cmwpLCBlLnVuaWZvcm0xZihjLnVuaWZvcm1zLmR0LCBSZSksIGUudW5pZm9ybTFpKGMudW5pZm9ybXMudVZlbG9jaXR5LCAwKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTApLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBoKS5yZWFkLnRleCksIGUudW5pZm9ybTFpKGMudW5pZm9ybXMudUN1cmwsIDEpLCBlLmFjdGl2ZVRleHR1cmUoZS5URVhUVVJFMSksIGUuYmluZFRleHR1cmUoZS5URVhUVVJFXzJELCB0KHRoaXMsIFgpLnRleCksIEQodCh0aGlzLCBoKS53cml0ZS5mYm8pLCB0KHRoaXMsIGgpLnN3YXAoKSwgdCh0aGlzLCB4KS5tb3ZlZCAmJiAoZC5iaW5kKCksIGUudW5pZm9ybTFmKGQudW5pZm9ybXMuYXNwZWN0UmF0aW8sIHQodGhpcywgRSkgLyB0KHRoaXMsIGIpKSwgZS51bmlmb3JtMmYoZC51bmlmb3Jtcy5wb2ludCwgdCh0aGlzLCB4KS54ICogdCh0aGlzLCB3KSAvIHQodGhpcywgRSksIDEgLSB0KHRoaXMsIHgpLnkgKiB0KHRoaXMsIHcpIC8gdCh0aGlzLCBiKSksIGUudW5pZm9ybTFmKGQudW5pZm9ybXMucmFkaXVzLCByLnNwbGF0UmFkaXVzKSwgZS51bmlmb3JtMWkoZC51bmlmb3Jtcy51VGFyZ2V0LCAwKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTApLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBoKS5yZWFkLnRleCksIGUudW5pZm9ybTNmKGQudW5pZm9ybXMuY29sb3IsIHQodGhpcywgeCkuZHggKiByLnNwbGF0Rm9yY2UsIC10KHRoaXMsIHgpLmR5ICogci5zcGxhdEZvcmNlLCAwKSwgRCh0KHRoaXMsIGgpLndyaXRlLmZibyksIHQodGhpcywgaCkuc3dhcCgpLCBlLmFjdGl2ZVRleHR1cmUoZS5URVhUVVJFMCksIGUuYmluZFRleHR1cmUoZS5URVhUVVJFXzJELCB0KHRoaXMsIFIpLnJlYWQudGV4KSwgZS51bmlmb3JtM2YoZC51bmlmb3Jtcy5jb2xvciwgMSwgMSwgMSksIEQodCh0aGlzLCBSKS53cml0ZS5mYm8pLCB0KHRoaXMsIFIpLnN3YXAoKSwgdCh0aGlzLCB4KS5tb3ZlZCA9ICExKSwgcy5iaW5kKCksIGUudW5pZm9ybTJmKHMudW5pZm9ybXMudGV4ZWxTaXplLCAxIC8geSwgMSAvIGYpLCBlLnVuaWZvcm0xaShzLnVuaWZvcm1zLnVWZWxvY2l0eSwgMCksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUwKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgaCkucmVhZC50ZXgpLCBlLnVuaWZvcm0xaShzLnVuaWZvcm1zLnVPYnN0YWNsZSwgMSksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUxKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgVSkpLCBEKHQodGhpcywgQykuZmJvKSwgbi5iaW5kKCksIGUudW5pZm9ybTJmKG4udW5pZm9ybXMudGV4ZWxTaXplLCAxIC8geSwgMSAvIGYpLCBlLnVuaWZvcm0xaShuLnVuaWZvcm1zLnVEaXZlcmdlbmNlLCAwKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTApLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBDKS50ZXgpLCBlLnVuaWZvcm0xaShuLnVuaWZvcm1zLnVPYnN0YWNsZSwgMSksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUxKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgVSkpOwogIGZvciAobGV0IEggPSAwOyBIIDwgci5wcmVzc3VyZUl0ZXJhdGlvbnM7IEgrKykKICAgIGUudW5pZm9ybTFpKG4udW5pZm9ybXMudVByZXNzdXJlLCAyKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTIpLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBGKS5yZWFkLnRleCksIEQodCh0aGlzLCBGKS53cml0ZS5mYm8pLCB0KHRoaXMsIEYpLnN3YXAoKTsKICB1LmJpbmQoKSwgZS51bmlmb3JtMmYodS51bmlmb3Jtcy50ZXhlbFNpemUsIDEgLyB5LCAxIC8gZiksIGUudW5pZm9ybTFpKHUudW5pZm9ybXMudVByZXNzdXJlLCAwKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTApLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBGKS5yZWFkLnRleCksIGUudW5pZm9ybTFpKHUudW5pZm9ybXMudVZlbG9jaXR5LCAxKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTEpLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBoKS5yZWFkLnRleCksIGUudW5pZm9ybTFpKHUudW5pZm9ybXMudU9ic3RhY2xlLCAyKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTIpLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBVKSksIEQodCh0aGlzLCBoKS53cml0ZS5mYm8pLCB0KHRoaXMsIGgpLnN3YXAoKSwgZS52aWV3cG9ydCgwLCAwLCB0KHRoaXMsIEUpLCB0KHRoaXMsIGIpKSwgZS5iaW5kRnJhbWVidWZmZXIoZS5GUkFNRUJVRkZFUiwgbnVsbCksIGUuY2xlYXIoZS5DT0xPUl9CVUZGRVJfQklUKSwgbS5iaW5kKCksIGUudW5pZm9ybTJmKG0udW5pZm9ybXMudGV4ZWxTaXplLCAxIC8gdCh0aGlzLCBFKSwgMSAvIHQodGhpcywgYikpLCBlLnVuaWZvcm0zZnYobS51bmlmb3Jtcy51V2F0ZXJDb2xvciwgci53YXRlckNvbG9yKSwgZS51bmlmb3JtM2Z2KG0udW5pZm9ybXMudUdsb3dDb2xvciwgci5nbG93Q29sb3IpLCBlLnVuaWZvcm0xZihtLnVuaWZvcm1zLnVSZWZyYWN0aW9uLCByLnJlZnJhY3Rpb24pLCBlLnVuaWZvcm0xZihtLnVuaWZvcm1zLnVTcGVjdWxhckV4cCwgci5zcGVjdWxhckV4cCksIGUudW5pZm9ybTFmKG0udW5pZm9ybXMudVNoaW5lLCByLnNoaW5lKSwgZS51bmlmb3JtMWYobS51bmlmb3Jtcy51V2FycFN0cmVuZ3RoLCByLndhcnBTdHJlbmd0aCA/PyAwLjAxNSksIGUudW5pZm9ybTFpKG0udW5pZm9ybXMudUFsZ29yaXRobSwgTmVbci5hbGdvcml0aG1dID8/IDApLCBlLmFjdGl2ZVRleHR1cmUoZS5URVhUVVJFMCksIGUuYmluZFRleHR1cmUoZS5URVhUVVJFXzJELCB0KHRoaXMsIFIpLnJlYWQudGV4KSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTEpLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBVKSksIGUuYWN0aXZlVGV4dHVyZShlLlRFWFRVUkUyKSwgZS5iaW5kVGV4dHVyZShlLlRFWFRVUkVfMkQsIHQodGhpcywgTSkpLCBlLmFjdGl2ZVRleHR1cmUoZS5URVhUVVJFMyksIGUuYmluZFRleHR1cmUoZS5URVhUVVJFXzJELCB0KHRoaXMsIEIpKSwgZS5hY3RpdmVUZXh0dXJlKGUuVEVYVFVSRTQpLCBlLmJpbmRUZXh0dXJlKGUuVEVYVFVSRV8yRCwgdCh0aGlzLCBoKS5yZWFkLnRleCksIGUudW5pZm9ybTFpKG0udW5pZm9ybXMudVRleHR1cmUsIDApLCBlLnVuaWZvcm0xaShtLnVuaWZvcm1zLnVPYnN0YWNsZSwgMSksIGUudW5pZm9ybTFpKG0udW5pZm9ybXMudUJhY2tncm91bmQsIDIpLCBlLnVuaWZvcm0xaShtLnVuaWZvcm1zLnVDb3ZlcmFnZSwgMyksIGUudW5pZm9ybTFpKG0udW5pZm9ybXMudVZlbG9jaXR5LCA0KSwgRChudWxsKTsKfTsKbGV0IHYgPSBudWxsOwpzZWxmLm9ubWVzc2FnZSA9IGFzeW5jIChpKSA9PiB7CiAgY29uc3QgeyB0eXBlOiBlLCAuLi5yIH0gPSBpLmRhdGE7CiAgdHJ5IHsKICAgIHN3aXRjaCAoZSkgewogICAgICBjYXNlICJpbml0IjogewogICAgICAgIGNvbnN0IHsgY2FudmFzOiBvLCB3aWR0aDogcywgaGVpZ2h0OiBuLCBjb25maWc6IHUsIGRwcjogZCwgcXVhbGl0eTogcCB9ID0gcjsKICAgICAgICBvLndpZHRoID0gcywgby5oZWlnaHQgPSBuLCB2ID0gbmV3ICRlKG8sIHUsIHAgPz8ge30pLCB2LnJlc2l6ZShzLCBuLCBkIHx8IDEpLCBzZWxmLnBvc3RNZXNzYWdlKHsgdHlwZTogInJlYWR5IiB9KTsKICAgICAgICBicmVhazsKICAgICAgfQogICAgICBjYXNlICJzZXRUZXh0U291cmNlIjogewogICAgICAgIGlmICghdikKICAgICAgICAgIHJldHVybjsKICAgICAgICB2LnNldFRleHRTb3VyY2Uoci5vcHRzKTsKICAgICAgICBicmVhazsKICAgICAgfQogICAgICBjYXNlICJzZXRJbWFnZVNvdXJjZSI6IHsKICAgICAgICBpZiAoIXYpCiAgICAgICAgICByZXR1cm47CiAgICAgICAgYXdhaXQgdi5zZXRJbWFnZVNvdXJjZSgKICAgICAgICAgIHIuc3JjLAogICAgICAgICAgci5lZmZlY3QsCiAgICAgICAgICByLnNpemUKICAgICAgICApOwogICAgICAgIGJyZWFrOwogICAgICB9CiAgICAgIGNhc2UgInNldEltYWdlQml0bWFwIjogewogICAgICAgIGlmICghdikKICAgICAgICAgIHJldHVybjsKICAgICAgICB2LnNldEltYWdlQml0bWFwKAogICAgICAgICAgci5iaXRtYXAsCiAgICAgICAgICByLmVmZmVjdCwKICAgICAgICAgIHIuc2l6ZQogICAgICAgICk7CiAgICAgICAgYnJlYWs7CiAgICAgIH0KICAgICAgY2FzZSAic2V0QmFja2dyb3VuZCI6IHsKICAgICAgICBpZiAoIXYpCiAgICAgICAgICByZXR1cm47CiAgICAgICAgdi5zZXRCYWNrZ3JvdW5kKHIuYml0bWFwLCByLnNpemUpOwogICAgICAgIGJyZWFrOwogICAgICB9CiAgICAgIGNhc2UgInNwbGF0IjogewogICAgICAgIGlmICghdikKICAgICAgICAgIHJldHVybjsKICAgICAgICB2LnNwbGF0KAogICAgICAgICAgci54LAogICAgICAgICAgci55LAogICAgICAgICAgci52eCwKICAgICAgICAgIHIudnksCiAgICAgICAgICByLnN0cmVuZ3RoID8/IDEKICAgICAgICApOwogICAgICAgIGJyZWFrOwogICAgICB9CiAgICAgIGNhc2UgIm1vdmUiOiB7CiAgICAgICAgaWYgKCF2KQogICAgICAgICAgcmV0dXJuOwogICAgICAgIHYuaGFuZGxlTW92ZShyLngsIHIueSwgci5zdHJlbmd0aCA/PyAxKTsKICAgICAgICBicmVhazsKICAgICAgfQogICAgICBjYXNlICJyZXNpemUiOiB7CiAgICAgICAgaWYgKCF2KQogICAgICAgICAgcmV0dXJuOwogICAgICAgIHYucmVzaXplKHIud2lkdGgsIHIuaGVpZ2h0LCByLmRwcik7CiAgICAgICAgYnJlYWs7CiAgICAgIH0KICAgICAgY2FzZSAidXBkYXRlUXVhbGl0eSI6IHsKICAgICAgICBpZiAoIXYpCiAgICAgICAgICByZXR1cm47CiAgICAgICAgdi51cGRhdGVRdWFsaXR5KHIucXVhbGl0eSk7CiAgICAgICAgYnJlYWs7CiAgICAgIH0KICAgICAgY2FzZSAidXBkYXRlQ29uZmlnIjogewogICAgICAgIGlmICghdikKICAgICAgICAgIHJldHVybjsKICAgICAgICB2LnVwZGF0ZUNvbmZpZyhyLmNvbmZpZyk7CiAgICAgICAgYnJlYWs7CiAgICAgIH0KICAgICAgY2FzZSAiZGVzdHJveSI6IHsKICAgICAgICB2ID09IG51bGwgfHwgdi5kZXN0cm95KCksIHYgPSBudWxsOwogICAgICAgIGJyZWFrOwogICAgICB9CiAgICAgIGRlZmF1bHQ6CiAgICAgICAgY29uc29sZS53YXJuKCJbZmx1aWRpdHktanMgd29ya2VyXSBVbmtub3duIG1lc3NhZ2UgdHlwZToiLCBlKTsKICAgIH0KICB9IGNhdGNoIChvKSB7CiAgICBzZWxmLnBvc3RNZXNzYWdlKHsgdHlwZTogImVycm9yIiwgbWVzc2FnZTogKG8gPT0gbnVsbCA/IHZvaWQgMCA6IG8ubWVzc2FnZSkgPz8gU3RyaW5nKG8pIH0pOwogIH0KfTsK", JI = typeof window < "u" && window.Blob && new Blob([atob(PI)], { type: "text/javascript;charset=utf-8" });
680
+ function Gg() {
675
681
  let C;
676
682
  try {
677
- if (C = FI && (window.URL || window.webkitURL).createObjectURL(FI), !C)
683
+ if (C = JI && (window.URL || window.webkitURL).createObjectURL(JI), !C)
678
684
  throw "";
679
685
  return new Worker(C);
680
686
  } catch {
681
- return new Worker("data:application/javascript;base64," + zI, { type: "module" });
687
+ return new Worker("data:application/javascript;base64," + PI, { type: "module" });
682
688
  } finally {
683
689
  C && (window.URL || window.webkitURL).revokeObjectURL(C);
684
690
  }
685
691
  }
686
- const rg = typeof Worker < "u" && typeof OffscreenCanvas < "u";
687
- var p, k, aI, wI, yI, QI;
688
- class ug {
689
- constructor(I, { isWorkerEnabled: i = !0, config: e = {} } = {}) {
692
+ const hg = typeof Worker < "u" && typeof OffscreenCanvas < "u";
693
+ var p, U, ZI, T, O, XI, _I;
694
+ class Bg {
695
+ constructor(I, {
696
+ isWorkerEnabled: i = !0,
697
+ quality: e = {},
698
+ config: s = {}
699
+ } = {}) {
690
700
  // ---------------------------------------------------------------------------
691
701
  // Private
692
702
  // ---------------------------------------------------------------------------
693
- n(this, yI);
694
- n(this, p, null);
695
- n(this, k, null);
696
- n(this, aI, void 0);
697
- n(this, wI, void 0);
698
- o(this, wI, I), o(this, aI, i && rg), g(this, aI) ? b(this, yI, QI).call(this, I, e) : o(this, k, new fI(I, e));
703
+ m(this, XI);
704
+ m(this, p, null);
705
+ m(this, U, null);
706
+ m(this, ZI, void 0);
707
+ m(this, T, void 0);
708
+ m(this, O, void 0);
709
+ l(this, T, Math.max(0.1, Math.min(1, e.dpr ?? 1))), l(this, O, Math.max(0.1, Math.min(1, e.sim ?? 0.5))), l(this, ZI, i && hg), g(this, ZI) ? b(this, XI, _I).call(this, I, s) : l(this, U, new EI(I, s, { dpr: g(this, T), sim: g(this, O) }));
699
710
  }
700
711
  // ---------------------------------------------------------------------------
701
712
  // Source setters
702
713
  // ---------------------------------------------------------------------------
703
714
  setTextSource(I) {
704
- g(this, p) ? g(this, p).postMessage({ type: "setTextSource", opts: I }) : g(this, k).setTextSource(I);
715
+ g(this, p) ? g(this, p).postMessage({ type: "setTextSource", opts: I }) : g(this, U).setTextSource(I);
705
716
  }
706
- setImageSource(I, i = q.effect, e = q.imageSize) {
717
+ setImageSource(I, i = iI.effect, e = iI.imageSize) {
707
718
  if (g(this, p)) {
708
- const l = new URL(I, location.href).href;
709
- g(this, p).postMessage({ type: "setImageSource", src: l, effect: i, size: e });
719
+ const s = new URL(I, location.href).href;
720
+ g(this, p).postMessage({ type: "setImageSource", src: s, effect: i, size: e });
710
721
  } else
711
- g(this, k).setImageSource(I, i, e);
722
+ g(this, U).setImageSource(I, i, e);
712
723
  }
713
724
  setBackground(I, i = "cover") {
714
725
  var e;
715
726
  if (g(this, p)) {
716
- const l = I ? [I] : [];
717
- g(this, p).postMessage({ type: "setBackground", bitmap: I ?? null, size: i }, l);
727
+ const s = I ? [I] : [];
728
+ g(this, p).postMessage({ type: "setBackground", bitmap: I ?? null, size: i }, s);
718
729
  } else
719
- (e = g(this, k)) == null || e.setBackground(I ?? null, i);
730
+ (e = g(this, U)) == null || e.setBackground(I ?? null, i);
720
731
  }
721
732
  // ---------------------------------------------------------------------------
722
733
  // Interaction
@@ -725,165 +736,185 @@ class ug {
725
736
  * Immediately injects one splat at (x, y) with explicit velocity (vx, vy).
726
737
  * Safe to call multiple times per frame. See FluidSimulation.splat for details.
727
738
  */
728
- splat(I, i, e, l, s = 1) {
729
- g(this, p) ? g(this, p).postMessage({ type: "splat", x: I, y: i, vx: e, vy: l, strength: s }) : g(this, k).splat(I, i, e, l, s);
739
+ splat(I, i, e, s, o = 1) {
740
+ g(this, p) ? g(this, p).postMessage({ type: "splat", x: I, y: i, vx: e, vy: s, strength: o }) : g(this, U).splat(I, i, e, s, o);
730
741
  }
731
742
  handleMove(I, i, e = 1) {
732
- g(this, p) ? g(this, p).postMessage({ type: "move", x: I, y: i, strength: e }) : g(this, k).handleMove(I, i, e);
743
+ g(this, p) ? g(this, p).postMessage({ type: "move", x: I, y: i, strength: e }) : g(this, U).handleMove(I, i, e);
733
744
  }
734
745
  // ---------------------------------------------------------------------------
735
746
  // Config + control
736
747
  // ---------------------------------------------------------------------------
748
+ updateQuality(I) {
749
+ l(this, T, Math.max(0.1, Math.min(1, I.dpr ?? g(this, T)))), l(this, O, Math.max(0.1, Math.min(1, I.sim ?? g(this, O)))), g(this, p) ? g(this, p).postMessage({ type: "updateQuality", quality: { dpr: g(this, T), sim: g(this, O) } }) : g(this, U).updateQuality(I);
750
+ }
737
751
  updateConfig(I) {
738
- g(this, p) ? g(this, p).postMessage({ type: "updateConfig", config: I }) : g(this, k).updateConfig(I);
752
+ g(this, p) ? g(this, p).postMessage({ type: "updateConfig", config: I }) : g(this, U).updateConfig(I);
739
753
  }
740
754
  resize(I, i) {
741
- if (g(this, p)) {
742
- const e = typeof window < "u" && window.devicePixelRatio || 1;
743
- g(this, p).postMessage({ type: "resize", width: I, height: i, dpr: e });
744
- } else
745
- g(this, k).resize(I, i);
755
+ const e = (typeof window < "u" && window.devicePixelRatio || 1) * g(this, T);
756
+ g(this, p) ? g(this, p).postMessage({ type: "resize", width: I, height: i, dpr: e }) : g(this, U).resize(I, i, e);
746
757
  }
747
758
  destroy() {
748
759
  var I;
749
760
  if (g(this, p)) {
750
761
  const i = g(this, p);
751
- o(this, p, null), i.postMessage({ type: "destroy" }), setTimeout(() => i.terminate(), 50);
762
+ l(this, p, null), i.postMessage({ type: "destroy" }), setTimeout(() => i.terminate(), 50);
752
763
  } else
753
- (I = g(this, k)) == null || I.destroy(), o(this, k, null);
764
+ (I = g(this, U)) == null || I.destroy(), l(this, U, null);
754
765
  }
755
766
  }
756
- p = new WeakMap(), k = new WeakMap(), aI = new WeakMap(), wI = new WeakMap(), yI = new WeakSet(), QI = function(I, i) {
757
- const e = typeof window < "u" && window.devicePixelRatio || 1, l = Math.round(I.clientWidth * e), s = Math.round(I.clientHeight * e);
758
- I.width = l, I.height = s;
767
+ p = new WeakMap(), U = new WeakMap(), ZI = new WeakMap(), T = new WeakMap(), O = new WeakMap(), XI = new WeakSet(), _I = function(I, i) {
768
+ const e = (typeof window < "u" && window.devicePixelRatio || 1) * g(this, T), s = Math.round(I.clientWidth * e), o = Math.round(I.clientHeight * e);
769
+ I.width = s, I.height = o;
759
770
  let t;
760
771
  try {
761
772
  t = I.transferControlToOffscreen();
762
773
  } catch {
763
774
  console.warn(
764
775
  "[fluidity-js] OffscreenCanvas transfer failed — falling back to main-thread mode. This is expected in React StrictMode development."
765
- ), o(this, aI, !1), o(this, k, new fI(I, i));
776
+ ), l(this, ZI, !1), l(this, U, new EI(I, i, { dpr: g(this, T), sim: g(this, O) }));
766
777
  return;
767
778
  }
768
- const m = o(this, p, new ng());
769
- m.onerror = (r) => {
770
- console.error("[fluidity-js] Worker error:", r.message);
771
- }, m.onmessage = (r) => {
772
- r.data.type === "error" && console.error("[fluidity-js] Simulation error:", r.data.message);
773
- }, m.postMessage({ type: "init", canvas: t, width: l, height: s, config: i, dpr: e }, [t]);
779
+ const u = l(this, p, new Gg());
780
+ u.onerror = (a) => {
781
+ console.error("[fluidity-js] Worker error:", a.message);
782
+ }, u.onmessage = (a) => {
783
+ a.data.type === "error" && console.error("[fluidity-js] Simulation error:", a.data.message);
784
+ }, u.postMessage(
785
+ { type: "init", canvas: t, width: s, height: o, config: i, dpr: e, quality: { dpr: g(this, T), sim: g(this, O) } },
786
+ [t]
787
+ );
774
788
  };
775
- function OI(C, { isWorkerEnabled: I = !0, config: i = {} } = {}) {
776
- const e = bI(null), l = bI({ isWorkerEnabled: I, config: i });
777
- return Q(() => {
778
- const s = C.current;
779
- if (!s)
789
+ function qI(C, { isWorkerEnabled: I = !0, quality: i = {}, config: e = {} } = {}) {
790
+ const s = mI(null), o = mI({ isWorkerEnabled: I, quality: i, config: e }), t = mI(Math.max(0.1, Math.min(1, i.dpr ?? 1))), u = mI({
791
+ dpr: i.dpr,
792
+ sim: i.sim
793
+ });
794
+ return j(() => {
795
+ const a = C.current;
796
+ if (!a)
780
797
  return;
781
- const t = document.createElement("canvas");
782
- t.style.cssText = "position:absolute;inset:0;width:100%;height:100%;display:block;", s.appendChild(t);
783
- const m = window.devicePixelRatio || 1, r = s.getBoundingClientRect(), a = Math.round((r.width || s.clientWidth) * m) || 0, u = Math.round((r.height || s.clientHeight) * m) || 0;
784
- a > 0 && (t.width = a, t.height = u);
785
- const { isWorkerEnabled: S, config: A } = l.current, h = new ug(t, { isWorkerEnabled: S, config: A });
786
- e.current = h;
787
- const v = new ResizeObserver((Y) => {
788
- for (const K of Y) {
789
- const D = window.devicePixelRatio || 1, { inlineSize: c, blockSize: w } = K.contentBoxSize[0];
790
- h.resize(Math.round(c * D), Math.round(w * D));
798
+ const c = document.createElement("canvas");
799
+ c.style.cssText = "position:absolute;inset:0;width:100%;height:100%;display:block;", a.appendChild(c);
800
+ const { isWorkerEnabled: n, quality: G, config: d } = o.current, h = (window.devicePixelRatio || 1) * t.current, W = a.getBoundingClientRect(), x = Math.round((W.width || a.clientWidth) * h) || 0, F = Math.round((W.height || a.clientHeight) * h) || 0;
801
+ x > 0 && (c.width = x, c.height = F), F === 0 && console.warn(
802
+ "[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
+ );
804
+ const w = new Bg(c, { isWorkerEnabled: n, quality: G, config: d });
805
+ s.current = w;
806
+ const D = new ResizeObserver((A) => {
807
+ for (const y of A) {
808
+ const r = (window.devicePixelRatio || 1) * t.current, { inlineSize: Z, blockSize: V } = y.contentBoxSize[0];
809
+ w.resize(Math.round(Z * r), Math.round(V * r));
791
810
  }
792
811
  });
793
- return v.observe(s), () => {
794
- v.disconnect(), h.destroy(), t.remove(), e.current = null;
812
+ return D.observe(a), () => {
813
+ D.disconnect(), w.destroy(), c.remove(), s.current = null;
795
814
  };
796
- }, []), e;
815
+ }, []), j(() => {
816
+ t.current = Math.max(0.1, Math.min(1, i.dpr ?? 1));
817
+ const a = u.current;
818
+ u.current = { dpr: i.dpr, sim: i.sim };
819
+ const c = s.current, n = C.current;
820
+ if (!c || !n || a.dpr === i.dpr && a.sim === i.sim)
821
+ return;
822
+ c.updateQuality(i);
823
+ const G = (window.devicePixelRatio || 1) * t.current, d = n.clientWidth, h = n.clientHeight;
824
+ d > 0 && h > 0 && c.resize(Math.round(d * G), Math.round(h * G));
825
+ }, [i.dpr, i.sim]), s;
797
826
  }
798
- const hg = xI(function({
827
+ const Lg = zI(function({
799
828
  text: I,
800
- fontSize: i = j.fontSize,
801
- color: e = j.color,
802
- fontFamily: l = j.fontFamily,
803
- fontWeight: s = j.fontWeight,
829
+ fontSize: i = $.fontSize,
830
+ color: e = $.color,
831
+ fontFamily: s = $.fontFamily,
832
+ fontWeight: o = $.fontWeight,
804
833
  className: t,
805
- style: m,
806
- config: r,
807
- preset: a,
808
- algorithm: u,
809
- backgroundColor: S = j.backgroundColor,
810
- backgroundSrc: A,
811
- backgroundSize: h = j.backgroundSize,
812
- isMouseEnabled: v = j.isMouseEnabled,
813
- isWorkerEnabled: Y = j.isWorkerEnabled
814
- }, K) {
815
- const D = bI(null), c = OI(D, {
816
- isWorkerEnabled: Y,
817
- config: uI({ ...r, ...u ? { algorithm: u } : {} }, a, kI)
834
+ style: u,
835
+ config: a,
836
+ preset: c,
837
+ algorithm: n,
838
+ backgroundColor: G = $.backgroundColor,
839
+ backgroundSrc: d,
840
+ backgroundSize: h = $.backgroundSize,
841
+ isMouseEnabled: W = $.isMouseEnabled,
842
+ isWorkerEnabled: x = $.isWorkerEnabled,
843
+ quality: F = HI.quality
844
+ }, w) {
845
+ const D = mI(null), A = qI(D, {
846
+ isWorkerEnabled: x,
847
+ quality: F,
848
+ config: pI({ ...a, ...n ? { algorithm: n } : {} }, c, TI)
818
849
  });
819
- MI(
820
- K,
850
+ QI(
851
+ w,
821
852
  () => ({
822
853
  reset() {
823
- var d;
824
- (d = c.current) == null || d.setTextSource({ text: I, fontSize: i, color: e, fontFamily: l, fontWeight: s });
854
+ var r;
855
+ (r = A.current) == null || r.setTextSource({ text: I, fontSize: i, color: e, fontFamily: s, fontWeight: o });
825
856
  },
826
- move({ x: d, y: G, strength: L = 1 }) {
827
- var B;
828
- (B = c.current) == null || B.handleMove(d, G, L);
857
+ move(r, Z, V = 1) {
858
+ var S;
859
+ (S = A.current) == null || S.handleMove(r, Z, V);
829
860
  },
830
- splat(d, G, L, B, T = 1) {
831
- var O;
832
- (O = c.current) == null || O.splat(d, G, L, B, T);
861
+ splat(r, Z, V, S, M = 1) {
862
+ var q;
863
+ (q = A.current) == null || q.splat(r, Z, V, S, M);
833
864
  },
834
- updateConfig(d) {
835
- var G;
836
- (G = c.current) == null || G.updateConfig(d);
865
+ updateConfig(r) {
866
+ var Z;
867
+ (Z = A.current) == null || Z.updateConfig(r);
837
868
  }
838
869
  }),
839
- [I, i, e, l, s]
840
- ), Q(() => {
841
- var d;
842
- (d = c.current) == null || d.setTextSource({ text: I, fontSize: i, color: e, fontFamily: l, fontWeight: s });
843
- }, [I, i, e, l, s]);
844
- const w = JSON.stringify(r);
845
- return Q(() => {
846
- var d;
847
- (d = c.current) == null || d.updateConfig(
848
- uI({ ...r, ...u !== void 0 ? { algorithm: u } : {} }, a, kI)
870
+ [I, i, e, s, o]
871
+ ), j(() => {
872
+ var r;
873
+ (r = A.current) == null || r.setTextSource({ text: I, fontSize: i, color: e, fontFamily: s, fontWeight: o });
874
+ }, [I, i, e, s, o]);
875
+ const y = JSON.stringify(a);
876
+ return j(() => {
877
+ var r;
878
+ (r = A.current) == null || r.updateConfig(
879
+ pI({ ...a, ...n !== void 0 ? { algorithm: n } : {} }, c, TI)
849
880
  );
850
- }, [a, u, w]), Q(() => {
851
- var G;
852
- if (!A) {
853
- (G = c.current) == null || G.setBackground(null);
881
+ }, [c, n, y]), j(() => {
882
+ var Z;
883
+ if (!d) {
884
+ (Z = A.current) == null || Z.setBackground(null);
854
885
  return;
855
886
  }
856
- let d = !1;
857
- return XI(A).then((L) => {
858
- var B;
859
- if (d) {
860
- L.close();
887
+ let r = !1;
888
+ return kI(d).then((V) => {
889
+ var S;
890
+ if (r) {
891
+ V.close();
861
892
  return;
862
893
  }
863
- (B = c.current) == null || B.setBackground(L, h);
864
- }).catch((L) => console.error("[fluidity-js] backgroundSrc load failed:", L)), () => {
865
- d = !0;
894
+ (S = A.current) == null || S.setBackground(V, h);
895
+ }).catch((V) => console.error("[fluidity-js] backgroundSrc load failed:", V)), () => {
896
+ r = !0;
866
897
  };
867
- }, [A, h]), Q(() => {
868
- if (!v)
898
+ }, [d, h]), j(() => {
899
+ if (!W)
869
900
  return;
870
- const d = D.current;
871
- if (!d)
901
+ const r = D.current;
902
+ if (!r)
872
903
  return;
873
- const G = (B) => {
874
- var O;
875
- const T = d.getBoundingClientRect();
876
- (O = c.current) == null || O.handleMove(B.clientX - T.left, B.clientY - T.top, 2);
877
- }, L = (B) => {
878
- var UI;
879
- B.preventDefault();
880
- const T = d.getBoundingClientRect(), O = B.touches[0];
881
- (UI = c.current) == null || UI.handleMove(O.clientX - T.left, O.clientY - T.top, 1);
904
+ const Z = (S) => {
905
+ var q;
906
+ const M = r.getBoundingClientRect();
907
+ (q = A.current) == null || q.handleMove(S.clientX - M.left, S.clientY - M.top, 2);
908
+ }, V = (S) => {
909
+ var fI;
910
+ S.preventDefault();
911
+ const M = r.getBoundingClientRect(), q = S.touches[0];
912
+ (fI = A.current) == null || fI.handleMove(q.clientX - M.left, q.clientY - M.top, 1);
882
913
  };
883
- return d.addEventListener("mousemove", G), d.addEventListener("touchmove", L, { passive: !1 }), () => {
884
- d.removeEventListener("mousemove", G), d.removeEventListener("touchmove", L);
914
+ return r.addEventListener("mousemove", Z), r.addEventListener("touchmove", V, { passive: !1 }), () => {
915
+ r.removeEventListener("mousemove", Z), r.removeEventListener("touchmove", V);
885
916
  };
886
- }, [v]), /* @__PURE__ */ TI(
917
+ }, [W]), /* @__PURE__ */ NI(
887
918
  "div",
888
919
  {
889
920
  ref: D,
@@ -893,125 +924,127 @@ const hg = xI(function({
893
924
  display: "block",
894
925
  width: "100%",
895
926
  height: "100%",
896
- background: S,
897
- ...m
927
+ background: G,
928
+ ...u
898
929
  }
899
930
  }
900
931
  );
901
- }), Sg = xI(function({
932
+ }), Vg = zI(function({
902
933
  src: I,
903
- effect: i = q.effect,
904
- imageSize: e = q.imageSize,
905
- className: l,
906
- style: s,
934
+ effect: i = iI.effect,
935
+ imageSize: e = iI.imageSize,
936
+ className: s,
937
+ style: o,
907
938
  config: t,
908
- preset: m,
909
- algorithm: r,
910
- backgroundColor: a = q.backgroundColor,
911
- backgroundSrc: u,
912
- backgroundSize: S = q.backgroundSize,
913
- isMouseEnabled: A = q.isMouseEnabled,
914
- isWorkerEnabled: h = q.isWorkerEnabled
915
- }, v) {
916
- const Y = bI(null), K = OI(Y, {
939
+ preset: u,
940
+ algorithm: a,
941
+ backgroundColor: c = iI.backgroundColor,
942
+ backgroundSrc: n,
943
+ backgroundSize: G = iI.backgroundSize,
944
+ isMouseEnabled: d = iI.isMouseEnabled,
945
+ isWorkerEnabled: h = iI.isWorkerEnabled,
946
+ quality: W = HI.quality
947
+ }, x) {
948
+ const F = mI(null), w = qI(F, {
917
949
  isWorkerEnabled: h,
918
- config: uI({ ...t, ...r ? { algorithm: r } : {} }, m)
950
+ quality: W,
951
+ config: pI({ ...t, ...a ? { algorithm: a } : {} }, u)
919
952
  });
920
- MI(
921
- v,
953
+ QI(
954
+ x,
922
955
  () => ({
923
956
  reset() {
924
- var c;
925
- I && ((c = K.current) == null || c.setImageSource(I, i, e));
957
+ var A;
958
+ I && ((A = w.current) == null || A.setImageSource(I, i, e));
926
959
  },
927
- move({ x: c, y: w, strength: d = 1 }) {
928
- var G;
929
- (G = K.current) == null || G.handleMove(c, w, d);
960
+ move(A, y, r = 1) {
961
+ var Z;
962
+ (Z = w.current) == null || Z.handleMove(A, y, r);
930
963
  },
931
- splat(c, w, d, G, L = 1) {
932
- var B;
933
- (B = K.current) == null || B.splat(c, w, d, G, L);
964
+ splat(A, y, r, Z, V = 1) {
965
+ var S;
966
+ (S = w.current) == null || S.splat(A, y, r, Z, V);
934
967
  },
935
- updateConfig(c) {
936
- var w;
937
- (w = K.current) == null || w.updateConfig(c);
968
+ updateConfig(A) {
969
+ var y;
970
+ (y = w.current) == null || y.updateConfig(A);
938
971
  }
939
972
  }),
940
973
  [I, i, e]
941
- ), Q(() => {
942
- var c;
943
- I && ((c = K.current) == null || c.setImageSource(I, i, e));
974
+ ), j(() => {
975
+ var A;
976
+ I && ((A = w.current) == null || A.setImageSource(I, i, e));
944
977
  }, [I, i, e]);
945
978
  const D = JSON.stringify(t);
946
- return Q(() => {
947
- var c;
948
- (c = K.current) == null || c.updateConfig(
949
- uI({ ...t, ...r !== void 0 ? { algorithm: r } : {} }, m)
979
+ return j(() => {
980
+ var A;
981
+ (A = w.current) == null || A.updateConfig(
982
+ pI({ ...t, ...a !== void 0 ? { algorithm: a } : {} }, u)
950
983
  );
951
- }, [m, r, D]), Q(() => {
952
- var w;
953
- if (!u) {
954
- (w = K.current) == null || w.setBackground(null);
984
+ }, [u, a, D]), j(() => {
985
+ var y;
986
+ if (!n) {
987
+ (y = w.current) == null || y.setBackground(null);
955
988
  return;
956
989
  }
957
- let c = !1;
958
- return XI(u).then((d) => {
959
- var G;
960
- if (c) {
961
- d.close();
990
+ let A = !1;
991
+ return kI(n).then((r) => {
992
+ var Z;
993
+ if (A) {
994
+ r.close();
962
995
  return;
963
996
  }
964
- (G = K.current) == null || G.setBackground(d, S);
965
- }).catch((d) => console.error("[fluidity-js] backgroundSrc load failed:", d)), () => {
966
- c = !0;
997
+ (Z = w.current) == null || Z.setBackground(r, G);
998
+ }).catch((r) => console.error("[fluidity-js] backgroundSrc load failed:", r)), () => {
999
+ A = !0;
967
1000
  };
968
- }, [u, S]), Q(() => {
969
- if (!A)
1001
+ }, [n, G]), j(() => {
1002
+ if (!d)
970
1003
  return;
971
- const c = Y.current;
972
- if (!c)
1004
+ const A = F.current;
1005
+ if (!A)
973
1006
  return;
974
- const w = (G) => {
975
- var B;
976
- const L = c.getBoundingClientRect();
977
- (B = K.current) == null || B.handleMove(G.clientX - L.left, G.clientY - L.top, 2);
978
- }, d = (G) => {
979
- var T;
980
- G.preventDefault();
981
- const L = c.getBoundingClientRect(), B = G.touches[0];
982
- (T = K.current) == null || T.handleMove(B.clientX - L.left, B.clientY - L.top, 1);
1007
+ const y = (Z) => {
1008
+ var S;
1009
+ const V = A.getBoundingClientRect();
1010
+ (S = w.current) == null || S.handleMove(Z.clientX - V.left, Z.clientY - V.top, 2);
1011
+ }, r = (Z) => {
1012
+ var M;
1013
+ Z.preventDefault();
1014
+ const V = A.getBoundingClientRect(), S = Z.touches[0];
1015
+ (M = w.current) == null || M.handleMove(S.clientX - V.left, S.clientY - V.top, 1);
983
1016
  };
984
- return c.addEventListener("mousemove", w), c.addEventListener("touchmove", d, { passive: !1 }), () => {
985
- c.removeEventListener("mousemove", w), c.removeEventListener("touchmove", d);
1017
+ return A.addEventListener("mousemove", y), A.addEventListener("touchmove", r, { passive: !1 }), () => {
1018
+ A.removeEventListener("mousemove", y), A.removeEventListener("touchmove", r);
986
1019
  };
987
- }, [A]), /* @__PURE__ */ TI(
1020
+ }, [d]), /* @__PURE__ */ NI(
988
1021
  "div",
989
1022
  {
990
- ref: Y,
991
- className: l,
1023
+ ref: F,
1024
+ className: s,
992
1025
  style: {
993
1026
  position: "relative",
994
1027
  display: "block",
995
1028
  width: "100%",
996
1029
  height: "100%",
997
- background: a,
998
- ...s
1030
+ background: c,
1031
+ ...o
999
1032
  }
1000
1033
  }
1001
1034
  );
1002
1035
  });
1003
1036
  export {
1004
- JI as DEFAULT_CONFIG,
1005
- kI as DEFAULT_CONFIG_TEXT,
1006
- q as DEFAULT_PROPS_IMAGE,
1007
- EI as DEFAULT_PROPS_SHARED,
1008
- j as DEFAULT_PROPS_TEXT,
1009
- ug as FluidController,
1010
- Sg as FluidImage,
1011
- fI as FluidSimulation,
1012
- hg as FluidText,
1013
- _I as PRESETS,
1014
- XI as loadImageBitmap,
1015
- uI as mergeConfig,
1016
- OI as useFluid
1037
+ OI as DEFAULT_CONFIG,
1038
+ TI as DEFAULT_CONFIG_TEXT,
1039
+ iI as DEFAULT_PROPS_IMAGE,
1040
+ HI as DEFAULT_PROPS_SHARED,
1041
+ $ as DEFAULT_PROPS_TEXT,
1042
+ Bg as FluidController,
1043
+ Vg as FluidImage,
1044
+ EI as FluidSimulation,
1045
+ Lg as FluidText,
1046
+ ig as PRESETS,
1047
+ kI as loadImageBitmap,
1048
+ pI as mergeConfig,
1049
+ qI as useFluid
1017
1050
  };