惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

aimingoo的专栏
aimingoo的专栏
量子位
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
S
Schneier on Security
Cisco Talos Blog
Cisco Talos Blog
T
ThreatConnect
J
Java Code Geeks
博客园 - 司徒正美
A
Arctic Wolf
T
True Tiger Recordings
C
Cybersecurity and Infrastructure Security Agency CISA
Cyberwarzone
Cyberwarzone
Know Your Adversary
Know Your Adversary
T
Threat Research - Cisco Blogs
V
Vulnerabilities – Threatpost
Recorded Future
Recorded Future
P
Palo Alto Networks Blog
The Hacker News
The Hacker News
The Register - Security
The Register - Security
S
Securelist
www.infosecurity-magazine.com
www.infosecurity-magazine.com
C
CXSECURITY Database RSS Feed - CXSecurity.com
Application and Cybersecurity Blog
Application and Cybersecurity Blog
I
Intezer
P
Privacy & Cybersecurity Law Blog
Scott Helme
Scott Helme
K
Kaspersky official blog
博客园 - 聂微东
Last Week in AI
Last Week in AI
V
V2EX
小众软件
小众软件
F
Fox-IT International blog
Martin Fowler
Martin Fowler
Apple Machine Learning Research
Apple Machine Learning Research
T
Tenable Blog
F
Future of Privacy Forum
Microsoft Security Blog
Microsoft Security Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
腾讯CDC
Stack Overflow Blog
Stack Overflow Blog
C
Check Point Blog
阮一峰的网络日志
阮一峰的网络日志
GbyAI
GbyAI
T
Threatpost
I
InfoQ
P
Proofpoint News Feed
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
T
Tor Project blog
G
GRAHAM CLULEY
D
DataBreaches.Net

DEV Community

HTB — MonitorsFour | Writeup Fr 97. Embeddings and Vector Search: Semantic Search That Works ABAP Unit Testing with Test Doubles and Mocking Frameworks: A Senior Architects Guide to Isolating Dependencies in SAP S/4HANA LeetCode Solution: 5. Longest Palindromic Substring kovax-react 0.8: Tailwind v4 preset, FormField adapters, ColorModeScript, and Storybook I built an AI résumé tool that refuses to lie about your experience The hat Azure Entra ID User & Role Management — Step-by-Step Practical Guide With A Simple Excercise The AI-Native Company: How a Single Founder Can Build Global Organizations Powered by AWS and an Ecosystem of Artificial Intelligences Building a Lightweight Remote MCP Knowledge Base on Cloudflare Workers Why I built Trinavo for the MENA merchants Western platforms ignore The N+1 Query That Killed Our Database, And How I Fixed It Docstrings vs Markdown Docs: What Should Developers Actually Write? Training Data Provenance: The Manifest Diff That Explains the Hash Add SVGIcons MCP to Claude Code and Find SVG Icons from Your Terminal 3 CLI Tools You Can Buy with Crypto — No KYC, No Subscriptions COSS Weekly: OpenClaw competitor NanoClaw Raises $12M, Dust Raises $40M, Sonar Acquires Gitar, and more How to know if you actually need mobile proxies (without buying any) Building Cursor for Community: A Buildathon Built on Time Pressure How we built a PII masking layer for LLM APIs — local detection, reversible tokens, one line to integrate Why MLFQ Was Way Ahead of Its Time Add Runtime Limits to Claude Agent Workflows I Built a Prompt Injection Detector with 98% Recall on Unseen Attacks. Here's Why Data Beat Architecture. 8 Vite Config Options Every Developer Should Know (Vite 8) Feature Flags That Forgot to Leave Why Trust Infrastructure Is Becoming the Hidden Layer of Donation Platforms XyPriss: Rethinking Core Performance and Zero-Trust Architecture in Modern Backends Designing Configuration for Scalable Treasure Hunts SSH Login Delays: The 10-Second Wait That Drives Us Crazy Building Production Multi-Agent Workflows in n8n: What 50 Deployments Taught Us A 3-layer memory system that gives Claude Code persistent context across sessions. Trishul SNMP Suite 2.0.1: Better MIBs, Traps, and SNMP Labs How I built a production AI SaaS as a solo developer Auto-labelling 1.2M robotics frames with VLMs: a failover story India’s Laws Were Not Built for AI — And Courts Are Filling the Gap skill-insp: A Skill That Scores Other Skills Clprolf Minimalist Messaging in the Age of AI What's actually in a good .cursorrules file? I built 10 of them — here's what I learned Building Strong Python Basics – Loops, Functions and Logic How to Choose the Right Tech Stack for Your Project I built a free multi-tab JSON editor — here's what I learned HTTP Headers Every Developer Should Know (2026) Building Cross-Platform Digital Products: Challenges and Best Practices Data Privacy in the Age of AI: How Product Teams Can Build Trust with Users What Would WordPress Look Like If It Were Designed Today? Why Backup Success Does Not Mean Database Recoverability Local AI Office Assistant That Never Sends Your Documents to the Cloud Building TaskForge: Translating Enterprise Chaos into an Open-Source Scheduler Tesla P40 in a Homelab: 24GB of Inference on a Budget Llama 4: Meta's Latest — Scout, Maverick, and the MoE Revolution George Hotz called AI code 'slop.' He's half right. Como Construir um Fluxo de Trabalho Baseado em Engenharia de Prompt e Automação We Audited Our Agent Tool-Call Traces. Half Our Eval Data Was Garbage. The Hidden Cost of Downtime: How SRE Error Budgets Protect National Economic Infrastructure Getting started with openHUMANS can be an exciting venture for developers looking to create innovative applications in the realm of human-ce Stack Overflow: A Powerful Community for Developers and Learners From Language Models to Humanoid Minds ✨ Road to Senior #2: How Computers Think in Numbers Why LLM debugging fails on fragmented repository context How to Deploy a LangGraph Agent on AWS Bedrock AgentCore An outreach kit for solo founders whose drafts can't hallucinate Open Satchel is live Amy Kwalwasser and the Growing Importance of Quantum Risk Modeling I Built ShellReq - A Native API Client for VS Code & Terminal If Microsoft and Uber can't afford AI coding, what chance do the rest of us have? MADCAP: Building a Multi-Agent Debate CLI That Argues With Itself So You Don't Have To Why most AI fails at IDOR (and how AMAS fixes it with causal reasoning) How to Audit a Laravel Codebase You've Inherited LangGraph 워크플로우 템플릿 (v34) BugBench: a developer origin story and practical guide for VS Code / Kiro users A solution to messy token systems for Next.js A NestJS reference app that proves the nest-native stack under realistic backend pressure Observability for AI Systems: Monitoring Drift, Hallucinations, and Reliability in Production I Thought “Data Analyst” Was the Whole Game… Then I Entered the Data Avengers Office 👀 Create and configure network security groups How to analyze the cost of Kafka? How I Shipped 2,500+ Commits With AI Agents Using a 12-Phase Workflow [Boost] We built MDCMS, a Markdown-first CMS for teams using AI agents Zero Heap Allocations at 1.18 GB/s: Deep Dive into ForgeZero 4.0.x The Minimum Viable Test Suite for Working with Agents Why Perplexity Started Citing My Blog: 5 Changes That Actually Worked Sync Supabase via OAuth: No Connection String Needed I asked three AI models the same API question. Only one had it right. Implementing Saga Pattern With Lambda Durable Function Why does AI forget what you said (and how to fix it) I built a daily Wordle-style game for AI tools - Here's how Mapping Polish company structures: querying KRS direct via API Built tmpdrop — a tiny self-hosted ephemeral file drop Running Local LLM - 0$ Personal Agentic AI Assistant - Part 3 LLD Object-Oriented Design: Interfaces & Abstract Classes (Designing Contracts) The Smaller Ship: Vitalik, the Ethereum Foundation's Restructuring, and What It Leaves for Investors Looking for 4 people to build something weird with me Building a Local-Only RAG System with Ollama and TypeScript The False Positive Tax: a 1:1 TP:FP analysis of eslint-plugin-security What's new in Data Preprocessor 1.5.x — R codegen, Robust Scaler, and a deadlock post-mortem How I self-hosted my Flask app on an old laptop for almost free I built a free DSA interview prep site because I was tired of the existing options I built an AI agent that migrates Next.js Pages Router to App Router
Deep Dive: Building "Gravity Paint" - A Tactile Physics Instrument with React, Matter.js, and p5.js
Harish Kotra · 2026-05-26 · via DEV Community

Physics-based browser games are nothing new, but creating something that feels truly organic, interactive, and sensorily satisfying remains a rewarding engineering challenge. In this technical blog post, we look under the hood of Gravity Paint – a gravity-aware sandbox puzzle where players paint rigid rope cables to funnel a stream of colorful glass, steel, and bubble marbles into an target collection bucket.

We’ll dissect how to seamlessly combine the state-management of React 19, the performant rigid-body simulations of Matter.js, the smooth visual canvas loops of p5.js, and an in-house Web Audio API synthesizer.

The Three-Headed Architecture: React, P5, & Matter.js

When creating complex interactive canvases, the biggest design issue developers face is Architectural Coordination.

  • React is superb at managing structured data, layout trees, level progression, and sidebars. However, React's virtual-DOM rendering is completely unsuited for high-frequency (60fps) physics steps.
  • Matter.js runs an analytical engine representing worlds, vectors, collisions, shapes, forces, and constraint joints.
  • p5.js (instance-mode) handles graphic pipelines, drawing pixels, mouse inputs, clear frames, and UI animations.

Using these together requires a meticulous boundary of concerns:

+------------------------------------------+
|                 REACT UI                 |
| (Campaign Progress, Sound Settings, HUD) |
+--------------------+---------------------+
                     |
         Dispatches Window Event
                     |
                     v
+--------------------+---------------------+
|              P5.js INSTANCE              |
|  (Renders pixel world, manages inputs)   |
+--------------------+---------------------+
                     |
             Updates Coordinates
                     |
                     v
+--------------------+---------------------+
|             MATTER.JS WORLD              |
|  (Evaluates collisions, forces, joints)  |
+------------------------------------------+

Enter fullscreen mode Exit fullscreen mode

To prevent memory leaks and coordinate clean slate shifts (like when a level resets or chains are cleared), we wrap the entire canvas setup inside a single React useEffect() ref hook container. Communication across the react wrapper and inner loop coordinates cleanly through custom Window events (CustomEvent), which avoids messy React re-renders interfering with the physics cycle.

The Physics of Paintable Ropes (Chains)

How do we build paintable rope constraints that drape, sag, and sag under weight?

In Matter.js, there is no built-in "rope" body. Instead, a rope is represented as a composite sequence of circular bodies linked sequentially via distance constraints.

1. Tracking Drag Inputs

When a player clicks and drags on the canvas, we record coordinate arrays inside p5:

p.mouseDragged = () => {
  const lastPt = tempPoints[tempPoints.length - 1];
  const distToPrev = p.dist(p.mouseX, p.mouseY, lastPt.x, lastPt.y);

  // Add a new link point only if the mouse has moved far enough (dense brush optimization)
  if (distToPrev > 16) {
    tempPoints.push({ x: p.mouseX, y: p.mouseY });
  }
};

Enter fullscreen mode Exit fullscreen mode

2. Spawning Composite Structures

On mouse release, we convert these floating coordinates into interconnected Matter.js bodies:

const segmentCount = tempPoints.length;
const links: Matter.Body[] = [];
const constraints: Matter.Constraint[] = [];

// Create the link circles
for (let i = 0; i < segmentCount; i++) {
  const pt = tempPoints[i];
  const link = Matter.Bodies.circle(pt.x, pt.y, 4, {
    density: 0.005, // Dictates sag behavior
    friction: 0.08,
    restitution: 0.4,
    collisionFilter: { group: Matter.Body.nextGroup(true) } // Disable self-collisions within the rope
  });
  links.push(link);
  Matter.World.add(world, link);
}

// Interconnect the circles via spring constraints
for (let i = 0; i < segmentCount - 1; i++) {
  const con = Matter.Constraint.create({
    bodyA: links[i],
    bodyB: links[i + 1],
    stiffness: 0.85, 
    length: p.dist(tempPoints[i].x, tempPoints[i].y, tempPoints[i+1].x, tempPoints[i+1].y),
    render: { visible: false }
  });
  constraints.push(con);
  Matter.World.add(world, con);
}

Enter fullscreen mode Exit fullscreen mode

The Gravity Snapping Algorithm

If a rope is left unanchored, gravity causes it to plunge into the abyss. To make it a puzzle, we introduce Golden Anchors. These are coordinate locks.

During mouse-down (drag start) and mouse-up (drag end), we check for proximity to preset anchors using traditional Euclidean space calculations:

const findClosestAnchor = (x: number, y: number, radiusThreshold: number = 32) => {
  let closest: { x: number; y: number } | null = null;
  let minDist = radiusThreshold;

  currentLevel.anchors.forEach(anc => {
    const d = p.dist(x, y, anc.x, anc.y);
    if (d < minDist) {
      minDist = d;
      closest = anc;
    }
  });
  return closest;
};

Enter fullscreen mode Exit fullscreen mode

If a snap coordinate is found within range:

  1. We align the starting/ending rope link precisely to the Golden Anchor Center.
  2. We create an absolute anchor constraint linking the segment directly to the static Matter world coordinate (represented by a static body).
// Pin rope end firmly to Golden Anchor
const staticAnchorBody = Matter.Bodies.circle(snappedAnchor.x, snappedAnchor.y, 1, { isStatic: true });
Matter.World.add(world, staticAnchorBody);

const anchorConstraint = Matter.Constraint.create({
  bodyA: TargetRopeEndBody,
  bodyB: staticAnchorBody,
  stiffness: 0.95,
  length: 0
});
Matter.World.add(world, anchorConstraint);

Enter fullscreen mode Exit fullscreen mode

This creates the gorgeous suspension bridge effect where ropes sway and sag under the impact of rushing marbles!

Creative Sandbox UX: "Clear Chains" Feature

During playtesting, we observed that players wanting to fine-tune their designs became frustrated with full system resets. Resetting cleared their current level collection score and restarted level progress.

To fix this, we implemented Iterative Design Clearing.

We introduced a Clear Chains feature. It targets and clears only the user-painted Matter composite chains, leaving the physics progress of flowing particles, level timers, and collection tally scores completely intact:

// Define clear logic cleanly within Matter physics scope
const handleClearChainsEvent = () => {
  if (!world) return;

  // Recursively dismount current chains from Matter.World
  chains.forEach(c => {
    if (c) {
      if (c.links) c.links.forEach(l => l && Matter.World.remove(world, l));
      if (c.constraints) c.constraints.forEach(con => con && Matter.World.remove(world, con));
    }
  });
  chains = []; // Clear local tracker list
  setChainsCount(0); // Sync React state counter
};

Enter fullscreen mode Exit fullscreen mode

Mitigating React Closure Pitfalls in High-Hz Canvas Loops

One of the trickiest bugs solved in this runtime was the Sloppy Run State Capture Freeze.

The Symptom:

When a player dropped more than 5 marbles, a "Sloppy Run! Game Over" card appeared on the screen, prompting a retry. Even though the overlay was active, the marble faucet spigot continued spawning balls, and the fallen ball counter kept increasing, rendering the UI state incorrect.

The Root Cause:

Inside P5's high-frequency p.draw loop (running at 60fps), referencing the React state value showSloppyMessage returns a closure value captured at P5 initialization time. The canvas loop was checking a stale state, meaning it didn't know the game over modal was shown!

The Solution:

We implemented Synchronized Mutable Refs to act as hot-wiring bridges between React and the P5 paint routine:

// 1. Declare state and tracking Ref
const [showSloppyMessage, setShowSloppyMessage] = useState<boolean>(false);
const showSloppyMessageRef = useRef(showSloppyMessage);

// 2. Hot-swap state into hot-ref on state changes securely
useEffect(() => { 
  showSloppyMessageRef.current = showSloppyMessage; 
}, [showSloppyMessage]);

// 3. Inside p5 loop drawer, evaluate strictly from raw Ref:
if (faucetActiveRef.current && activeState === "playing" && !showSloppyMessageRef.current) {
  if (p.frameCount % streamSpeedRef.current === 0) {
    spawnMarble(); // Faucet spawning stops instantly!
  }
}

Enter fullscreen mode Exit fullscreen mode

By querying the Hot-Ref inside p.draw(), the physics loop reacts instantaneously to state changes without experiencing closure lags or forcing canvas re-mounts.


Sound Synthesis: Interactive Web Audio API

To provide sensory satisfaction, Gravity Paint bypasses bulky .mp3 assets in favor of dynamic Web Audio Synthesizers. Whenever a marble collides or drops, we initiate synthetic tone triggers:

export class Synthesizer {
  private ctx: AudioContext | null = null;

  playPing(index: number) {
    if (!this.ctx) this.ctx = new (window.AudioContext || (window as any).webkitAudioContext)();

    const osc = this.ctx.createOscillator();
    const gain = this.ctx.createGain();

    // Choose dynamic pentatonic scaling for musical feedback
    const baseFreq = 220;
    const pentatonicScales = [1, 1.125, 1.25, 1.5, 1.667, 2, 2.25, 2.5, 3];
    const multiplier = pentatonicScales[index % pentatonicScales.length];

    osc.frequency.setValueAtTime(baseFreq * multiplier, this.ctx.currentTime);
    osc.type = 'triangle'; // Smooth, woody marimba timbre

    gain.gain.setValueAtTime(0.2, this.ctx.currentTime);
    gain.gain.exponentialRampToValueAtTime(0.001, this.ctx.currentTime + 0.45);

    osc.connect(gain);
    gain.connect(this.ctx.destination);

    osc.start();
    osc.stop(this.ctx.currentTime + 0.5);
  }
}

Enter fullscreen mode Exit fullscreen mode

This creates a generative visual-instrument experience out of a simple canvas sandbox!

Combining rigid-body physics engines like Matter.js with p5.js layouts inside highly reactive frameworks like React 19 can be challenging. By:

  1. Segregating physics calculations from React renders,
  2. Bridging dynamic loops with hot-ref registers, and
  3. Using decentralized sound generation via the Web Audio API,

we built a high-performance web game. It runs cleanly on everything from standard production servers to sandboxed mobile frames.

Explore the source code, check out the game, and get drawing!

Screenshots

How to play 1

How to play 2

How to play 3

Code and more: https://www.dailybuild.xyz/project/143-gravity-paint