<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Jack Lee</title>
    <description>The latest articles on DEV Community by Jack Lee (@linb).</description>
    <link>https://dev.to/linb</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.us-east-2.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F167185%2Fa03d48fd-2235-4fef-aa0a-9f5a2b9ea784.jpeg</url>
      <title>DEV Community: Jack Lee</title>
      <link>https://dev.to/linb</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/linb"/>
    <language>en</language>
    <item>
      <title>Your AI writes the logic. Studio tunes the UI. They don't compete.</title>
      <dc:creator>Jack Lee</dc:creator>
      <pubDate>Mon, 22 Jun 2026 16:43:44 +0000</pubDate>
      <link>https://dev.to/linb/your-ai-writes-the-logic-studio-tunes-the-ui-they-dont-compete-4jf5</link>
      <guid>https://dev.to/linb/your-ai-writes-the-logic-studio-tunes-the-ui-they-dont-compete-4jf5</guid>
      <description>&lt;p&gt;I was building a dashboard. The card layout was almost right — just needed a bit more breathing room, a tighter border radius on the inner rows, and the label font to feel slightly lighter. Three small things.&lt;/p&gt;

&lt;p&gt;I typed the description into Cursor. It came back with 180 lines rewritten. Every &lt;code&gt;sx&lt;/code&gt; prop touched, four components restructured, a new wrapper &lt;code&gt;div&lt;/code&gt; appeared. The layout looked the same. The diff was enormous.&lt;/p&gt;

&lt;p&gt;I hit &lt;code&gt;Ctrl+Z&lt;/code&gt; and went back to staring at the code.&lt;/p&gt;

&lt;p&gt;That was the moment I stopped thinking of AI as a universal tool and started thinking of it as a &lt;em&gt;specific&lt;/em&gt; tool — an exceptionally good one, for specific things.&lt;/p&gt;

&lt;h2&gt;
  
  
  What AI is actually good at
&lt;/h2&gt;

&lt;p&gt;AI editors are brilliant at the parts of React development that are fundamentally about &lt;em&gt;thinking&lt;/em&gt;: data flow, state architecture, API wiring, conditional rendering logic, form validation, accessibility semantics, refactoring towards patterns. These are tasks where the right answer involves reasoning across the whole codebase, holding context, understanding intent.&lt;/p&gt;

&lt;p&gt;When I ask an AI to "add optimistic update to this mutation handler" or "extract this form logic into a custom hook" — I get back something genuinely useful, usually on the first try.&lt;/p&gt;

&lt;h2&gt;
  
  
  What AI is genuinely bad at
&lt;/h2&gt;

&lt;p&gt;UI polish is a different category. Not because AI can't read CSS — it clearly can. But because getting a layout &lt;em&gt;exactly right&lt;/em&gt; is an iterative, visual, trial-and-error process. You need to &lt;em&gt;see&lt;/em&gt; the result immediately. You need to tweak a value, check it, tweak again.&lt;/p&gt;

&lt;p&gt;The AI-for-UI workflow is fundamentally broken:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Describe what you want in words (already lossy)&lt;/li&gt;
&lt;li&gt;Wait for the model to generate a response&lt;/li&gt;
&lt;li&gt;Review 200 lines of diff&lt;/li&gt;
&lt;li&gt;Wait for the build&lt;/li&gt;
&lt;li&gt;See the result — probably not right&lt;/li&gt;
&lt;li&gt;Repeat, burning tokens each round&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;The bigger the model's context about your component, the more it rewrites. It doesn't know how to make a surgical one-property change, because its job is to understand and regenerate, not to minimally diff.&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;AI is expensive at precision. It's cheap at breadth. UI tuning needs precision. Logic needs breadth. Match the tool to the task.&lt;/p&gt;
&lt;/blockquote&gt;

&lt;h2&gt;
  
  
  Where each tool belongs
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;AI Editor (Cursor / Claude / Copilot)&lt;/strong&gt;&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Component architecture and refactoring&lt;/li&gt;
&lt;li&gt;State management and data flow&lt;/li&gt;
&lt;li&gt;API integration and async logic&lt;/li&gt;
&lt;li&gt;Accessibility and semantics&lt;/li&gt;
&lt;li&gt;Test generation&lt;/li&gt;
&lt;li&gt;Large-scale rewrites and migrations&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;strong&gt;CrossUI Studio&lt;/strong&gt;&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Spacing, sizing, layout tweaks&lt;/li&gt;
&lt;li&gt;MUI &lt;code&gt;sx&lt;/code&gt; and responsive breakpoints&lt;/li&gt;
&lt;li&gt;Color, typography, visual hierarchy&lt;/li&gt;
&lt;li&gt;Deep-layer UI drill-down and isolation&lt;/li&gt;
&lt;li&gt;Rapid iterative visual feedback&lt;/li&gt;
&lt;li&gt;Surgical one-property diffs&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Notice there's no overlap. That's not accidental — these tools solve genuinely different problems. The goal was never to have one tool do everything. It was to cover the full surface of React development without gaps.&lt;/p&gt;

&lt;h2&gt;
  
  
  The pairing in practice
&lt;/h2&gt;

&lt;p&gt;Since we shipped Local Folder access in v0.9.6, the workflow is concrete: point both your AI editor &lt;em&gt;and&lt;/em&gt; Studio at the same folder on your machine. They work on the same files simultaneously. One writes logic, the other tunes UI. The diff in Git is always clean because every change is surgical.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Step 1 — AI builds the component&lt;/strong&gt;&lt;br&gt;
Cursor or Claude writes the data-fetching hook, the conditional render, the event handlers. Full context, full reasoning, exactly what it's good at.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Step 2 — Studio opens the same folder&lt;/strong&gt;&lt;br&gt;
No upload, no sync, no token. Studio reads the files directly off disk — the canvas shows your component live.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Step 3 — You tune visually&lt;/strong&gt;&lt;br&gt;
Adjust spacing, colors, and layout directly on the canvas. The code updates in real-time. One property changed = one property in the diff.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Step 4 — PR lands clean&lt;/strong&gt;&lt;br&gt;
Logic changes from AI. UI changes from Studio. Both are minimal, reviewable, and traceable. No 200-line rewrites for a border-radius tweak.&lt;/p&gt;

&lt;h2&gt;
  
  
  What this means for "AI replacing developers"
&lt;/h2&gt;

&lt;p&gt;There's a lot of anxiety right now about AI eating programming jobs. Some of it is warranted — the volume of boilerplate code a developer needs to write by hand has dropped dramatically, and it's not coming back.&lt;/p&gt;

&lt;p&gt;But the anxiety assumes AI is a single homogeneous capability that scales linearly. It isn't. There are things AI does cheaply and well (reasoning, pattern-matching, generation at scale) and things it does poorly or expensively (precise iterative feedback loops, visual judgment, knowing when to stop).&lt;/p&gt;

&lt;p&gt;The developer who understands this boundary — and builds a workflow around it — is faster than the developer who either ignores AI entirely &lt;em&gt;or&lt;/em&gt; tries to use it for everything. The job doesn't disappear. It shifts: less time writing boilerplate, more time designing systems and judging quality.&lt;/p&gt;

&lt;p&gt;Studio is a bet on that shift. Not a replacement for AI. Not a rejection of it. A tool for the part of your work that AI handles badly — so you can let AI fully own the part it handles well.&lt;/p&gt;




&lt;p&gt;If you've been using an AI editor and hitting the same wall on UI iteration — try the pairing. Open Studio, point it at your project folder, and see if the division of labor clicks for you.&lt;/p&gt;

&lt;p&gt;No account needed — &lt;a href="https://studio.crossui.com/app" rel="noopener noreferrer"&gt;Guest mode&lt;/a&gt; opens a workspace instantly. Or connect a local folder and work on your real files.&lt;/p&gt;




&lt;p&gt;&lt;strong&gt;About CrossUI Studio&lt;/strong&gt; — A visual IDE for React &amp;amp; MUI. Code and canvas stay in two-way sync on the same AST: edit code and the canvas updates live; click an element on the canvas, edit its props visually, and the code changes with a surgical one-line diff. No build, no localhost — it runs in the browser, works on your real Git repo or a local folder, with no vendor lock-in.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://studio.crossui.com" rel="noopener noreferrer"&gt;Try it free → studio.crossui.com&lt;/a&gt;&lt;/p&gt;

</description>
      <category>react</category>
      <category>ai</category>
      <category>cursor</category>
      <category>dx</category>
    </item>
    <item>
      <title>Don't just find the broken import. Bypass it.</title>
      <dc:creator>Jack Lee</dc:creator>
      <pubDate>Mon, 22 Jun 2026 16:07:49 +0000</pubDate>
      <link>https://dev.to/linb/dont-just-find-the-broken-import-bypass-it-56hf</link>
      <guid>https://dev.to/linb/dont-just-find-the-broken-import-bypass-it-56hf</guid>
      <description>&lt;p&gt;A dependency five levels down throws, and your whole canvas goes white. CrossUI Studio's Dependency Graph still draws — zero build — points you straight at the blinking node, and then lets you inject a mock for the broken module to bring the page back. No file edit. No rebuild.&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;The Dependency Graph is built from static AST parsing, not a build — open a file and the import tree renders instantly, even when the canvas has completely crashed. The failing module blinks with its exact line/column. Then the part that's actually new: from the inspector you can set an interceptor or override to swap the broken dependency for a mock, hit Apply Dependency Injection, and the render comes back — rules auto-persist per entry file.&lt;/p&gt;
&lt;/blockquote&gt;




&lt;h2&gt;
  
  
  The blank-screen problem, five levels deep
&lt;/h2&gt;

&lt;p&gt;A component blows up, but the component is never the culprit. Here's a real one. The canvas throws inside &lt;code&gt;AIPanel.jsx&lt;/code&gt;:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;Error (Canvas render — runtime)
Parser engine initialization aborted.
  in Studio/src/components/panels/AIPanel.jsx

  at initializeParser (Engine/src/TreeSitter.js:32:11)
  at async Engine/src/TreeSitter.js:36:31
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;But &lt;code&gt;AIPanel&lt;/code&gt; isn't where it broke. The real failure is in &lt;code&gt;TreeSitter.js&lt;/code&gt;, line 32 — &lt;strong&gt;five imports down&lt;/strong&gt; the chain:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;AIPanel.jsx → AIJsxResponseReconciler.js → JSXParserEngine.js
            → CodeFormatter.js → TreeSitter.js   ← throws here
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Normally this is where the afternoon disappears: rebuild to reproduce, scatter &lt;code&gt;console.log&lt;/code&gt;s down the chain, guess which import actually threw. And the tools built to help have all just gone dark — React DevTools shows nothing (no component mounted), the profiler has no render to profile, the error overlay gives you a line but not the &lt;em&gt;shape&lt;/em&gt; of the code around it. They all need the app to run first. It didn't.&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;The one moment you most need to understand your code's structure is the one moment your code refuses to produce it.&lt;/p&gt;
&lt;/blockquote&gt;




&lt;h2&gt;
  
  
  Zero build, instant display
&lt;/h2&gt;

&lt;p&gt;The Dependency Graph skips the build step entirely. Open a file and it parses the &lt;code&gt;import&lt;/code&gt; tree to an AST and draws it in real time — no bundling, no dev-server round-trip, no waiting. That alone is the everyday time-saver: you see the whole reachable module graph the instant you ask for it, not after a rebuild.&lt;/p&gt;

&lt;p&gt;And because the map is parsed from &lt;em&gt;source&lt;/em&gt; rather than observed from a &lt;em&gt;run&lt;/em&gt;, it has a property no runtime tool can match: it still renders when the canvas has completely crashed. A broken page isn't a dead end — it's the graph's primary entry point.&lt;/p&gt;




&lt;h2&gt;
  
  
  Two ways to know a codebase
&lt;/h2&gt;

&lt;p&gt;There are only two ways for a tool to learn the shape of your project, and the difference decides everything about when the tool is available to you.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Runtime introspection&lt;/strong&gt; — needs a successful render. React DevTools, profilers, error overlays. Reads the live fiber tree after mount, knows real prop values — but only if they exist. Goes completely dark when the render throws, and can't see a module that failed to initialize.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Static AST analysis&lt;/strong&gt; — needs nothing to run. CrossUI's Dependency Graph parses each file to an AST and reads its imports. Knows structure — every edge, before execution — indifferent to whether the app renders, and maps the broken module &lt;em&gt;and&lt;/em&gt; its neighbors.&lt;/p&gt;

&lt;p&gt;Runtime tools are richer when they work. But they are &lt;strong&gt;conditional on success&lt;/strong&gt;. Static analysis is poorer in some ways — it can't tell you what a variable held at 2:04pm — but it is &lt;strong&gt;unconditional&lt;/strong&gt;. It works on a codebase that has never once rendered cleanly. That property is the whole point.&lt;/p&gt;




&lt;h2&gt;
  
  
  Reading the graph
&lt;/h2&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.us-east-2.amazonaws.com%2Fuploads%2Farticles%2F8trzo849e37o6yu348n4.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.us-east-2.amazonaws.com%2Fuploads%2Farticles%2F8trzo849e37o6yu348n4.png" alt="Dependency Graph" width="800" height="475"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Set any file as the &lt;code&gt;ENTRY&lt;/code&gt; and the panel scans outward from it. Every node is tagged so you can place it at a glance: &lt;code&gt;Entry&lt;/code&gt;, &lt;code&gt;📄 Local&lt;/code&gt;, &lt;code&gt;🌐 Ext&lt;/code&gt;, and &lt;code&gt;⟳ Cyclic&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Hover or select a node and animated lines trace its relationships in two directions — and the direction is the whole point:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Solid lines — downstream.&lt;/strong&gt; The modules this file imports.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Dashed lines — upstream.&lt;/strong&gt; The parents that import &lt;em&gt;this&lt;/em&gt; file.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;The search box highlights matching nodes live; the failing module &lt;strong&gt;blinks&lt;/strong&gt; so your eye lands on it without hunting.&lt;/p&gt;

&lt;p&gt;Click any node and the inspector slides out — your troubleshooting console for that file. It shows the resolved path, the raw &lt;code&gt;source&lt;/code&gt; string and its &lt;code&gt;resolvedSource&lt;/code&gt;, and, when parsing hit trouble, the exact error in red with line and column:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;Inspector · TreeSitter.js                     📄 Local
  specifiers       { parserInstance }
  source           ./TreeSitter.js
  resolvedSource   Engine/src/TreeSitter.js
  error            Parser engine initialization aborted · Line 32, Column 12
  interceptor      [ intercept source… ]
  override         [ override resolved path… ]
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Navigation note: mouse-wheel pans vertically, &lt;code&gt;Shift&lt;/code&gt;+wheel pans horizontally, &lt;code&gt;Ctrl&lt;/code&gt;+wheel zooms on the cursor — so a huge graph stays fast to move through.&lt;/p&gt;

&lt;h3&gt;
  
  
  Cycles, in red and one click away
&lt;/h3&gt;

&lt;p&gt;The engine detects circular references and draws the offending edge in red. A &lt;strong&gt;Show cycles&lt;/strong&gt; filter in the bottom-left isolates just the nodes caught in a loop. Inside the inspector, a node in a cycle shows the full closed-loop chain — and hovering the chain highlights every related node on the canvas, while clicking a name in the chain selects and focuses that file. Circular imports are the quiet cause of a whole genre of bugs — a module that's &lt;code&gt;undefined&lt;/code&gt; at first access, lazy chunks that won't split, "works on the second hot-reload" gremlins. The graph just shows you the loop instead of making you find it by accident.&lt;/p&gt;




&lt;h2&gt;
  
  
  The part that's actually new: bypass the broken module
&lt;/h2&gt;

&lt;p&gt;Finding the blinking node is good. But the Dependency Graph goes one step further than any read-only map: it lets you &lt;strong&gt;swap the broken dependency for a mock and bring the render back&lt;/strong&gt; — without editing the file and without a rebuild. Two injection levers, both in the inspector:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Interceptor&lt;/strong&gt; — match on the &lt;code&gt;import … from 'yyy'&lt;/code&gt; declaration. The &lt;code&gt;source&lt;/code&gt; string shown on the node is exactly the word you intercept.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Override&lt;/strong&gt; — match on the final physical path the system resolved to. The &lt;code&gt;resolvedSource&lt;/code&gt; on the node is exactly the path you override.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Configure either, then click the highlighted &lt;strong&gt;Apply Dependency Injection&lt;/strong&gt; button in the top toolbar. Studio injects your rule into the current rendering context and attempts to restore the visualization — the deep, crashing module is now standing in for itself with something that renders. And the rules &lt;strong&gt;auto-persist, keyed by the entry file&lt;/strong&gt;: reopen the same page tomorrow and your injection is already in place.&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;A read-only graph tells you where it broke. This one lets you route around the break and keep working — the file untouched, the page alive.&lt;/p&gt;
&lt;/blockquote&gt;

&lt;h3&gt;
  
  
  The render-error resolution flow
&lt;/h3&gt;

&lt;ol&gt;
&lt;li&gt;
&lt;strong&gt;Locate the blinking node.&lt;/strong&gt; Open the graph from the error overlay's &lt;strong&gt;Show Dependency Graph&lt;/strong&gt; link. The module that crashed the render is already blinking.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Investigate the root cause.&lt;/strong&gt; Follow the line/column on the node, double-click to open that exact file, and find the offending code.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Isolate with injection.&lt;/strong&gt; If a deep dependency is the cause, set an &lt;code&gt;interceptor&lt;/code&gt; (on &lt;code&gt;source&lt;/code&gt;) or &lt;code&gt;override&lt;/code&gt; (on &lt;code&gt;resolvedSource&lt;/code&gt;) in the inspector to swap it for a mock.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Apply to restore the render.&lt;/strong&gt; Click &lt;strong&gt;Apply Dependency Injection&lt;/strong&gt;. The rule is injected into the rendering context and the page attempts to come back — no rebuild.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Persistence handles the rest.&lt;/strong&gt; Your rules are saved locally against this entry file and restored automatically next time. Re-enter nothing.&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;  &lt;iframe src="https://www.youtube.com/embed/HZ7T4MFcqg8"&gt;
  &lt;/iframe&gt;
&lt;/p&gt;

&lt;h2&gt;
  
  
  Why this is the same thesis as everything else
&lt;/h2&gt;

&lt;p&gt;If you read &lt;a href="https://blog.crossui.com/2026/05/why-we-rejected-the-export-and-fork-model-for-visual-ui-editing" rel="noopener noreferrer"&gt;our first post&lt;/a&gt;, this will feel familiar. The reason a CrossUI visual edit produces a one-line git diff is the same reason the Dependency Graph survives a crash and can inject around it: &lt;strong&gt;we treat your parsed source as the source of truth, and every surface is just a view over that AST.&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;The canvas is a view over the AST. The prop inspector is a view over the AST. The Dependency Graph is a view over the AST &lt;em&gt;across files&lt;/em&gt; — and injection is a controlled rewrite of how one edge resolves. None of it depends on your app successfully running, because none of it is built on the runtime. That's a constraint we chose on purpose — it's harder to build, and it's exactly what keeps the tooling alive in the moments runtime tools abandon you.&lt;/p&gt;




&lt;h2&gt;
  
  
  What it can't do — honestly
&lt;/h2&gt;

&lt;p&gt;Static analysis buys unconditional availability by giving up runtime knowledge. The boundaries are real and we'd rather name them:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Fully dynamic imports are unresolvable.&lt;/strong&gt; A dynamic &lt;code&gt;import('./pages/' + name)&lt;/code&gt; with a runtime-computed path can't be resolved statically — we mark the edge rather than guess wrong.&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;It shows structure, not values.&lt;/strong&gt; The graph tells you &lt;code&gt;AIPanel&lt;/code&gt; reaches &lt;code&gt;TreeSitter&lt;/code&gt;; it can't tell you what the parser returned this particular render. Injection mocks the dependency; it doesn't replay the runtime state.&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Exotic resolution needs config.&lt;/strong&gt; Non-standard bundler aliases or monorepo path magic resolve best when we can read your &lt;code&gt;tsconfig&lt;/code&gt; / jsconfig. Without it, some edges fall back to raw specifiers.&lt;/p&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;None of those change the core promise: the graph is there when you're stuck, and it can route you around the break — because it never needed your code to run.&lt;/p&gt;

&lt;p&gt;→ &lt;a href="https://studio.crossui.com" rel="noopener noreferrer"&gt;Open Studio&lt;/a&gt; — crash a deep dependency, watch the graph point at it, then inject a mock and click Apply.&lt;/p&gt;




&lt;p&gt;&lt;strong&gt;About CrossUI Studio&lt;/strong&gt; — A visual IDE for React &amp;amp; MUI. Code and canvas stay in two-way sync on the same AST: edit code and the canvas updates live; click an element on the canvas, edit its props visually, and the code changes with a surgical one-line diff. No build, no localhost — it runs in the browser, works on your real Git repo or a local folder, with no vendor lock-in.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://studio.crossui.com" rel="noopener noreferrer"&gt;Try it free → studio.crossui.com&lt;/a&gt;&lt;/p&gt;

</description>
      <category>react</category>
      <category>development</category>
      <category>webdev</category>
      <category>programming</category>
    </item>
    <item>
      <title>Add "Try in Studio" to Your React Tutorials in 5 Minutes</title>
      <dc:creator>Jack Lee</dc:creator>
      <pubDate>Mon, 22 Jun 2026 14:40:38 +0000</pubDate>
      <link>https://dev.to/linb/add-try-in-studio-to-your-react-tutorials-in-5-minutes-5f68</link>
      <guid>https://dev.to/linb/add-try-in-studio-to-your-react-tutorials-in-5-minutes-5f68</guid>
      <description>&lt;p&gt;&lt;em&gt;~7 min read  ·TUTORIAL&lt;/em&gt;&lt;/p&gt;




&lt;p&gt;Your readers copy-paste your code examples. Wouldn't it be better if they could try them live, in the browser, right from your article? Add an "Open in Studio" button with a single script tag. No build, no local setup, no dependencies.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fblog.crossui.com%2Fimages%2Fguest-snapshot.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fblog.crossui.com%2Fimages%2Fguest-snapshot.png" alt="CrossUI Studio — visual IDE for React and MUI" width="800" height="488"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Bridge is how Studio reaches your audience. If you write React tutorials, teach courses, sell component templates, or build UIs for clients — one button turns your static code examples into live, editable Studio workspaces. This is how you set it up.&lt;/p&gt;

&lt;h2&gt;
  
  
  FOR PARTNERS
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;This is how partners bring Studio to their audience.&lt;/strong&gt; Add one line of code and your tutorials, courses, or templates get a visual IDE built in. Your readers learn better. Your students skip setup. Your templates sell themselves. Your clients sign off faster.&lt;/p&gt;

&lt;h2&gt;
  
  
  Why Bridge?
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;You already have great content, courses, or templates.&lt;/strong&gt; Bridge makes them more valuable with zero effort on your side:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;For Tutorial &amp;amp; Content Partners:&lt;/strong&gt; Transform static snippets into live, tweakable playgrounds. Readers experiment instead of just reading. No hosting, no infra — it's free and maintenance-free.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;For Template Authors:&lt;/strong&gt; Offer buyers a "Try Before You Buy" visual preview. They visually edit layouts without touching React code, drastically reducing support tickets. Templates sell themselves.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;For Instructors &amp;amp; Bootcamps:&lt;/strong&gt; Your students get free Pro for 1 month per cohort. They skip npm install, environment hell, localhost issues. Jump straight into visual React learning. Your course feels more premium.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;For Agencies:&lt;/strong&gt; Co-design UIs with clients in real-time. Visual edits immediately generate clean, PR-ready React code. Sign-offs faster, less back-and-forth.&lt;/li&gt;
&lt;/ul&gt;

&lt;blockquote&gt;
&lt;p&gt;One button. Your audience gets a better experience. You get more value from the work you've already done.&lt;/p&gt;
&lt;/blockquote&gt;

&lt;h2&gt;
  
  
  Three integration paths
&lt;/h2&gt;

&lt;p&gt;Pick one based on your site's tech stack:&lt;/p&gt;

&lt;h3&gt;
  
  
  Path 1: HTML Form (Simplest)
&lt;/h3&gt;

&lt;p&gt;Works on static blogs, markdown CMS, or anywhere you can paste raw HTML. Just a form with a button.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Step 1: Copy this HTML&lt;/strong&gt;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight html"&gt;&lt;code&gt;&lt;span class="nt"&gt;&amp;lt;form&lt;/span&gt; &lt;span class="na"&gt;action=&lt;/span&gt;&lt;span class="s"&gt;"https://studio.crossui.com/public/bridge.php"&lt;/span&gt; &lt;span class="na"&gt;method=&lt;/span&gt;&lt;span class="s"&gt;"POST"&lt;/span&gt; &lt;span class="na"&gt;target=&lt;/span&gt;&lt;span class="s"&gt;"_blank"&lt;/span&gt;&lt;span class="nt"&gt;&amp;gt;&lt;/span&gt;
  &lt;span class="nt"&gt;&amp;lt;textarea&lt;/span&gt; &lt;span class="na"&gt;name=&lt;/span&gt;&lt;span class="s"&gt;"code"&lt;/span&gt; &lt;span class="na"&gt;style=&lt;/span&gt;&lt;span class="s"&gt;"display:none;"&lt;/span&gt;&lt;span class="nt"&gt;&amp;gt;&lt;/span&gt;
import React from 'react';
import { Button } from '@mui/material';

export default function Demo() {
  return &lt;span class="nt"&gt;&amp;lt;Button&lt;/span&gt; &lt;span class="na"&gt;variant=&lt;/span&gt;&lt;span class="s"&gt;"contained"&lt;/span&gt;&lt;span class="nt"&gt;&amp;gt;&lt;/span&gt;Hello!&lt;span class="nt"&gt;&amp;lt;/Button&amp;gt;&lt;/span&gt;;
}
  &lt;span class="nt"&gt;&amp;lt;/textarea&amp;gt;&lt;/span&gt;

  &lt;span class="nt"&gt;&amp;lt;input&lt;/span&gt; &lt;span class="na"&gt;type=&lt;/span&gt;&lt;span class="s"&gt;"hidden"&lt;/span&gt; &lt;span class="na"&gt;name=&lt;/span&gt;&lt;span class="s"&gt;"ref"&lt;/span&gt; &lt;span class="na"&gt;value=&lt;/span&gt;&lt;span class="s"&gt;"YOUR_REF_CODE"&lt;/span&gt;&lt;span class="nt"&gt;&amp;gt;&lt;/span&gt;

  &lt;span class="nt"&gt;&amp;lt;button&lt;/span&gt; &lt;span class="na"&gt;type=&lt;/span&gt;&lt;span class="s"&gt;"submit"&lt;/span&gt;&lt;span class="nt"&gt;&amp;gt;&lt;/span&gt;Open in CrossUI Studio&lt;span class="nt"&gt;&amp;lt;/button&amp;gt;&lt;/span&gt;
&lt;span class="nt"&gt;&amp;lt;/form&amp;gt;&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;strong&gt;Step 2: Customize the button (optional)&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;Style it to match your site. It's just a standard form button — use your own CSS. Add an &lt;code&gt;origin&lt;/code&gt; parameter if you want to track which page sends traffic (helpful for knowing what's working).&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight html"&gt;&lt;code&gt;&lt;span class="nt"&gt;&amp;lt;input&lt;/span&gt; &lt;span class="na"&gt;type=&lt;/span&gt;&lt;span class="s"&gt;"hidden"&lt;/span&gt; &lt;span class="na"&gt;name=&lt;/span&gt;&lt;span class="s"&gt;"origin"&lt;/span&gt; &lt;span class="na"&gt;value=&lt;/span&gt;&lt;span class="s"&gt;"my-react-tutorial-page"&lt;/span&gt;&lt;span class="nt"&gt;&amp;gt;&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;strong&gt;Step 3: Paste the React code&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;Replace the &lt;code&gt;import React...&lt;/code&gt; block with your actual component. That's it. You're done.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Styling?&lt;/strong&gt; Add CSS to the button element. It's just a standard form button. You can style it however you want.&lt;/p&gt;

&lt;p&gt;👉 &lt;a href="https://studio.crossui.com/public/bridge-demo-html.html" rel="noopener noreferrer"&gt;Live demo — HTML Form&lt;/a&gt;&lt;/p&gt;

&lt;h3&gt;
  
  
  Path 2: JS SDK (Most Automatic)
&lt;/h3&gt;

&lt;p&gt;For documentation sites or blogs where you have many code blocks. The SDK automatically finds them, injects buttons, and handles the rest.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Step 1: Add the script&lt;/strong&gt;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight html"&gt;&lt;code&gt;&lt;span class="nt"&gt;&amp;lt;script &lt;/span&gt;&lt;span class="na"&gt;src=&lt;/span&gt;&lt;span class="s"&gt;"https://studio.crossui.com/public/CrossUI-Bridge-Client.js"&lt;/span&gt;&lt;span class="nt"&gt;&amp;gt;&amp;lt;/script&amp;gt;&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Place it near the end of your &lt;code&gt;&amp;lt;body&amp;gt;&lt;/code&gt; tag.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Step 2: Initialize with your config&lt;/strong&gt;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight html"&gt;&lt;code&gt;&lt;span class="nt"&gt;&amp;lt;script&amp;gt;&lt;/span&gt;
&lt;span class="nx"&gt;CrossUIBridge&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;init&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt;
  &lt;span class="na"&gt;selector&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;pre code.language-jsx&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;  &lt;span class="c1"&gt;// Target JSX blocks&lt;/span&gt;
  &lt;span class="na"&gt;ref&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;YOUR_REF_CODE&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;              &lt;span class="c1"&gt;// Your partner code&lt;/span&gt;
  &lt;span class="na"&gt;origin&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;my-blog&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;                  &lt;span class="c1"&gt;// Optional: analytics&lt;/span&gt;
&lt;span class="p"&gt;});&lt;/span&gt;
&lt;span class="nt"&gt;&amp;lt;/script&amp;gt;&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;The script scans your page for code blocks matching that CSS selector and adds a button to each one.&lt;/p&gt;

&lt;p&gt;👉 &lt;a href="https://studio.crossui.com/public/bridge-demo-js-sdk.html" rel="noopener noreferrer"&gt;Live demo — JS SDK&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Markdown blog?&lt;/strong&gt; Most markdown renders code as &lt;code&gt;&amp;lt;pre&amp;gt;&amp;lt;code class="language-jsx"&amp;gt;...&amp;lt;/code&amp;gt;&amp;lt;/pre&amp;gt;&lt;/code&gt;. Set &lt;code&gt;selector: 'pre code.language-jsx'&lt;/code&gt; (or your markdown's class). The SDK does the rest.&lt;/p&gt;

&lt;h3&gt;
  
  
  Path 3: Server-Side (Most Flexible)
&lt;/h3&gt;

&lt;p&gt;For premium template marketplaces or enterprise sites where you want to validate the user's purchase before showing them the code. Your backend receives the request, checks the user's access, and if OK, injects your partner code before redirecting to Studio.&lt;/p&gt;

&lt;p&gt;See the &lt;a href="https://github.com/linb/CrossUI-Studio-Bridge" rel="noopener noreferrer"&gt;GitHub bridge folder&lt;/a&gt; for full examples in Node, PHP, C#, and Java.&lt;/p&gt;

&lt;h2&gt;
  
  
  What do your readers / students / clients get?
&lt;/h2&gt;

&lt;p&gt;One click. Zero setup. They land in Studio with full IDE power:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Instant feedback:&lt;/strong&gt; Edit code, see changes live on the canvas in real-time (SCD — Synchronized Code &amp;amp; Design)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;No environment hell:&lt;/strong&gt; No npm install, no localhost, no waiting. Works in any browser, on any device.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Visual + Code:&lt;/strong&gt; Drill down any layer in the component tree, edit styles visually, tweak props, understand the structure.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Guest access:&lt;/strong&gt; No sign-up required. They get the full IDE experience immediately.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;strong&gt;Result:&lt;/strong&gt; Your readers learn better. Your students skip friction. Your templates show themselves. Your clients buy with confidence.&lt;/p&gt;

&lt;h2&gt;
  
  
  Ready to bring Studio to your audience?
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;Three easy steps:&lt;/strong&gt;&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;
&lt;strong&gt;Pick your integration path&lt;/strong&gt; — HTML Form (simplest), JS SDK (most automatic), or Server-Side (most control)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Read the full guide:&lt;/strong&gt; See the &lt;a href="https://github.com/linb/CrossUI-Studio-Bridge" rel="noopener noreferrer"&gt;GitHub bridge folder&lt;/a&gt; for all three paths with code examples in Node, PHP, C#, and Java.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Apply to become a partner:&lt;/strong&gt; &lt;a href="https://studio.crossui.com/partners" rel="noopener noreferrer"&gt;Fill out the partnership form&lt;/a&gt; and we'll get back to you in 48 hours with setup docs and everything you need.&lt;/li&gt;
&lt;/ol&gt;




&lt;h2&gt;
  
  
  Bring Studio to your audience.
&lt;/h2&gt;

&lt;p&gt;Make your content interactive. Make your courses friction-free. Make your templates sell themselves.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://studio.crossui.com/partners" rel="noopener noreferrer"&gt;Become a Partner →&lt;/a&gt;&lt;/p&gt;




&lt;p&gt;&lt;strong&gt;About CrossUI Studio&lt;/strong&gt; — A visual IDE for React &amp;amp; MUI. Code and canvas stay in two-way sync on the same AST: edit code and the canvas updates live; click an element on the canvas, edit its props visually, and the code changes with a surgical one-line diff. No build, no localhost — it runs in the browser, works on your real Git repo or a local folder, with no vendor lock-in.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://studio.crossui.com" rel="noopener noreferrer"&gt;Try it free → studio.crossui.com&lt;/a&gt;&lt;/p&gt;

</description>
      <category>react</category>
      <category>tutorial</category>
      <category>bridge</category>
      <category>crossui</category>
    </item>
    <item>
      <title>Why we rejected the export-and-fork model for visual UI editing</title>
      <dc:creator>Jack Lee</dc:creator>
      <pubDate>Mon, 22 Jun 2026 13:16:56 +0000</pubDate>
      <link>https://dev.to/linb/why-we-rejected-the-export-and-fork-model-for-visual-ui-editing-d5g</link>
      <guid>https://dev.to/linb/why-we-rejected-the-export-and-fork-model-for-visual-ui-editing-d5g</guid>
      <description>&lt;p&gt;&lt;em&gt;~10 min read · Engineering&lt;/em&gt;&lt;/p&gt;




&lt;p&gt;There are roughly three honest ways a visual editing tool can relate to your codebase.&lt;/p&gt;

&lt;p&gt;The first is &lt;strong&gt;export and fork&lt;/strong&gt;: you design in the tool, it generates code, you paste it into your project, and from that moment forward the tool and your repo are strangers. This is how most AI code generators (v0, Bolt, Lovable) work — and for greenfield components, it's fine. You generate, you customize, you ship.&lt;/p&gt;

&lt;p&gt;The second is &lt;strong&gt;SDK ownership&lt;/strong&gt;: the tool becomes a runtime dependency. Your components are stored in a proprietary model, rendered via the tool's SDK, and deployed through the tool's infrastructure. Leave the tool, and you're migrating a schema, not just moving files. This is roughly how Plasmic and Builder.io work — and again, for the right use case (marketing pages, designer-editable CMS content), it makes sense.&lt;/p&gt;

&lt;p&gt;The third — the one we bet CrossUI Studio on — doesn't have a clean industry name yet. We call it &lt;strong&gt;Symmetric Collaborative Development (SCD)&lt;/strong&gt;: your React source code is the single source of truth, and the visual canvas is a peer editor of that source. Every visual change is written back to your file as a precise AST patch. No export step. No SDK. No lock-in. Stop using the tool tomorrow and nothing breaks.&lt;/p&gt;

&lt;p&gt;This post explains why we think the third model is the right one for senior engineers maintaining real codebases — and what it actually takes to build it.&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;&lt;strong&gt;TL;DR.&lt;/strong&gt; If your existing repo is the source of truth and you want a visual layer over it, the editor has to write back through the AST — atomically, one property at a time — or it will lose against your code reviewer. Everything else is a different product for a different team.&lt;/p&gt;
&lt;/blockquote&gt;




&lt;h2&gt;
  
  
  The git diff test
&lt;/h2&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.us-east-2.amazonaws.com%2Fuploads%2Farticles%2Fv4u4g82qkag5u3a1kghu.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.us-east-2.amazonaws.com%2Fuploads%2Farticles%2Fv4u4g82qkag5u3a1kghu.png" alt="Export Fork" width="799" height="380"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Here's the simplest way to evaluate any visual editor. Three steps:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Open the tool with a real MUI component already in your codebase.&lt;/li&gt;
&lt;li&gt;Change &lt;code&gt;variant="text"&lt;/code&gt; to &lt;code&gt;variant="contained"&lt;/code&gt; on a button.&lt;/li&gt;
&lt;li&gt;Run &lt;code&gt;git diff&lt;/code&gt;.&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;Every "design-to-code" tool I evaluated over the last five years failed this test, in one of two ways. Either the file was &lt;strong&gt;re-emitted entirely&lt;/strong&gt; — comments stripped, formatting normalized, imports reshuffled, hand-tuned &lt;code&gt;useMemo&lt;/code&gt; blocks subtly rearranged — or the tool refused to touch existing files at all and emitted a parallel "design document" we then had to glue back into the project.&lt;/p&gt;

&lt;p&gt;Neither was acceptable for a team that takes diff hygiene seriously. We rejected three tools in a row on this basis.&lt;/p&gt;

&lt;p&gt;The third time, I decided the category itself was wrong.&lt;/p&gt;




&lt;h2&gt;
  
  
  Why "re-emit the file" is such a hard habit to break
&lt;/h2&gt;

&lt;p&gt;If you read how most visual editors are architected, the root assumption is: &lt;strong&gt;the tool owns the rendering model&lt;/strong&gt;. The tool maintains its own representation of the component tree — either in a proprietary schema or in an in-memory tree — and "saving" means serializing that representation back to JSX.&lt;/p&gt;

&lt;p&gt;Serialization is lossy by nature. The tool's model doesn't know about your comment. It doesn't know why you named a variable &lt;code&gt;isSaving&lt;/code&gt;. It doesn't know that your &lt;code&gt;/* Main content area */&lt;/code&gt; block comment is something a human will read in code review. So when it writes back, it writes what it knows — the component structure, the props, the styles — and silently discards everything it doesn't.&lt;/p&gt;

&lt;p&gt;The result is a visually identical file that is a &lt;em&gt;conceptually different&lt;/em&gt; file. Same output, different authorship. If you're a senior engineer who cares about your codebase, that's enough reason to never use the tool again.&lt;/p&gt;

&lt;p&gt;The only way to avoid this is to not use serialization at all. Which means working at the AST level from the start — and the AST is hard.&lt;/p&gt;




&lt;h2&gt;
  
  
  What AST-level sync actually requires
&lt;/h2&gt;

&lt;p&gt;Abstract Syntax Trees are how JavaScript parsers represent code internally. Every identifier, every JSX attribute, every function call has a position in the tree. If you want to change &lt;code&gt;variant="outlined"&lt;/code&gt; to &lt;code&gt;variant="contained"&lt;/code&gt;, you don't re-serialize the whole file — you find the &lt;code&gt;JSXAttribute&lt;/code&gt; node whose name is &lt;code&gt;variant&lt;/code&gt;, update its &lt;code&gt;StringLiteral&lt;/code&gt; value, and write back only the characters that changed.&lt;/p&gt;

&lt;p&gt;This is what CrossUI Studio's engine does. The practical consequences:&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Formatting is preserved.&lt;/strong&gt; We don't pass the file through Prettier on every visual edit (though we do run Prettier on deliberate code saves). The unchanged bytes don't move.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Comments survive.&lt;/strong&gt; AST nodes have attached comment ranges. We track them and leave them where they are.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Business logic is untouched.&lt;/strong&gt; The engine identifies which AST node corresponds to the visual change and patches only that node. Your custom hooks, your &lt;code&gt;useCallback&lt;/code&gt; wrappers, your conditional rendering logic above and below the component — we don't touch any of it.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;The undo stack is unified.&lt;/strong&gt; Whether you typed in the code editor, dragged a component on the canvas, or changed a value in the inspector, every operation is a reversible AST mutation. &lt;code&gt;Ctrl+Z&lt;/code&gt; rolls back the code and the canvas in sync.&lt;/p&gt;

&lt;p&gt;The visible result for the engineer:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight diff"&gt;&lt;code&gt;   {/* keep until the new spec lands — Jordan, Apr 14 */}
   const total = useMemo(() =&amp;gt; sum(order.items), [order])
   return (
     &amp;lt;Card sx={{ p: 2 }}&amp;gt;
&lt;span class="gd"&gt;-      &amp;lt;Button variant="text" onClick={handle}&amp;gt;Pay&amp;lt;/Button&amp;gt;
&lt;/span&gt;&lt;span class="gi"&gt;+      &amp;lt;Button variant="contained" onClick={handle}&amp;gt;Pay&amp;lt;/Button&amp;gt;
&lt;/span&gt;     &amp;lt;/Card&amp;gt;
   )
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;One line removed, one line added. The comment, the &lt;code&gt;useMemo&lt;/code&gt;, the surrounding JSX — byte-identical to before. &lt;strong&gt;The diff that lands in code review is exactly the change the engineer would have typed.&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;  &lt;iframe src="https://www.youtube.com/embed/TOiYf7aN4yo"&gt;
  &lt;/iframe&gt;
&lt;/p&gt;




&lt;h2&gt;
  
  
  The depth problem: why &lt;code&gt;.map&lt;/code&gt; is the real test
&lt;/h2&gt;

&lt;p&gt;Even among tools that do some form of code-aware editing, almost all share the same limitation: they can only operate on the outermost JSX block.&lt;/p&gt;

&lt;p&gt;Consider this common pattern:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight jsx"&gt;&lt;code&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="nx"&gt;items&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;map&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;item&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;(&lt;/span&gt;
  &lt;span class="p"&gt;&amp;lt;&lt;/span&gt;&lt;span class="nc"&gt;Card&lt;/span&gt; &lt;span class="na"&gt;key&lt;/span&gt;&lt;span class="p"&gt;=&lt;/span&gt;&lt;span class="si"&gt;{&lt;/span&gt;&lt;span class="nx"&gt;item&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;id&lt;/span&gt;&lt;span class="si"&gt;}&lt;/span&gt; &lt;span class="na"&gt;elevation&lt;/span&gt;&lt;span class="p"&gt;=&lt;/span&gt;&lt;span class="si"&gt;{&lt;/span&gt;&lt;span class="nx"&gt;item&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;featured&lt;/span&gt; &lt;span class="p"&gt;?&lt;/span&gt; &lt;span class="mi"&gt;3&lt;/span&gt; &lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mi"&gt;1&lt;/span&gt;&lt;span class="si"&gt;}&lt;/span&gt;&lt;span class="p"&gt;&amp;gt;&lt;/span&gt;
    &lt;span class="p"&gt;&amp;lt;&lt;/span&gt;&lt;span class="nc"&gt;CardContent&lt;/span&gt;&lt;span class="p"&gt;&amp;gt;&lt;/span&gt;
      &lt;span class="p"&gt;&amp;lt;&lt;/span&gt;&lt;span class="nc"&gt;Typography&lt;/span&gt; &lt;span class="na"&gt;variant&lt;/span&gt;&lt;span class="p"&gt;=&lt;/span&gt;&lt;span class="s"&gt;"h6"&lt;/span&gt;&lt;span class="p"&gt;&amp;gt;&lt;/span&gt;&lt;span class="si"&gt;{&lt;/span&gt;&lt;span class="nx"&gt;item&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;title&lt;/span&gt;&lt;span class="si"&gt;}&lt;/span&gt;&lt;span class="p"&gt;&amp;lt;/&lt;/span&gt;&lt;span class="nc"&gt;Typography&lt;/span&gt;&lt;span class="p"&gt;&amp;gt;&lt;/span&gt;
    &lt;span class="p"&gt;&amp;lt;/&lt;/span&gt;&lt;span class="nc"&gt;CardContent&lt;/span&gt;&lt;span class="p"&gt;&amp;gt;&lt;/span&gt;
  &lt;span class="p"&gt;&amp;lt;/&lt;/span&gt;&lt;span class="nc"&gt;Card&lt;/span&gt;&lt;span class="p"&gt;&amp;gt;&lt;/span&gt;
&lt;span class="p"&gt;))}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;A typical visual editor sees the &lt;code&gt;map&lt;/code&gt; expression. It cannot edit the &lt;code&gt;Card&lt;/code&gt; inside it visually — because to render a canvas for the &lt;code&gt;Card&lt;/code&gt;, it needs to know what &lt;code&gt;item&lt;/code&gt; is, and &lt;code&gt;item&lt;/code&gt; only exists at runtime inside the callback.&lt;/p&gt;

&lt;p&gt;CrossUI's &lt;strong&gt;Layer Drill-down Engine&lt;/strong&gt; solves this with a Test Data Injector. When you drill into a &lt;code&gt;.map&lt;/code&gt; callback, Studio asks: &lt;em&gt;"what's the shape of &lt;code&gt;item&lt;/code&gt;?"&lt;/em&gt; You provide a mock value — or it infers one from surrounding usage — and Studio renders an isolated canvas for the inner block, with &lt;code&gt;item&lt;/code&gt; injected as a real prop. You edit the &lt;code&gt;Card&lt;/code&gt; visually. The changes are written back to the callback body as precise AST patches. The outer &lt;code&gt;items.map(...)&lt;/code&gt; expression is untouched.&lt;/p&gt;

&lt;p&gt;The same mechanism works for ternary branches (&lt;code&gt;cond ? &amp;lt;A/&amp;gt; : &amp;lt;B/&amp;gt;&lt;/code&gt;) and &lt;code&gt;children&lt;/code&gt; render slots. You can drill &lt;em&gt;any&lt;/em&gt; depth your component tree requires.&lt;/p&gt;

&lt;p&gt;This is the feature that separates a toy from a tool you'd actually use on a production codebase.&lt;/p&gt;




&lt;h2&gt;
  
  
  MUI isn't generic React
&lt;/h2&gt;

&lt;p&gt;One more thing the "re-emit" tools get wrong: they treat all React components the same.&lt;/p&gt;

&lt;p&gt;MUI components have a richer prop surface than native HTML. &lt;code&gt;variant&lt;/code&gt;, &lt;code&gt;size&lt;/code&gt;, &lt;code&gt;color&lt;/code&gt;, &lt;code&gt;elevation&lt;/code&gt;, &lt;code&gt;sx&lt;/code&gt; — these aren't just &lt;code&gt;className&lt;/code&gt; strings. They're typed, enumerated, and in some cases highly structured. The &lt;code&gt;sx&lt;/code&gt; prop accepts a deep object with pseudo-class keys, state-based keys, and responsive breakpoint objects like &lt;code&gt;{ xs: 'small', md: 'large' }&lt;/code&gt;. Generic React visual editors render an untyped text field for &lt;code&gt;sx&lt;/code&gt; and call it done.&lt;/p&gt;

&lt;p&gt;CrossUI Studio ships a dedicated MUI panel with:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Enumeration completions&lt;/strong&gt; for every MUI prop value (correct values, no guessing).&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;&lt;code&gt;sx&lt;/code&gt; tree editor&lt;/strong&gt; — nested objects, pseudo-classes, and media queries in a structured visual tree.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Responsive property promotion&lt;/strong&gt; — click the lightning icon next to any MUI system prop and it splits into a breakpoint object. Change the value at LG and the code becomes &lt;code&gt;size={{ xs: 'small', lg: 'medium' }}&lt;/code&gt; automatically.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Design Token access&lt;/strong&gt; — your custom &lt;code&gt;createTheme()&lt;/code&gt; tokens surface in the inspector.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;These aren't features you can retrofit onto a generic visual editor. They require knowing what MUI is, how it works, and where the edge cases are.&lt;/p&gt;




&lt;h2&gt;
  
  
  What we gave up
&lt;/h2&gt;

&lt;p&gt;The SCD model has real costs. It's why most tools don't do it. We owe an honest accounting:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;We don't work on every React codebase.&lt;/strong&gt; If your components are deeply entangled with SSR edge cases, or your bundler produces non-standard output, the AST layer may not parse cleanly. We document the boundaries instead of pretending they don't exist.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Focus is our superpower.&lt;/strong&gt; No Vue, no Svelte, no Angular — not on the roadmap. We'd rather be the best possible tool for one ecosystem than a mediocre tool for five. We are dedicated exclusively to the React ecosystem, deeply supporting leading design systems like MUI and shadcn/ui, with expandable support for Joy UI, Ant Design, and beyond.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;We don't generate apps.&lt;/strong&gt; If you want to prompt your way to a working dashboard, use v0 or Bolt. They're genuinely good at that. We're for engineers who already have a codebase and want a canvas that respects it.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;We don't host your code.&lt;/strong&gt; No CrossUI-side storage. Git operations are direct browser-to-provider. That's a feature, but it also means we can't offer cloud workspaces, team collaboration on a shared document, or any of the conveniences a hosted model enables.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;We evolve alongside the ecosystem.&lt;/strong&gt; We closely follow major updates of our supported component libraries. While we prioritize empowering teams on the latest versions with robust, modern alignment, we continually evaluate expanding our backwards compatibility. If you are locked into legacy framework versions, we might not be the right tool for you — yet.&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  The decision
&lt;/h2&gt;

&lt;p&gt;Three honest recommendations, no false modesty:&lt;/p&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;If you're…&lt;/th&gt;
&lt;th&gt;…use&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;Building a marketing site that designers need to edit&lt;/td&gt;
&lt;td&gt;&lt;strong&gt;Plasmic&lt;/strong&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Prototyping from scratch and want AI generation speed&lt;/td&gt;
&lt;td&gt;
&lt;strong&gt;v0&lt;/strong&gt; or &lt;strong&gt;Bolt&lt;/strong&gt;
&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Handing Figma files off to engineers, once&lt;/td&gt;
&lt;td&gt;
&lt;strong&gt;Locofy&lt;/strong&gt; or &lt;strong&gt;Anima&lt;/strong&gt;
&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Maintaining a React + MUI codebase, care about your git history, want a visual canvas that treats your code as the source of truth&lt;/td&gt;
&lt;td&gt;&lt;strong&gt;CrossUI Studio&lt;/strong&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;

&lt;p&gt;The Playground is free, works immediately, and doesn't require connecting a repo. Paste any React snippet and see it on the canvas.&lt;/p&gt;




&lt;p&gt;&lt;strong&gt;About CrossUI Studio&lt;/strong&gt; — A visual IDE for React &amp;amp; MUI. Code and canvas stay in two-way sync on the same AST: edit code and the canvas updates live; click an element on the canvas, edit its props visually, and the code changes with a surgical one-line diff. No build, no localhost — it runs in the browser, works on your real Git repo or a local folder, with no vendor lock-in.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://studio.crossui.com" rel="noopener noreferrer"&gt;Try it free → studio.crossui.com&lt;/a&gt;&lt;/p&gt;

</description>
      <category>ast</category>
      <category>react</category>
      <category>scd</category>
      <category>mui</category>
    </item>
    <item>
      <title>CrossUI Web App Builder</title>
      <dc:creator>Jack Lee</dc:creator>
      <pubDate>Mon, 13 May 2019 21:46:10 +0000</pubDate>
      <link>https://dev.to/linb/crossui-builder-je4</link>
      <guid>https://dev.to/linb/crossui-builder-je4</guid>
      <description>&lt;p&gt;&lt;a href="https://crossui.com/RADGithub" rel="noopener noreferrer"&gt;https://crossui.com/RADGithub&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Explore the benefits of the no code app builder&lt;br&gt;
You no longer need to hire an app developer to create an application for you. Create a fully interactive and responsive HTML5 application without writing a single line of code with CrossUI Front-end Builder. It’s much faster, easier and more fun to do app development without coding. Just choose the necessary template, modify all the elements you need and launch. Whether you need to create an interactive prototype, database management app or build a management information system, you can do it with the Cross UI app builder within several days.&lt;/p&gt;

&lt;p&gt;What you can do with CrossUI no code app builder&lt;br&gt;
If you hired a developer, you would have spent as much time explaining the concept as it took you to do it yourself with CrossUI. With CrossUI, every person who is familiar with business logic can build an application. Just make sure that your data can be stored in a database, entered with a form and displayed in common charts, tables, or rows. Our builder has an easy-to-navigate menu and a set of features that will allow you to design and build your HTML5 application within a few days. Just create what you imagine. Here are the things you can do with the CrossUI builder:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Use built-in templates relevant to the type of application you want to build.&lt;/li&gt;
&lt;li&gt;Drag and drop the objects to the desired location.&lt;/li&gt;
&lt;li&gt;Use and edit 100+ UI widgets. Among them are buttons, inputs, lists, tabs, shapes, diagrams, charts and other.&lt;/li&gt;
&lt;li&gt;Create customized reusable modules.&lt;/li&gt;
&lt;li&gt;Bring in interaction by using events actions that can be easily configured.&lt;/li&gt;
&lt;li&gt;Call Remote Service EndPoint. No coding is required.&lt;/li&gt;
&lt;li&gt;Publish your app where you want, on any desktop (Linux, Windows, Mac PC/Mobile), or any website.&lt;/li&gt;
&lt;li&gt;Embed the WYSIWYG designer, our designer product, into the application you already have. The exported APIs and Events offer customization flexibility to developers.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Benefits of no code apps&lt;br&gt;
Create an app without coding to quickly get custom solutions to your business needs. According to a Quick Base report, no code app builders can help save money, increase productivity, and work efficiency. Use CrossUI to save time and remain 100% focused on your business. It is especially beneficial for small businesses that don’t require complex solutions. If you need to organize inventory information, keep track of your business contacts, manage customer data, you can do it with an app, developed with an app builder specifically for your case. CrossUI front-end builder has a user-friendly dashboard and admin panel that make updates a breeze. You can entirely customize your application's behavior without writing code. Its API tools allow connecting apps to your own back-end software. The library of pre-built application elements reduce development time and accelerate production. CrossUI builder can be used for commercial projects as well as non-profit ones. If you are a student who wants to learn how to create an app with no coding, or if you have personal, not-for-profit projects, we can offer you a license at a special price. Check out our prices on the Purchase page. Keep in mind, that each license is bound to one PC only. If you are interested in reselling our products, please contact our sales department. Build an app without code which is powerful enough to support your most ambitious ideas!&lt;/p&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.us-east-2.amazonaws.com%2Fuploads%2Farticles%2F3fvzbikllzka1hc663gr.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.us-east-2.amazonaws.com%2Fuploads%2Farticles%2F3fvzbikllzka1hc663gr.png" width="680" height="360"&gt;&lt;/a&gt;&lt;/p&gt;

</description>
      <category>javascript</category>
      <category>design</category>
      <category>github</category>
      <category>webdev</category>
    </item>
  </channel>
</rss>
