How I Moved a React Component Across the DOM Without Losing Its State — A Checkout Story — DeepSeek Blog | Neura Market
    Neura MarketNeura Market/DeepSeek
    ChatGPTChatGPTClaudeClaudeGeminiGeminiCursorCursorGrokGrokPerplexityPerplexityDeepSeekDeepSeek
    CoPilotCoPilotStable DiffusionStable DiffusionMidjourneyMidjourney
    View All Directories
    OverviewRulesPromptsMCPsAgentsBlogVideosGuidesCoursesCommunityTrendingGenerate
    DeepSeekBlogHow I Moved a React Component Across the DOM Without Losing Its State — A Checkout Story
    Back to Blog
    How I Moved a React Component Across the DOM Without Losing Its State — A Checkout Story
    frontend

    How I Moved a React Component Across the DOM Without Losing Its State — A Checkout Story

    Satish Gowda March 23, 2026
    0 views

    How I used React Portals and JavaScript media queries to solve a real-world layout problem on a...

    --- title: How I Moved a React Component Across the DOM Without Losing Its State — A Checkout Story published: true description: tags: # cover_image: https://dev-to-uploads.s3.amazonaws.com/uploads/articles/ysfw8a3a1f8l2b9dhjbt.png # Use a ratio of 100:42 for best results. # published_at: 2026-03-23 05:44 +0000 --- ![Image description](https://dev-to-uploads.s3.amazonaws.com/uploads/articles/37zo4t6b3t1u7f865si5.png) _How I used React Portals and JavaScript media queries to solve a real-world layout problem on a checkout page_ --- ## The Problem Nobody Warns You About You're building a checkout page. On desktop, you have the classic **Holy Grail layout**, a wide left column with the main form content and a right sidebar with supporting information. Your designer places a coupon/offers component in that right sidebar. Clean. Elegant. Makes sense. Then you open the mobile view. That right sidebar? Gone, because it collapses into a linear single-column layout on smaller screens. The coupon component now needs to live somewhere in the middle of the left column's content flow on mobile. Not at the bottom. Not at the top. Somewhere _specific_, visible to the user in the first fold. And here's where it gets messier than just "desktop vs mobile." These UI decisions aren't just about two breakpoints. A tablet in portrait mode behaves like a **mobile** device — single column, linear flow. Rotate it to landscape and suddenly it's **desktop** territory sidebar back, Holy Grail restored. Same device, same user, two completely different layout expectations within seconds. This is where CSS `display: none` / `display: block` tricks fall apart. You could render the component **twice** — once for landscape (**desktop**) and once for portrait (**mobile**) — and toggle visibility. But now you have two instances of the same component, two sets of state, two API connections, and a maintenance headache. There had to be a better way. --- ## What Are React Portals? React Portals let you render a component's output into **any DOM node you choose**, even one that exists completely outside your component's parent hierarchy while maintaining the state from the parent component. ```tsx import { createPortal } from 'react-dom'; const MyComponent = () => { return createPortal( <div className="modal">Hello from a portal!</div>, document.body ); }; ``` Portals are typically used for modals, tooltips, and dropdowns. Cases where you need to escape a parent's `overflow: hidden` or `z-index` stacking context. But they're capable of a lot more. **The key insight: while the DOM node is placed elsewhere, the React component tree and all its state stays intact.** One component instance, one state, rendered wherever you need it. --- ## The Solution: `PortComponent` The idea is straightforward, Build a wrapper component that: 1. Accepts two DOM target selectors, one for desktop and one for mobile. 2. Listens to the viewport width using a JavaScript media query. 3. Uses `createPortal` to teleport the child component to the correct target. ```tsx import { ReactElement, useEffect, useRef } from "react"; import { createPortal } from "react-dom"; import { useMediaQuery } from "react-responsive"; interface PortComponentProps { deskElem: string; // CSS selector for the desktop target container mobElem: string; // CSS selector for the mobile target container Component: ReactElement; } const PortComponent = ({ deskElem, mobElem, Component }: PortComponentProps) => { const isDesktop = useMediaQuery({ query: "(min-width: 1024px)" }); const elem = useRef<HTMLDivElement>(document.createElement("div")); useEffect(() => { if (isDesktop) { document.querySelector(deskElem)?.appendChild(elem.current); } else { document.querySelector(mobElem)?.appendChild(elem.current); } }, [isDesktop, deskElem, mobElem]); return createPortal(Component, elem.current); }; export default PortComponent; ``` And here is how it looks at the call site: ```tsx <PortComponent deskElem="#deskElem" mobElem="#mobileElem" Component={createElement(CouponWidget, { onCouponApplied: () => refetchPaymentMode(), coupons: couponList, })} /> ``` --- ## Breaking Down the Magic ### `useMediaQuery`, Reactive Viewport Detection ```tsx const isDesktop = useMediaQuery({ query: "(min-width: 1024px)" }); ``` From the `react-responsive` library, this isn't a one-time check, it subscribes to a `MediaQueryList` event listener under the hood. When the viewport crosses 1024px, `isDesktop` flips, the component re-renders, and the `useEffect` fires again. The component _responds_ to layout changes in real time. ### `useRef` for the Portal Host ```tsx const elem = useRef<HTMLDivElement>(document.createElement("div")); ``` This creates a plain `<div>` that acts as the portal's mounting point. Using `useRef` means it's created **once** and persists across renders. Moving it between DOM containers doesn't destroy or recreate it — and crucially, the React component tree inside it (state, API calls, everything) stays completely alive. We're moving the container, not remounting the component. ### `useEffect` — The DOM Transplant ```tsx useEffect(() => { if (isDesktop) { document.querySelector(deskElem)?.appendChild(elem.current); } else { document.querySelector(mobElem)?.appendChild(elem.current); } }, [isDesktop, deskElem, mobElem]); ``` `appendChild` on a node that already exists in the DOM _moves_ it, it doesn't clone it. The browser handles the transplant natively, and React's virtual DOM stays consistent because `createPortal` always points to the same `elem.current` ref. --- ## The Target Containers in Your Layout You need placeholder elements in your JSX to serve as anchor points: ```tsx // Desktop layout (right sidebar) <aside className="checkout-sidebar"> <div id="deskElem" /> {/* Coupon component lands here on desktop */} <PriceDetails /> <OrderSummary /> </aside> // Mobile layout (main content column) <main className="checkout-main"> <ShippingAddress /> <BillingAddress /> <div id="mobileElem" /> {/* Coupon component lands here on mobile */} <WalletSection /> <PaymentMethods /> </main> ``` These empty divs are invisible in themselves but act as precise positional anchors for the teleporting component. --- ## Why Not Just Use CSS? You could render two instances and toggle with CSS: ```css .coupon-desktop { display: none; } .coupon-mobile { display: block; } @media (min-width: 1024px) { .coupon-desktop { display: block; } .coupon-mobile { display: none; } } ``` For a stateless display component, that's perfectly fine. But this approach has a deeper problem that goes beyond just API calls — **it doesn't preserve state**. Imagine a user on a tablet who rotates their device mid-session. The coupon they just applied? Gone! because the hidden instance was never in sync. The input they half-typed into a promo code field? Cleared. Any loading or error state? Lost. When you toggle between two separate component instances, each one has its own isolated React state. There's no shared memory between them. Whichever one was hidden was effectively dead. ```tsx {/* Both mount independently, both have their own state */} <div className="coupon-desktop"> <CouponWidget /> {/* State lives here — applied coupon, input value, loading */} </div> <div className="coupon-mobile"> <CouponWidget /> {/* Completely separate state — knows nothing about the above */} </div> ``` Beyond state, two instances also means double the API calls on mount, synchronisation problems if one instance triggers a side effect, and twice as much to maintain when the component logic changes. The Portal approach sidesteps all of this. There is only ever **one instance** of the component in memory. Its state is never destroyed when the layout shifts — the component physically moves to a new DOM location, but React keeps its internal state, refs, and context connections completely intact. The user's applied coupon, their half-typed promo code, any in-flight API request — all of it survives a breakpoint change seamlessly. That's the real win: **one instance, one state, one source of truth** — displayed wherever the layout demands. --- ## Potential Improvements This pattern works well in production, but here are a few things worth considering for a more robust version: **Cleanup on unmount:** The current implementation doesn't remove `elem.current` from the DOM when the component unmounts: ```tsx useEffect(() => { const target = document.querySelector(isDesktop ? deskElem : mobElem); target?.appendChild(elem.current); return () => { elem.current.remove(); }; }, [isDesktop, deskElem, mobElem]); ``` **SSR compatibility:** `document.createElement` runs at initialisation time, which will throw in server-side rendering environments like Next.js. A `useEffect`-based setup would be needed there. **Configurable breakpoint:** Externalising `1024px` as a prop makes `PortComponent` reusable across different layout breakpoints in the same project. --- ## Takeaways React Portals are often introduced as a tool for modals and dropdowns, but their real power is giving you **precise control over where in the DOM your components live**. Independent of where they sit in your component tree. When combined with a reactive media query hook, you get a component that is genuinely **layout-aware**: it doesn't just look different at different screen sizes, it _lives_ in a different place. That's a meaningful distinction for complex layouts like checkout pages, dashboards, or any UI that fundamentally restructures itself between breakpoints. If you've been hacking around this with CSS `display:none`, there's a cleaner way — and your users will thank you. --- _In the next article, I'll show how I'd rewrite this today using Radix UI — and whether it's actually worth it._ _Have you used React Portals in unconventional ways? Drop a comment — I'd love to hear about it._

    Tags

    frontendjavascriptreacttutorial

    Comments

    More Blog

    View all
    How I'm using ASTs and Gemini to solve the "Codebase Onboarding" problem 🧠ai

    How I'm using ASTs and Gemini to solve the "Codebase Onboarding" problem 🧠

    Hi everyone! 👋 I’m Tara, a Senior Software Engineer and Consultant. Over the years, I've jumped...

    T
    tworrell
    Local AI Will Save Us All (The Math Says So, Trust Me)ai

    Local AI Will Save Us All (The Math Says So, Trust Me)

    Every few weeks a take goes viral in tech circles making the case for ditching cloud AI and running...

    S
    Sebastian Schürmann
    Lost in the AI Hype, I Started Smallai

    Lost in the AI Hype, I Started Small

    And it helped me get back into tech without drowning TL;DR at the end Coming back to...

    R
    Rohini Gaonkar
    Building a Replay-Tested Interactive Brokers Client in Gogo

    Building a Replay-Tested Interactive Brokers Client in Go

    I wanted an IBKR library that felt like Go and had testing I could trust. So I wrote one.

    T
    Thomas Marcelis
    Playwright in Pictures: Fully Parallel Modeplaywright

    Playwright in Pictures: Fully Parallel Mode

    Playwright’s fullyParallel mode is often treated as a simple performance switch. In practice, it...

    V
    Vitaliy Potapov
    Designing a CLI for Both Humans and Agentscli

    Designing a CLI for Both Humans and Agents

    Learn how Alpic designed its CLI for both human developers and AI agents — covering tradeoffs like polling, context windows, interactivity, and statelessness.

    J
    Julien Vallini

    Stay up to date

    Get the latest DeepSeek prompts, rules, and resources delivered to your inbox weekly.

    Neura Market LogoNeura Market

    Discover the best AI prompts, plugins, and resources for DeepSeek and more.

    Content Types

    • Rules
    • Prompts
    • MCPs
    • Agents
    • Guides

    Platforms

    • ChatGPT Directory
    • Claude Directory
    • Gemini Directory
    • Cursor Directory
    • Grok Directory
    • Perplexity Directory
    • DeepSeek Directory
    • CoPilot Directory
    • Stable Diffusion Directory
    • Midjourney Directory
    • All Directories

    Resources

    • Blog
    • Documentation
    • Help Center
    • Marketplace

    Legal

    • Privacy Policy
    • Terms of Service

    © 2026 Neura Market. All rights reserved.

    |

    Not affiliated with any AI platform vendors.