前端嘛 Logo
前端嘛
写点“悄悄话”

写点“悄悄话”

2025-10-31
<!DOCTYPE html>
<html lang="zh">
  <head>
    <meta charset="UTF-8" />
    <title>Fluid motion #webgpu version</title>
    <meta
      name="viewport"
      content="width=device-width,initial-scale=1,minimal-ui,shrink-to-fit=no,viewport-fit=cover"
    />
    <meta name="apple-mobile-web-app-capable" content="yes" />
    <meta name="mobile-web-app-capable" content="yes" />
    <meta
      name="description"
      content="Fluid motion with webGPU, which makes it more performant and future proof"
    />
    <link
      rel="stylesheet"
      href="https://public.codepenassets.com/css/normalize-5.0.0.min.css"
    />
    <style>
      html,
      body {
        overflow: clip;
      }

      :root {
        --c-glass: #bbbbbc;
        --c-light: #fff;
        --c-dark: #000;

        --c-content: #224;
        --c-action: #0052f5;

        --c-bg: #fff;

        --glass-reflex-dark: 1;
        --glass-reflex-light: 1;

        --saturation: 150%;

        font-size: 20px;
        font-family: sans-serif;
        font-optical-sizing: auto;

        transition: background 400ms cubic-bezier(1, 0, 0.4, 1),
          color 400ms cubic-bezier(1, 0, 0.4, 1);
      }

      body {
        margin: 0;
        position: absolute;
        width: 100%;
        height: 100%;
      }

      #container {
        margin: 0;
        padding: 0;
        display: flex;
        flex-direction: column;
        justify-content: center;
        align-items: center;
        width: 100%;
        height: 100%;
        background: var(--c-bg);
        color: var(--c-content);
      }

      .a-title {
        position: absolute;
        color: transparent;
        -webkit-background-clip: text;
        -webkit-text-fill-color: transparent;
        background-image: conic-gradient(#ed0101, blue);
        pointer-events: none;
        mix-blend-mode: difference;
        filter: drop-shadow(2px 4px 6px black);
      }
      .a-second-title {
        position: absolute;
        margin-top: 25vh;
        pointer-events: none;
        -webkit-text-stroke: 1.3px white;
        letter-spacing: 1.125px;
        font-size: -webkit-xxx-large;
        font-weight: 900;
        mix-blend-mode: color-burn;
        :has(input[value="dark"]:checked) & {
          mix-blend-mode: color-dodge;
        }
      }

      canvas {
        width: 100%;
        height: 100%;
        background: var(--c-bg);
      }

      /*glass switcher*/
      :has(input[value="dark"]:checked) {
        --c-glass: #bbbbbc;
        --c-light: #fff;
        --c-dark: #000;

        --c-content: #e1e1e1;
        --c-action: #ffdc03;

        --c-bg: #000;

        --glass-reflex-dark: 2;
        --glass-reflex-light: 0.3;

        --saturation: 150%;
      }

      .switcher {
        position: fixed;
        z-index: 2;
        top: 40px;
        left: 50%;
        translate: -50%;
        display: flex;
        align-items: center;
        gap: 8px;
        width: 168px; /* Adjusted for two options */
        max-width: 168px; /* Adjusted for two options */
        height: 70px;
        box-sizing: border-box;
        padding: 8px 12px 10px;
        margin: 0 auto;
        border: none;
        border-radius: 99em;
        font-size: var(--fz);
        background-color: color-mix(in srgb, var(--c-glass) 12%, transparent);
        backdrop-filter: blur(8px) url(#switcher) saturate(var(--saturation));
        -webkit-backdrop-filter: blur(8px) saturate(var(--saturation));
        box-shadow: inset 0 0 0 1px
            color-mix(
              in srgb,
              var(--c-light) calc(var(--glass-reflex-light) * 10%),
              transparent
            ),
          inset 1.8px 3px 0px -2px color-mix(in srgb, var(--c-light)
                calc(var(--glass-reflex-light) * 90%), transparent),
          inset -2px -2px 0px -2px color-mix(in srgb, var(--c-light)
                calc(var(--glass-reflex-light) * 80%), transparent),
          inset -3px -8px 1px -6px color-mix(in srgb, var(--c-light)
                calc(var(--glass-reflex-light) * 60%), transparent),
          inset -0.3px -1px 4px 0px
            color-mix(
              in srgb,
              var(--c-dark) calc(var(--glass-reflex-dark) * 12%),
              transparent
            ),
          inset -1.5px 2.5px 0px -2px
            color-mix(
              in srgb,
              var(--c-dark) calc(var(--glass-reflex-dark) * 20%),
              transparent
            ),
          inset 0px 3px 4px -2px color-mix(in srgb, var(--c-dark)
                calc(var(--glass-reflex-dark) * 20%), transparent),
          inset 2px -6.5px 1px -4px
            color-mix(
              in srgb,
              var(--c-dark) calc(var(--glass-reflex-dark) * 10%),
              transparent
            ),
          0px 1px 5px 0px
            color-mix(
              in srgb,
              var(--c-dark) calc(var(--glass-reflex-dark) * 10%),
              transparent
            ),
          0px 6px 16px 0px
            color-mix(
              in srgb,
              var(--c-dark) calc(var(--glass-reflex-dark) * 8%),
              transparent
            );
        transition: background-color 400ms cubic-bezier(1, 0, 0.4, 1),
          box-shadow 400ms cubic-bezier(1, 0, 0.4, 1);
        -webkit-backdrop-filter: blur(5px);
        backdrop-filter: blur(5px);
      }

      .switcher__legend {
        position: absolute;
        width: 1px;
        height: 1px;
        margin: -1px;
        border: 0;
        padding: 0;
        white-space: nowrap;
        -webkit-clip-path: inset(100%);
        clip-path: inset(100%);
        clip: rect(0 0 0 0);
        overflow: hidden;
      }

      .switcher__input {
        clip: rect(0 0 0 0);
        -webkit-clip-path: inset(100%);
        clip-path: inset(100%);
        height: 1px;
        width: 1px;
        overflow: hidden;
        position: absolute;
        white-space: nowrap;
      }

      .switcher__icon {
        display: block;
        width: 100%;
        transition: scale 200ms cubic-bezier(0.5, 0, 0, 1);
      }

      .switcher__filter {
        position: absolute;
        width: 0;
        height: 0;
        z-index: -1;
      }

      .switcher__option {
        --c: var(--c-content);
        display: flex;
        justify-content: center;
        align-items: center;
        padding: 0 16px;
        width: 68px;
        height: 100%;
        box-sizing: border-box;
        border-radius: 99em;
        opacity: 1;
        transition: all 160ms;
      }

      .switcher__option:hover {
        --c: var(--c-action);
        cursor: pointer;
      }

      .switcher__option:hover .switcher__icon {
        scale: 1.2;
      }

      .switcher__option:has(input:checked) {
        --c: var(--c-content);
        cursor: auto;
      }

      .switcher__option:has(input:checked) .switcher__icon {
        scale: 1;
      }

      .switcher::after {
        content: "";
        position: absolute;
        left: 4px;
        top: 4px;
        display: block;
        width: 84px;
        height: calc(100% - 10px);
        border-radius: 99em;
        background-color: color-mix(in srgb, var(--c-glass) 36%, transparent);
        z-index: -1;
        box-shadow: inset 0 0 0 1px
            color-mix(
              in srgb,
              var(--c-light) calc(var(--glass-reflex-light) * 10%),
              transparent
            ),
          inset 2px 1px 0px -1px color-mix(in srgb, var(--c-light)
                calc(var(--glass-reflex-light) * 90%), transparent),
          inset -1.5px -1px 0px -1px color-mix(in srgb, var(--c-light)
                calc(var(--glass-reflex-light) * 80%), transparent),
          inset -2px -6px 1px -5px color-mix(in srgb, var(--c-light)
                calc(var(--glass-reflex-light) * 60%), transparent),
          inset -1px 2px 3px -1px
            color-mix(
              in srgb,
              var(--c-dark) calc(var(--glass-reflex-dark) * 20%),
              transparent
            ),
          inset 0px -4px 1px -2px
            color-mix(
              in srgb,
              var(--c-dark) calc(var(--glass-reflex-dark) * 10%),
              transparent
            ),
          0px 3px 6px 0px
            color-mix(
              in srgb,
              var(--c-dark) calc(var(--glass-reflex-dark) * 8%),
              transparent
            );
      }

      .switcher:has(input[c-option="1"]:checked)::after {
        translate: 0 0;
        transform-origin: right;
        transition: background-color 400ms cubic-bezier(1, 0, 0.4, 1),
          box-shadow 400ms cubic-bezier(1, 0, 0.4, 1),
          translate 400ms cubic-bezier(1, 0, 0.4, 1);
        -webkit-animation: scaleToggle 440ms ease;
        animation: scaleToggle 440ms ease;
      }

      .switcher:has(input[c-option="2"]:checked)::after {
        translate: 76px 0;
        transition: background-color 400ms cubic-bezier(1, 0, 0.4, 1),
          box-shadow 400ms cubic-bezier(1, 0, 0.4, 1),
          translate 400ms cubic-bezier(1, 0, 0.4, 1);
        -webkit-animation: scaleToggle2 440ms ease;
        animation: scaleToggle2 440ms ease;
        transform-origin: left; /* Set transform-origin for the second option */
      }

      @-webkit-keyframes scaleToggle {
        0% {
          scale: 1 1;
        }
        50% {
          scale: 1.1 1;
        }
        100% {
          scale: 1 1;
        }
      }

      @keyframes scaleToggle {
        0% {
          scale: 1 1;
        }
        50% {
          scale: 1.1 1;
        }
        100% {
          scale: 1 1;
        }
      }

      @-webkit-keyframes scaleToggle2 {
        0% {
          scale: 1 1;
        }
        50% {
          scale: 1.1 1; /* Adjusted scale for consistency */
        }
        100% {
          scale: 1 1;
        }
      }

      @keyframes scaleToggle2 {
        0% {
          scale: 1 1;
        }
        50% {
          scale: 1.1 1; /* Adjusted scale for consistency */
        }
        100% {
          scale: 1 1;
        }
      }
    </style>
  </head>

  <body>
    <section id="container">
      <h1 class="a-title">在这里涂鸦</h1>
      <h2 class="a-second-title">偷偷写点悄悄话在这里</h2>
      <canvas id="fluid-webgpu"></canvas>
    </section>

    <!-- switcher light-dark-->
    <fieldset class="switcher">
      <legend class="switcher__legend">选择主题</legend>
      <label class="switcher__option">
        <input
          class="switcher__input"
          type="radio"
          name="theme"
          value="light"
          c-option="1"
          checked
        />
        <svg
          class="switcher__icon"
          xmlns="http://www.w3.org/2000/svg"
          fill="none"
          viewBox="0 0 36 36"
        >
          <path
            fill="var(--c)"
            fill-rule="evenodd"
            d="M18 12a6 6 0 1 1 0 12 6 6 0 0 1 0-12Zm0 2a4 4 0 1 0 0 8 4 4 0 0 0 0-8Z"
            clip-rule="evenodd"
          />
          <path
            fill="var(--c)"
            d="M17 6.038a1 1 0 1 1 2 0v3a1 1 0 0 1-2 0v-3ZM24.244 7.742a1 1 0 1 1 1.618 1.176L24.1 11.345a1 1 0 1 1-1.618-1.176l1.763-2.427ZM29.104 13.379a1 1 0 0 1 .618 1.902l-2.854.927a1 1 0 1 1-.618-1.902l2.854-.927ZM29.722 20.795a1 1 0 0 1-.619 1.902l-2.853-.927a1 1 0 1 1 .618-1.902l2.854.927ZM25.862 27.159a1 1 0 0 1-1.618 1.175l-1.763-2.427a1 1 0 1 1 1.618-1.175l1.763 2.427ZM19 30.038a1 1 0 0 1-2 0v-3a1 1 0 1 1 2 0v3ZM11.755 28.334a1 1 0 0 1-1.618-1.175l1.764-2.427a1 1 0 1 1 1.618 1.175l-1.764 2.427ZM6.896 22.697a1 1 0 1 1-.618-1.902l2.853-.927a1 1 0 1 1 .618 1.902l-2.853.927ZM6.278 15.28a1 1 0 1 1 .618-1.901l2.853.927a1 1 0 1 1-.618 1.902l-2.853-.927ZM10.137 8.918a1 1 0 0 1 1.618-1.176l1.764 2.427a1 1 0 0 1-1.618 1.176l-1.764-2.427Z"
          />
        </svg>
      </label>
      <label class="switcher__option">
        <input
          class="switcher__input"
          type="radio"
          name="theme"
          value="dark"
          c-option="2"
        />
        <svg
          class="switcher__icon"
          xmlns="http://www.w3.org/2000/svg"
          fill="none"
          viewBox="0 0 36 36"
        >
          <path
            fill="var(--c)"
            d="M12.5 8.473a10.968 10.968 0 0 1 8.785-.97 7.435 7.435 0 0 0-3.737 4.672l-.09.373A7.454 7.454 0 0 0 28.732 20.4a10.97 10.97 0 0 1-5.232 7.125l-.497.27c-5.014 2.566-11.175.916-14.234-3.813l-.295-.483C5.53 18.403 7.13 11.93 12.017 8.77l.483-.297Zm4.234.616a8.946 8.946 0 0 0-2.805.883l-.429.234A9 9 0 0 0 10.206 22.5l.241.395A9 9 0 0 0 22.5 25.794l.416-.255a8.94 8.94 0 0 0 2.167-1.99 9.433 9.433 0 0 1-2.782-.313c-5.043-1.352-8.036-6.535-6.686-11.578l.147-.491c.242-.745.573-1.44.972-2.078Z"
          />
        </svg>
      </label>

      <!--   <div class="switcher__toggle"></div> -->
      <div class="switcher__filter">
        <svg>
          <filter id="switcher" primitiveUnits="objectBoundingBox">
            <feImage
              result="map"
              width="100%"
              height="100%"
              x="0"
              y="0"
              href=""
            />

            <feGaussianBlur
              in="SourceGraphic"
              stdDeviation="0.04"
              result="blur"
            />
            <feDisplacementMap
              id="disp"
              in="blur"
              in2="map"
              scale="0.5"
              xChannelSelector="R"
              yChannelSelector="G"
            ></feDisplacementMap>
          </filter>

          <filter id="toggler" primitiveUnits="objectBoundingBox">
            <feImage
              result="map"
              width="100%"
              height="100%"
              x="0"
              y="0"
              href=""
            />

            <feGaussianBlur
              in="SourceGraphic"
              stdDeviation="0.01"
              result="blur"
            />
            <feDisplacementMap
              id="disp"
              in="blur"
              in2="map"
              scale="0.5"
              xChannelSelector="R"
              yChannelSelector="G"
            ></feDisplacementMap>
          </filter>
        </svg>
      </div>
    </fieldset>
    <script>
      /*main*/
      let settings = {
        grid_size: 64,
        dye_size: 256,
        sim_speed: 5,
        contain_fluid: true,
        velocity_add_intensity: 0.28,
        velocity_add_radius: 0.001,
        velocity_diffusion: 1,
        dye_add_intensity: 0.8,
        dye_add_radius: 0.0035,
        dye_diffusion: 0.96204,
        viscosity: 0,
        vorticity: 0,
        pressure_iterations: 8,
        buffer_view: "dye",
        input_symmetry: "none",
      };

      let device, presentationFormat, canvas, context;

      const mouseInfos = {
        current: null,
        last: null,
        velocity: null,
      };

      // Buffers
      let velocity,
        velocity0,
        dye,
        dye0,
        divergence,
        divergence0,
        pressure,
        pressure0,
        vorticity;

      // Uniforms
      const globalUniforms = {};
      let time,
        dt,
        mouse,
        grid,
        uSimSpeed,
        vel_force,
        vel_radius,
        vel_diff,
        dye_force,
        dye_radius,
        dye_diff;
      let viscosity, uVorticity, containFluid, uSymmetry, uRenderIntensity;

      // Programs
      let checkerProgram,
        updateDyeProgram,
        updateProgram,
        advectProgram,
        boundaryProgram,
        divergenceProgram;
      let boundaryDivProgram,
        pressureProgram,
        boundaryPressureProgram,
        gradientSubtractProgram,
        advectDyeProgram;
      let clearPressureProgram,
        vorticityProgram,
        vorticityConfinmentProgram,
        renderProgram;

      function handlePointerMove(e) {
        const pointer = e.touches ? e.touches[0] : e;
        const rect = canvas.getBoundingClientRect();

        if (!mouseInfos.current) mouseInfos.current = [];
        mouseInfos.current[0] = (pointer.clientX - rect.left) / rect.width;
        mouseInfos.current[1] = 1 - (pointer.clientY - rect.top) / rect.height; // Invert Y
      }

      function onWebGPUDetectionError(error) {
        console.log("Could not initialize WebGPU: " + error);
        document.querySelector(".webgpu-not-supported").style.visibility =
          "visible";
        return false;
      }

      // Init the WebGPU context by checking first if everything is supported
      // Returns true on init success, false otherwise
      async function initContext() {
        if (navigator.gpu == null)
          return onWebGPUDetectionError("WebGPU NOT Supported");

        const adapter = await navigator.gpu.requestAdapter();
        if (!adapter) return onWebGPUDetectionError("No adapter found");

        device = await adapter.requestDevice();

        canvas = document.getElementById("fluid-webgpu");
        context = canvas.getContext("webgpu");
        if (!context)
          return onWebGPUDetectionError("Canvas does not support WebGPU");

        // If we got here, WebGPU seems to be supported

        // Init canvas
        canvas.style.width = "100%";
        canvas.style.height = "100%";
        canvas.addEventListener("mousemove", handlePointerMove);
        canvas.addEventListener("touchmove", (e) => {
          e.preventDefault();
          handlePointerMove(e);
        });
        canvas.addEventListener("touchstart", (e) => {
          e.preventDefault();
          handlePointerMove(e);
          mouseInfos.last = [...mouseInfos.current];
        });

        // Init  context
        presentationFormat = navigator.gpu.getPreferredCanvasFormat(adapter);

        context.configure({
          device,
          format: presentationFormat,
          usage: GPUTextureUsage.RENDER_ATTACHMENT,
          alphaMode: "premultiplied",
        });

        // Init buffer sizes
        initSizes();

        // Resize event
        let resizeTimeout;
        window.addEventListener("resize", () => {
          clearTimeout(resizeTimeout);
          resizeTimeout = setTimeout(refreshSizes, 150);
        });

        return true;
      }

      function refreshSizes() {
        initSizes();
        initBuffers();
        initPrograms();
        globalUniforms.gridSize.needsUpdate = [
          settings.grid_w,
          settings.grid_h,
          settings.dye_w,
          settings.dye_h,
          settings.dx,
          settings.rdx,
          settings.dyeRdx,
        ];
      }

      // Init buffer & canvas dimensions to fit the screen while keeping the aspect ratio
      // and downscaling the dimensions if they exceed the browsers capabilities
      function initSizes() {
        const dpr = window.devicePixelRatio || 1;
        const aspectRatio = window.innerWidth / window.innerHeight;
        const maxBufferSize = device.limits.maxStorageBufferBindingSize;
        const maxCanvasSize = device.limits.maxTextureDimension2D;

        // Fit to screen while keeping the aspect ratio
        const getPreferredDimensions = (baseSize) => {
          let w, h;
          const scaledBaseSize = baseSize * dpr;

          if (aspectRatio > 1) {
            h = scaledBaseSize;
            w = Math.floor(h * aspectRatio);
          } else {
            w = scaledBaseSize;
            h = Math.floor(w / aspectRatio);
          }

          return getValidDimensions(w, h);
        };

        // Downscale if necessary to prevent crashes
        const getValidDimensions = (w, h) => {
          let downRatio = 1;

          // Prevent buffer size overflow
          if (w * h * 4 >= maxBufferSize)
            downRatio = Math.sqrt(maxBufferSize / (w * h * 4));

          // Prevent canvas size overflow
          if (w > maxCanvasSize) downRatio = maxCanvasSize / w;
          else if (h > maxCanvasSize) downRatio = maxCanvasSize / h;

          return {
            w: Math.floor(w * downRatio),
            h: Math.floor(h * downRatio),
          };
        };

        // Calculate simulation buffer dimensions
        let gridSize = getPreferredDimensions(settings.grid_size);
        settings.grid_w = gridSize.w;
        settings.grid_h = gridSize.h;

        // Calculate dye & canvas buffer dimensions
        let dyeSize = getPreferredDimensions(settings.dye_size);
        settings.dye_w = dyeSize.w;
        settings.dye_h = dyeSize.h;

        // Useful values for the simulation
        settings.rdx = settings.grid_size * 4;
        settings.dyeRdx = settings.dye_size * 4;
        settings.dx = 1 / settings.rdx;

        // Resize the canvas
        canvas.width = settings.dye_w;
        canvas.height = settings.dye_h;
      }

      /*shaders*/
      const STRUCT_GRID_SIZE = `
struct GridSize {
  w : f32,
  h : f32,
  dyeW: f32,
  dyeH: f32,
  dx : f32,
  rdx : f32,
  dyeRdx : f32
}`;

      const STRUCT_MOUSE = `
struct Mouse {
  pos: vec2<f32>,
  vel: vec2<f32>,
}`;

      // This code initialize the pos and index variables and target only interior cells
      const COMPUTE_START = `
var pos = vec2<f32>(global_id.xy);

if (pos.x == 0 || pos.y == 0 || pos.x >= uGrid.w - 1 || pos.y >= uGrid.h - 1) {
    return;
}      

let index = ID(pos.x, pos.y);`;

      const COMPUTE_START_DYE = `
var pos = vec2<f32>(global_id.xy);

if (pos.x == 0 || pos.y == 0 || pos.x >= uGrid.dyeW - 1 || pos.y >= uGrid.dyeH - 1) {
    return;
}      

let index = ID(pos.x, pos.y);`;

      // This code initialize the pos and index variables and target all cells
      const COMPUTE_START_ALL = `    
var pos = vec2<f32>(global_id.xy);

if (pos.x >= uGrid.w || pos.y >= uGrid.h) {
    return;
}      

let index = ID(pos.x, pos.y);`;

      const SPLAT_CODE = `
var m = uMouse.pos;
var v = uMouse.vel*2.;

var splat = createSplat(p, m, v, uRadius);
if (uSymmetry == 1. || uSymmetry == 3.) {splat += createSplat(p, vec2(1. - m.x, m.y), v * vec2(-1., 1.), uRadius);}
if (uSymmetry == 2. || uSymmetry == 3.) {splat += createSplat(p, vec2(m.x, 1. - m.y), v * vec2(1., -1.), uRadius);}
if (uSymmetry == 3. || uSymmetry == 4.) {splat += createSplat(p, vec2(1. - m.x, 1. - m.y), v * vec2(-1., -1.), uRadius);}
`;

      /// APPLY FORCE SHADER ///

      const updateVelocityShader = /* wgsl */ `

${STRUCT_GRID_SIZE}

struct Mouse {
  pos: vec2<f32>,
  vel: vec2<f32>,
}
@group(0) @binding(0) var<storage, read> x_in : array<f32>;
@group(0) @binding(1) var<storage, read> y_in : array<f32>;
@group(0) @binding(2) var<storage, read_write> x_out : array<f32>;
@group(0) @binding(3) var<storage, read_write> y_out : array<f32>;
@group(0) @binding(4) var<uniform> uGrid: GridSize;
@group(0) @binding(5) var<uniform> uMouse: Mouse;
@group(0) @binding(6) var<uniform> uForce : f32;
@group(0) @binding(7) var<uniform> uRadius : f32;
@group(0) @binding(8) var<uniform> uDiffusion : f32;
@group(0) @binding(9) var<uniform> uDt : f32;
@group(0) @binding(10) var<uniform> uTime : f32;
@group(0) @binding(11) var<uniform> uSymmetry : f32;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.w); }
fn inBetween(x : f32, lower : f32, upper : f32) -> bool {
  return x > lower && x < upper;
}
fn inBounds(pos : vec2<f32>, xMin : f32, xMax : f32, yMin: f32, yMax : f32) -> bool {
  return inBetween(pos.x, xMin * uGrid.w, xMax * uGrid.w) && inBetween(pos.y, yMin * uGrid.h, yMax * uGrid.h);
}

fn createSplat(pos : vec2<f32>, splatPos : vec2<f32>, vel : vec2<f32>, radius : f32) -> vec2<f32> {
  var p = pos - splatPos;
  p.x *= uGrid.w / uGrid.h;
  var v = vel;
  v.x *= uGrid.w / uGrid.h;
  var splat = exp(-dot(p, p) / radius) * v;
  return splat;
}

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {
    
    ${COMPUTE_START}

    let tmpT = uTime;
    var p = pos/vec2(uGrid.w, uGrid.h);

    ${SPLAT_CODE}
    
    splat *= uForce * uDt * 200.;

    x_out[index] = x_in[index]*uDiffusion + splat.x;
    y_out[index] = y_in[index]*uDiffusion + splat.y;
}`;

      const updateDyeShader = /* wgsl */ `

${STRUCT_GRID_SIZE}

struct Mouse {
  pos: vec2<f32>,
  vel: vec2<f32>,
}
@group(0) @binding(0) var<storage, read> x_in : array<f32>;
@group(0) @binding(1) var<storage, read> y_in : array<f32>;
@group(0) @binding(2) var<storage, read> z_in : array<f32>;
@group(0) @binding(3) var<storage, read_write> x_out : array<f32>;
@group(0) @binding(4) var<storage, read_write> y_out : array<f32>;
@group(0) @binding(5) var<storage, read_write> z_out : array<f32>;
@group(0) @binding(6) var<uniform> uGrid: GridSize;
@group(0) @binding(7) var<uniform> uMouse: Mouse;
@group(0) @binding(8) var<uniform> uForce : f32;
@group(0) @binding(9) var<uniform> uRadius : f32;
@group(0) @binding(10) var<uniform> uDiffusion : f32;
@group(0) @binding(11) var<uniform> uTime : f32;
@group(0) @binding(12) var<uniform> uDt : f32;
@group(0) @binding(13) var<uniform> uSymmetry : f32;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.dyeW); }
fn inBetween(x : f32, lower : f32, upper : f32) -> bool {
  return x > lower && x < upper;
}
fn inBounds(pos : vec2<f32>, xMin : f32, xMax : f32, yMin: f32, yMax : f32) -> bool {
  return inBetween(pos.x, xMin * uGrid.dyeW, xMax * uGrid.dyeW) && inBetween(pos.y, yMin * uGrid.dyeH, yMax * uGrid.dyeH);
}
// cosine based palette, 4 vec3 params
fn palette(t : f32, a : vec3<f32>, b : vec3<f32>, c : vec3<f32>, d : vec3<f32> ) -> vec3<f32> {
    return a + b*cos( 6.28318*(c*t+d) );
}

fn createSplat(pos : vec2<f32>, splatPos : vec2<f32>, vel : vec2<f32>, radius : f32) -> vec3<f32> {
  var p = pos - splatPos;
  p.x *= uGrid.w / uGrid.h;
  var v = vel;
  v.x *= uGrid.w / uGrid.h;
  var splat = exp(-dot(p, p) / radius) * length(v);
  return vec3(splat);
}

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {

    ${COMPUTE_START_DYE}

    let col_incr = 0.15;
    let col_start = palette(uTime/8., vec3(1), vec3(0.5), vec3(1), vec3(0, col_incr, col_incr*2.));

    var p = pos/vec2(uGrid.dyeW, uGrid.dyeH);

    ${SPLAT_CODE}

    splat *= col_start * uForce * uDt * 100.;

    x_out[index] = max(0., x_in[index]*uDiffusion + splat.x);
    y_out[index] = max(0., y_in[index]*uDiffusion + splat.y);
    z_out[index] = max(0., z_in[index]*uDiffusion + splat.z);
}`;

      /// ADVECT SHADER ///

      const advectShader = /* wgsl */ `

${STRUCT_GRID_SIZE}

@group(0) @binding(0) var<storage, read> x_in : array<f32>;
@group(0) @binding(1) var<storage, read> y_in : array<f32>;
@group(0) @binding(2) var<storage, read> x_vel : array<f32>;
@group(0) @binding(3) var<storage, read> y_vel : array<f32>;
@group(0) @binding(4) var<storage, read_write> x_out : array<f32>;
@group(0) @binding(5) var<storage, read_write> y_out : array<f32>;
@group(0) @binding(6) var<uniform> uGrid : GridSize;
@group(0) @binding(7) var<uniform> uDt : f32;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.w); }
fn in(x : f32, y : f32) -> vec2<f32> { let id = ID(x, y); return vec2(x_in[id], y_in[id]); }

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {
  
    ${COMPUTE_START}
    
    var x = pos.x - uDt * uGrid.rdx * x_vel[index];
    var y = pos.y - uDt * uGrid.rdx * y_vel[index];

    if (x < 0) { x = 0; }
    else if (x >= uGrid.w - 1) { x = uGrid.w - 1; }
    if (y < 0) { y = 0; }
    else if (y >= uGrid.h - 1) { y = uGrid.h - 1; }

    let x1 = floor(x);
    let y1 = floor(y);
    let x2 = x1 + 1;
    let y2 = y1 + 1;

    let TL = in(x1, y2);
    let TR = in(x2, y2);
    let BL = in(x1, y1);
    let BR = in(x2, y1);

    let xMod = fract(x);
    let yMod = fract(y);
    
    let bilerp = mix( mix(BL, BR, xMod), mix(TL, TR, xMod), yMod );

    x_out[index] = bilerp.x;
    y_out[index] = bilerp.y;
}`;

      const advectDyeShader = /* wgsl */ `

${STRUCT_GRID_SIZE}

@group(0) @binding(0) var<storage, read> x_in : array<f32>;
@group(0) @binding(1) var<storage, read> y_in : array<f32>;
@group(0) @binding(2) var<storage, read> z_in : array<f32>;
@group(0) @binding(3) var<storage, read> x_vel : array<f32>;
@group(0) @binding(4) var<storage, read> y_vel : array<f32>;
@group(0) @binding(5) var<storage, read_write> x_out : array<f32>;
@group(0) @binding(6) var<storage, read_write> y_out : array<f32>;
@group(0) @binding(7) var<storage, read_write> z_out : array<f32>;
@group(0) @binding(8) var<uniform> uGrid : GridSize;
@group(0) @binding(9) var<uniform> uDt : f32;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.dyeW); }
fn in(x : f32, y : f32) -> vec3<f32> { let id = ID(x, y); return vec3(x_in[id], y_in[id], z_in[id]); }
fn vel(x : f32, y : f32) -> vec2<f32> { 
  let id = u32(i32(x) + i32(y) * i32(uGrid.w));
  return vec2(x_vel[id], y_vel[id]);
}

fn vel_bilerp(x0 : f32, y0 : f32) -> vec2<f32> {
    var x = x0 * uGrid.w / uGrid.dyeW;
    var y = y0 * uGrid.h / uGrid.dyeH;

    if (x < 0) { x = 0; }
    else if (x >= uGrid.w - 1) { x = uGrid.w - 1; }
    if (y < 0) { y = 0; }
    else if (y >= uGrid.h - 1) { y = uGrid.h - 1; }

    let x1 = floor(x);
    let y1 = floor(y);
    let x2 = x1 + 1;
    let y2 = y1 + 1;

    let TL = vel(x1, y2);
    let TR = vel(x2, y2);
    let BL = vel(x1, y1);
    let BR = vel(x2, y1);

    let xMod = fract(x);
    let yMod = fract(y);

    return mix( mix(BL, BR, xMod), mix(TL, TR, xMod), yMod );
}

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {

    ${COMPUTE_START_DYE}

    let V = vel_bilerp(pos.x, pos.y);

    var x = pos.x - uDt * uGrid.dyeRdx * V.x;
    var y = pos.y - uDt * uGrid.dyeRdx * V.y;

    if (x < 0) { x = 0; }
    else if (x >= uGrid.dyeW - 1) { x = uGrid.dyeW - 1; }
    if (y < 0) { y = 0; }
    else if (y >= uGrid.dyeH - 1) { y = uGrid.dyeH - 1; }

    let x1 = floor(x);
    let y1 = floor(y);
    let x2 = x1 + 1;
    let y2 = y1 + 1;

    let TL = in(x1, y2);
    let TR = in(x2, y2);
    let BL = in(x1, y1);
    let BR = in(x2, y1);

    let xMod = fract(x);
    let yMod = fract(y);

    let bilerp = mix( mix(BL, BR, xMod), mix(TL, TR, xMod), yMod );

    x_out[index] = bilerp.x;
    y_out[index] = bilerp.y;
    z_out[index] = bilerp.z;
}`;

      /// DIVERGENCE SHADER ///

      const divergenceShader = /* wgsl */ `   

${STRUCT_GRID_SIZE}

@group(0) @binding(0) var<storage, read> x_vel : array<f32>;
@group(0) @binding(1) var<storage, read> y_vel : array<f32>;
@group(0) @binding(2) var<storage, read_write> div : array<f32>;
@group(0) @binding(3) var<uniform> uGrid : GridSize;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.w); }
fn vel(x : f32, y : f32) -> vec2<f32> { let id = ID(x, y); return vec2(x_vel[id], y_vel[id]); }

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {

  ${COMPUTE_START}

  let L = vel(pos.x - 1, pos.y).x;
  let R = vel(pos.x + 1, pos.y).x;
  let B = vel(pos.x, pos.y - 1).y;
  let T = vel(pos.x, pos.y + 1).y;

  div[index] = 0.5 * uGrid.rdx * ((R - L) + (T - B));
}`;

      /// PRESSURE SHADER ///

      const pressureShader = /* wgsl */ `      

${STRUCT_GRID_SIZE}

@group(0) @binding(0) var<storage, read> pres_in : array<f32>;
@group(0) @binding(1) var<storage, read> div : array<f32>;
@group(0) @binding(2) var<storage, read_write> pres_out : array<f32>;
@group(0) @binding(3) var<uniform> uGrid : GridSize;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.w); }
fn in(x : f32, y : f32) -> f32 { let id = ID(x, y); return pres_in[id]; }

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {

  ${COMPUTE_START}
        
  let L = pos - vec2(1, 0);
  let R = pos + vec2(1, 0);
  let B = pos - vec2(0, 1);
  let T = pos + vec2(0, 1);

  let Lx = in(L.x, L.y);
  let Rx = in(R.x, R.y);
  let Bx = in(B.x, B.y);
  let Tx = in(T.x, T.y);

  let bC = div[index];

  let alpha = -(uGrid.dx * uGrid.dx);
  let rBeta = .25;

  pres_out[index] = (Lx + Rx + Bx + Tx + alpha * bC) * rBeta;
}`;

      /// GRADIENT SUBTRACT SHADER ///

      const gradientSubtractShader = /* wgsl */ `      

${STRUCT_GRID_SIZE}

@group(0) @binding(0) var<storage, read> pressure : array<f32>;
@group(0) @binding(1) var<storage, read> x_vel : array<f32>;
@group(0) @binding(2) var<storage, read> y_vel : array<f32>;
@group(0) @binding(3) var<storage, read_write> x_out : array<f32>;
@group(0) @binding(4) var<storage, read_write> y_out : array<f32>;
@group(0) @binding(5) var<uniform> uGrid : GridSize;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.w); }
fn pres(x : f32, y : f32) -> f32 { let id = ID(x, y); return pressure[id]; }

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {

  ${COMPUTE_START}

  let L = pos - vec2(1, 0);
  let R = pos + vec2(1, 0);
  let B = pos - vec2(0, 1);
  let T = pos + vec2(0, 1);

  let xL = pres(L.x, L.y);
  let xR = pres(R.x, R.y);
  let yB = pres(B.x, B.y);
  let yT = pres(T.x, T.y);
  
  let finalX = x_vel[index] - .5 * uGrid.rdx * (xR - xL);
  let finalY = y_vel[index] - .5 * uGrid.rdx * (yT - yB);

  x_out[index] = finalX;
  y_out[index] = finalY;
}`;

      /// VORTICITY SHADER ///

      const vorticityShader = /* wgsl */ `      

${STRUCT_GRID_SIZE}

@group(0) @binding(0) var<storage, read> x_vel : array<f32>;
@group(0) @binding(1) var<storage, read> y_vel : array<f32>;
@group(0) @binding(2) var<storage, read_write> vorticity : array<f32>;
@group(0) @binding(3) var<uniform> uGrid : GridSize;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.w); }
fn vel(x : f32, y : f32) -> vec2<f32> { let id = ID(x, y); return vec2(x_vel[id], y_vel[id]); }

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {

  ${COMPUTE_START}

  let Ly = vel(pos.x - 1, pos.y).y;
  let Ry = vel(pos.x + 1, pos.y).y;
  let Bx = vel(pos.x, pos.y - 1).x;
  let Tx = vel(pos.x, pos.y + 1).x;

  vorticity[index] = 0.5 * uGrid.rdx * ((Ry - Ly) - (Tx - Bx));
}`;

      /// VORTICITY CONFINMENT SHADER ///

      const vorticityConfinmentShader = /* wgsl */ `      

${STRUCT_GRID_SIZE}

@group(0) @binding(0) var<storage, read> x_vel_in : array<f32>;
@group(0) @binding(1) var<storage, read> y_vel_in : array<f32>;
@group(0) @binding(2) var<storage, read> vorticity : array<f32>;
@group(0) @binding(3) var<storage, read_write> x_vel_out : array<f32>;
@group(0) @binding(4) var<storage, read_write> y_vel_out : array<f32>;
@group(0) @binding(5) var<uniform> uGrid : GridSize;
@group(0) @binding(6) var<uniform> uDt : f32;
@group(0) @binding(7) var<uniform> uVorticity : f32;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.w); }
fn vort(x : f32, y : f32) -> f32 { let id = ID(x, y); return vorticity[id]; }

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {

  ${COMPUTE_START}

  let L = vort(pos.x - 1, pos.y);
  let R = vort(pos.x + 1, pos.y);
  let B = vort(pos.x, pos.y - 1);
  let T = vort(pos.x, pos.y + 1);
  let C = vorticity[index];

  var force = 0.5 * uGrid.rdx * vec2(abs(T) - abs(B), abs(R) - abs(L));

  let epsilon = 2.4414e-4;
  let magSqr = max(epsilon, dot(force, force));

  force = force / sqrt(magSqr);
  force *= uGrid.dx * uVorticity * uDt * C * vec2(1, -1);

  x_vel_out[index] = x_vel_in[index] + force.x;
  y_vel_out[index] = y_vel_in[index] + force.y;
}`;

      /// CLEAR PRESSURE SHADER ///

      const clearPressureShader = /* wgsl */ `  

${STRUCT_GRID_SIZE}

@group(0) @binding(0) var<storage, read> x_in : array<f32>;
@group(0) @binding(1) var<storage, read_write> x_out : array<f32>;
@group(0) @binding(2) var<uniform> uGrid : GridSize;
@group(0) @binding(3) var<uniform> uVisc : f32;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.w); }

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {

  ${COMPUTE_START_ALL}

  x_out[index] = x_in[index]*uVisc;
}`;

      /// BOUNDARY SHADER ///

      const boundaryShader = /* wgsl */ `

${STRUCT_GRID_SIZE}

@group(0) @binding(0) var<storage, read> x_in : array<f32>;
@group(0) @binding(1) var<storage, read> y_in : array<f32>;
@group(0) @binding(2) var<storage, read_write> x_out : array<f32>;
@group(0) @binding(3) var<storage, read_write> y_out : array<f32>;
@group(0) @binding(4) var<uniform> uGrid : GridSize;
@group(0) @binding(5) var<uniform> containFluid : f32;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.w); }

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {

  ${COMPUTE_START_ALL}

  // disable scale to disable contained bounds
  var scaleX = 1.;
  var scaleY = 1.;

  if (pos.x == 0) { pos.x += 1; scaleX = -1.; }
  else if (pos.x == uGrid.w - 1) { pos.x -= 1; scaleX = -1.; }
  if (pos.y == 0) { pos.y += 1; scaleY = -1.; }
  else if (pos.y == uGrid.h - 1) { pos.y -= 1; scaleY = -1.; }

  if (containFluid == 0.) {
    scaleX = 1.;
    scaleY = 1.;
  }

  x_out[index] = x_in[ID(pos.x, pos.y)] * scaleX;
  y_out[index] = y_in[ID(pos.x, pos.y)] * scaleY;
}`;

      /// BOUNDARY PRESSURE SHADER ///

      const boundaryPressureShader = /* wgsl */ `    

${STRUCT_GRID_SIZE}

@group(0) @binding(0) var<storage, read> x_in : array<f32>;
@group(0) @binding(1) var<storage, read_write> x_out : array<f32>;
@group(0) @binding(2) var<uniform> uGrid : GridSize;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.w); }

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {

  ${COMPUTE_START_ALL}

  if (pos.x == 0) { pos.x += 1; }
  else if (pos.x == uGrid.w - 1) { pos.x -= 1; }
  if (pos.y == 0) { pos.y += 1; }
  else if (pos.y == uGrid.h - 1) { pos.y -= 1; }

  x_out[index] = x_in[ID(pos.x, pos.y)];
}`;

      const checkerboardShader = /* wgsl */ `    

${STRUCT_GRID_SIZE}

@group(0) @binding(0) var<storage, read_write> x_out : array<f32>;
@group(0) @binding(1) var<storage, read_write> y_out : array<f32>;
@group(0) @binding(2) var<storage, read_write> z_out : array<f32>;
@group(0) @binding(3) var<uniform> uGrid : GridSize;
@group(0) @binding(4) var<uniform> uTime : f32;

fn ID(x : f32, y : f32) -> u32 { return u32(x + y * uGrid.dyeW); }

fn noise(p_ : vec3<f32>) -> f32 {
  var p = p_;
	var ip=floor(p);
  p-=ip; 
  var s=vec3(7.,157.,113.);
  var h=vec4(0.,s.y, s.z,s.y+s.z)+dot(ip,s);
  p=p*p*(3. - 2.*p); 
  h=mix(fract(sin(h)*43758.5),fract(sin(h+s.x)*43758.5),p.x);
  var r=mix(h.xz,h.yw,p.y);
  h.x = r.x;
  h.y = r.y;
  return mix(h.x,h.y,p.z); 
}

fn fbm(p_ : vec3<f32>, octaveNum : i32) -> vec2<f32> {
  var p=p_;
	var acc = vec2(0.);	
	var freq = 1.0;
	var amp = 0.5;
  var shift = vec3(100.);
	for (var i = 0; i < octaveNum; i++) {
		acc += vec2(noise(p), noise(p + vec3(0.,0.,10.))) * amp;
    p = p * 2.0 + shift;
    amp *= 0.5;
	}
	return acc;
}

@compute @workgroup_size(8, 8)
fn main(@builtin(global_invocation_id) global_id : vec3<u32>) {

  ${COMPUTE_START_DYE}

  var uv = pos/vec2(uGrid.dyeW, uGrid.dyeH);
  var zoom = 4.;

  var smallNoise = fbm(vec3(uv.x*zoom*2., uv.y*zoom*2., uTime+2.145), 7) - .5;
  var bigNoise = fbm(vec3(uv.x*zoom, uv.y*zoom, uTime*.1+30.), 7) - .5;

  var noise = max(length(bigNoise) * 0.035, 0.);
  var noise2 = max(length(smallNoise) * 0.035, 0.);

  noise = noise + noise2 * .05;

  var czoom = 4.;
  var n = fbm(vec3(uv.x*czoom, uv.y*czoom, uTime*.1+63.1), 7)*.75+.25;
  var n2 = fbm(vec3(uv.x*czoom, uv.y*czoom, uTime*.1+23.4), 7)*.75+.25;
  
  var col = vec3(1.);

  x_out[index] += noise * col.x;
  y_out[index] += noise * col.y;
  z_out[index] += noise * col.z;
}`;

      /*render*/
      const renderShader = /* wgsl */ `
${STRUCT_GRID_SIZE}

struct VertexOut {
  @builtin(position) position : vec4<f32>,
  @location(1) uv : vec2<f32>,
};

@group(0) @binding(0) var<storage, read> fieldX : array<f32>;
@group(0) @binding(1) var<storage, read> fieldY : array<f32>;
@group(0) @binding(2) var<storage, read> fieldZ : array<f32>;
@group(0) @binding(3) var<uniform> uGrid : GridSize;
@group(0) @binding(4) var<uniform> multiplier : f32;


@vertex
fn vertex_main(@location(0) position: vec4<f32>) -> VertexOut
{
    var output : VertexOut;
    output.position = position;
    output.uv = position.xy*.5+.5;
    return output;
}

@fragment
fn fragment_main(fragData : VertexOut) -> @location(0) vec4<f32>
{
    var w = uGrid.dyeW;
    var h = uGrid.dyeH;

    let fuv = vec2<f32>((floor(fragData.uv*vec2(w, h))));
    let id = u32(fuv.x + fuv.y * w);

    let r = fieldX[id];
    let g = fieldY[id];
    let b = fieldZ[id];
    let col = vec3(r, g, b);

    let alpha = clamp(length(col), 0.0, 1.0);
    return vec4(col * multiplier, alpha);
}
`;

      // Renders 3 (r, g, b) storage buffers to the canvas
      class RenderProgram {
        constructor() {
          const vertices = new Float32Array([
            -1, -1, 0, 1, -1, 1, 0, 1, 1, -1, 0, 1, 1, -1, 0, 1, -1, 1, 0, 1, 1,
            1, 0, 1,
          ]);

          this.vertexBuffer = device.createBuffer({
            size: vertices.byteLength,
            usage: GPUBufferUsage.VERTEX | GPUBufferUsage.COPY_DST,
            mappedAtCreation: true,
          });
          new Float32Array(this.vertexBuffer.getMappedRange()).set(vertices);
          this.vertexBuffer.unmap();

          const vertexBuffersDescriptors = [
            {
              attributes: [
                {
                  shaderLocation: 0,
                  offset: 0,
                  format: "float32x4",
                },
              ],
              arrayStride: 16,
              stepMode: "vertex",
            },
          ];

          const shaderModule = device.createShaderModule({
            code: renderShader,
          });

          this.renderPipeline = device.createRenderPipeline({
            layout: "auto",
            vertex: {
              module: shaderModule,
              entryPoint: "vertex_main",
              buffers: vertexBuffersDescriptors,
            },
            fragment: {
              module: shaderModule,
              entryPoint: "fragment_main",
              targets: [
                {
                  format: presentationFormat,
                },
              ],
            },
            primitive: {
              topology: "triangle-list",
            },
          });

          // The r,g,b buffer containing the data to render
          this.buffer = new DynamicBuffer({
            dims: 3,
            w: settings.dye_w,
            h: settings.dye_h,
          });

          // Uniforms
          const entries = [
            ...this.buffer.buffers,
            globalUniforms.gridSize.buffer,
            globalUniforms.render_intensity_multiplier.buffer,
          ].map((b, i) => ({
            binding: i,
            resource: { buffer: b },
          }));

          this.renderBindGroup = device.createBindGroup({
            layout: this.renderPipeline.getBindGroupLayout(0),
            entries,
          });

          this.renderPassDescriptor = {
            colorAttachments: [
              {
                clearValue: { r: 0.0, g: 0.0, b: 0.0, a: 0.0 },
                loadOp: "clear",
                storeOp: "store",
              },
            ],
          };
        }

        // Dispatch a draw command to render on the canvas
        dispatch(commandEncoder) {
          this.renderPassDescriptor.colorAttachments[0].view = context
            .getCurrentTexture()
            .createView();

          const renderPassEncoder = commandEncoder.beginRenderPass(
            this.renderPassDescriptor
          );

          renderPassEncoder.setPipeline(this.renderPipeline);
          renderPassEncoder.setBindGroup(0, this.renderBindGroup);
          renderPassEncoder.setVertexBuffer(0, this.vertexBuffer);
          renderPassEncoder.draw(6);
          renderPassEncoder.end();
        }
      }

      /*utils*/
      // Creates and manage multi-dimensional buffers by creating a buffer for each dimension
      class DynamicBuffer {
        constructor({
          dims = 1, // Number of dimensions
          w = settings.grid_w, // Buffer width
          h = settings.grid_h, // Buffer height
        } = {}) {
          this.dims = dims;
          this.bufferSize = w * h * 4;
          this.w = w;
          this.h = h;
          this.buffers = new Array(dims).fill().map((_) =>
            device.createBuffer({
              size: this.bufferSize,
              usage:
                GPUBufferUsage.STORAGE |
                GPUBufferUsage.COPY_SRC |
                GPUBufferUsage.COPY_DST,
            })
          );
        }

        // Copy each buffer to another DynamicBuffer's buffers.
        // If the dimensions don't match, the last non-empty dimension will be copied instead
        copyTo(buffer, commandEncoder) {
          for (let i = 0; i < Math.max(this.dims, buffer.dims); i++) {
            commandEncoder.copyBufferToBuffer(
              this.buffers[Math.min(i, this.buffers.length - 1)],
              0,
              buffer.buffers[Math.min(i, buffer.buffers.length - 1)],
              0,
              this.bufferSize
            );
          }
        }

        // Reset all the buffers
        clear(queue) {
          for (let i = 0; i < this.dims; i++) {
            queue.writeBuffer(
              this.buffers[i],
              0,
              new Float32Array(this.w * this.h)
            );
          }
        }
      }

      // Manage uniform buffers relative to the compute shaders
      class Uniform {
        constructor(name, { size, value } = {}) {
          this.name = name;
          this.size =
            size ?? (value && typeof value === "object" ? value.length : 1);
          this.needsUpdate = false;

          if (this.size === 1) {
            if (settings[name] == null) {
              settings[name] = value ?? 0;
              this.alwaysUpdate = true;
            }
          }

          if (this.size === 1 || value != null) {
            this.buffer = device.createBuffer({
              mappedAtCreation: true,
              size: this.size * 4,
              usage: GPUBufferUsage.UNIFORM | GPUBufferUsage.COPY_DST,
            });

            const arrayBuffer = this.buffer.getMappedRange();
            const sourceValue = value ?? [settings[this.name]];
            const sourceArray =
              typeof sourceValue === "number"
                ? [sourceValue]
                : Array.isArray(sourceValue)
                ? sourceValue
                : [0]; // Default to [0] if value is invalid
            new Float32Array(arrayBuffer).set(new Float32Array(sourceArray));
            this.buffer.unmap();
          } else {
            this.buffer = device.createBuffer({
              size: this.size * 4,
              usage: GPUBufferUsage.UNIFORM | GPUBufferUsage.COPY_DST,
            });
          }

          globalUniforms[name] = this;
        }

        setValue(value) {
          settings[this.name] = value;
          this.needsUpdate = true;
        }

        update(queue, value) {
          if (this.needsUpdate || this.alwaysUpdate || value != null) {
            if (typeof this.needsUpdate !== "boolean") value = this.needsUpdate;
            queue.writeBuffer(
              this.buffer,
              0,
              new Float32Array(value ?? [parseFloat(settings[this.name])]),
              0,
              this.size
            );
            this.needsUpdate = false;
          }
        }
      }

      // Creates a shader module, compute pipeline & bind group to use with the GPU
      class Program {
        constructor({
          buffers = [], // Storage buffers
          uniforms = [], // Uniform buffers
          shader, // WGSL Compute Shader as a string
          dispatchX = settings.grid_w, // Dispatch workers width
          dispatchY = settings.grid_h, // Dispatch workers height
        }) {
          this.computePipeline = device.createComputePipeline({
            layout: "auto",
            compute: {
              module: device.createShaderModule({ code: shader }),
              entryPoint: "main",
            },
          });

          const storageEntries = buffers.map((b) => b.buffers).flat();
          const uniformEntries = uniforms
            .filter((u) => u && u.buffer)
            .map((u) => u.buffer);

          const allEntries = [...storageEntries, ...uniformEntries].map(
            (buffer, i) => ({
              binding: i,
              resource: { buffer },
            })
          );

          this.bindGroup = device.createBindGroup({
            layout: this.computePipeline.getBindGroupLayout(0),
            entries: allEntries,
          });

          this.dispatchX = dispatchX;
          this.dispatchY = dispatchY;
        }

        dispatch(passEncoder) {
          passEncoder.setPipeline(this.computePipeline);
          passEncoder.setBindGroup(0, this.bindGroup);
          passEncoder.dispatchWorkgroups(
            Math.ceil(this.dispatchX / 8),
            Math.ceil(this.dispatchY / 8)
          );
        }
      }

      /// Useful classes for cleaner understanding of the input and output buffers
      /// used in the declarations of programs & fluid simulation steps

      class AdvectProgram extends Program {
        constructor({
          in_quantity,
          in_velocity,
          out_quantity,
          uniforms,
          shader = advectShader,
          ...props
        }) {
          uniforms ??= [globalUniforms.gridSize];
          super({
            buffers: [in_quantity, in_velocity, out_quantity],
            uniforms,
            shader,
            ...props,
          });
        }
      }

      class DivergenceProgram extends Program {
        constructor({
          in_velocity,
          out_divergence,
          uniforms,
          shader = divergenceShader,
        }) {
          uniforms ??= [globalUniforms.gridSize];
          super({ buffers: [in_velocity, out_divergence], uniforms, shader });
        }
      }

      class PressureProgram extends Program {
        constructor({
          in_pressure,
          in_divergence,
          out_pressure,
          uniforms,
          shader = pressureShader,
        }) {
          uniforms ??= [globalUniforms.gridSize];
          super({
            buffers: [in_pressure, in_divergence, out_pressure],
            uniforms,
            shader,
          });
        }
      }

      class GradientSubtractProgram extends Program {
        constructor({
          in_pressure,
          in_velocity,
          out_velocity,
          uniforms,
          shader = gradientSubtractShader,
        }) {
          uniforms ??= [globalUniforms.gridSize];
          super({
            buffers: [in_pressure, in_velocity, out_velocity],
            uniforms,
            shader,
          });
        }
      }

      class BoundaryProgram extends Program {
        constructor({
          in_quantity,
          out_quantity,
          uniforms,
          shader = boundaryShader,
        }) {
          uniforms ??= [globalUniforms.gridSize];
          super({ buffers: [in_quantity, out_quantity], uniforms, shader });
        }
      }

      class UpdateProgram extends Program {
        constructor({
          in_quantity,
          out_quantity,
          uniforms,
          shader = updateVelocityShader,
          ...props
        }) {
          uniforms ??= [globalUniforms.gridSize];
          super({
            buffers: [in_quantity, out_quantity],
            uniforms,
            shader,
            ...props,
          });
        }
      }

      class VorticityProgram extends Program {
        constructor({
          in_velocity,
          out_vorticity,
          uniforms,
          shader = vorticityShader,
          ...props
        }) {
          uniforms ??= [globalUniforms.gridSize];
          super({
            buffers: [in_velocity, out_vorticity],
            uniforms,
            shader,
            ...props,
          });
        }
      }

      class VorticityConfinmentProgram extends Program {
        constructor({
          in_velocity,
          in_vorticity,
          out_velocity,
          uniforms,
          shader = vorticityConfinmentShader,
          ...props
        }) {
          uniforms ??= [globalUniforms.gridSize];
          super({
            buffers: [in_velocity, in_vorticity, out_velocity],
            uniforms,
            shader,
            ...props,
          });
        }
      }

      function initBuffers() {
        velocity = new DynamicBuffer({ dims: 2 });
        velocity0 = new DynamicBuffer({ dims: 2 });

        dye = new DynamicBuffer({
          dims: 3,
          w: settings.dye_w,
          h: settings.dye_h,
        });
        dye0 = new DynamicBuffer({
          dims: 3,
          w: settings.dye_w,
          h: settings.dye_h,
        });

        divergence = new DynamicBuffer();
        divergence0 = new DynamicBuffer();

        pressure = new DynamicBuffer();
        pressure0 = new DynamicBuffer();

        vorticity = new DynamicBuffer();
      }

      function initUniforms() {
        time = new Uniform("time");
        dt = new Uniform("dt");
        mouse = new Uniform("mouseInfos", { size: 4 });
        grid = new Uniform("gridSize", {
          size: 7,
          value: [
            settings.grid_w,
            settings.grid_h,
            settings.dye_w,
            settings.dye_h,
            settings.dx,
            settings.rdx,
            settings.dyeRdx,
          ],
        });
        uSimSpeed = new Uniform("sim_speed", { value: settings.sim_speed });
        vel_force = new Uniform("velocity_add_intensity", {
          value: settings.velocity_add_intensity,
        });
        vel_radius = new Uniform("velocity_add_radius", {
          value: settings.velocity_add_radius,
        });
        vel_diff = new Uniform("velocity_diffusion", {
          value: settings.velocity_diffusion,
        });
        dye_force = new Uniform("dye_add_intensity", {
          value: settings.dye_add_intensity,
        });
        dye_radius = new Uniform("dye_add_radius", {
          value: settings.dye_add_radius,
        });
        dye_diff = new Uniform("dye_diffusion", {
          value: settings.dye_diffusion,
        });
        viscosity = new Uniform("viscosity", {
          value: settings.viscosity,
        });
        uVorticity = new Uniform("vorticity", {
          value: settings.vorticity,
        });
        containFluid = new Uniform("contain_fluid", {
          value: settings.contain_fluid,
        });
        uSymmetry = new Uniform("mouse_type", { value: 0 });
        uRenderIntensity = new Uniform("render_intensity_multiplier", {
          value: 1,
        });
      }

      function initPrograms() {
        checkerProgram = new Program({
          buffers: [dye],
          shader: checkerboardShader,
          dispatchX: settings.dye_w,
          dispatchY: settings.dye_h,
          uniforms: [grid, time],
        });

        updateDyeProgram = new UpdateProgram({
          in_quantity: dye,
          out_quantity: dye0,
          uniforms: [
            grid,
            mouse,
            dye_force,
            dye_radius,
            dye_diff,
            time,
            dt,
            uSymmetry,
          ],
          dispatchX: settings.dye_w,
          dispatchY: settings.dye_h,
          shader: updateDyeShader,
        });

        updateProgram = new UpdateProgram({
          in_quantity: velocity,
          out_quantity: velocity0,
          uniforms: [
            grid,
            mouse,
            vel_force,
            vel_radius,
            vel_diff,
            dt,
            time,
            uSymmetry,
          ],
        });

        advectProgram = new AdvectProgram({
          in_quantity: velocity0,
          in_velocity: velocity0,
          out_quantity: velocity,
          uniforms: [grid, dt],
        });

        boundaryProgram = new BoundaryProgram({
          in_quantity: velocity,
          out_quantity: velocity0,
          uniforms: [grid, containFluid],
        });

        divergenceProgram = new DivergenceProgram({
          in_velocity: velocity0,
          out_divergence: divergence0,
        });

        boundaryDivProgram = new BoundaryProgram({
          in_quantity: divergence0,
          out_quantity: divergence,
          shader: boundaryPressureShader,
        });

        pressureProgram = new PressureProgram({
          in_pressure: pressure,
          in_divergence: divergence,
          out_pressure: pressure0,
        });

        boundaryPressureProgram = new BoundaryProgram({
          in_quantity: pressure0,
          out_quantity: pressure,
          shader: boundaryPressureShader,
        });

        gradientSubtractProgram = new GradientSubtractProgram({
          in_pressure: pressure,
          in_velocity: velocity0,
          out_velocity: velocity,
        });

        advectDyeProgram = new AdvectProgram({
          in_quantity: dye0,
          in_velocity: velocity,
          out_quantity: dye,
          uniforms: [grid, dt],
          dispatchX: settings.dye_w,
          dispatchY: settings.dye_h,
          shader: advectDyeShader,
        });

        clearPressureProgram = new UpdateProgram({
          in_quantity: pressure,
          out_quantity: pressure0,
          uniforms: [grid, viscosity],
          shader: clearPressureShader,
        });

        vorticityProgram = new VorticityProgram({
          in_velocity: velocity,
          out_vorticity: vorticity,
        });

        vorticityConfinmentProgram = new VorticityConfinmentProgram({
          in_velocity: velocity,
          in_vorticity: vorticity,
          out_velocity: velocity0,
          uniforms: [grid, dt, uVorticity],
        });

        renderProgram = new RenderProgram();
      }

      async function main() {
        // Init WebGPU Context
        const initializationSuccess = await initContext();
        if (!initializationSuccess) return;

        // Init buffers, uniforms and programs
        initBuffers();
        initUniforms();
        initPrograms();

        // Simulation reset
        function reset() {
          velocity.clear(device.queue);
          dye.clear(device.queue);
          pressure.clear(device.queue);

          settings.time = 0;
        }
        settings.reset = reset;

        // Fluid simulation step
        function dispatchComputePipeline(passEncoder) {
          // Add velocity and dye at the mouse position
          updateDyeProgram.dispatch(passEncoder);
          updateProgram.dispatch(passEncoder);

          // Advect the velocity field through itself
          advectProgram.dispatch(passEncoder);
          boundaryProgram.dispatch(passEncoder); // boundary conditions

          // Compute the divergence
          divergenceProgram.dispatch(passEncoder);
          boundaryDivProgram.dispatch(passEncoder); // boundary conditions

          // Solve the jacobi-pressure equation
          for (let i = 0; i < settings.pressure_iterations; i++) {
            pressureProgram.dispatch(passEncoder);
            boundaryPressureProgram.dispatch(passEncoder); // boundary conditions
          }

          // Subtract the pressure from the velocity field
          gradientSubtractProgram.dispatch(passEncoder);
          clearPressureProgram.dispatch(passEncoder);

          // Compute & apply vorticity confinment
          vorticityProgram.dispatch(passEncoder);
          vorticityConfinmentProgram.dispatch(passEncoder);

          // Advect the dye through the velocity field
          advectDyeProgram.dispatch(passEncoder);
        }

        let lastFrame = performance.now();

        // Render loop
        async function step() {
          requestAnimationFrame(step);

          // Update time
          const now = performance.now();
          settings.dt =
            Math.min(1 / 60, (now - lastFrame) / 1000) * settings.sim_speed;
          settings.time += settings.dt;
          lastFrame = now;

          // Update uniforms
          Object.values(globalUniforms).forEach((u) => u.update(device.queue));

          // Update custom uniform
          if (mouseInfos.current) {
            let dx = mouseInfos.last
              ? mouseInfos.current[0] - mouseInfos.last[0]
              : 0;
            let dy = mouseInfos.last
              ? mouseInfos.current[1] - mouseInfos.last[1]
              : 0;

            const isMobile =
              "ontouchstart" in window || navigator.maxTouchPoints > 0;
            if (isMobile) {
              const touchStrengthMultiplier = 0.2;
              dx *= touchStrengthMultiplier;
              dy *= touchStrengthMultiplier;
            }

            mouseInfos.velocity = [dx, dy];

            mouse.update(device.queue, [
              ...mouseInfos.current,
              ...mouseInfos.velocity,
            ]);
            mouseInfos.last = [...mouseInfos.current];
          }

          // Compute fluid
          const commandEncoder = device.createCommandEncoder();
          const passEncoder = commandEncoder.beginComputePass();
          dispatchComputePipeline(passEncoder);
          passEncoder.end();

          velocity0.copyTo(velocity, commandEncoder);
          pressure0.copyTo(pressure, commandEncoder);
          dye.copyTo(renderProgram.buffer, commandEncoder);

          // Draw fluid
          renderProgram.dispatch(commandEncoder);

          // Send commands to the GPU
          const gpuCommands = commandEncoder.finish();
          device.queue.submit([gpuCommands]);
        }

        step();
      }

      main();
    </script>
  </body>
</html>