-
Notifications
You must be signed in to change notification settings - Fork 7.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat(joshwcomeau): add new Routes josh w comeau #18071
base: master
Are you sure you want to change the base?
Conversation
Successfully generated as following: http://localhost:1200/joshwcomeau/latest/javascript - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>JavaScript | Articles and Tutorials | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com/javascript</link>
<atom:link href="http://localhost:1200/joshwcomeau/latest/javascript" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on JavaScript | - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Wed, 08 Jan 2025 07:00:28 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>Promises From The Ground Up</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/javascript/promises/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/javascript/promises/#why-would-they-design-it-this-way-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Why would they design it this way??</a><a href="https://www.joshwcomeau.com/javascript/promises/#callbacks-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Callbacks</a><a href="https://www.joshwcomeau.com/javascript/promises/#introducing-promises-3" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Introducing Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#working-with-promises-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Working with Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#creating-our-own-promises-5" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Creating our own Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#chaining-promises-6" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Chaining Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#passing-data-7" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Passing data</a><a href="https://www.joshwcomeau.com/javascript/promises/#rejected-promises-8" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Rejected Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#async--await-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Async / Await</a><a href="https://www.joshwcomeau.com/javascript/promises/#more-to-come-10" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">More to come!</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">There are a lot of speed bumps and potholes on the road to JavaScript proficiency. One of the biggest and most daunting is <em style="color:inherit" class="w10oesj0">Promises.</em></p>
<p class="prg881n">In order to understand Promises, we need a surprisingly deep understanding of how JavaScript works and what its limitations are. Without that context, Promises won’t really make much sense.</p>
<p class="prg881n">It can be frustrating because the Promises API is <i>so important</i> nowadays. It’s become the de facto way of working with asynchronous code. Modern web APIs are built on top of Promises. There’s no getting around it: if we want to be productive with JavaScript, it really helps to understand Promises.</p>
<p class="prg881n">So, in this tutorial, we’re going to learn about Promises, but we’ll start at the beginning. I’ll share all of the critical bits of context that took me years to understand. And by the end, hopefully, you’ll have a much deeper understanding of what Promises are and how to use them effectively. ✨</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-intended-audience" class="t1xqw7pl">Intended audience</strong><div class="c690lmu"><p class="prg881n">This blog post is intended for beginner-to-intermediate JavaScript developers. Some knowledge of basic JavaScript syntax is assumed.</p></div></aside>
<h2 id="why-would-they-design-it-this-way" data-element-type="ContentHeading" class="hrk0sy8"><a id="why-would-they-design-it-this-way-1" href="https://www.joshwcomeau.com/javascript/promises/#why-would-they-design-it-this-way-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Why would they design it this way??</h2>
<p class="prg881n">Suppose we wanted to build a <em style="color:inherit" class="w10oesj0">Happy New Year!</em> countdown, something like this:</p>
<!-- --><div class="w52b1qq"><p class="v793fuv">––</p><button data-disabled="false" class="b1b5r9ns dj4vswd">Start Countdown</button></div>
<p class="prg881n">If JavaScript was like most other programming languages, we could solve the problem like this:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA">function</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> newYearsCountdown</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">()</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">3</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">2</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">1</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">Happy New Year! 🎉</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">}</span></span></code></pre></div></div>
<p class="prg881n">In this hypothetical code snippet, the program would pause when it hits a <code class="i165vvr1">sleep()</code> call, and then resume after the specified amount of time has passed.</p>
<p class="prg881n">Unfortunately, there is no <code class="i165vvr1">sleep</code> function in JavaScript, because it’s a <em style="color:inherit" class="w10oesj0">single-threaded</em> language.<button class="t1cz8cv8"><span class="a1ni5s6f">*</span><span class="e2cu4f7"></span></button><span data-use-adaptive-colors="true" id=":R1gneiavfelb:" data-direction="n" role="tooltip" style="flex-direction:column" class="p1lgzcrp l17lpp8k"><span class="p833mxt"><span class="pj0hw36"><span class="cffclov">Technically, modern JavaScript has access to multiple threads via Web Workers, but those extra threads don't have access to the DOM, so they can’t really be used in most situations.</span></span></span><svg xmlns="http://www.w3.org/2000/svg" width="24" height="6" fill="none" viewBox="0 0 24 6" style="--origin-y:0%" class="s1vq0yeh s1k9z937"><path d="
M 0 0
C 6 0
7.199999999999999 6
12 6
C 16.8 6
18 0
24 0
Z
"></path></svg></span> A “thread” is a long-running process that executes code. JavaScript only has one thread, and so it can only do one thing at a time. It can’t multitask. This is a problem because if our lone JavaScript thread is busy managing this countdown timer, it can’t do anything <i>else</i>.</p>
<p class="prg881n">When I was first learning about this stuff, it wasn’t immediately obvious to me why this was a problem. If the countdown timer is the only thing happening right now, isn’t it fine if the JS thread was fully occupied during that time??</p>
<p class="prg881n">Well, even though JavaScript doesn’t have a <code class="i165vvr1">sleep</code> function, it <i>does</i> have some other functions that occupy the main thread for an extended amount of time. We can use those other methods to get a glimpse into what it would be like if JavaScript had a <code class="i165vvr1">sleep</code> function.</p>
<p class="prg881n">For example, <code class="i165vvr1">window.prompt()</code>. This function is used to gather information from the user, and it halts execution of our code much like our hypothetical <code class="i165vvr1">sleep()</code> function would.</p>
<p class="prg881n">Click the button in this playground, and then <strong>try to interact with the page</strong> while the prompt is open:</p>
<!-- --><div data-stacked="false" class="il1gxx9"><div style="--color-text:hsl(210deg 10% 90%);--color-background:hsl(210deg 15% 6%);--color-blurred-background:hsl(210deg 15% 6% / 0.75);--color-muted-background:hsl(210deg 38% 15% / 0.85);--color-action:hsl(240deg 95% 62%);--color-primary:hsl(225deg 100% 75%);--color-secondary:hsl(333deg 100% 55%);--color-tertiary:hsl(280deg 100% 85%);--color-decorative:hsl(200deg 50% 60%);--color-info-100:hsl(214deg 40% 15%);--color-info-200:hsl(218deg 32% 20%);--color-info-300:hsl(218deg 30% 25%);--color-info-400:hsl(218deg 45% 40%);--color-info-500:hsl(225deg 100% 60%);--color-info-700:hsl(215deg 100% 72%);--color-warning-100:hsl(30deg 25% 11%);--color-warning-200:hsl(32deg 20% 15%);--color-warning-300:hsl(32deg 25% 20%);--color-warning-400:hsl(35deg 45% 35%);--color-warning-500:hsl(40deg 100% 50%);--color-warning-700:hsl(43deg 100% 72%);--color-success-100:hsl(176deg 35% 10%);--color-success-200:hsl(176deg 26% 14%);--color-success-300:hsl(176deg 28% 20%);--color-success-400:hsl(176deg 45% 30%);--color-success-500:hsl(160deg 100% 40%);--color-success-700:hsl(160deg 80% 65%);--color-cloud-100:hsl(210deg 15% 6%);--color-cloud-300:hsl(212deg 40% 9%);--color-cloud-400:hsl(213deg 40% 10%);--color-cloud-500:hsl(213deg 40% 12%);--color-sky-from:hsl(214deg 40% 11%);--color-sky-to:hsl(200deg 50% 30%);--color-sky-subtle:hsl(210deg 40% 16%);--color-gray-50:hsl(210deg 19% 9%);--color-gray-100:hsl(210deg 15% 12%);--color-gray-200:hsl(210deg 15% 18%);--color-gray-300:hsl(210deg 10% 30%);--color-gray-400:hsl(210deg 9% 40%);--color-gray-500:hsl(210deg 8% 50%);--color-gray-600:hsl(210deg 12% 55%);--color-gray-700:hsl(210deg 14% 66%);--color-gray-800:hsl(210deg 20% 77%);--color-gray-900:hsl(210deg 25% 88%);--color-gray-1000:hsl(210deg 25% 96%);--color-adaptive-white:hsl(210deg 25% 92%);--syntax-bg:hsl(210deg 15% 6%);--syntax-highlight:hsl(210deg 30% 18%);--syntax-txt:hsl(0deg 0% 100%);--syntax-comment:hsl(200deg 18% 51%);--syntax-prop:hsl(326deg 100% 61%);--syntax-bool:hsl(50deg 100% 50%);--syntax-val:hsl(210deg 12% 65%);--syntax-str:hsl(259deg 100% 71%);--syntax-name:hsl(280deg 100% 66%);--syntax-del:hsl(0deg 100% 67%);--syntax-regex:hsl(50deg 100% 50%);--syntax-fn:hsl(195deg 100% 50%);--color-info:hsl(225deg 100% 80%);--color-info-background:hsl(225deg 100% 80% / 0.1);--color-error:hsl(340deg 95% 60%);--color-error-background:hsl(340deg 95% 43% / 0.1);--color-success:hsl(160deg 100% 40%);--color-success-background:hsl(160deg 100% 40% / 0.1);--color-alert:hsl(40deg 100% 50%);--color-alert-background:hsl(40deg 100% 50% / 0.1);--color-page-background:hsl(210deg 15% 6%);--color-page-primary:hsl(225deg 100% 75%);--kbd-background-color:hsl(210deg 9% 40%);--kbd-border-color:hsl(210deg 10% 30%);--color-code-bg:hsl(210deg 15% 12%);--color-content-outline:hsl(210deg 10% 30%)" class="full-bleed i1j6zvew s1hfctpl d16kq653"><header class="w14j10eb"><p class="tkoi9ui">Code Playground</p><div class="au0cxa1"><button class="wsn2i55 a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7;overflow:visible" class="s1gvsyfu"><g style="transform:rotate(-45deg)" class="w15rf7yg"><rect x="0" y="8" width="24" height="6" rx="2" fill="var(--color-gray-100)"></rect><line x1="18" y1="8" x2="18" y2="14"></line></g></svg><span class="wlgbkjj">Format code using Prettier</span></button><abbr style="display:inline-block"><div class="wy3w6yt"><button title="Reset code" class="a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7" class="s1gvsyfu"><mask id="skip-icon-mask-:R39h1eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="black"></rect><rect x="9" y="0" width="16" height="24" fill="white"></rect></mask><line x1="5" y1="19" x2="5" y2="5"></line><polygon points="19 20 9 12 19 4 19 20" mask="url(#skip-icon-mask-:R39h1eiavfelb:)"></polygon></svg><span class="wlgbkjj">Reset Code</span></button><div class="pszdhfz"><span style="gap:1.5px" class="w15sqd9b"><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span></span></div></div></abbr></div></header><div></div><div class="w198elmk"><div class="c1jztmaw"><div style="flex:547.4" class="f15xiyga"><div style="min-height:100%" class="whrzwqb"><div class="h1nlnw7n"><p class="tbdao9h">HTML</p></div><div in="false" style="--max-height:80vh" class="w1bgwmtf"><button class="f9ibg9h"><span class="wlgbkjj">Focus the editor. This will trap focus until you press Escape.</span></button><label><span class="wlgbkjj">Code editor:</span><div translate="no" class="c18okpq0" style="position:relative;text-align:left;box-sizing:border-box;padding:0;overflow:hidden"><pre aria-hidden="true" style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:relative;pointer-events:none;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0"><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">script</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">function</span><span class="token script language-javascript"> </span><span class="token script language-javascript function" style="color:var(--syntax-fn)">askForName</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript"> </span><span class="token script language-javascript punctuation">{</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">const</span><span class="token script language-javascript"> name </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript dom variable" style="color:var(--syntax-str)">window</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript method function property-access" style="color:var(--syntax-fn)">prompt</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">'What is your name?'</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript punctuation">;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">const</span><span class="token script language-javascript"> elem </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript dom variable" style="color:var(--syntax-str)">document</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript method function property-access" style="color:var(--syntax-fn)">querySelector</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">'#greeting'</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript punctuation">;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> elem</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript property-access">innerText</span><span class="token script language-javascript"> </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript template-string template-punctuation string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">`</span><span class="token script language-javascript template-string string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">Hi </span><span class="token script language-javascript template-string interpolation interpolation-punctuation punctuation">${</span><span class="token script language-javascript template-string interpolation">name</span><span class="token script language-javascript template-string interpolation interpolation-punctuation punctuation">}</span><span class="token script language-javascript template-string string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">!</span><span class="token script language-javascript template-string template-punctuation string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">`</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript punctuation">}</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">script</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain" style="display:inline-block">
</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">button</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag special-attr attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">onclick</span><span class="token tag special-attr attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag special-attr attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag special-attr attr-value value javascript language-javascript function" style="color:var(--syntax-fn);font-weight:var(--font-weight-medium)">askForName</span><span class="token tag special-attr attr-value value javascript language-javascript punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">(</span><span class="token tag special-attr attr-value value javascript language-javascript punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">)</span><span class="token tag special-attr attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"> Click me</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">button</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">id</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">greeting</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span></div></pre><textarea style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:absolute;top:0;left:0;height:100%;width:100%;resize:none;color:inherit;overflow:hidden;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;-webkit-text-fill-color:transparent;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0" class="npm__react-simple-code-editor__textarea" autocapitalize="off" autocomplete="off" autocorrect="off" spellcheck="false" data-gramm="false">&lt;script&gt;
function askForName() {
const name = window.prompt('What is your name?');
const elem = document.querySelector('#greeting');
elem.innerText = `Hi ${name}!`
}
&lt;/script&gt;
&lt;button onclick="askForName()"&gt;
Click me
&lt;/button&gt;
&lt;div id="greeting"&gt;&lt;/div&gt;</textarea><style>
/**
* Reset the text fill color so that placeholder is visible
*/
.npm__react-simple-code-editor__textarea:empty {
-webkit-text-fill-color: inherit !important;
}
/**
* Hack to apply on some CSS on IE10 and IE11
*/
@media all and (-ms-high-contrast: none), (-ms-high-contrast: active) {
/**
* IE doesn't support '-webkit-text-fill-color'
* So we use 'color: transparent' to make the text transparent on IE
* Unlike other browsers, it doesn't affect caret color in IE
*/
.npm__react-simple-code-editor__textarea {
color: transparent !important;
}
.npm__react-simple-code-editor__textarea::selection {
background-color: #accef7 !important;
color: transparent !important;
}
}
</style></div></label></div></div></div><button class="dijulrw"><span class="wlgbkjj">Resize editor. Use left/right arrows.</span></button><div style="flex:234.60000000000002" class="s1c0ceq0"><div style="height:100%" class="whrzwqb"><div class="h1nlnw7n"><p class="tbdao9h">Result</p><abbr title="Refresh pane"><button style="transform:rotate(0deg);opacity:0.7" class="w11udlg3 ah084m5"><svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-refresh-cw"><path d="M3 12a9 9 0 0 1 9-9 9.75 9.75 0 0 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path><path d="M21 12a9 9 0 0 1-9 9 9.75 9.75 0 0 1-6.74-2.74L3 16"></path><path d="M8 16H3v5"></path></svg><span class="wlgbkjj">Refresh results pane</span></button></abbr></div><div style="--height:100%;--flex:1" class="wqqdb70"><iframe height="100%" width="100%" title="example" frameborder="0" loading="lazy" class="fxdlgbu" referrerpolicy="no-referrer"></iframe></div></div></div></div></div></div></div>
<p class="prg881n"><strong>Notice that while the prompt is open, the page is totally unresponsive?</strong> You can't scroll, click any links, or select any text! The JavaScript thread is busy waiting for us to provide a value so that it can finish running that code. While it’s waiting, it can’t do anything <i>else</i>, and so the browser locks down the UI.</p>
<p class="prg881n">Other languages have multiple threads, and so it's no big deal if one of them gets preoccupied for a while. In JavaScript, though, we only have the one, and it’s used for everything: handling events, managing network requests, updating the UI, etc.</p>
<p class="prg881n">If we want to create a countdown, we need to find a way to do it without blocking the thread.</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-but-why-is-the-entire-ui-frozen" class="t1xqw7pl">But why is the entire UI frozen??</strong><div class="c690lmu"><p class="prg881n">In the example above with <code class="i165vvr1">window.prompt()</code>, the entire UI becomes unresponsive while the browser waits for us to provide a value.</p><p class="prg881n">This is kinda strange… the browser doesn’t rely on JavaScript to scroll the page, or to select text. So why can’t we do any of those things?</p><p class="prg881n">I think browsers work this way to prevent bugs. Scrolling the page, for example, triggers “scroll” events which can be caught and handled with JavaScript. If the JS thread is occupied while the scroll event happens, that code never runs, which could lead to bugs if the developer assumed that scroll events would always be handled.</p><p class="prg881n">It could also be a UX thing; maybe the browser disables the UI so that the user can’t ignore the prompt. Either way, though, I suspect a native <code class="i165vvr1">sleep</code> function would need to work the same way to prevent bugs.</p></div></aside>
<h2 id="callbacks" data-element-type="ContentHeading" class="hrk0sy8"><a id="callbacks-2" href="https://www.joshwcomeau.com/javascript/promises/#callbacks-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Callbacks</h2>
<p class="prg881n">The main tool in our toolbox for solving these sorts of problems is <code class="i165vvr1">setTimeout</code>. <code class="i165vvr1">setTimeout</code> is a function which accepts two arguments:</p>
<ol class="oc96fnv">
<li class="wp18jjy"><div class="l1n7gg9f">A chunk of work to do, at some point in the future.</div></li>
<li class="wp18jjy"><div class="l1n7gg9f">The amount of time to wait for.</div></li>
</ol>
<p class="prg881n">Here's an example:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">Start</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">setTimeout</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> ()</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA"> =&gt;</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF"> console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">After one second</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> },</span></span>
<span class="line"><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF"> 1000</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span></code></pre></div></div>
<p class="prg881n">The chunk of work is passed in through a function. This pattern is known as a <em style="color:inherit" class="w10oesj0">callback.</em></p>
<p class="prg881n">The hypothetical <code class="i165vvr1">sleep()</code> function we saw before is like calling a company and waiting on hold for the next available representative. <code class="i165vvr1">setTimeout()</code> is like pressing <i>1</i> to have them <em style="color:inherit" class="w10oesj0">call you back</em> when the representative is available. You can hang up the phone and get on with your life.</p>
<p class="prg881n"><code class="i165vvr1">setTimeout()</code> is known as an <em style="color:inherit" class="w10oesj0">asynchronous</em> function. This means that it doesn’t block the thread. By contrast, <code class="i165vvr1">window.prompt()</code> is <em style="color:inherit" class="w10oesj0">synchronous</em>, because the JavaScript thread can't do anything else while it’s waiting.</p>
<p class="prg881n">The big downside with asynchronous code is that it means our code won't always run in a linear order. Consider the following setup:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">1. Before setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">setTimeout</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">()</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA"> =&gt;</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF"> console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">2. Inside setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">},</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF"> 500</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">3. After setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span></code></pre></div></div>
<p class="prg881n">You might expect these logs to fire in order from top to bottom: <code class="i165vvr1">1</code> &gt; <code class="i165vvr1">2</code> &gt; <code class="i165vvr1">3</code>. <strong>But remember, the whole idea with callbacks is that we’re scheduling a call back.</strong> The JavaScript thread doesn’t sit around and ... |
http://localhost:1200/joshwcomeau/latest - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>Articles and Tutorials | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com</link>
<atom:link href="http://localhost:1200/joshwcomeau/latest" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on React, CSS, Animation, and more! - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Wed, 08 Jan 2025 07:00:30 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>Next-level frosted glass with backdrop-filter</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/css/backdrop-filter/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#css-filters-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">CSS filters</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#the-issue-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">The Issue</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#pointer-events-3" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Pointer events</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#flickering-top-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Flickering top</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#thicker-glass-5" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Thicker glass</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#browser-support-6" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Browser support</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#glassy-edge-7" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Glassy edge</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#the-final-code-8" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">The final code</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#continue-learning-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Continue learning</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">One of my all-time favourite CSS tricks is using <code class="i165vvr1">backdrop-filter: blur()</code> to create a frosted glass effect. I use it in just about every project I work on, including this blog!</p>
<p class="prg881n">Here’s a quick demo, to show what I’m talking about:</p>
<!-- --><div class="o11h7o12 w1bd8cgi" style="overflow:hidden"><div class="hduekzx"><div class="t1xdhus0"><div class="t1vhg4x9"><div class="rptyea t8ndwpz"></div><div class="y1qao05t t8ndwpz"></div><div class="g1v2ob21 t8ndwpz"></div></div><div class="tj6926w"><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="lep27c6 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(0deg);transform-origin:center center"></path></svg><span>Some Example Website</span><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="r2mz821 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(90deg);transform-origin:center center"></path></svg></div></div><div class="a13tcou3"><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-left"><path d="m12 19-7-7 7-7"></path><path d="M19 12H5"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-right"><path d="M5 12h14"></path><path d="m12 5 7 7-7 7"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-rotate-cw"><path d="M21 12a9 9 0 1 1-9-9c2.52 0 4.93 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path></svg></div><div class="a1booqr"><span>https://www.joshwcomeau.com/example-website</span></div></div></div><div style="overflow:clip"><div class="wi9xlz4 s1hfctpl d16kq653"><div class="hpv2kfm"><div class="b1dh7eg0"></div><div class="bi95upp"></div><p class="l19j3i1d">Some Website</p></div><div class="cufzfud"><img src="https://www.joshwcomeau.com/images/backdrop-filter/cupcake.png" style="animation-play-state:paused" class="c19tvtvc" referrerpolicy="no-referrer"><p>This is an example website showing how I typically use<!-- --> <code class="i165vvr1">backdrop-filter</code> to create glassy headers.</p><p class="o1fnc03u">Notice that as the cupcake moves behind the header, it appears blurry, as it would if it was passing behind frosted glass.</p><div class="atopy0n"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1t96eke w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><span style="width:20px;height:20px;--duration:1422.2222222222222ms;--size:20px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div></div></div>
<p class="prg881n">This effect helps us add depth and realism to our projects. It’s lovely.</p>
<p class="prg881n"><strong>But when I see this effect in the wild, it’s almost always missing some crucial optimizations.</strong> A couple of small changes can make our frosted glass <em style="color:inherit" class="w10oesj0">so</em> much more lush and realistic!</p>
<p class="prg881n">In this post, you’ll learn how to make the slickest frosted glass ever ✨. We’ll also learn quite a bit about CSS filters along the way!</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-credit-where-credit-is-due" class="t1xqw7pl">Credit where credit is due</strong><div class="c690lmu"><p class="prg881n">I learned about this effect from <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://expensive.toys/">Artur<!-- --> <span class="p7a6phg">Bien’s<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R2qm8deiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R2qm8deiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a> incredible demos. He credits <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://www.jamiegray.net/">Jamie<!-- --> <span class="p7a6phg">Gray<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R2sm8deiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R2sm8deiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a> for the original concept.</p></div></aside>
<h2 id="css-filters" data-element-type="ContentHeading" class="hrk0sy8"><a id="css-filters-1" href="https://www.joshwcomeau.com/css/backdrop-filter/#css-filters-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>CSS filters</h2>
<p class="prg881n">To briefly explain the underlying concept: CSS gives us quick and easy access to SVG filters via the <code class="i165vvr1">filter</code> property.</p>
<p class="prg881n">For example, we can give elements a Gaussian blur with <code class="i165vvr1">filter: blur()</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="wskxkqa"><img alt="My 3D mascot smiling with his tongue out. It’s very blurry, but dragging the slider makes it clearer" src="https://www.joshwcomeau.com/images/josh-originals/josh-tongue-dark.png" loading="lazy" style="filter:blur(16px)" class="j5sqeuc" referrerpolicy="no-referrer"><div class="c162apj7"></div></div><div style="--template-cols:1fr 1fr;grid-template-columns:1fr" class="c1j2qstn"><div style="max-width:400px;width:100%;margin-inline:auto" class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgleiavfelb:_radius" class="lpsyt9t">Blur radius<!-- -->:</label><span class="w1194iyg">16px</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgleiavfelb:_radius" min="0" max="16" class="i1i8zjy7" value="16"></div></div></div></div></div>
<p class="prg881n">There are lots of fun filter options, the sorts of things you’d find in image-editing software. Like, rotating the hue of all the colors:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1d4uhch"><img alt="My 3D mascot making a puzzled face. Dragging the slider causes him to become unnatural shades of green and pink." src="https://www.joshwcomeau.com/images/newsletter/joy-of-react-mascot.png" loading="lazy" style="filter:hue-rotate(0deg)" class="m1wtl9k0" referrerpolicy="no-referrer"><div class="c1kud80y"></div></div><div style="--template-cols:1fr 1fr;grid-template-columns:1fr" class="c1j2qstn"><div style="max-width:400px;width:100%;margin-inline:auto" class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgpeiavfelb:_rotation" class="lpsyt9t">Hue rotation<!-- -->:</label><span class="w1194iyg">0deg</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgpeiavfelb:_rotation" min="0" max="360" class="i1i8zjy7" value="0"></div></div></div></div></div>
<p class="prg881n">In these examples, I’m applying the filters to an <code class="i165vvr1">&lt;img&gt;</code> tag, but we can apply them to standard DOM nodes as well:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1icc7dx"><div style="filter:blur(0px) sepia(0%)" class="c1l5p6hh"><p>Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged.</p></div><div class="c1koe0ch"></div></div><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_radius" class="lpsyt9t">Blur radius<!-- -->:</label><span class="w1194iyg">0px</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_radius" min="0" max="16" step="0.1" class="i1i8zjy7" value="0"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_sepia" class="lpsyt9t">Sepia percentage<!-- -->:</label><span class="w1194iyg">0%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_sepia" min="0" max="100" class="i1i8zjy7" value="0"></div></div></div></div></div>
<p class="prg881n">Pretty neat, right?</p>
<p class="prg881n">Things get even cooler with <code class="i165vvr1">backdrop-filter</code>. This property lets us apply these same filters to the stuff <em style="color:inherit" class="w10oesj0">behind</em> a given element.</p>
<p class="prg881n">For example:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1qszusq"><div class="isvxvbh"><img src="https://www.joshwcomeau.com/images/backdrop-filter/tokyo.jpg" loading="lazy" class="tvcyp0e" referrerpolicy="no-referrer"><div style="top:80%;left:50%;--scale:2;-webkit-backdrop-filter:sepia(50%) brightness(130%);backdrop-filter:sepia(50%) brightness(130%)" class="ctfyewy"></div></div><div class="ck6v2m7"></div></div><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rh5eiavfelb:_sepia" class="lpsyt9t">Sepia<!-- -->:</label><span class="w1194iyg">50%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rh5eiavfelb:_sepia" min="0" max="100" class="i1i8zjy7" value="50"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rh5eiavfelb:_brightness" class="lpsyt9t">Brightness<!-- -->:</label><span class="w1194iyg">130%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rh5eiavfelb:_brightness" min="50" max="300" class="i1i8zjy7" value="130"></div></div></div></div></div>
<p class="prg881n">In this demo, the <code class="i165vvr1">.magic-ring</code> element sits in front of a photo (<a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://unsplash.com/photos/assorted-signages-on-buildings-nTBW1cOY1qI"> <span class="p7a6phg">source<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:Rn17eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:Rn17eiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a>). It uses the <code class="i165vvr1">backdrop-filter</code> property to apply some filtering to everything behind it, which can be used for some pretty artistic effects.</p>
<p class="prg881n">In practice, I pretty much only use <code class="i165vvr1">backdrop-filter</code> for one use case: blurring everything behind an element, usually a header, to create the “frosted glass” effect I mentioned earlier:</p>
<!-- --><div class="o11h7o12 w1bd8cgi" style="overflow:hidden"><div class="hduekzx"><div class="t1xdhus0"><div class="t1vhg4x9"><div class="rptyea t8ndwpz"></div><div class="y1qao05t t8ndwpz"></div><div class="g1v2ob21 t8ndwpz"></div></div><div class="tj6926w"><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="lep27c6 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(0deg);transform-origin:center center"></path></svg><span>Some Example Website</span><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="r2mz821 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(90deg);transform-origin:center center"></path></svg></div></div><div class="a13tcou3"><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-left"><path d="m12 19-7-7 7-7"></path><path d="M19 12H5"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-right"><path d="M5 12h14"></path><path d="m12 5 7 7-7 7"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-rotate-cw"><path d="M21 12a9 9 0 1 1-9-9c2.52 0 4.93 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path></svg></div><div class="a1booqr"><span>https://www.joshwcomeau.com/example-website</span></div></div></div><div style="overflow:clip"><div class="wi9xlz4 s1hfctpl d16kq653"><div class="hpv2kfm"><div class="b1dh7eg0"></div><div class="bi95upp"></div><p class="l19j3i1d">Some Website</p></div><div class="cufzfud"><img src="https://www.joshwcomeau.com/images/backdrop-filter/cupcake.png" style="animation-play-state:paused" class="c19tvtvc" referrerpolicy="no-referrer"><p>This is an example website showing how I typically use<!-- --> <code class="i165vvr1">backdrop-filter</code> to create glassy headers.</p><p class="o1fnc03u">Notice that as the cupcake moves behind the header, it appears blurry, as it would if it was passing behind frosted glass.</p><div class="atopy0n"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1t96eke w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><span style="width:20px;height:20px;--duration:1422.2222222222222ms;--size:20px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div></div></div>
<p class="prg881n">Alright. Let’s talk about the thing most developers miss.</p>
<h2 id="the-issue" data-element-type="ContentHeading" class="hrk0sy8"><a id="the-issue-2" href="https://www.joshwcomeau.com/css/backdrop-filter/#the-issue-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>The Issue</h2>
<p class="prg881n"><strong>Here’s the problem:</strong> The <code class="i165vvr1">backdrop-filter</code> algorithm only considers the pixels that are <i>directly behind</i> the element.</p>
<p class="prg881n">For filters like <code class="i165vvr1">brightness</code> or <code class="i165vvr1">hue-rotate</code>, that makes perfect sense. With blur, though, we actually want to consider pixels that are <em style="color:inherit" class="w10oesj0">near</em> the element too.</p>
<p class="prg881n">This is one of those things where a demo is worth a thousand words. Check out the difference:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="i1df5wci s360ott"><div id=":Rhneiavfelb:_mode" class="w73h7tw"><div class="r1cct0ab"><button value="default" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">Default</span></button><button value="fixed" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">Tweaked</span></button></div></div></div></div><div class="wr5us8n s1hfctpl d16kq653"><div class="hhqru8"><div class="g8qz4zc"></div></div><div style="animation-play-state:running" class="b1uj7lhw"></div></div><div class="awqze7l"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1rsl36b w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-pause"><rect x="14" y="4" width="4" height="16" rx="1"></rect><rect x="6" y="4" width="4" height="16" rx="1"></rect></svg></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div>
<p class="prg881n">By default, the gaussian blur algorithm is applied to all of the pixels behind the element. This means that if a big colorful element is <em style="color:inherit" class="w10oesj0">near</em> the element, it won’t have any effect.</p>
<p class="prg881n">That’s not really how frosted glass works in real life though. Light bounces off of objects and then goes through the glass. It looks <em style="color:inherit" class="w10oesj0">so much better</em> when the blurring algorithm includes nearby content.</p>
<p class="prg881n">Unfortunately, this isn’t something we can configure directly. Instead, we need to be a bit crafty.</p>
<p class="prg881n">Here’s the code:</p>
<!-- --><div data-stacked="false" class="il1gxx9"><div style="--color-text:hsl(210deg 10% 90%);--color-background:hsl(210deg 15% 6%);--color-blurred-background:hsl(210deg 15% 6% / 0.75);--color-muted-background:hsl(210deg 38% 15% / 0.85);--color-action:hsl(240deg 95% 62%);--color-primary:hsl(225deg 100% 75%);--color-secondary:hsl(333deg 100% 55%);--color-tertiary:hsl(280deg 100% 85%);--color-decorative:hsl(200deg 50% 60%);--color-info-100:hsl(214deg 40% 15%);--color-info-200:hsl(218deg 32% 20%);--color-info-300:hsl(218deg 30% 25%);--color-info-400:hsl(218deg 45% 40%);--color-info-500:hsl(225deg 100% 60%);--color-info-700:hsl(215deg 100% 72%);--color-warning-100:hsl(30deg 25% 11%);--color-warning-200:hsl(32deg 20% 15%);--color-warning-300:hsl(32deg 25% 20%);--color-warning-400:hsl(35deg 45% 35%);--color-warning-500:hsl(40deg 100% 50%);--color-warning-700:hsl(43deg 100% 72%);--color-success-100:hsl(176deg 35% 10%);--color-success-200:hsl(176deg 26% 14%);--color-success-300:hsl(176deg 28% 20%);--color-success-400:hsl(176deg 45% 30%);--color-success-500:hsl(160deg 100% 40%);--color-success-700:hsl(160deg 80% 65%);--color-cloud-100:hsl(210deg 15% 6%);--color-cloud-300:hsl(212deg 40% 9%);--color-cloud-400:hsl(213deg 40% 10%);--color-cloud-500:hsl(213deg 40% 12%);--color-sky-from:hsl(214deg 40% 11%);--color-sky-to:hsl(200deg 50% 30%);--color-sky-subtle:hsl(210deg 40% 16%);--color-gray-50:hsl(210deg 19% 9%);--color-gray-100:hsl(210deg 15% 12%);--color-gray-200:hsl(210deg 15% 18%);--color-gray-300:hsl(210deg 10% 30%);--color-gray-400:hsl(210deg 9% 40%);--color-gray-500:hsl(210deg 8% 50%);--color-gray-600:hsl(210deg 12% 55%);--color-gray-700:hsl(210deg 14% 66%);--color-gray-800:hsl(210deg 20% 77%);--color-gray-900:hsl(210deg 25% 88%);--color-gray-1000:hsl(210deg 25% 96%);--color-adaptive-white:hsl(210deg 25% 92%);--syntax-bg:hsl(210deg 15% 6%);--syntax-highlight:hsl(210deg 30% 18%);--syntax-txt:hsl(0deg 0% 100%);--syntax-comment:hsl(200deg 18% 51%);--syntax-prop:hsl(326deg 100% 61%);--syntax-bool:hsl(50deg 100% 50%);--syntax-val:hsl(210deg 12% 65%);--syntax-str:hsl(259deg 100% 71%);--syntax-name:hsl(280deg 100% 66%);--syntax-del:hsl(0deg 100% 67%);--syntax-regex:hsl(50deg 100% 50%);--syntax-fn:hsl(195deg 100% 50%);--color-info:hsl(225deg 100% 80%);--color-info-background:hsl(225deg 100% 80% / 0.1);--color-error:hsl(340deg 95% 60%);--color-error-background:hsl(340deg 95% 43% / 0.1);--color-success:hsl(160deg 100% 40%);--color-success-background:hsl(160deg 100% 40% / 0.1);--color-alert:hsl(40deg 100% 50%);--color-alert-background:hsl(40deg 100% 50% / 0.1);--color-page-background:hsl(210deg 15% 6%);--color-page-primary:hsl(225deg 100% 75%);--kbd-background-color:hsl(210deg 9% 40%);--kbd-border-color:hsl(210deg 10% 30%);--color-code-bg:hsl(210deg 15% 12%);--color-content-outline:hsl(210deg 10% 30%)" class="full-bleed i1j6zvew s1hfctpl d16kq653"><header class="w14j10eb"><p class="tkoi9ui">Code Playground</p><div class="au0cxa1"><button class="wsn2i55 a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7;overflow:visible" class="s1gvsyfu"><g style="transform:rotate(-45deg)" class="w15rf7yg"><rect x="0" y="8" width="24" height="6" rx="2" fill="var(--color-gray-100)"></rect><line x1="18" y1="8" x2="18" y2="14"></line></g></svg><span class="wlgbkjj">Format code using Prettier</span></button><abbr style="display:inline-block"><div class="wy3w6yt"><button title="Reset code" class="a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7" class="s1gvsyfu"><mask id="skip-icon-mask-:R39i1eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="black"></rect><rect x="9" y="0" width="16" height="24" fill="white"></rect></mask><line x1="5" y1="19" x2="5" y2="5"></line><polygon points="19 20 9 12 19 4 19 20" mask="url(#skip-icon-mask-:R39i1eiavfelb:)"></polygon></svg><span class="wlgbkjj">Reset Code</span></button><div class="pszdhfz"><span style="gap:1.5px" class="w15sqd9b"><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span></span></div></div></abbr></div></header><div></div><div class="w198elmk"><div class="c1jztmaw"><div style="flex:516.12" class="f15xiyga"><div class="w16o0eiw"><div class="tt2ystl"><button style="--weight:var(--font-weight-bold);--color:var(--color-text)" class="ts2dciv">HTML</button><button style="--weight:var(--font-weight-normal);--color:var(--color-gray-500)" class="ts2dciv">CSS</button></div><div in="false" style="--max-height:80vh" class="w1bgwmtf"><button class="f9ibg9h"><span class="wlgbkjj">Focus the editor. This will trap focus until you press Escape.</span></button><label><span class="wlgbkjj">Code editor:</span><div translate="no" class="c18okpq0" style="position:relative;text-align:left;box-sizing:border-box;padding:0;overflow:hidden"><pre aria-hidden="true" style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:relative;pointer-events:none;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0"><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">style</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css selector" style="color:var(--syntax-name)">header</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">{</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">position</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> relative</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">}</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css selector class" style="color:var(--syntax-name)">.backdrop</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">{</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">position</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> absolute</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">inset</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">0</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">height</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">200</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">backdrop-filter</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css function" style="color:var(--syntax-fn)">blur</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">(</span><span class="token style language-css number" style="color:var(--syntax-bool)">16</span><span class="token style language-css unit" style="color:var(--syntax-str)">px</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">)</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">mask-image</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css function" style="color:var(--syntax-fn)">linear-gradient</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">(</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> to bottom</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">,</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css color" style="color:var(--syntax-str)">black</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">0</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">50</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">,</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css color" style="color:var(--syntax-str)">transparent</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">50</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">100</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">)</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">}</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">style</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain" style="display:inline-block">
</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">header</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"> </span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">class</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">backdrop</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">header</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">class</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">ball</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medi ... |
http://localhost:1200/joshwcomeau/popular/false - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>Popular Content | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com</link>
<atom:link href="http://localhost:1200/joshwcomeau/popular/false" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on React, CSS, Animation, and more! - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Wed, 08 Jan 2025 07:00:32 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>How To Center a Div</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/css/center-a-div/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with auto margins</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering within the viewport</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-elements-with-unknown-sizes-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering elements with unknown sizes</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-css-grid-5" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with CSS Grid</a><a href="https://www.joshwcomeau.com/css/center-a-div/#differences-from-flexbox-6" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Differences from Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-a-stack-of-elements-7" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering a stack of elements</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-text-8" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering text</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-in-the-future-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering in the future</a><a href="https://www.joshwcomeau.com/css/center-a-div/#going-beyond-the-patterns-10" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Going beyond the patterns</a><a href="https://www.joshwcomeau.com/css/center-a-div/#when-to-use-which-method-11" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">When to use which method</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">For a long time, centering an element within its parent was a surprisingly tricky thing to do. As CSS has evolved, we've been granted more and more tools we can use to solve this problem. These days, we're spoiled for choice!</p>
<p class="prg881n">I decided to create this tutorial to help you understand the trade-offs between different approaches, and to give you an arsenal of strategies you can use, to handle centering in all sorts of scenarios.</p>
<p class="prg881n">Honestly, this turned out to be <em style="color:inherit" class="w10oesj0">way more interesting</em> than I initially thought&nbsp;😅. Even if you've been using CSS for a while, I bet you'll learn at least 1 new strategy!</p>
<h2 id="centering-with-auto-margins" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-auto-margins-1" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with auto margins</h2>
<p class="prg881n">The first strategy we'll look at is one of the oldest. If we want to center an element horizontally, we can do so using margins set to the special value <code class="i165vvr1">auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgbeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgbeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-right</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-left:auto;margin-right:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgbeiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgbeiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">First, we need to constrain the element's width; by default, elements in Flow layout will expand horizontally to fill the available space, and we can't really center something that is full-width.</p>
<p class="prg881n">I <em style="color:inherit" class="w10oesj0">could</em> constrain the width with a fixed value (eg. <code class="i165vvr1">200px</code>), but really what I want in this case is for the element to shrinkwrap around its content. <code class="i165vvr1">fit-content</code> is a magical value that does exactly this. Essentially, it makes “width” behave like “height”, so that the element’s size is determined by its contents.</p>
<p class="prg881n"><strong>Why am I setting <code class="i165vvr1">max-width</code> instead of <code class="i165vvr1">width</code>?</strong> Well, my goal is to stop the element from expanding horizontally. I want to clamp its maximum size. If I used <code class="i165vvr1">width</code> instead, it would lock it to that size, and the element would overflow when the container is really narrow. If you drag that “Container Width” slider all the way to the left, you can see that the element shrinks with its container.</p>
<p class="prg881n">Now that our element is constrained, we can center it with <em style="color:inherit" class="w10oesj0">auto margins</em>.</p>
<p class="prg881n">I like to think of auto margins like <i>Hungry Hungry Hippos</i>. Each auto margin will try to gobble up as much space as possible. For example, check out what happens if we <em style="color:inherit" class="w10oesj0">only</em> set <code class="i165vvr1">margin-left: auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgneiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgneiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div></div><div style="margin-left:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgneiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgneiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">When <code class="i165vvr1">margin-left</code> is the only side with auto margins, <i>all</i> of the extra space gets applied as margin to that side. When we set both <code class="i165vvr1">margin-left: auto</code> <i>and</i> <code class="i165vvr1">margin-right: auto</code>, the two hippos each gobble up an equal amount of space. This forces the element to the center.</p>
<p class="prg881n"><em style="color:inherit" class="w10oesj0">Also:</em> I've been using <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> because they're familiar, but there's a better, more-modern way to do this:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line highlighted"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-inline</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-inline:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgteiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgteiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n"><code class="i165vvr1">margin-inline</code> will set both <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> to the same value (<code class="i165vvr1">auto</code>). It has <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://caniuse.com/mdn-css_properties_margin-inline">very good browser<!-- --> <span class="p7a6phg">support<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R1e8veiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R1e8veiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a>, having landed in all major browsers several years ago.</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-logical-properties" class="t1xqw7pl">Logical properties</strong><div class="c690lmu"><p class="prg881n"><code class="i165vvr1">margin-inline</code> is more than just a convenient shorthand for <code class="i165vvr1">margin-left</code> + <code class="i165vvr1">margin-right</code>. It's part of a collection of <em style="color:inherit" class="w10oesj0">logical properties</em>, designed to make it easier to internationalize the web.</p><p class="prg881n">In English, characters are written in a horizontal line from left to right. Those characters are composed into words and sentences, and assembled into “blocks” (paragraphs, headings, lists, etc). Blocks are stacked vertically, from top to bottom. We can think of this as the <i>orientation</i> of English-language websites.</p><p class="prg881n">This isn't universal, though! Some languages, like Arabic and Hebrew, are written from right to left. Other languages, like Chinese, have historically been written vertically, with characters running from top to bottom, and <i>blocks</i> running side to side.<button class="t1cz8cv8"><span class="a1ni5s6f">*</span><span class="e2cu4f7"></span></button><span data-use-adaptive-colors="true" id=":Rhm91eiavfelb:" data-direction="n" role="tooltip" style="flex-direction:column" class="p1lgzcrp l17lpp8k"><span class="p833mxt"><span class="pj0hw36"><span class="cffclov">Interestingly, this has been changing over the past few decades, and most East Asian languages these days are written horizontally from left to right, especially on the web.</span></span></span><svg xmlns="http://www.w3.org/2000/svg" width="24" height="6" fill="none" viewBox="0 0 24 6" style="--origin-y:0%" class="s1vq0yeh s1k9z937"><path d="
M 0 0
C 6 0
7.199999999999999 6
12 6
C 16.8 6
18 0
24 0
Z
"></path></svg></span></p><p class="prg881n">The primary goal of logical properties is to create an abstraction that sits above these differences. Instead of setting <code class="i165vvr1">margin-left</code> for left-to-right languages and flipping it to <code class="i165vvr1">margin-right</code> for right-to-left languages, we can instead use <code class="i165vvr1">margin-inline-start</code>. The margin will automatically be applied to the correct side, depending on the page's language.</p></div></aside>
<p class="prg881n">Even though this centering method has been around forever, I still find myself reaching for it on a regular basis! It's particularly useful when we want to center a single child, without affecting any of its siblings (for example, an image in-between paragraphs in a blog post).</p>
<p class="prg881n">Let's continue on our centering journey.</p>
<h2 id="centering-with-flexbox" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-flexbox-2" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with Flexbox</h2>
<p class="prg881n">Flexbox is designed to give us a <em style="color:inherit" class="w10oesj0">ton</em> of control when it comes to distributing a group of items along a primary axis. It offers some <em style="color:inherit" class="w10oesj0">really</em> powerful tools for centering!</p>
<p class="prg881n">Let's start by centering a single element, both horizontally and vertically:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div class="rcwtqfe"><div class="c9vkmf2"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="djxxsdm c9vkmf2"><div data-use-adaptive-colors="true" style="width:100%;height:200px" class="c17eovmj l17lpp8k"><div class="f1vkqrq wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div></div></div>
<p class="prg881n">The really cool thing about Flexbox centering is that it works <i>even when the children don’t fit in their container!</i> Try shrinking the width/height, and notice that the element overflows symmetrically.</p>
<p class="prg881n">It also works for <em style="color:inherit" class="w10oesj0">multiple</em> children. We can control how they stack with the <code class="i165vvr1">flex-direction</code> property:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Flex Direction<!-- -->:</span><div id=":Rhjeiavfelb:_flexDirection" class="ray0b30 w73h7tw"><div class="r1cct0ab"><button value="row" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">row</span></button><button value="column" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column</span></button><button value="row-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">row-reverse</span></button><button value="column-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column-reverse</span></button></div></div></div></div><div class="r1xmilea"><div class="cgnjx9w c2tip9q"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> flex-direction</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> row</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> gap</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 4</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d1iw6ryr c2tip9q"><div data-use-adaptive-colors="true" style="width:100%;min-height:212px;flex-direction:row" class="c1oa8tf1 l17lpp8k"><div class="bt1ddka wythvuy"><span class="t16m3o3k">1</span></div><div style="z-index:2" class="bt1ddka wythvuy"><span class="t16m3o3k">2</span></div><div class="bt1ddka wythvuy"><span class="t16m3o3k">3</span></div></div></div></div></div></div>
<p class="prg881n">Out of all the centering patterns we'll explore in this tutorial, this is probably the one I use the most. It's a great jack-of-all-trades, a great default option.</p>
<h2 id="centering-within-the-viewport" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-within-the-viewport-3" href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering within the viewport</h2>
<p class="prg881n">So far, we've been looking at how to center an element within its parent container. But what if we want to center an element in a different context? Certain elements like dialogs, prompts, and GDPR banners need to be centered within the viewport.</p>
<p class="prg881n">This is the domain of <em style="color:inherit" class="w10oesj0">positioned layout,</em> a layout mode used when we want to take something out of flow and anchor it to something else.</p>
<p class="prg881n">Here's what this looks like:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerWidth" min="30" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="rd1irhw"><div class="c1fq67g4"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> inset</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 0</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 5</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">vw</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">dvh</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dougf27 c1fq67g4"><div data-use-adaptive-colors="true" style="width:100%;height:17.625rem" class="c8q6jp9 l17lpp8k"><div class="bfhwaun"><div class="hiqwc5b"><span style="width:3rem" class="hktm3y9 bx3nfyo"></span><span style="width:5rem" class="hktm3y9 bx3nfyo"></span><span style="width:1rem" class="hktm3y9 bx3nfyo"></span><span style="width:4rem" class="hktm3y9 bx3nfyo"></span><span style="width:7rem" class="hktm3y9 bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:1rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2rem" class="bx3nfyo"></span><span style="width:4.75rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:7rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:5rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div></div><div class="hpafenu m148wj37"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="h1b4sgtf m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="h1b4sgtf m101ijsn"></div></div><div class="vm87swv m148wj37"><div style="clip-path:polygon(0% 0%, 100% 0%, 100% 0%, 0% 0%)" class="v15iwzns m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(0% 100%, 100% 100%, 100% 100%, 0% 100%)" class="v15iwzns m101ijsn"></div></div><div class="m1b7mso0 p1j9axfr"></div></div></div></div></div></div></div>
<p class="prg881n">Of all the strategies we'll discuss, this one is probably the most complex. Let's break it down.</p>
<p class="prg881n">We're using <code class="i165vvr1">position: fixed</code>, which anchors this element to the viewport. I like to think of the viewport like a pane of glass that sits in front of the website, like the window of a train that shows the landscape scrolling by. An element with <code class="i165vvr1">position: fixed</code> is like a ladybug that lands on the window.</p>
<p class="prg881n">Next, we're setting <code class="i165vvr1">inset: 0px</code>, which is a shorthand that sets <code class="i165vvr1">top</code>, <code class="i165vvr1">left</code>, <code class="i165vvr1">right</code>, and <code class="i165vvr1">bottom</code> all to the same value, <code class="i165vvr1">0px</code>.</p>
<p class="prg881n">With only these two properties, the element would stretch to fill the entire viewport, growing so that it's 0px from each edge. This can be useful in some contexts, but it's not what we're going for here. We need to constrain it.</p>
<p class="prg881n">The exact values we pick will vary on the specifics of each situation, but in general we want to set default values (with <code class="i165vvr1">width</code> and <code class="i165vvr1">height</code>), as well as max values (<code class="i165vvr1">max-width</code> and <code class="i165vvr1">max-height</code>), so that the element doesn't overflow on smaller viewports.</p>
<p class="prg881n"><strong>There's something interesting here:</strong> we've set up an impossible condition. Our element can't be 0px from the left <i>and</i> 0px from the right <i>and</i> only 12rem wide (assuming the viewport is wider than 12rem). We can only pick 2:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Pick two<!-- -->:</span><div id=":Rideiavfelb:_values" class="w73h7tw"><div class="r1cct0ab"><button value="left: 0px" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">left: 0px</span></button><button value="right: 0px" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">right: 0px</span></button><button value="width: 12rem" tabindex="-1" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">width: 12rem</span></button></div></div></div></div><div class="r3vmhao"><div class="c7zul9b"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d19iwc1r c7zul9b"><div data-use-adaptive-colors="true" class="ctnf89z l17lpp8k"><div class="b14ie1by"><div class="fujgo4n"><span style="width:3rem" class="h1vx44g7 byjv7xx"></span><span style="width:5rem" class="h1vx44g7 byjv7xx"></span><span style="width:1rem" class="h1vx44g7 byjv7xx"></span><span style="width:4rem" class="h1vx44g7 byjv7xx"></span><span style="width:7rem" class="h1vx44g7 byjv7xx"></span></div><div class="f1p79fid"><span style="width:1rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2rem" class="byjv7xx"></span><span style="width:4.75rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:7rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:5rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:2.1rem" class="byjv7xx"></span><span style="width:0.785rem" class="byjv7xx"></span><span style="width:2.5rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:3.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span></div><div class="f1p79fid"><span style="width:0.785rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:2.1rem" class="byjv7xx"></sp ... |
http://localhost:1200/joshwcomeau/popular - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>Popular Content | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com</link>
<atom:link href="http://localhost:1200/joshwcomeau/popular" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on React, CSS, Animation, and more! - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Wed, 08 Jan 2025 07:00:32 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>How To Center a Div</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/css/center-a-div/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with auto margins</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering within the viewport</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-elements-with-unknown-sizes-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering elements with unknown sizes</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-css-grid-5" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with CSS Grid</a><a href="https://www.joshwcomeau.com/css/center-a-div/#differences-from-flexbox-6" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Differences from Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-a-stack-of-elements-7" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering a stack of elements</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-text-8" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering text</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-in-the-future-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering in the future</a><a href="https://www.joshwcomeau.com/css/center-a-div/#going-beyond-the-patterns-10" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Going beyond the patterns</a><a href="https://www.joshwcomeau.com/css/center-a-div/#when-to-use-which-method-11" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">When to use which method</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">For a long time, centering an element within its parent was a surprisingly tricky thing to do. As CSS has evolved, we've been granted more and more tools we can use to solve this problem. These days, we're spoiled for choice!</p>
<p class="prg881n">I decided to create this tutorial to help you understand the trade-offs between different approaches, and to give you an arsenal of strategies you can use, to handle centering in all sorts of scenarios.</p>
<p class="prg881n">Honestly, this turned out to be <em style="color:inherit" class="w10oesj0">way more interesting</em> than I initially thought&nbsp;😅. Even if you've been using CSS for a while, I bet you'll learn at least 1 new strategy!</p>
<h2 id="centering-with-auto-margins" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-auto-margins-1" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with auto margins</h2>
<p class="prg881n">The first strategy we'll look at is one of the oldest. If we want to center an element horizontally, we can do so using margins set to the special value <code class="i165vvr1">auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgbeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgbeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-right</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-left:auto;margin-right:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgbeiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgbeiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">First, we need to constrain the element's width; by default, elements in Flow layout will expand horizontally to fill the available space, and we can't really center something that is full-width.</p>
<p class="prg881n">I <em style="color:inherit" class="w10oesj0">could</em> constrain the width with a fixed value (eg. <code class="i165vvr1">200px</code>), but really what I want in this case is for the element to shrinkwrap around its content. <code class="i165vvr1">fit-content</code> is a magical value that does exactly this. Essentially, it makes “width” behave like “height”, so that the element’s size is determined by its contents.</p>
<p class="prg881n"><strong>Why am I setting <code class="i165vvr1">max-width</code> instead of <code class="i165vvr1">width</code>?</strong> Well, my goal is to stop the element from expanding horizontally. I want to clamp its maximum size. If I used <code class="i165vvr1">width</code> instead, it would lock it to that size, and the element would overflow when the container is really narrow. If you drag that “Container Width” slider all the way to the left, you can see that the element shrinks with its container.</p>
<p class="prg881n">Now that our element is constrained, we can center it with <em style="color:inherit" class="w10oesj0">auto margins</em>.</p>
<p class="prg881n">I like to think of auto margins like <i>Hungry Hungry Hippos</i>. Each auto margin will try to gobble up as much space as possible. For example, check out what happens if we <em style="color:inherit" class="w10oesj0">only</em> set <code class="i165vvr1">margin-left: auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgneiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgneiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div></div><div style="margin-left:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgneiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgneiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">When <code class="i165vvr1">margin-left</code> is the only side with auto margins, <i>all</i> of the extra space gets applied as margin to that side. When we set both <code class="i165vvr1">margin-left: auto</code> <i>and</i> <code class="i165vvr1">margin-right: auto</code>, the two hippos each gobble up an equal amount of space. This forces the element to the center.</p>
<p class="prg881n"><em style="color:inherit" class="w10oesj0">Also:</em> I've been using <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> because they're familiar, but there's a better, more-modern way to do this:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line highlighted"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-inline</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-inline:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgteiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgteiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n"><code class="i165vvr1">margin-inline</code> will set both <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> to the same value (<code class="i165vvr1">auto</code>). It has <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://caniuse.com/mdn-css_properties_margin-inline">very good browser<!-- --> <span class="p7a6phg">support<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R1e8veiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R1e8veiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a>, having landed in all major browsers several years ago.</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-logical-properties" class="t1xqw7pl">Logical properties</strong><div class="c690lmu"><p class="prg881n"><code class="i165vvr1">margin-inline</code> is more than just a convenient shorthand for <code class="i165vvr1">margin-left</code> + <code class="i165vvr1">margin-right</code>. It's part of a collection of <em style="color:inherit" class="w10oesj0">logical properties</em>, designed to make it easier to internationalize the web.</p><p class="prg881n">In English, characters are written in a horizontal line from left to right. Those characters are composed into words and sentences, and assembled into “blocks” (paragraphs, headings, lists, etc). Blocks are stacked vertically, from top to bottom. We can think of this as the <i>orientation</i> of English-language websites.</p><p class="prg881n">This isn't universal, though! Some languages, like Arabic and Hebrew, are written from right to left. Other languages, like Chinese, have historically been written vertically, with characters running from top to bottom, and <i>blocks</i> running side to side.<button class="t1cz8cv8"><span class="a1ni5s6f">*</span><span class="e2cu4f7"></span></button><span data-use-adaptive-colors="true" id=":Rhm91eiavfelb:" data-direction="n" role="tooltip" style="flex-direction:column" class="p1lgzcrp l17lpp8k"><span class="p833mxt"><span class="pj0hw36"><span class="cffclov">Interestingly, this has been changing over the past few decades, and most East Asian languages these days are written horizontally from left to right, especially on the web.</span></span></span><svg xmlns="http://www.w3.org/2000/svg" width="24" height="6" fill="none" viewBox="0 0 24 6" style="--origin-y:0%" class="s1vq0yeh s1k9z937"><path d="
M 0 0
C 6 0
7.199999999999999 6
12 6
C 16.8 6
18 0
24 0
Z
"></path></svg></span></p><p class="prg881n">The primary goal of logical properties is to create an abstraction that sits above these differences. Instead of setting <code class="i165vvr1">margin-left</code> for left-to-right languages and flipping it to <code class="i165vvr1">margin-right</code> for right-to-left languages, we can instead use <code class="i165vvr1">margin-inline-start</code>. The margin will automatically be applied to the correct side, depending on the page's language.</p></div></aside>
<p class="prg881n">Even though this centering method has been around forever, I still find myself reaching for it on a regular basis! It's particularly useful when we want to center a single child, without affecting any of its siblings (for example, an image in-between paragraphs in a blog post).</p>
<p class="prg881n">Let's continue on our centering journey.</p>
<h2 id="centering-with-flexbox" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-flexbox-2" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with Flexbox</h2>
<p class="prg881n">Flexbox is designed to give us a <em style="color:inherit" class="w10oesj0">ton</em> of control when it comes to distributing a group of items along a primary axis. It offers some <em style="color:inherit" class="w10oesj0">really</em> powerful tools for centering!</p>
<p class="prg881n">Let's start by centering a single element, both horizontally and vertically:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div class="rcwtqfe"><div class="c9vkmf2"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="djxxsdm c9vkmf2"><div data-use-adaptive-colors="true" style="width:100%;height:200px" class="c17eovmj l17lpp8k"><div class="f1vkqrq wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div></div></div>
<p class="prg881n">The really cool thing about Flexbox centering is that it works <i>even when the children don’t fit in their container!</i> Try shrinking the width/height, and notice that the element overflows symmetrically.</p>
<p class="prg881n">It also works for <em style="color:inherit" class="w10oesj0">multiple</em> children. We can control how they stack with the <code class="i165vvr1">flex-direction</code> property:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Flex Direction<!-- -->:</span><div id=":Rhjeiavfelb:_flexDirection" class="ray0b30 w73h7tw"><div class="r1cct0ab"><button value="row" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">row</span></button><button value="column" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column</span></button><button value="row-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">row-reverse</span></button><button value="column-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column-reverse</span></button></div></div></div></div><div class="r1xmilea"><div class="cgnjx9w c2tip9q"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> flex-direction</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> row</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> gap</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 4</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d1iw6ryr c2tip9q"><div data-use-adaptive-colors="true" style="width:100%;min-height:212px;flex-direction:row" class="c1oa8tf1 l17lpp8k"><div class="bt1ddka wythvuy"><span class="t16m3o3k">1</span></div><div style="z-index:2" class="bt1ddka wythvuy"><span class="t16m3o3k">2</span></div><div class="bt1ddka wythvuy"><span class="t16m3o3k">3</span></div></div></div></div></div></div>
<p class="prg881n">Out of all the centering patterns we'll explore in this tutorial, this is probably the one I use the most. It's a great jack-of-all-trades, a great default option.</p>
<h2 id="centering-within-the-viewport" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-within-the-viewport-3" href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering within the viewport</h2>
<p class="prg881n">So far, we've been looking at how to center an element within its parent container. But what if we want to center an element in a different context? Certain elements like dialogs, prompts, and GDPR banners need to be centered within the viewport.</p>
<p class="prg881n">This is the domain of <em style="color:inherit" class="w10oesj0">positioned layout,</em> a layout mode used when we want to take something out of flow and anchor it to something else.</p>
<p class="prg881n">Here's what this looks like:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerWidth" min="30" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="rd1irhw"><div class="c1fq67g4"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> inset</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 0</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 5</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">vw</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">dvh</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dougf27 c1fq67g4"><div data-use-adaptive-colors="true" style="width:100%;height:17.625rem" class="c8q6jp9 l17lpp8k"><div class="bfhwaun"><div class="hiqwc5b"><span style="width:3rem" class="hktm3y9 bx3nfyo"></span><span style="width:5rem" class="hktm3y9 bx3nfyo"></span><span style="width:1rem" class="hktm3y9 bx3nfyo"></span><span style="width:4rem" class="hktm3y9 bx3nfyo"></span><span style="width:7rem" class="hktm3y9 bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:1rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2rem" class="bx3nfyo"></span><span style="width:4.75rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:7rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:5rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div></div><div class="hpafenu m148wj37"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="h1b4sgtf m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="h1b4sgtf m101ijsn"></div></div><div class="vm87swv m148wj37"><div style="clip-path:polygon(0% 0%, 100% 0%, 100% 0%, 0% 0%)" class="v15iwzns m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(0% 100%, 100% 100%, 100% 100%, 0% 100%)" class="v15iwzns m101ijsn"></div></div><div class="m1b7mso0 p1j9axfr"></div></div></div></div></div></div></div>
<p class="prg881n">Of all the strategies we'll discuss, this one is probably the most complex. Let's break it down.</p>
<p class="prg881n">We're using <code class="i165vvr1">position: fixed</code>, which anchors this element to the viewport. I like to think of the viewport like a pane of glass that sits in front of the website, like the window of a train that shows the landscape scrolling by. An element with <code class="i165vvr1">position: fixed</code> is like a ladybug that lands on the window.</p>
<p class="prg881n">Next, we're setting <code class="i165vvr1">inset: 0px</code>, which is a shorthand that sets <code class="i165vvr1">top</code>, <code class="i165vvr1">left</code>, <code class="i165vvr1">right</code>, and <code class="i165vvr1">bottom</code> all to the same value, <code class="i165vvr1">0px</code>.</p>
<p class="prg881n">With only these two properties, the element would stretch to fill the entire viewport, growing so that it's 0px from each edge. This can be useful in some contexts, but it's not what we're going for here. We need to constrain it.</p>
<p class="prg881n">The exact values we pick will vary on the specifics of each situation, but in general we want to set default values (with <code class="i165vvr1">width</code> and <code class="i165vvr1">height</code>), as well as max values (<code class="i165vvr1">max-width</code> and <code class="i165vvr1">max-height</code>), so that the element doesn't overflow on smaller viewports.</p>
<p class="prg881n"><strong>There's something interesting here:</strong> we've set up an impossible condition. Our element can't be 0px from the left <i>and</i> 0px from the right <i>and</i> only 12rem wide (assuming the viewport is wider than 12rem). We can only pick 2:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Pick two<!-- -->:</span><div id=":Rideiavfelb:_values" class="w73h7tw"><div class="r1cct0ab"><button value="left: 0px" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">left: 0px</span></button><button value="right: 0px" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">right: 0px</span></button><button value="width: 12rem" tabindex="-1" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">width: 12rem</span></button></div></div></div></div><div class="r3vmhao"><div class="c7zul9b"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d19iwc1r c7zul9b"><div data-use-adaptive-colors="true" class="ctnf89z l17lpp8k"><div class="b14ie1by"><div class="fujgo4n"><span style="width:3rem" class="h1vx44g7 byjv7xx"></span><span style="width:5rem" class="h1vx44g7 byjv7xx"></span><span style="width:1rem" class="h1vx44g7 byjv7xx"></span><span style="width:4rem" class="h1vx44g7 byjv7xx"></span><span style="width:7rem" class="h1vx44g7 byjv7xx"></span></div><div class="f1p79fid"><span style="width:1rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2rem" class="byjv7xx"></span><span style="width:4.75rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:7rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:5rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:2.1rem" class="byjv7xx"></span><span style="width:0.785rem" class="byjv7xx"></span><span style="width:2.5rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:3.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span></div><div class="f1p79fid"><span style="width:0.785rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:2.1rem" class="byjv7xx"></span><span sty |
note that there's a built-in feed for the latest articles: https://www.joshwcomeau.com/rss.xml |
Yes, and its not full feed |
Co-authored-by: Tony <TonyRL@users.noreply.github.com>
Co-authored-by: Tony <TonyRL@users.noreply.github.com>
Successfully generated as following: http://localhost:1200/joshwcomeau/latest/javascript - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>JavaScript | Articles and Tutorials | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com/javascript</link>
<atom:link href="http://localhost:1200/joshwcomeau/latest/javascript" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on JavaScript | - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Mon, 13 Jan 2025 03:49:22 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>Promises From The Ground Up</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/javascript/promises/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/javascript/promises/#why-would-they-design-it-this-way-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Why would they design it this way??</a><a href="https://www.joshwcomeau.com/javascript/promises/#callbacks-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Callbacks</a><a href="https://www.joshwcomeau.com/javascript/promises/#introducing-promises-3" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Introducing Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#working-with-promises-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Working with Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#creating-our-own-promises-5" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Creating our own Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#chaining-promises-6" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Chaining Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#passing-data-7" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Passing data</a><a href="https://www.joshwcomeau.com/javascript/promises/#rejected-promises-8" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Rejected Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#async--await-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Async / Await</a><a href="https://www.joshwcomeau.com/javascript/promises/#more-to-come-10" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">More to come!</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">There are a lot of speed bumps and potholes on the road to JavaScript proficiency. One of the biggest and most daunting is <em style="color:inherit" class="w10oesj0">Promises.</em></p>
<p class="prg881n">In order to understand Promises, we need a surprisingly deep understanding of how JavaScript works and what its limitations are. Without that context, Promises won’t really make much sense.</p>
<p class="prg881n">It can be frustrating because the Promises API is <i>so important</i> nowadays. It’s become the de facto way of working with asynchronous code. Modern web APIs are built on top of Promises. There’s no getting around it: if we want to be productive with JavaScript, it really helps to understand Promises.</p>
<p class="prg881n">So, in this tutorial, we’re going to learn about Promises, but we’ll start at the beginning. I’ll share all of the critical bits of context that took me years to understand. And by the end, hopefully, you’ll have a much deeper understanding of what Promises are and how to use them effectively. ✨</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-intended-audience" class="t1xqw7pl">Intended audience</strong><div class="c690lmu"><p class="prg881n">This blog post is intended for beginner-to-intermediate JavaScript developers. Some knowledge of basic JavaScript syntax is assumed.</p></div></aside>
<h2 id="why-would-they-design-it-this-way" data-element-type="ContentHeading" class="hrk0sy8"><a id="why-would-they-design-it-this-way-1" href="https://www.joshwcomeau.com/javascript/promises/#why-would-they-design-it-this-way-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Why would they design it this way??</h2>
<p class="prg881n">Suppose we wanted to build a <em style="color:inherit" class="w10oesj0">Happy New Year!</em> countdown, something like this:</p>
<!-- --><div class="w52b1qq"><p class="v793fuv">––</p><button data-disabled="false" class="b1b5r9ns dj4vswd">Start Countdown</button></div>
<p class="prg881n">If JavaScript was like most other programming languages, we could solve the problem like this:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA">function</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> newYearsCountdown</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">()</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">3</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">2</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">1</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">Happy New Year! 🎉</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">}</span></span></code></pre></div></div>
<p class="prg881n">In this hypothetical code snippet, the program would pause when it hits a <code class="i165vvr1">sleep()</code> call, and then resume after the specified amount of time has passed.</p>
<p class="prg881n">Unfortunately, there is no <code class="i165vvr1">sleep</code> function in JavaScript, because it’s a <em style="color:inherit" class="w10oesj0">single-threaded</em> language.<button class="t1cz8cv8"><span class="a1ni5s6f">*</span><span class="e2cu4f7"></span></button><span data-use-adaptive-colors="true" id=":R1gneiavfelb:" data-direction="n" role="tooltip" style="flex-direction:column" class="p1lgzcrp l17lpp8k"><span class="p833mxt"><span class="pj0hw36"><span class="cffclov">Technically, modern JavaScript has access to multiple threads via Web Workers, but those extra threads don't have access to the DOM, so they can’t really be used in most situations.</span></span></span><svg xmlns="http://www.w3.org/2000/svg" width="24" height="6" fill="none" viewBox="0 0 24 6" style="--origin-y:0%" class="s1vq0yeh s1k9z937"><path d="
M 0 0
C 6 0
7.199999999999999 6
12 6
C 16.8 6
18 0
24 0
Z
"></path></svg></span> A “thread” is a long-running process that executes code. JavaScript only has one thread, and so it can only do one thing at a time. It can’t multitask. This is a problem because if our lone JavaScript thread is busy managing this countdown timer, it can’t do anything <i>else</i>.</p>
<p class="prg881n">When I was first learning about this stuff, it wasn’t immediately obvious to me why this was a problem. If the countdown timer is the only thing happening right now, isn’t it fine if the JS thread was fully occupied during that time??</p>
<p class="prg881n">Well, even though JavaScript doesn’t have a <code class="i165vvr1">sleep</code> function, it <i>does</i> have some other functions that occupy the main thread for an extended amount of time. We can use those other methods to get a glimpse into what it would be like if JavaScript had a <code class="i165vvr1">sleep</code> function.</p>
<p class="prg881n">For example, <code class="i165vvr1">window.prompt()</code>. This function is used to gather information from the user, and it halts execution of our code much like our hypothetical <code class="i165vvr1">sleep()</code> function would.</p>
<p class="prg881n">Click the button in this playground, and then <strong>try to interact with the page</strong> while the prompt is open:</p>
<!-- --><div data-stacked="false" class="il1gxx9"><div style="--color-text:hsl(210deg 10% 90%);--color-background:hsl(210deg 15% 6%);--color-blurred-background:hsl(210deg 15% 6% / 0.75);--color-muted-background:hsl(210deg 38% 15% / 0.85);--color-action:hsl(240deg 95% 62%);--color-primary:hsl(225deg 100% 75%);--color-secondary:hsl(333deg 100% 55%);--color-tertiary:hsl(280deg 100% 85%);--color-decorative:hsl(200deg 50% 60%);--color-info-100:hsl(214deg 40% 15%);--color-info-200:hsl(218deg 32% 20%);--color-info-300:hsl(218deg 30% 25%);--color-info-400:hsl(218deg 45% 40%);--color-info-500:hsl(225deg 100% 60%);--color-info-700:hsl(215deg 100% 72%);--color-warning-100:hsl(30deg 25% 11%);--color-warning-200:hsl(32deg 20% 15%);--color-warning-300:hsl(32deg 25% 20%);--color-warning-400:hsl(35deg 45% 35%);--color-warning-500:hsl(40deg 100% 50%);--color-warning-700:hsl(43deg 100% 72%);--color-success-100:hsl(176deg 35% 10%);--color-success-200:hsl(176deg 26% 14%);--color-success-300:hsl(176deg 28% 20%);--color-success-400:hsl(176deg 45% 30%);--color-success-500:hsl(160deg 100% 40%);--color-success-700:hsl(160deg 80% 65%);--color-cloud-100:hsl(210deg 15% 6%);--color-cloud-300:hsl(212deg 40% 9%);--color-cloud-400:hsl(213deg 40% 10%);--color-cloud-500:hsl(213deg 40% 12%);--color-sky-from:hsl(214deg 40% 11%);--color-sky-to:hsl(200deg 50% 30%);--color-sky-subtle:hsl(210deg 40% 16%);--color-gray-50:hsl(210deg 19% 9%);--color-gray-100:hsl(210deg 15% 12%);--color-gray-200:hsl(210deg 15% 18%);--color-gray-300:hsl(210deg 10% 30%);--color-gray-400:hsl(210deg 9% 40%);--color-gray-500:hsl(210deg 8% 50%);--color-gray-600:hsl(210deg 12% 55%);--color-gray-700:hsl(210deg 14% 66%);--color-gray-800:hsl(210deg 20% 77%);--color-gray-900:hsl(210deg 25% 88%);--color-gray-1000:hsl(210deg 25% 96%);--color-adaptive-white:hsl(210deg 25% 92%);--syntax-bg:hsl(210deg 15% 6%);--syntax-highlight:hsl(210deg 30% 18%);--syntax-txt:hsl(0deg 0% 100%);--syntax-comment:hsl(200deg 18% 51%);--syntax-prop:hsl(326deg 100% 61%);--syntax-bool:hsl(50deg 100% 50%);--syntax-val:hsl(210deg 12% 65%);--syntax-str:hsl(259deg 100% 71%);--syntax-name:hsl(280deg 100% 66%);--syntax-del:hsl(0deg 100% 67%);--syntax-regex:hsl(50deg 100% 50%);--syntax-fn:hsl(195deg 100% 50%);--color-info:hsl(225deg 100% 80%);--color-info-background:hsl(225deg 100% 80% / 0.1);--color-error:hsl(340deg 95% 60%);--color-error-background:hsl(340deg 95% 43% / 0.1);--color-success:hsl(160deg 100% 40%);--color-success-background:hsl(160deg 100% 40% / 0.1);--color-alert:hsl(40deg 100% 50%);--color-alert-background:hsl(40deg 100% 50% / 0.1);--color-page-background:hsl(210deg 15% 6%);--color-page-primary:hsl(225deg 100% 75%);--kbd-background-color:hsl(210deg 9% 40%);--kbd-border-color:hsl(210deg 10% 30%);--color-code-bg:hsl(210deg 15% 12%);--color-content-outline:hsl(210deg 10% 30%)" class="i1j6zvew s1hfctpl d16kq653"><header class="w14j10eb"><p class="tkoi9ui">Code Playground</p><div class="au0cxa1"><button class="wsn2i55 a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7;overflow:visible" class="s1gvsyfu"><g style="transform:rotate(-45deg)" class="w15rf7yg"><rect x="0" y="8" width="24" height="6" rx="2" fill="var(--color-gray-100)"></rect><line x1="18" y1="8" x2="18" y2="14"></line></g></svg><span class="wlgbkjj">Format code using Prettier</span></button><abbr style="display:inline-block"><div class="wy3w6yt"><button title="Reset code" class="a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7" class="s1gvsyfu"><mask id="skip-icon-mask-:R39h1eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="black"></rect><rect x="9" y="0" width="16" height="24" fill="white"></rect></mask><line x1="5" y1="19" x2="5" y2="5"></line><polygon points="19 20 9 12 19 4 19 20" mask="url(#skip-icon-mask-:R39h1eiavfelb:)"></polygon></svg><span class="wlgbkjj">Reset Code</span></button><div class="pszdhfz"><span style="gap:1.5px" class="w15sqd9b"><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span></span></div></div></abbr></div></header><div></div><div class="w198elmk"><div class="c1jztmaw"><div style="flex:547.4" class="f15xiyga"><div style="min-height:100%" class="whrzwqb"><div class="h1nlnw7n"><p class="tbdao9h">HTML</p></div><div in="false" style="--max-height:80vh" class="w1bgwmtf"><button class="f9ibg9h"><span class="wlgbkjj">Focus the editor. This will trap focus until you press Escape.</span></button><label><span class="wlgbkjj">Code editor:</span><div translate="no" class="c18okpq0" style="position:relative;text-align:left;box-sizing:border-box;padding:0;overflow:hidden"><pre aria-hidden="true" style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:relative;pointer-events:none;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0"><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">script</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">function</span><span class="token script language-javascript"> </span><span class="token script language-javascript function" style="color:var(--syntax-fn)">askForName</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript"> </span><span class="token script language-javascript punctuation">{</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">const</span><span class="token script language-javascript"> name </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript dom variable" style="color:var(--syntax-str)">window</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript method function property-access" style="color:var(--syntax-fn)">prompt</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">'What is your name?'</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript punctuation">;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">const</span><span class="token script language-javascript"> elem </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript dom variable" style="color:var(--syntax-str)">document</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript method function property-access" style="color:var(--syntax-fn)">querySelector</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">'#greeting'</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript punctuation">;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> elem</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript property-access">innerText</span><span class="token script language-javascript"> </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript template-string template-punctuation string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">`</span><span class="token script language-javascript template-string string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">Hi </span><span class="token script language-javascript template-string interpolation interpolation-punctuation punctuation">${</span><span class="token script language-javascript template-string interpolation">name</span><span class="token script language-javascript template-string interpolation interpolation-punctuation punctuation">}</span><span class="token script language-javascript template-string string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">!</span><span class="token script language-javascript template-string template-punctuation string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">`</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript punctuation">}</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">script</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain" style="display:inline-block">
</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">button</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag special-attr attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">onclick</span><span class="token tag special-attr attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag special-attr attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag special-attr attr-value value javascript language-javascript function" style="color:var(--syntax-fn);font-weight:var(--font-weight-medium)">askForName</span><span class="token tag special-attr attr-value value javascript language-javascript punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">(</span><span class="token tag special-attr attr-value value javascript language-javascript punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">)</span><span class="token tag special-attr attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"> Click me</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">button</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">id</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">greeting</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span></div></pre><textarea style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:absolute;top:0;left:0;height:100%;width:100%;resize:none;color:inherit;overflow:hidden;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;-webkit-text-fill-color:transparent;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0" class="npm__react-simple-code-editor__textarea" autocapitalize="off" autocomplete="off" autocorrect="off" spellcheck="false" data-gramm="false">&lt;script&gt;
function askForName() {
const name = window.prompt('What is your name?');
const elem = document.querySelector('#greeting');
elem.innerText = `Hi ${name}!`
}
&lt;/script&gt;
&lt;button onclick="askForName()"&gt;
Click me
&lt;/button&gt;
&lt;div id="greeting"&gt;&lt;/div&gt;</textarea><style>
/**
* Reset the text fill color so that placeholder is visible
*/
.npm__react-simple-code-editor__textarea:empty {
-webkit-text-fill-color: inherit !important;
}
/**
* Hack to apply on some CSS on IE10 and IE11
*/
@media all and (-ms-high-contrast: none), (-ms-high-contrast: active) {
/**
* IE doesn't support '-webkit-text-fill-color'
* So we use 'color: transparent' to make the text transparent on IE
* Unlike other browsers, it doesn't affect caret color in IE
*/
.npm__react-simple-code-editor__textarea {
color: transparent !important;
}
.npm__react-simple-code-editor__textarea::selection {
background-color: #accef7 !important;
color: transparent !important;
}
}
</style></div></label></div></div></div><button class="dijulrw"><span class="wlgbkjj">Resize editor. Use left/right arrows.</span></button><div style="flex:234.60000000000002" class="s1c0ceq0"><div style="height:100%" class="whrzwqb"><div class="h1nlnw7n"><p class="tbdao9h">Result</p><abbr title="Refresh pane"><button style="transform:rotate(0deg);opacity:0.7" class="w11udlg3 ah084m5"><svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-refresh-cw"><path d="M3 12a9 9 0 0 1 9-9 9.75 9.75 0 0 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path><path d="M21 12a9 9 0 0 1-9 9 9.75 9.75 0 0 1-6.74-2.74L3 16"></path><path d="M8 16H3v5"></path></svg><span class="wlgbkjj">Refresh results pane</span></button></abbr></div><div style="--height:100%;--flex:1" class="wqqdb70"><iframe height="100%" width="100%" title="example" frameborder="0" loading="lazy" class="fxdlgbu" referrerpolicy="no-referrer"></iframe></div></div></div></div></div></div></div>
<p class="prg881n"><strong>Notice that while the prompt is open, the page is totally unresponsive?</strong> You can't scroll, click any links, or select any text! The JavaScript thread is busy waiting for us to provide a value so that it can finish running that code. While it’s waiting, it can’t do anything <i>else</i>, and so the browser locks down the UI.</p>
<p class="prg881n">Other languages have multiple threads, and so it's no big deal if one of them gets preoccupied for a while. In JavaScript, though, we only have the one, and it’s used for everything: handling events, managing network requests, updating the UI, etc.</p>
<p class="prg881n">If we want to create a countdown, we need to find a way to do it without blocking the thread.</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-but-why-is-the-entire-ui-frozen" class="t1xqw7pl">But why is the entire UI frozen??</strong><div class="c690lmu"><p class="prg881n">In the example above with <code class="i165vvr1">window.prompt()</code>, the entire UI becomes unresponsive while the browser waits for us to provide a value.</p><p class="prg881n">This is kinda strange… the browser doesn’t rely on JavaScript to scroll the page, or to select text. So why can’t we do any of those things?</p><p class="prg881n">I think browsers work this way to prevent bugs. Scrolling the page, for example, triggers “scroll” events which can be caught and handled with JavaScript. If the JS thread is occupied while the scroll event happens, that code never runs, which could lead to bugs if the developer assumed that scroll events would always be handled.</p><p class="prg881n">It could also be a UX thing; maybe the browser disables the UI so that the user can’t ignore the prompt. Either way, though, I suspect a native <code class="i165vvr1">sleep</code> function would need to work the same way to prevent bugs.</p></div></aside>
<h2 id="callbacks" data-element-type="ContentHeading" class="hrk0sy8"><a id="callbacks-2" href="https://www.joshwcomeau.com/javascript/promises/#callbacks-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Callbacks</h2>
<p class="prg881n">The main tool in our toolbox for solving these sorts of problems is <code class="i165vvr1">setTimeout</code>. <code class="i165vvr1">setTimeout</code> is a function which accepts two arguments:</p>
<ol class="oc96fnv">
<li class="wp18jjy"><div class="l1n7gg9f">A chunk of work to do, at some point in the future.</div></li>
<li class="wp18jjy"><div class="l1n7gg9f">The amount of time to wait for.</div></li>
</ol>
<p class="prg881n">Here's an example:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">Start</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">setTimeout</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> ()</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA"> =&gt;</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF"> console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">After one second</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> },</span></span>
<span class="line"><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF"> 1000</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span></code></pre></div></div>
<p class="prg881n">The chunk of work is passed in through a function. This pattern is known as a <em style="color:inherit" class="w10oesj0">callback.</em></p>
<p class="prg881n">The hypothetical <code class="i165vvr1">sleep()</code> function we saw before is like calling a company and waiting on hold for the next available representative. <code class="i165vvr1">setTimeout()</code> is like pressing <i>1</i> to have them <em style="color:inherit" class="w10oesj0">call you back</em> when the representative is available. You can hang up the phone and get on with your life.</p>
<p class="prg881n"><code class="i165vvr1">setTimeout()</code> is known as an <em style="color:inherit" class="w10oesj0">asynchronous</em> function. This means that it doesn’t block the thread. By contrast, <code class="i165vvr1">window.prompt()</code> is <em style="color:inherit" class="w10oesj0">synchronous</em>, because the JavaScript thread can't do anything else while it’s waiting.</p>
<p class="prg881n">The big downside with asynchronous code is that it means our code won't always run in a linear order. Consider the following setup:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">1. Before setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">setTimeout</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">()</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA"> =&gt;</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF"> console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">2. Inside setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">},</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF"> 500</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">3. After setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span></code></pre></div></div>
<p class="prg881n">You might expect these logs to fire in order from top to bottom: <code class="i165vvr1">1</code> &gt; <code class="i165vvr1">2</code> &gt; <code class="i165vvr1">3</code>. <strong>But remember, the whole idea with callbacks is that we’re scheduling a call back.</strong> The JavaScript thread doesn’t sit around and wait, it ke ... |
http://localhost:1200/joshwcomeau/latest - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>Articles and Tutorials | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com</link>
<atom:link href="http://localhost:1200/joshwcomeau/latest" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on React, CSS, Animation, and more! - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Mon, 13 Jan 2025 03:49:24 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>Next-level frosted glass with backdrop-filter</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/css/backdrop-filter/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#css-filters-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">CSS filters</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#the-issue-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">The Issue</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#pointer-events-3" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Pointer events</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#flickering-top-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Flickering top</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#thicker-glass-5" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Thicker glass</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#browser-support-6" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Browser support</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#glassy-edge-7" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Glassy edge</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#the-final-code-8" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">The final code</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#continue-learning-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Continue learning</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">One of my all-time favourite CSS tricks is using <code class="i165vvr1">backdrop-filter: blur()</code> to create a frosted glass effect. I use it in just about every project I work on, including this blog!</p>
<p class="prg881n">Here’s a quick demo, to show what I’m talking about:</p>
<!-- --><div class="o11h7o12 w1bd8cgi" style="overflow:hidden"><div class="hduekzx"><div class="t1xdhus0"><div class="t1vhg4x9"><div class="rptyea t8ndwpz"></div><div class="y1qao05t t8ndwpz"></div><div class="g1v2ob21 t8ndwpz"></div></div><div class="tj6926w"><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="lep27c6 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(0deg);transform-origin:center center"></path></svg><span>Some Example Website</span><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="r2mz821 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(90deg);transform-origin:center center"></path></svg></div></div><div class="a13tcou3"><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-left"><path d="m12 19-7-7 7-7"></path><path d="M19 12H5"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-right"><path d="M5 12h14"></path><path d="m12 5 7 7-7 7"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-rotate-cw"><path d="M21 12a9 9 0 1 1-9-9c2.52 0 4.93 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path></svg></div><div class="a1booqr"><span>https://www.joshwcomeau.com/example-website</span></div></div></div><div style="overflow:clip"><div class="wi9xlz4 s1hfctpl d16kq653"><div class="hpv2kfm"><div class="b1dh7eg0"></div><div class="bi95upp"></div><p class="l19j3i1d">Some Website</p></div><div class="cufzfud"><img src="https://www.joshwcomeau.com/images/backdrop-filter/cupcake.png" style="animation-play-state:paused" class="c19tvtvc" referrerpolicy="no-referrer"><p>This is an example website showing how I typically use<!-- --> <code class="i165vvr1">backdrop-filter</code> to create glassy headers.</p><p class="o1fnc03u">Notice that as the cupcake moves behind the header, it appears blurry, as it would if it was passing behind frosted glass.</p><div class="atopy0n"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1t96eke w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><span style="width:20px;height:20px;--duration:1422.2222222222222ms;--size:20px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div></div></div>
<p class="prg881n">This effect helps us add depth and realism to our projects. It’s lovely.</p>
<p class="prg881n"><strong>But when I see this effect in the wild, it’s almost always missing some crucial optimizations.</strong> A couple of small changes can make our frosted glass <em style="color:inherit" class="w10oesj0">so</em> much more lush and realistic!</p>
<p class="prg881n">In this post, you’ll learn how to make the slickest frosted glass ever ✨. We’ll also learn quite a bit about CSS filters along the way!</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-credit-where-credit-is-due" class="t1xqw7pl">Credit where credit is due</strong><div class="c690lmu"><p class="prg881n">I learned about this effect from <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://expensive.toys/">Artur<!-- --> <span class="p7a6phg">Bien’s<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R2qm8deiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R2qm8deiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a> incredible demos. He credits <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://www.jamiegray.net/">Jamie<!-- --> <span class="p7a6phg">Gray<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R2sm8deiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R2sm8deiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a> for the original concept.</p></div></aside>
<h2 id="css-filters" data-element-type="ContentHeading" class="hrk0sy8"><a id="css-filters-1" href="https://www.joshwcomeau.com/css/backdrop-filter/#css-filters-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>CSS filters</h2>
<p class="prg881n">To briefly explain the underlying concept: CSS gives us quick and easy access to SVG filters via the <code class="i165vvr1">filter</code> property.</p>
<p class="prg881n">For example, we can give elements a Gaussian blur with <code class="i165vvr1">filter: blur()</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="wskxkqa"><img alt="My 3D mascot smiling with his tongue out. It’s very blurry, but dragging the slider makes it clearer" src="https://www.joshwcomeau.com/images/josh-originals/josh-tongue-dark.png" loading="lazy" style="filter:blur(16px)" class="j5sqeuc" referrerpolicy="no-referrer"><div class="c162apj7"></div></div><div style="--template-cols:1fr 1fr;grid-template-columns:1fr" class="c1j2qstn"><div style="max-width:400px;width:100%;margin-inline:auto" class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgleiavfelb:_radius" class="lpsyt9t">Blur radius<!-- -->:</label><span class="w1194iyg">16px</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgleiavfelb:_radius" min="0" max="16" class="i1i8zjy7" value="16"></div></div></div></div></div>
<p class="prg881n">There are lots of fun filter options, the sorts of things you’d find in image-editing software. Like, rotating the hue of all the colors:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1d4uhch"><img alt="My 3D mascot making a puzzled face. Dragging the slider causes him to become unnatural shades of green and pink." src="https://www.joshwcomeau.com/images/newsletter/joy-of-react-mascot.png" loading="lazy" style="filter:hue-rotate(0deg)" class="m1wtl9k0" referrerpolicy="no-referrer"><div class="c1kud80y"></div></div><div style="--template-cols:1fr 1fr;grid-template-columns:1fr" class="c1j2qstn"><div style="max-width:400px;width:100%;margin-inline:auto" class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgpeiavfelb:_rotation" class="lpsyt9t">Hue rotation<!-- -->:</label><span class="w1194iyg">0deg</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgpeiavfelb:_rotation" min="0" max="360" class="i1i8zjy7" value="0"></div></div></div></div></div>
<p class="prg881n">In these examples, I’m applying the filters to an <code class="i165vvr1">&lt;img&gt;</code> tag, but we can apply them to standard DOM nodes as well:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1icc7dx"><div style="filter:blur(0px) sepia(0%)" class="c1l5p6hh"><p>Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged.</p></div><div class="c1koe0ch"></div></div><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_radius" class="lpsyt9t">Blur radius<!-- -->:</label><span class="w1194iyg">0px</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_radius" min="0" max="16" step="0.1" class="i1i8zjy7" value="0"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_sepia" class="lpsyt9t">Sepia percentage<!-- -->:</label><span class="w1194iyg">0%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_sepia" min="0" max="100" class="i1i8zjy7" value="0"></div></div></div></div></div>
<p class="prg881n">Pretty neat, right?</p>
<p class="prg881n">Things get even cooler with <code class="i165vvr1">backdrop-filter</code>. This property lets us apply these same filters to the stuff <em style="color:inherit" class="w10oesj0">behind</em> a given element.</p>
<p class="prg881n">For example:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1qszusq"><div class="isvxvbh"><img src="https://www.joshwcomeau.com/images/backdrop-filter/tokyo.jpg" loading="lazy" class="tvcyp0e" referrerpolicy="no-referrer"><div style="top:80%;left:50%;--scale:2;-webkit-backdrop-filter:sepia(50%) brightness(130%);backdrop-filter:sepia(50%) brightness(130%)" class="ctfyewy"></div></div><div class="ck6v2m7"></div></div><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rh5eiavfelb:_sepia" class="lpsyt9t">Sepia<!-- -->:</label><span class="w1194iyg">50%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rh5eiavfelb:_sepia" min="0" max="100" class="i1i8zjy7" value="50"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rh5eiavfelb:_brightness" class="lpsyt9t">Brightness<!-- -->:</label><span class="w1194iyg">130%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rh5eiavfelb:_brightness" min="50" max="300" class="i1i8zjy7" value="130"></div></div></div></div></div>
<p class="prg881n">In this demo, the <code class="i165vvr1">.magic-ring</code> element sits in front of a photo (<a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://unsplash.com/photos/assorted-signages-on-buildings-nTBW1cOY1qI"> <span class="p7a6phg">source<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:Rn17eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:Rn17eiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a>). It uses the <code class="i165vvr1">backdrop-filter</code> property to apply some filtering to everything behind it, which can be used for some pretty artistic effects.</p>
<p class="prg881n">In practice, I pretty much only use <code class="i165vvr1">backdrop-filter</code> for one use case: blurring everything behind an element, usually a header, to create the “frosted glass” effect I mentioned earlier:</p>
<!-- --><div class="o11h7o12 w1bd8cgi" style="overflow:hidden"><div class="hduekzx"><div class="t1xdhus0"><div class="t1vhg4x9"><div class="rptyea t8ndwpz"></div><div class="y1qao05t t8ndwpz"></div><div class="g1v2ob21 t8ndwpz"></div></div><div class="tj6926w"><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="lep27c6 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(0deg);transform-origin:center center"></path></svg><span>Some Example Website</span><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="r2mz821 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(90deg);transform-origin:center center"></path></svg></div></div><div class="a13tcou3"><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-left"><path d="m12 19-7-7 7-7"></path><path d="M19 12H5"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-right"><path d="M5 12h14"></path><path d="m12 5 7 7-7 7"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-rotate-cw"><path d="M21 12a9 9 0 1 1-9-9c2.52 0 4.93 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path></svg></div><div class="a1booqr"><span>https://www.joshwcomeau.com/example-website</span></div></div></div><div style="overflow:clip"><div class="wi9xlz4 s1hfctpl d16kq653"><div class="hpv2kfm"><div class="b1dh7eg0"></div><div class="bi95upp"></div><p class="l19j3i1d">Some Website</p></div><div class="cufzfud"><img src="https://www.joshwcomeau.com/images/backdrop-filter/cupcake.png" style="animation-play-state:paused" class="c19tvtvc" referrerpolicy="no-referrer"><p>This is an example website showing how I typically use<!-- --> <code class="i165vvr1">backdrop-filter</code> to create glassy headers.</p><p class="o1fnc03u">Notice that as the cupcake moves behind the header, it appears blurry, as it would if it was passing behind frosted glass.</p><div class="atopy0n"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1t96eke w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><span style="width:20px;height:20px;--duration:1422.2222222222222ms;--size:20px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div></div></div>
<p class="prg881n">Alright. Let’s talk about the thing most developers miss.</p>
<h2 id="the-issue" data-element-type="ContentHeading" class="hrk0sy8"><a id="the-issue-2" href="https://www.joshwcomeau.com/css/backdrop-filter/#the-issue-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>The Issue</h2>
<p class="prg881n"><strong>Here’s the problem:</strong> The <code class="i165vvr1">backdrop-filter</code> algorithm only considers the pixels that are <i>directly behind</i> the element.</p>
<p class="prg881n">For filters like <code class="i165vvr1">brightness</code> or <code class="i165vvr1">hue-rotate</code>, that makes perfect sense. With blur, though, we actually want to consider pixels that are <em style="color:inherit" class="w10oesj0">near</em> the element too.</p>
<p class="prg881n">This is one of those things where a demo is worth a thousand words. Check out the difference:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="i1df5wci s360ott"><div id=":Rhneiavfelb:_mode" class="w73h7tw"><div class="r1cct0ab"><button value="default" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">Default</span></button><button value="fixed" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">Tweaked</span></button></div></div></div></div><div class="wr5us8n s1hfctpl d16kq653"><div class="hhqru8"><div class="g8qz4zc"></div></div><div style="animation-play-state:running" class="b1uj7lhw"></div></div><div class="awqze7l"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1rsl36b w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-pause"><rect x="14" y="4" width="4" height="16" rx="1"></rect><rect x="6" y="4" width="4" height="16" rx="1"></rect></svg></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div>
<p class="prg881n">By default, the gaussian blur algorithm is applied to all of the pixels behind the element. This means that if a big colorful element is <em style="color:inherit" class="w10oesj0">near</em> the element, it won’t have any effect.</p>
<p class="prg881n">That’s not really how frosted glass works in real life though. Light bounces off of objects and then goes through the glass. It looks <em style="color:inherit" class="w10oesj0">so much better</em> when the blurring algorithm includes nearby content.</p>
<p class="prg881n">Unfortunately, this isn’t something we can configure directly. Instead, we need to be a bit crafty.</p>
<p class="prg881n">Here’s the code:</p>
<!-- --><div data-stacked="false" class="il1gxx9"><div style="--color-text:hsl(210deg 10% 90%);--color-background:hsl(210deg 15% 6%);--color-blurred-background:hsl(210deg 15% 6% / 0.75);--color-muted-background:hsl(210deg 38% 15% / 0.85);--color-action:hsl(240deg 95% 62%);--color-primary:hsl(225deg 100% 75%);--color-secondary:hsl(333deg 100% 55%);--color-tertiary:hsl(280deg 100% 85%);--color-decorative:hsl(200deg 50% 60%);--color-info-100:hsl(214deg 40% 15%);--color-info-200:hsl(218deg 32% 20%);--color-info-300:hsl(218deg 30% 25%);--color-info-400:hsl(218deg 45% 40%);--color-info-500:hsl(225deg 100% 60%);--color-info-700:hsl(215deg 100% 72%);--color-warning-100:hsl(30deg 25% 11%);--color-warning-200:hsl(32deg 20% 15%);--color-warning-300:hsl(32deg 25% 20%);--color-warning-400:hsl(35deg 45% 35%);--color-warning-500:hsl(40deg 100% 50%);--color-warning-700:hsl(43deg 100% 72%);--color-success-100:hsl(176deg 35% 10%);--color-success-200:hsl(176deg 26% 14%);--color-success-300:hsl(176deg 28% 20%);--color-success-400:hsl(176deg 45% 30%);--color-success-500:hsl(160deg 100% 40%);--color-success-700:hsl(160deg 80% 65%);--color-cloud-100:hsl(210deg 15% 6%);--color-cloud-300:hsl(212deg 40% 9%);--color-cloud-400:hsl(213deg 40% 10%);--color-cloud-500:hsl(213deg 40% 12%);--color-sky-from:hsl(214deg 40% 11%);--color-sky-to:hsl(200deg 50% 30%);--color-sky-subtle:hsl(210deg 40% 16%);--color-gray-50:hsl(210deg 19% 9%);--color-gray-100:hsl(210deg 15% 12%);--color-gray-200:hsl(210deg 15% 18%);--color-gray-300:hsl(210deg 10% 30%);--color-gray-400:hsl(210deg 9% 40%);--color-gray-500:hsl(210deg 8% 50%);--color-gray-600:hsl(210deg 12% 55%);--color-gray-700:hsl(210deg 14% 66%);--color-gray-800:hsl(210deg 20% 77%);--color-gray-900:hsl(210deg 25% 88%);--color-gray-1000:hsl(210deg 25% 96%);--color-adaptive-white:hsl(210deg 25% 92%);--syntax-bg:hsl(210deg 15% 6%);--syntax-highlight:hsl(210deg 30% 18%);--syntax-txt:hsl(0deg 0% 100%);--syntax-comment:hsl(200deg 18% 51%);--syntax-prop:hsl(326deg 100% 61%);--syntax-bool:hsl(50deg 100% 50%);--syntax-val:hsl(210deg 12% 65%);--syntax-str:hsl(259deg 100% 71%);--syntax-name:hsl(280deg 100% 66%);--syntax-del:hsl(0deg 100% 67%);--syntax-regex:hsl(50deg 100% 50%);--syntax-fn:hsl(195deg 100% 50%);--color-info:hsl(225deg 100% 80%);--color-info-background:hsl(225deg 100% 80% / 0.1);--color-error:hsl(340deg 95% 60%);--color-error-background:hsl(340deg 95% 43% / 0.1);--color-success:hsl(160deg 100% 40%);--color-success-background:hsl(160deg 100% 40% / 0.1);--color-alert:hsl(40deg 100% 50%);--color-alert-background:hsl(40deg 100% 50% / 0.1);--color-page-background:hsl(210deg 15% 6%);--color-page-primary:hsl(225deg 100% 75%);--kbd-background-color:hsl(210deg 9% 40%);--kbd-border-color:hsl(210deg 10% 30%);--color-code-bg:hsl(210deg 15% 12%);--color-content-outline:hsl(210deg 10% 30%)" class="i1j6zvew s1hfctpl d16kq653"><header class="w14j10eb"><p class="tkoi9ui">Code Playground</p><div class="au0cxa1"><button class="wsn2i55 a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7;overflow:visible" class="s1gvsyfu"><g style="transform:rotate(-45deg)" class="w15rf7yg"><rect x="0" y="8" width="24" height="6" rx="2" fill="var(--color-gray-100)"></rect><line x1="18" y1="8" x2="18" y2="14"></line></g></svg><span class="wlgbkjj">Format code using Prettier</span></button><abbr style="display:inline-block"><div class="wy3w6yt"><button title="Reset code" class="a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7" class="s1gvsyfu"><mask id="skip-icon-mask-:R39i1eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="black"></rect><rect x="9" y="0" width="16" height="24" fill="white"></rect></mask><line x1="5" y1="19" x2="5" y2="5"></line><polygon points="19 20 9 12 19 4 19 20" mask="url(#skip-icon-mask-:R39i1eiavfelb:)"></polygon></svg><span class="wlgbkjj">Reset Code</span></button><div class="pszdhfz"><span style="gap:1.5px" class="w15sqd9b"><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span></span></div></div></abbr></div></header><div></div><div class="w198elmk"><div class="c1jztmaw"><div style="flex:516.12" class="f15xiyga"><div class="w16o0eiw"><div class="tt2ystl"><button style="--weight:var(--font-weight-bold);--color:var(--color-text)" class="ts2dciv">HTML</button><button style="--weight:var(--font-weight-normal);--color:var(--color-gray-500)" class="ts2dciv">CSS</button></div><div in="false" style="--max-height:80vh" class="w1bgwmtf"><button class="f9ibg9h"><span class="wlgbkjj">Focus the editor. This will trap focus until you press Escape.</span></button><label><span class="wlgbkjj">Code editor:</span><div translate="no" class="c18okpq0" style="position:relative;text-align:left;box-sizing:border-box;padding:0;overflow:hidden"><pre aria-hidden="true" style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:relative;pointer-events:none;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0"><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">style</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css selector" style="color:var(--syntax-name)">header</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">{</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">position</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> relative</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">}</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css selector class" style="color:var(--syntax-name)">.backdrop</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">{</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">position</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> absolute</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">inset</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">0</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">height</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">200</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">backdrop-filter</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css function" style="color:var(--syntax-fn)">blur</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">(</span><span class="token style language-css number" style="color:var(--syntax-bool)">16</span><span class="token style language-css unit" style="color:var(--syntax-str)">px</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">)</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">mask-image</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css function" style="color:var(--syntax-fn)">linear-gradient</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">(</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> to bottom</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">,</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css color" style="color:var(--syntax-str)">black</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">0</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">50</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">,</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css color" style="color:var(--syntax-str)">transparent</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">50</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">100</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">)</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">}</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">style</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain" style="display:inline-block">
</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">header</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"> </span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">class</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">backdrop</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">header</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">class</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">ball</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"&g ... |
http://localhost:1200/joshwcomeau/popular/false - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>Popular Content | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com</link>
<atom:link href="http://localhost:1200/joshwcomeau/popular/false" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on React, CSS, Animation, and more! - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Mon, 13 Jan 2025 03:49:25 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>How To Center a Div</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/css/center-a-div/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with auto margins</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering within the viewport</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-elements-with-unknown-sizes-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering elements with unknown sizes</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-css-grid-5" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with CSS Grid</a><a href="https://www.joshwcomeau.com/css/center-a-div/#differences-from-flexbox-6" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Differences from Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-a-stack-of-elements-7" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering a stack of elements</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-text-8" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering text</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-in-the-future-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering in the future</a><a href="https://www.joshwcomeau.com/css/center-a-div/#going-beyond-the-patterns-10" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Going beyond the patterns</a><a href="https://www.joshwcomeau.com/css/center-a-div/#when-to-use-which-method-11" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">When to use which method</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">For a long time, centering an element within its parent was a surprisingly tricky thing to do. As CSS has evolved, we've been granted more and more tools we can use to solve this problem. These days, we're spoiled for choice!</p>
<p class="prg881n">I decided to create this tutorial to help you understand the trade-offs between different approaches, and to give you an arsenal of strategies you can use, to handle centering in all sorts of scenarios.</p>
<p class="prg881n">Honestly, this turned out to be <em style="color:inherit" class="w10oesj0">way more interesting</em> than I initially thought&nbsp;😅. Even if you've been using CSS for a while, I bet you'll learn at least 1 new strategy!</p>
<h2 id="centering-with-auto-margins" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-auto-margins-1" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with auto margins</h2>
<p class="prg881n">The first strategy we'll look at is one of the oldest. If we want to center an element horizontally, we can do so using margins set to the special value <code class="i165vvr1">auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgbeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgbeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-right</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-left:auto;margin-right:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgbeiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgbeiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">First, we need to constrain the element's width; by default, elements in Flow layout will expand horizontally to fill the available space, and we can't really center something that is full-width.</p>
<p class="prg881n">I <em style="color:inherit" class="w10oesj0">could</em> constrain the width with a fixed value (eg. <code class="i165vvr1">200px</code>), but really what I want in this case is for the element to shrinkwrap around its content. <code class="i165vvr1">fit-content</code> is a magical value that does exactly this. Essentially, it makes “width” behave like “height”, so that the element’s size is determined by its contents.</p>
<p class="prg881n"><strong>Why am I setting <code class="i165vvr1">max-width</code> instead of <code class="i165vvr1">width</code>?</strong> Well, my goal is to stop the element from expanding horizontally. I want to clamp its maximum size. If I used <code class="i165vvr1">width</code> instead, it would lock it to that size, and the element would overflow when the container is really narrow. If you drag that “Container Width” slider all the way to the left, you can see that the element shrinks with its container.</p>
<p class="prg881n">Now that our element is constrained, we can center it with <em style="color:inherit" class="w10oesj0">auto margins</em>.</p>
<p class="prg881n">I like to think of auto margins like <i>Hungry Hungry Hippos</i>. Each auto margin will try to gobble up as much space as possible. For example, check out what happens if we <em style="color:inherit" class="w10oesj0">only</em> set <code class="i165vvr1">margin-left: auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgneiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgneiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div></div><div style="margin-left:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgneiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgneiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">When <code class="i165vvr1">margin-left</code> is the only side with auto margins, <i>all</i> of the extra space gets applied as margin to that side. When we set both <code class="i165vvr1">margin-left: auto</code> <i>and</i> <code class="i165vvr1">margin-right: auto</code>, the two hippos each gobble up an equal amount of space. This forces the element to the center.</p>
<p class="prg881n"><em style="color:inherit" class="w10oesj0">Also:</em> I've been using <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> because they're familiar, but there's a better, more-modern way to do this:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line highlighted"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-inline</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-inline:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgteiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgteiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n"><code class="i165vvr1">margin-inline</code> will set both <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> to the same value (<code class="i165vvr1">auto</code>). It has <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://caniuse.com/mdn-css_properties_margin-inline">very good browser<!-- --> <span class="p7a6phg">support<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R1e8veiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R1e8veiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a>, having landed in all major browsers several years ago.</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-logical-properties" class="t1xqw7pl">Logical properties</strong><div class="c690lmu"><p class="prg881n"><code class="i165vvr1">margin-inline</code> is more than just a convenient shorthand for <code class="i165vvr1">margin-left</code> + <code class="i165vvr1">margin-right</code>. It's part of a collection of <em style="color:inherit" class="w10oesj0">logical properties</em>, designed to make it easier to internationalize the web.</p><p class="prg881n">In English, characters are written in a horizontal line from left to right. Those characters are composed into words and sentences, and assembled into “blocks” (paragraphs, headings, lists, etc). Blocks are stacked vertically, from top to bottom. We can think of this as the <i>orientation</i> of English-language websites.</p><p class="prg881n">This isn't universal, though! Some languages, like Arabic and Hebrew, are written from right to left. Other languages, like Chinese, have historically been written vertically, with characters running from top to bottom, and <i>blocks</i> running side to side.<button class="t1cz8cv8"><span class="a1ni5s6f">*</span><span class="e2cu4f7"></span></button><span data-use-adaptive-colors="true" id=":Rhm91eiavfelb:" data-direction="n" role="tooltip" style="flex-direction:column" class="p1lgzcrp l17lpp8k"><span class="p833mxt"><span class="pj0hw36"><span class="cffclov">Interestingly, this has been changing over the past few decades, and most East Asian languages these days are written horizontally from left to right, especially on the web.</span></span></span><svg xmlns="http://www.w3.org/2000/svg" width="24" height="6" fill="none" viewBox="0 0 24 6" style="--origin-y:0%" class="s1vq0yeh s1k9z937"><path d="
M 0 0
C 6 0
7.199999999999999 6
12 6
C 16.8 6
18 0
24 0
Z
"></path></svg></span></p><p class="prg881n">The primary goal of logical properties is to create an abstraction that sits above these differences. Instead of setting <code class="i165vvr1">margin-left</code> for left-to-right languages and flipping it to <code class="i165vvr1">margin-right</code> for right-to-left languages, we can instead use <code class="i165vvr1">margin-inline-start</code>. The margin will automatically be applied to the correct side, depending on the page's language.</p></div></aside>
<p class="prg881n">Even though this centering method has been around forever, I still find myself reaching for it on a regular basis! It's particularly useful when we want to center a single child, without affecting any of its siblings (for example, an image in-between paragraphs in a blog post).</p>
<p class="prg881n">Let's continue on our centering journey.</p>
<h2 id="centering-with-flexbox" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-flexbox-2" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with Flexbox</h2>
<p class="prg881n">Flexbox is designed to give us a <em style="color:inherit" class="w10oesj0">ton</em> of control when it comes to distributing a group of items along a primary axis. It offers some <em style="color:inherit" class="w10oesj0">really</em> powerful tools for centering!</p>
<p class="prg881n">Let's start by centering a single element, both horizontally and vertically:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div class="rcwtqfe"><div class="c9vkmf2"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="djxxsdm c9vkmf2"><div data-use-adaptive-colors="true" style="width:100%;height:200px" class="c17eovmj l17lpp8k"><div class="f1vkqrq wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div></div></div>
<p class="prg881n">The really cool thing about Flexbox centering is that it works <i>even when the children don’t fit in their container!</i> Try shrinking the width/height, and notice that the element overflows symmetrically.</p>
<p class="prg881n">It also works for <em style="color:inherit" class="w10oesj0">multiple</em> children. We can control how they stack with the <code class="i165vvr1">flex-direction</code> property:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Flex Direction<!-- -->:</span><div id=":Rhjeiavfelb:_flexDirection" class="ray0b30 w73h7tw"><div class="r1cct0ab"><button value="row" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">row</span></button><button value="column" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column</span></button><button value="row-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">row-reverse</span></button><button value="column-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column-reverse</span></button></div></div></div></div><div class="r1xmilea"><div class="cgnjx9w c2tip9q"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> flex-direction</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> row</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> gap</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 4</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d1iw6ryr c2tip9q"><div data-use-adaptive-colors="true" style="width:100%;min-height:212px;flex-direction:row" class="c1oa8tf1 l17lpp8k"><div class="bt1ddka wythvuy"><span class="t16m3o3k">1</span></div><div style="z-index:2" class="bt1ddka wythvuy"><span class="t16m3o3k">2</span></div><div class="bt1ddka wythvuy"><span class="t16m3o3k">3</span></div></div></div></div></div></div>
<p class="prg881n">Out of all the centering patterns we'll explore in this tutorial, this is probably the one I use the most. It's a great jack-of-all-trades, a great default option.</p>
<h2 id="centering-within-the-viewport" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-within-the-viewport-3" href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering within the viewport</h2>
<p class="prg881n">So far, we've been looking at how to center an element within its parent container. But what if we want to center an element in a different context? Certain elements like dialogs, prompts, and GDPR banners need to be centered within the viewport.</p>
<p class="prg881n">This is the domain of <em style="color:inherit" class="w10oesj0">positioned layout,</em> a layout mode used when we want to take something out of flow and anchor it to something else.</p>
<p class="prg881n">Here's what this looks like:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerWidth" min="30" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="rd1irhw"><div class="c1fq67g4"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> inset</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 0</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 5</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">vw</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">dvh</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dougf27 c1fq67g4"><div data-use-adaptive-colors="true" style="width:100%;height:17.625rem" class="c8q6jp9 l17lpp8k"><div class="bfhwaun"><div class="hiqwc5b"><span style="width:3rem" class="hktm3y9 bx3nfyo"></span><span style="width:5rem" class="hktm3y9 bx3nfyo"></span><span style="width:1rem" class="hktm3y9 bx3nfyo"></span><span style="width:4rem" class="hktm3y9 bx3nfyo"></span><span style="width:7rem" class="hktm3y9 bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:1rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2rem" class="bx3nfyo"></span><span style="width:4.75rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:7rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:5rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div></div><div class="hpafenu m148wj37"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="h1b4sgtf m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="h1b4sgtf m101ijsn"></div></div><div class="vm87swv m148wj37"><div style="clip-path:polygon(0% 0%, 100% 0%, 100% 0%, 0% 0%)" class="v15iwzns m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(0% 100%, 100% 100%, 100% 100%, 0% 100%)" class="v15iwzns m101ijsn"></div></div><div class="m1b7mso0 p1j9axfr"></div></div></div></div></div></div></div>
<p class="prg881n">Of all the strategies we'll discuss, this one is probably the most complex. Let's break it down.</p>
<p class="prg881n">We're using <code class="i165vvr1">position: fixed</code>, which anchors this element to the viewport. I like to think of the viewport like a pane of glass that sits in front of the website, like the window of a train that shows the landscape scrolling by. An element with <code class="i165vvr1">position: fixed</code> is like a ladybug that lands on the window.</p>
<p class="prg881n">Next, we're setting <code class="i165vvr1">inset: 0px</code>, which is a shorthand that sets <code class="i165vvr1">top</code>, <code class="i165vvr1">left</code>, <code class="i165vvr1">right</code>, and <code class="i165vvr1">bottom</code> all to the same value, <code class="i165vvr1">0px</code>.</p>
<p class="prg881n">With only these two properties, the element would stretch to fill the entire viewport, growing so that it's 0px from each edge. This can be useful in some contexts, but it's not what we're going for here. We need to constrain it.</p>
<p class="prg881n">The exact values we pick will vary on the specifics of each situation, but in general we want to set default values (with <code class="i165vvr1">width</code> and <code class="i165vvr1">height</code>), as well as max values (<code class="i165vvr1">max-width</code> and <code class="i165vvr1">max-height</code>), so that the element doesn't overflow on smaller viewports.</p>
<p class="prg881n"><strong>There's something interesting here:</strong> we've set up an impossible condition. Our element can't be 0px from the left <i>and</i> 0px from the right <i>and</i> only 12rem wide (assuming the viewport is wider than 12rem). We can only pick 2:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Pick two<!-- -->:</span><div id=":Rideiavfelb:_values" class="w73h7tw"><div class="r1cct0ab"><button value="left: 0px" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">left: 0px</span></button><button value="right: 0px" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">right: 0px</span></button><button value="width: 12rem" tabindex="-1" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">width: 12rem</span></button></div></div></div></div><div class="r3vmhao"><div class="c7zul9b"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d19iwc1r c7zul9b"><div data-use-adaptive-colors="true" class="ctnf89z l17lpp8k"><div class="b14ie1by"><div class="fujgo4n"><span style="width:3rem" class="h1vx44g7 byjv7xx"></span><span style="width:5rem" class="h1vx44g7 byjv7xx"></span><span style="width:1rem" class="h1vx44g7 byjv7xx"></span><span style="width:4rem" class="h1vx44g7 byjv7xx"></span><span style="width:7rem" class="h1vx44g7 byjv7xx"></span></div><div class="f1p79fid"><span style="width:1rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2rem" class="byjv7xx"></span><span style="width:4.75rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:7rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:5rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:2.1rem" class="byjv7xx"></span><span style="width:0.785rem" class="byjv7xx"></span><span style="width:2.5rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:3.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span></div><div class="f1p79fid"><span style="width:0.785rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></spa ... |
http://localhost:1200/joshwcomeau/popular - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>Popular Content | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com</link>
<atom:link href="http://localhost:1200/joshwcomeau/popular" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on React, CSS, Animation, and more! - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Mon, 13 Jan 2025 03:49:26 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>How To Center a Div</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/css/center-a-div/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with auto margins</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering within the viewport</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-elements-with-unknown-sizes-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering elements with unknown sizes</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-css-grid-5" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with CSS Grid</a><a href="https://www.joshwcomeau.com/css/center-a-div/#differences-from-flexbox-6" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Differences from Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-a-stack-of-elements-7" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering a stack of elements</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-text-8" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering text</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-in-the-future-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering in the future</a><a href="https://www.joshwcomeau.com/css/center-a-div/#going-beyond-the-patterns-10" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Going beyond the patterns</a><a href="https://www.joshwcomeau.com/css/center-a-div/#when-to-use-which-method-11" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">When to use which method</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">For a long time, centering an element within its parent was a surprisingly tricky thing to do. As CSS has evolved, we've been granted more and more tools we can use to solve this problem. These days, we're spoiled for choice!</p>
<p class="prg881n">I decided to create this tutorial to help you understand the trade-offs between different approaches, and to give you an arsenal of strategies you can use, to handle centering in all sorts of scenarios.</p>
<p class="prg881n">Honestly, this turned out to be <em style="color:inherit" class="w10oesj0">way more interesting</em> than I initially thought&nbsp;😅. Even if you've been using CSS for a while, I bet you'll learn at least 1 new strategy!</p>
<h2 id="centering-with-auto-margins" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-auto-margins-1" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with auto margins</h2>
<p class="prg881n">The first strategy we'll look at is one of the oldest. If we want to center an element horizontally, we can do so using margins set to the special value <code class="i165vvr1">auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgbeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgbeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-right</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-left:auto;margin-right:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgbeiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgbeiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">First, we need to constrain the element's width; by default, elements in Flow layout will expand horizontally to fill the available space, and we can't really center something that is full-width.</p>
<p class="prg881n">I <em style="color:inherit" class="w10oesj0">could</em> constrain the width with a fixed value (eg. <code class="i165vvr1">200px</code>), but really what I want in this case is for the element to shrinkwrap around its content. <code class="i165vvr1">fit-content</code> is a magical value that does exactly this. Essentially, it makes “width” behave like “height”, so that the element’s size is determined by its contents.</p>
<p class="prg881n"><strong>Why am I setting <code class="i165vvr1">max-width</code> instead of <code class="i165vvr1">width</code>?</strong> Well, my goal is to stop the element from expanding horizontally. I want to clamp its maximum size. If I used <code class="i165vvr1">width</code> instead, it would lock it to that size, and the element would overflow when the container is really narrow. If you drag that “Container Width” slider all the way to the left, you can see that the element shrinks with its container.</p>
<p class="prg881n">Now that our element is constrained, we can center it with <em style="color:inherit" class="w10oesj0">auto margins</em>.</p>
<p class="prg881n">I like to think of auto margins like <i>Hungry Hungry Hippos</i>. Each auto margin will try to gobble up as much space as possible. For example, check out what happens if we <em style="color:inherit" class="w10oesj0">only</em> set <code class="i165vvr1">margin-left: auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgneiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgneiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div></div><div style="margin-left:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgneiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgneiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">When <code class="i165vvr1">margin-left</code> is the only side with auto margins, <i>all</i> of the extra space gets applied as margin to that side. When we set both <code class="i165vvr1">margin-left: auto</code> <i>and</i> <code class="i165vvr1">margin-right: auto</code>, the two hippos each gobble up an equal amount of space. This forces the element to the center.</p>
<p class="prg881n"><em style="color:inherit" class="w10oesj0">Also:</em> I've been using <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> because they're familiar, but there's a better, more-modern way to do this:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line highlighted"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-inline</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-inline:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgteiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgteiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n"><code class="i165vvr1">margin-inline</code> will set both <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> to the same value (<code class="i165vvr1">auto</code>). It has <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://caniuse.com/mdn-css_properties_margin-inline">very good browser<!-- --> <span class="p7a6phg">support<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R1e8veiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R1e8veiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a>, having landed in all major browsers several years ago.</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-logical-properties" class="t1xqw7pl">Logical properties</strong><div class="c690lmu"><p class="prg881n"><code class="i165vvr1">margin-inline</code> is more than just a convenient shorthand for <code class="i165vvr1">margin-left</code> + <code class="i165vvr1">margin-right</code>. It's part of a collection of <em style="color:inherit" class="w10oesj0">logical properties</em>, designed to make it easier to internationalize the web.</p><p class="prg881n">In English, characters are written in a horizontal line from left to right. Those characters are composed into words and sentences, and assembled into “blocks” (paragraphs, headings, lists, etc). Blocks are stacked vertically, from top to bottom. We can think of this as the <i>orientation</i> of English-language websites.</p><p class="prg881n">This isn't universal, though! Some languages, like Arabic and Hebrew, are written from right to left. Other languages, like Chinese, have historically been written vertically, with characters running from top to bottom, and <i>blocks</i> running side to side.<button class="t1cz8cv8"><span class="a1ni5s6f">*</span><span class="e2cu4f7"></span></button><span data-use-adaptive-colors="true" id=":Rhm91eiavfelb:" data-direction="n" role="tooltip" style="flex-direction:column" class="p1lgzcrp l17lpp8k"><span class="p833mxt"><span class="pj0hw36"><span class="cffclov">Interestingly, this has been changing over the past few decades, and most East Asian languages these days are written horizontally from left to right, especially on the web.</span></span></span><svg xmlns="http://www.w3.org/2000/svg" width="24" height="6" fill="none" viewBox="0 0 24 6" style="--origin-y:0%" class="s1vq0yeh s1k9z937"><path d="
M 0 0
C 6 0
7.199999999999999 6
12 6
C 16.8 6
18 0
24 0
Z
"></path></svg></span></p><p class="prg881n">The primary goal of logical properties is to create an abstraction that sits above these differences. Instead of setting <code class="i165vvr1">margin-left</code> for left-to-right languages and flipping it to <code class="i165vvr1">margin-right</code> for right-to-left languages, we can instead use <code class="i165vvr1">margin-inline-start</code>. The margin will automatically be applied to the correct side, depending on the page's language.</p></div></aside>
<p class="prg881n">Even though this centering method has been around forever, I still find myself reaching for it on a regular basis! It's particularly useful when we want to center a single child, without affecting any of its siblings (for example, an image in-between paragraphs in a blog post).</p>
<p class="prg881n">Let's continue on our centering journey.</p>
<h2 id="centering-with-flexbox" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-flexbox-2" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with Flexbox</h2>
<p class="prg881n">Flexbox is designed to give us a <em style="color:inherit" class="w10oesj0">ton</em> of control when it comes to distributing a group of items along a primary axis. It offers some <em style="color:inherit" class="w10oesj0">really</em> powerful tools for centering!</p>
<p class="prg881n">Let's start by centering a single element, both horizontally and vertically:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div class="rcwtqfe"><div class="c9vkmf2"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="djxxsdm c9vkmf2"><div data-use-adaptive-colors="true" style="width:100%;height:200px" class="c17eovmj l17lpp8k"><div class="f1vkqrq wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div></div></div>
<p class="prg881n">The really cool thing about Flexbox centering is that it works <i>even when the children don’t fit in their container!</i> Try shrinking the width/height, and notice that the element overflows symmetrically.</p>
<p class="prg881n">It also works for <em style="color:inherit" class="w10oesj0">multiple</em> children. We can control how they stack with the <code class="i165vvr1">flex-direction</code> property:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Flex Direction<!-- -->:</span><div id=":Rhjeiavfelb:_flexDirection" class="ray0b30 w73h7tw"><div class="r1cct0ab"><button value="row" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">row</span></button><button value="column" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column</span></button><button value="row-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">row-reverse</span></button><button value="column-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column-reverse</span></button></div></div></div></div><div class="r1xmilea"><div class="cgnjx9w c2tip9q"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> flex-direction</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> row</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> gap</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 4</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d1iw6ryr c2tip9q"><div data-use-adaptive-colors="true" style="width:100%;min-height:212px;flex-direction:row" class="c1oa8tf1 l17lpp8k"><div class="bt1ddka wythvuy"><span class="t16m3o3k">1</span></div><div style="z-index:2" class="bt1ddka wythvuy"><span class="t16m3o3k">2</span></div><div class="bt1ddka wythvuy"><span class="t16m3o3k">3</span></div></div></div></div></div></div>
<p class="prg881n">Out of all the centering patterns we'll explore in this tutorial, this is probably the one I use the most. It's a great jack-of-all-trades, a great default option.</p>
<h2 id="centering-within-the-viewport" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-within-the-viewport-3" href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering within the viewport</h2>
<p class="prg881n">So far, we've been looking at how to center an element within its parent container. But what if we want to center an element in a different context? Certain elements like dialogs, prompts, and GDPR banners need to be centered within the viewport.</p>
<p class="prg881n">This is the domain of <em style="color:inherit" class="w10oesj0">positioned layout,</em> a layout mode used when we want to take something out of flow and anchor it to something else.</p>
<p class="prg881n">Here's what this looks like:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerWidth" min="30" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="rd1irhw"><div class="c1fq67g4"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> inset</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 0</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 5</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">vw</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">dvh</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dougf27 c1fq67g4"><div data-use-adaptive-colors="true" style="width:100%;height:17.625rem" class="c8q6jp9 l17lpp8k"><div class="bfhwaun"><div class="hiqwc5b"><span style="width:3rem" class="hktm3y9 bx3nfyo"></span><span style="width:5rem" class="hktm3y9 bx3nfyo"></span><span style="width:1rem" class="hktm3y9 bx3nfyo"></span><span style="width:4rem" class="hktm3y9 bx3nfyo"></span><span style="width:7rem" class="hktm3y9 bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:1rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2rem" class="bx3nfyo"></span><span style="width:4.75rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:7rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:5rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div></div><div class="hpafenu m148wj37"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="h1b4sgtf m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="h1b4sgtf m101ijsn"></div></div><div class="vm87swv m148wj37"><div style="clip-path:polygon(0% 0%, 100% 0%, 100% 0%, 0% 0%)" class="v15iwzns m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(0% 100%, 100% 100%, 100% 100%, 0% 100%)" class="v15iwzns m101ijsn"></div></div><div class="m1b7mso0 p1j9axfr"></div></div></div></div></div></div></div>
<p class="prg881n">Of all the strategies we'll discuss, this one is probably the most complex. Let's break it down.</p>
<p class="prg881n">We're using <code class="i165vvr1">position: fixed</code>, which anchors this element to the viewport. I like to think of the viewport like a pane of glass that sits in front of the website, like the window of a train that shows the landscape scrolling by. An element with <code class="i165vvr1">position: fixed</code> is like a ladybug that lands on the window.</p>
<p class="prg881n">Next, we're setting <code class="i165vvr1">inset: 0px</code>, which is a shorthand that sets <code class="i165vvr1">top</code>, <code class="i165vvr1">left</code>, <code class="i165vvr1">right</code>, and <code class="i165vvr1">bottom</code> all to the same value, <code class="i165vvr1">0px</code>.</p>
<p class="prg881n">With only these two properties, the element would stretch to fill the entire viewport, growing so that it's 0px from each edge. This can be useful in some contexts, but it's not what we're going for here. We need to constrain it.</p>
<p class="prg881n">The exact values we pick will vary on the specifics of each situation, but in general we want to set default values (with <code class="i165vvr1">width</code> and <code class="i165vvr1">height</code>), as well as max values (<code class="i165vvr1">max-width</code> and <code class="i165vvr1">max-height</code>), so that the element doesn't overflow on smaller viewports.</p>
<p class="prg881n"><strong>There's something interesting here:</strong> we've set up an impossible condition. Our element can't be 0px from the left <i>and</i> 0px from the right <i>and</i> only 12rem wide (assuming the viewport is wider than 12rem). We can only pick 2:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Pick two<!-- -->:</span><div id=":Rideiavfelb:_values" class="w73h7tw"><div class="r1cct0ab"><button value="left: 0px" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">left: 0px</span></button><button value="right: 0px" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">right: 0px</span></button><button value="width: 12rem" tabindex="-1" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">width: 12rem</span></button></div></div></div></div><div class="r3vmhao"><div class="c7zul9b"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d19iwc1r c7zul9b"><div data-use-adaptive-colors="true" class="ctnf89z l17lpp8k"><div class="b14ie1by"><div class="fujgo4n"><span style="width:3rem" class="h1vx44g7 byjv7xx"></span><span style="width:5rem" class="h1vx44g7 byjv7xx"></span><span style="width:1rem" class="h1vx44g7 byjv7xx"></span><span style="width:4rem" class="h1vx44g7 byjv7xx"></span><span style="width:7rem" class="h1vx44g7 byjv7xx"></span></div><div class="f1p79fid"><span style="width:1rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2rem" class="byjv7xx"></span><span style="width:4.75rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:7rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:5rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:2.1rem" class="byjv7xx"></span><span style="width:0.785rem" class="byjv7xx"></span><span style="width:2.5rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:3.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span></div><div class="f1p79fid"><span style="width:0.785rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span styl |
Successfully generated as following: http://localhost:1200/joshwcomeau/latest/javascript - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>JavaScript | Articles and Tutorials | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com/javascript</link>
<atom:link href="http://localhost:1200/joshwcomeau/latest/javascript" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on JavaScript | - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Sat, 18 Jan 2025 04:57:40 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>Promises From The Ground Up</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/javascript/promises/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/javascript/promises/#why-would-they-design-it-this-way-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Why would they design it this way??</a><a href="https://www.joshwcomeau.com/javascript/promises/#callbacks-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Callbacks</a><a href="https://www.joshwcomeau.com/javascript/promises/#introducing-promises-3" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Introducing Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#working-with-promises-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Working with Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#creating-our-own-promises-5" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Creating our own Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#chaining-promises-6" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Chaining Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#passing-data-7" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Passing data</a><a href="https://www.joshwcomeau.com/javascript/promises/#rejected-promises-8" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Rejected Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#async--await-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Async / Await</a><a href="https://www.joshwcomeau.com/javascript/promises/#more-to-come-10" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">More to come!</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">There are a lot of speed bumps and potholes on the road to JavaScript proficiency. One of the biggest and most daunting is <em style="color:inherit" class="w10oesj0">Promises.</em></p>
<p class="prg881n">In order to understand Promises, we need a surprisingly deep understanding of how JavaScript works and what its limitations are. Without that context, Promises won’t really make much sense.</p>
<p class="prg881n">It can be frustrating because the Promises API is <i>so important</i> nowadays. It’s become the de facto way of working with asynchronous code. Modern web APIs are built on top of Promises. There’s no getting around it: if we want to be productive with JavaScript, it really helps to understand Promises.</p>
<p class="prg881n">So, in this tutorial, we’re going to learn about Promises, but we’ll start at the beginning. I’ll share all of the critical bits of context that took me years to understand. And by the end, hopefully, you’ll have a much deeper understanding of what Promises are and how to use them effectively. ✨</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-intended-audience" class="t1xqw7pl">Intended audience</strong><div class="c690lmu"><p class="prg881n">This blog post is intended for beginner-to-intermediate JavaScript developers. Some knowledge of basic JavaScript syntax is assumed.</p></div></aside>
<h2 id="why-would-they-design-it-this-way" data-element-type="ContentHeading" class="hrk0sy8"><a id="why-would-they-design-it-this-way-1" href="https://www.joshwcomeau.com/javascript/promises/#why-would-they-design-it-this-way-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Why would they design it this way??</h2>
<p class="prg881n">Suppose we wanted to build a <em style="color:inherit" class="w10oesj0">Happy New Year!</em> countdown, something like this:</p>
<!-- --><div class="w52b1qq"><p class="v793fuv">––</p><button data-disabled="false" class="b1b5r9ns dj4vswd">Start Countdown</button></div>
<p class="prg881n">If JavaScript was like most other programming languages, we could solve the problem like this:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA">function</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> newYearsCountdown</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">()</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">3</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">2</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">1</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">Happy New Year! 🎉</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">}</span></span></code></pre></div></div>
<p class="prg881n">In this hypothetical code snippet, the program would pause when it hits a <code class="i165vvr1">sleep()</code> call, and then resume after the specified amount of time has passed.</p>
<p class="prg881n">Unfortunately, there is no <code class="i165vvr1">sleep</code> function in JavaScript, because it’s a <em style="color:inherit" class="w10oesj0">single-threaded</em> language.<button class="t1cz8cv8"><span class="a1ni5s6f">*</span><span class="e2cu4f7"></span></button><span data-use-adaptive-colors="true" id=":R1gneiavfelb:" data-direction="n" role="tooltip" style="flex-direction:column" class="p1lgzcrp l17lpp8k"><span class="p833mxt"><span class="pj0hw36"><span class="cffclov">Technically, modern JavaScript has access to multiple threads via Web Workers, but those extra threads don't have access to the DOM, so they can’t really be used in most situations.</span></span></span><svg xmlns="http://www.w3.org/2000/svg" width="24" height="6" fill="none" viewBox="0 0 24 6" style="--origin-y:0%" class="s1vq0yeh s1k9z937"><path d="
M 0 0
C 6 0
7.199999999999999 6
12 6
C 16.8 6
18 0
24 0
Z
"></path></svg></span> A “thread” is a long-running process that executes code. JavaScript only has one thread, and so it can only do one thing at a time. It can’t multitask. This is a problem because if our lone JavaScript thread is busy managing this countdown timer, it can’t do anything <i>else</i>.</p>
<p class="prg881n">When I was first learning about this stuff, it wasn’t immediately obvious to me why this was a problem. If the countdown timer is the only thing happening right now, isn’t it fine if the JS thread was fully occupied during that time??</p>
<p class="prg881n">Well, even though JavaScript doesn’t have a <code class="i165vvr1">sleep</code> function, it <i>does</i> have some other functions that occupy the main thread for an extended amount of time. We can use those other methods to get a glimpse into what it would be like if JavaScript had a <code class="i165vvr1">sleep</code> function.</p>
<p class="prg881n">For example, <code class="i165vvr1">window.prompt()</code>. This function is used to gather information from the user, and it halts execution of our code much like our hypothetical <code class="i165vvr1">sleep()</code> function would.</p>
<p class="prg881n">Click the button in this playground, and then <strong>try to interact with the page</strong> while the prompt is open:</p>
<!-- --><div data-stacked="false" class="il1gxx9"><div style="--color-text:hsl(210deg 10% 90%);--color-background:hsl(210deg 15% 6%);--color-blurred-background:hsl(210deg 15% 6% / 0.75);--color-muted-background:hsl(210deg 38% 15% / 0.85);--color-action:hsl(240deg 95% 62%);--color-primary:hsl(225deg 100% 75%);--color-secondary:hsl(333deg 100% 55%);--color-tertiary:hsl(280deg 100% 85%);--color-decorative:hsl(200deg 50% 60%);--color-info-100:hsl(214deg 40% 15%);--color-info-200:hsl(218deg 32% 20%);--color-info-300:hsl(218deg 30% 25%);--color-info-400:hsl(218deg 45% 40%);--color-info-500:hsl(225deg 100% 60%);--color-info-700:hsl(215deg 100% 72%);--color-warning-100:hsl(30deg 25% 11%);--color-warning-200:hsl(32deg 20% 15%);--color-warning-300:hsl(32deg 25% 20%);--color-warning-400:hsl(35deg 45% 35%);--color-warning-500:hsl(40deg 100% 50%);--color-warning-700:hsl(43deg 100% 72%);--color-success-100:hsl(176deg 35% 10%);--color-success-200:hsl(176deg 26% 14%);--color-success-300:hsl(176deg 28% 20%);--color-success-400:hsl(176deg 45% 30%);--color-success-500:hsl(160deg 100% 40%);--color-success-700:hsl(160deg 80% 65%);--color-cloud-100:hsl(210deg 15% 6%);--color-cloud-300:hsl(212deg 40% 9%);--color-cloud-400:hsl(213deg 40% 10%);--color-cloud-500:hsl(213deg 40% 12%);--color-sky-from:hsl(214deg 40% 11%);--color-sky-to:hsl(200deg 50% 30%);--color-sky-subtle:hsl(210deg 40% 16%);--color-gray-50:hsl(210deg 19% 9%);--color-gray-100:hsl(210deg 15% 12%);--color-gray-200:hsl(210deg 15% 18%);--color-gray-300:hsl(210deg 10% 30%);--color-gray-400:hsl(210deg 9% 40%);--color-gray-500:hsl(210deg 8% 50%);--color-gray-600:hsl(210deg 12% 55%);--color-gray-700:hsl(210deg 14% 66%);--color-gray-800:hsl(210deg 20% 77%);--color-gray-900:hsl(210deg 25% 88%);--color-gray-1000:hsl(210deg 25% 96%);--color-adaptive-white:hsl(210deg 25% 92%);--syntax-bg:hsl(210deg 15% 6%);--syntax-highlight:hsl(210deg 30% 18%);--syntax-txt:hsl(0deg 0% 100%);--syntax-comment:hsl(200deg 18% 51%);--syntax-prop:hsl(326deg 100% 61%);--syntax-bool:hsl(50deg 100% 50%);--syntax-val:hsl(210deg 12% 65%);--syntax-str:hsl(259deg 100% 71%);--syntax-name:hsl(280deg 100% 66%);--syntax-del:hsl(0deg 100% 67%);--syntax-regex:hsl(50deg 100% 50%);--syntax-fn:hsl(195deg 100% 50%);--color-info:hsl(225deg 100% 80%);--color-info-background:hsl(225deg 100% 80% / 0.1);--color-error:hsl(340deg 95% 60%);--color-error-background:hsl(340deg 95% 43% / 0.1);--color-success:hsl(160deg 100% 40%);--color-success-background:hsl(160deg 100% 40% / 0.1);--color-alert:hsl(40deg 100% 50%);--color-alert-background:hsl(40deg 100% 50% / 0.1);--color-page-background:hsl(210deg 15% 6%);--color-page-primary:hsl(225deg 100% 75%);--kbd-background-color:hsl(210deg 9% 40%);--kbd-border-color:hsl(210deg 10% 30%);--color-code-bg:hsl(210deg 15% 12%);--color-content-outline:hsl(210deg 10% 30%)" class="i1j6zvew s1hfctpl d16kq653"><header class="w14j10eb"><p class="tkoi9ui">Code Playground</p><div class="au0cxa1"><button class="wsn2i55 a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7;overflow:visible" class="s1gvsyfu"><g style="transform:rotate(-45deg)" class="w15rf7yg"><rect x="0" y="8" width="24" height="6" rx="2" fill="var(--color-gray-100)"></rect><line x1="18" y1="8" x2="18" y2="14"></line></g></svg><span class="wlgbkjj">Format code using Prettier</span></button><abbr style="display:inline-block"><div class="wy3w6yt"><button title="Reset code" class="a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7" class="s1gvsyfu"><mask id="skip-icon-mask-:R39h1eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="black"></rect><rect x="9" y="0" width="16" height="24" fill="white"></rect></mask><line x1="5" y1="19" x2="5" y2="5"></line><polygon points="19 20 9 12 19 4 19 20" mask="url(#skip-icon-mask-:R39h1eiavfelb:)"></polygon></svg><span class="wlgbkjj">Reset Code</span></button><div class="pszdhfz"><span style="gap:1.5px" class="w15sqd9b"><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span></span></div></div></abbr></div></header><div></div><div class="w198elmk"><div class="c1jztmaw"><div style="flex:547.4" class="f15xiyga"><div style="min-height:100%" class="whrzwqb"><div class="h1nlnw7n"><p class="tbdao9h">HTML</p></div><div in="false" style="--max-height:80vh" class="w1bgwmtf"><button class="f9ibg9h"><span class="wlgbkjj">Focus the editor. This will trap focus until you press Escape.</span></button><label><span class="wlgbkjj">Code editor:</span><div translate="no" class="c18okpq0" style="position:relative;text-align:left;box-sizing:border-box;padding:0;overflow:hidden"><pre aria-hidden="true" style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:relative;pointer-events:none;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0"><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">script</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">function</span><span class="token script language-javascript"> </span><span class="token script language-javascript function" style="color:var(--syntax-fn)">askForName</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript"> </span><span class="token script language-javascript punctuation">{</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">const</span><span class="token script language-javascript"> name </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript dom variable" style="color:var(--syntax-str)">window</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript method function property-access" style="color:var(--syntax-fn)">prompt</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">'What is your name?'</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript punctuation">;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">const</span><span class="token script language-javascript"> elem </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript dom variable" style="color:var(--syntax-str)">document</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript method function property-access" style="color:var(--syntax-fn)">querySelector</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">'#greeting'</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript punctuation">;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> elem</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript property-access">innerText</span><span class="token script language-javascript"> </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript template-string template-punctuation string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">`</span><span class="token script language-javascript template-string string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">Hi </span><span class="token script language-javascript template-string interpolation interpolation-punctuation punctuation">${</span><span class="token script language-javascript template-string interpolation">name</span><span class="token script language-javascript template-string interpolation interpolation-punctuation punctuation">}</span><span class="token script language-javascript template-string string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">!</span><span class="token script language-javascript template-string template-punctuation string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">`</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript punctuation">}</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">script</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain" style="display:inline-block">
</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">button</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag special-attr attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">onclick</span><span class="token tag special-attr attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag special-attr attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag special-attr attr-value value javascript language-javascript function" style="color:var(--syntax-fn);font-weight:var(--font-weight-medium)">askForName</span><span class="token tag special-attr attr-value value javascript language-javascript punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">(</span><span class="token tag special-attr attr-value value javascript language-javascript punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">)</span><span class="token tag special-attr attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"> Click me</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">button</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">id</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">greeting</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span></div></pre><textarea style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:absolute;top:0;left:0;height:100%;width:100%;resize:none;color:inherit;overflow:hidden;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;-webkit-text-fill-color:transparent;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0" class="npm__react-simple-code-editor__textarea" autocapitalize="off" autocomplete="off" autocorrect="off" spellcheck="false" data-gramm="false">&lt;script&gt;
function askForName() {
const name = window.prompt('What is your name?');
const elem = document.querySelector('#greeting');
elem.innerText = `Hi ${name}!`
}
&lt;/script&gt;
&lt;button onclick="askForName()"&gt;
Click me
&lt;/button&gt;
&lt;div id="greeting"&gt;&lt;/div&gt;</textarea><style>
/**
* Reset the text fill color so that placeholder is visible
*/
.npm__react-simple-code-editor__textarea:empty {
-webkit-text-fill-color: inherit !important;
}
/**
* Hack to apply on some CSS on IE10 and IE11
*/
@media all and (-ms-high-contrast: none), (-ms-high-contrast: active) {
/**
* IE doesn't support '-webkit-text-fill-color'
* So we use 'color: transparent' to make the text transparent on IE
* Unlike other browsers, it doesn't affect caret color in IE
*/
.npm__react-simple-code-editor__textarea {
color: transparent !important;
}
.npm__react-simple-code-editor__textarea::selection {
background-color: #accef7 !important;
color: transparent !important;
}
}
</style></div></label></div></div></div><button class="dijulrw"><span class="wlgbkjj">Resize editor. Use left/right arrows.</span></button><div style="flex:234.60000000000002" class="s1c0ceq0"><div style="height:100%" class="whrzwqb"><div class="h1nlnw7n"><p class="tbdao9h">Result</p><abbr title="Refresh pane"><button style="transform:rotate(0deg);opacity:0.7" class="w11udlg3 ah084m5"><svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-refresh-cw"><path d="M3 12a9 9 0 0 1 9-9 9.75 9.75 0 0 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path><path d="M21 12a9 9 0 0 1-9 9 9.75 9.75 0 0 1-6.74-2.74L3 16"></path><path d="M8 16H3v5"></path></svg><span class="wlgbkjj">Refresh results pane</span></button></abbr></div><div style="--height:100%;--flex:1" class="wqqdb70"><iframe height="100%" width="100%" title="example" frameborder="0" loading="lazy" class="fxdlgbu" referrerpolicy="no-referrer"></iframe></div></div></div></div></div></div></div>
<p class="prg881n"><strong>Notice that while the prompt is open, the page is totally unresponsive?</strong> You can't scroll, click any links, or select any text! The JavaScript thread is busy waiting for us to provide a value so that it can finish running that code. While it’s waiting, it can’t do anything <i>else</i>, and so the browser locks down the UI.</p>
<p class="prg881n">Other languages have multiple threads, and so it's no big deal if one of them gets preoccupied for a while. In JavaScript, though, we only have the one, and it’s used for everything: handling events, managing network requests, updating the UI, etc.</p>
<p class="prg881n">If we want to create a countdown, we need to find a way to do it without blocking the thread.</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-but-why-is-the-entire-ui-frozen" class="t1xqw7pl">But why is the entire UI frozen??</strong><div class="c690lmu"><p class="prg881n">In the example above with <code class="i165vvr1">window.prompt()</code>, the entire UI becomes unresponsive while the browser waits for us to provide a value.</p><p class="prg881n">This is kinda strange… the browser doesn’t rely on JavaScript to scroll the page, or to select text. So why can’t we do any of those things?</p><p class="prg881n">I think browsers work this way to prevent bugs. Scrolling the page, for example, triggers “scroll” events which can be caught and handled with JavaScript. If the JS thread is occupied while the scroll event happens, that code never runs, which could lead to bugs if the developer assumed that scroll events would always be handled.</p><p class="prg881n">It could also be a UX thing; maybe the browser disables the UI so that the user can’t ignore the prompt. Either way, though, I suspect a native <code class="i165vvr1">sleep</code> function would need to work the same way to prevent bugs.</p></div></aside>
<h2 id="callbacks" data-element-type="ContentHeading" class="hrk0sy8"><a id="callbacks-2" href="https://www.joshwcomeau.com/javascript/promises/#callbacks-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Callbacks</h2>
<p class="prg881n">The main tool in our toolbox for solving these sorts of problems is <code class="i165vvr1">setTimeout</code>. <code class="i165vvr1">setTimeout</code> is a function which accepts two arguments:</p>
<ol class="oc96fnv">
<li class="wp18jjy"><div class="l1n7gg9f">A chunk of work to do, at some point in the future.</div></li>
<li class="wp18jjy"><div class="l1n7gg9f">The amount of time to wait for.</div></li>
</ol>
<p class="prg881n">Here's an example:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">Start</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">setTimeout</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> ()</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA"> =&gt;</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF"> console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">After one second</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> },</span></span>
<span class="line"><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF"> 1000</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span></code></pre></div></div>
<p class="prg881n">The chunk of work is passed in through a function. This pattern is known as a <em style="color:inherit" class="w10oesj0">callback.</em></p>
<p class="prg881n">The hypothetical <code class="i165vvr1">sleep()</code> function we saw before is like calling a company and waiting on hold for the next available representative. <code class="i165vvr1">setTimeout()</code> is like pressing <i>1</i> to have them <em style="color:inherit" class="w10oesj0">call you back</em> when the representative is available. You can hang up the phone and get on with your life.</p>
<p class="prg881n"><code class="i165vvr1">setTimeout()</code> is known as an <em style="color:inherit" class="w10oesj0">asynchronous</em> function. This means that it doesn’t block the thread. By contrast, <code class="i165vvr1">window.prompt()</code> is <em style="color:inherit" class="w10oesj0">synchronous</em>, because the JavaScript thread can't do anything else while it’s waiting.</p>
<p class="prg881n">The big downside with asynchronous code is that it means our code won't always run in a linear order. Consider the following setup:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">1. Before setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">setTimeout</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">()</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA"> =&gt;</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF"> console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">2. Inside setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">},</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF"> 500</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">3. After setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span></code></pre></div></div>
<p class="prg881n">You might expect these logs to fire in order from top to bottom: <code class="i165vvr1">1</code> &gt; <code class="i165vvr1">2</code> &gt; <code class="i165vvr1">3</code>. <strong>But remember, the whole idea with callbacks is that we’re scheduling a call back.</strong> The JavaScript thread doesn’t sit around and wait, it ke ... |
http://localhost:1200/joshwcomeau/latest - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>Articles and Tutorials | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com</link>
<atom:link href="http://localhost:1200/joshwcomeau/latest" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on React, CSS, Animation, and more! - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Sat, 18 Jan 2025 04:57:41 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>Next-level frosted glass with backdrop-filter</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/css/backdrop-filter/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#css-filters-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">CSS filters</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#the-issue-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">The Issue</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#pointer-events-3" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Pointer events</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#flickering-top-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Flickering top</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#thicker-glass-5" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Thicker glass</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#browser-support-6" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Browser support</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#glassy-edge-7" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Glassy edge</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#the-final-code-8" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">The final code</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#continue-learning-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Continue learning</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">One of my all-time favourite CSS tricks is using <code class="i165vvr1">backdrop-filter: blur()</code> to create a frosted glass effect. I use it in just about every project I work on, including this blog!</p>
<p class="prg881n">Here’s a quick demo, to show what I’m talking about:</p>
<!-- --><div class="o11h7o12 w1bd8cgi" style="overflow:hidden"><div class="hduekzx"><div class="t1xdhus0"><div class="t1vhg4x9"><div class="rptyea t8ndwpz"></div><div class="y1qao05t t8ndwpz"></div><div class="g1v2ob21 t8ndwpz"></div></div><div class="tj6926w"><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="lep27c6 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(0deg);transform-origin:center center"></path></svg><span>Some Example Website</span><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="r2mz821 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(90deg);transform-origin:center center"></path></svg></div></div><div class="a13tcou3"><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-left"><path d="m12 19-7-7 7-7"></path><path d="M19 12H5"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-right"><path d="M5 12h14"></path><path d="m12 5 7 7-7 7"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-rotate-cw"><path d="M21 12a9 9 0 1 1-9-9c2.52 0 4.93 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path></svg></div><div class="a1booqr"><span>https://www.joshwcomeau.com/example-website</span></div></div></div><div style="overflow:clip"><div class="wi9xlz4 s1hfctpl d16kq653"><div class="hpv2kfm"><div class="b1dh7eg0"></div><div class="bi95upp"></div><p class="l19j3i1d">Some Website</p></div><div class="cufzfud"><img src="https://www.joshwcomeau.com/images/backdrop-filter/cupcake.png" style="animation-play-state:paused" class="c19tvtvc" referrerpolicy="no-referrer"><p>This is an example website showing how I typically use<!-- --> <code class="i165vvr1">backdrop-filter</code> to create glassy headers.</p><p class="o1fnc03u">Notice that as the cupcake moves behind the header, it appears blurry, as it would if it was passing behind frosted glass.</p><div class="atopy0n"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1t96eke w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><span style="width:20px;height:20px;--duration:1422.2222222222222ms;--size:20px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div></div></div>
<p class="prg881n">This effect helps us add depth and realism to our projects. It’s lovely.</p>
<p class="prg881n"><strong>But when I see this effect in the wild, it’s almost always missing some crucial optimizations.</strong> A couple of small changes can make our frosted glass <em style="color:inherit" class="w10oesj0">so</em> much more lush and realistic!</p>
<p class="prg881n">In this post, you’ll learn how to make the slickest frosted glass ever ✨. We’ll also learn quite a bit about CSS filters along the way!</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-credit-where-credit-is-due" class="t1xqw7pl">Credit where credit is due</strong><div class="c690lmu"><p class="prg881n">I learned about this effect from <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://expensive.toys/">Artur<!-- --> <span class="p7a6phg">Bien’s<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R2qm8deiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R2qm8deiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a> incredible demos. He credits <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://www.jamiegray.net/">Jamie<!-- --> <span class="p7a6phg">Gray<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R2sm8deiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R2sm8deiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a> for the original concept.</p></div></aside>
<h2 id="css-filters" data-element-type="ContentHeading" class="hrk0sy8"><a id="css-filters-1" href="https://www.joshwcomeau.com/css/backdrop-filter/#css-filters-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>CSS filters</h2>
<p class="prg881n">To briefly explain the underlying concept: CSS gives us quick and easy access to SVG filters via the <code class="i165vvr1">filter</code> property.</p>
<p class="prg881n">For example, we can give elements a Gaussian blur with <code class="i165vvr1">filter: blur()</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="wskxkqa"><img alt="My 3D mascot smiling with his tongue out. It’s very blurry, but dragging the slider makes it clearer" src="https://www.joshwcomeau.com/images/josh-originals/josh-tongue-dark.png" loading="lazy" style="filter:blur(16px)" class="j5sqeuc" referrerpolicy="no-referrer"><div class="c162apj7"></div></div><div style="--template-cols:1fr 1fr;grid-template-columns:1fr" class="c1j2qstn"><div style="max-width:400px;width:100%;margin-inline:auto" class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgleiavfelb:_radius" class="lpsyt9t">Blur radius<!-- -->:</label><span class="w1194iyg">16px</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgleiavfelb:_radius" min="0" max="16" class="i1i8zjy7" value="16"></div></div></div></div></div>
<p class="prg881n">There are lots of fun filter options, the sorts of things you’d find in image-editing software. Like, rotating the hue of all the colors:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1d4uhch"><img alt="My 3D mascot making a puzzled face. Dragging the slider causes him to become unnatural shades of green and pink." src="https://www.joshwcomeau.com/images/newsletter/joy-of-react-mascot.png" loading="lazy" style="filter:hue-rotate(0deg)" class="m1wtl9k0" referrerpolicy="no-referrer"><div class="c1kud80y"></div></div><div style="--template-cols:1fr 1fr;grid-template-columns:1fr" class="c1j2qstn"><div style="max-width:400px;width:100%;margin-inline:auto" class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgpeiavfelb:_rotation" class="lpsyt9t">Hue rotation<!-- -->:</label><span class="w1194iyg">0deg</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgpeiavfelb:_rotation" min="0" max="360" class="i1i8zjy7" value="0"></div></div></div></div></div>
<p class="prg881n">In these examples, I’m applying the filters to an <code class="i165vvr1">&lt;img&gt;</code> tag, but we can apply them to standard DOM nodes as well:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1icc7dx"><div style="filter:blur(0px) sepia(0%)" class="c1l5p6hh"><p>Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged.</p></div><div class="c1koe0ch"></div></div><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_radius" class="lpsyt9t">Blur radius<!-- -->:</label><span class="w1194iyg">0px</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_radius" min="0" max="16" step="0.1" class="i1i8zjy7" value="0"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_sepia" class="lpsyt9t">Sepia percentage<!-- -->:</label><span class="w1194iyg">0%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_sepia" min="0" max="100" class="i1i8zjy7" value="0"></div></div></div></div></div>
<p class="prg881n">Pretty neat, right?</p>
<p class="prg881n">Things get even cooler with <code class="i165vvr1">backdrop-filter</code>. This property lets us apply these same filters to the stuff <em style="color:inherit" class="w10oesj0">behind</em> a given element.</p>
<p class="prg881n">For example:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1qszusq"><div class="isvxvbh"><img src="https://www.joshwcomeau.com/images/backdrop-filter/tokyo.jpg" loading="lazy" class="tvcyp0e" referrerpolicy="no-referrer"><div style="top:80%;left:50%;--scale:2;-webkit-backdrop-filter:sepia(50%) brightness(130%);backdrop-filter:sepia(50%) brightness(130%)" class="ctfyewy"></div></div><div class="ck6v2m7"></div></div><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rh5eiavfelb:_sepia" class="lpsyt9t">Sepia<!-- -->:</label><span class="w1194iyg">50%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rh5eiavfelb:_sepia" min="0" max="100" class="i1i8zjy7" value="50"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rh5eiavfelb:_brightness" class="lpsyt9t">Brightness<!-- -->:</label><span class="w1194iyg">130%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rh5eiavfelb:_brightness" min="50" max="300" class="i1i8zjy7" value="130"></div></div></div></div></div>
<p class="prg881n">In this demo, the <code class="i165vvr1">.magic-ring</code> element sits in front of a photo (<a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://unsplash.com/photos/assorted-signages-on-buildings-nTBW1cOY1qI"> <span class="p7a6phg">source<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:Rn17eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:Rn17eiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a>). It uses the <code class="i165vvr1">backdrop-filter</code> property to apply some filtering to everything behind it, which can be used for some pretty artistic effects.</p>
<p class="prg881n">In practice, I pretty much only use <code class="i165vvr1">backdrop-filter</code> for one use case: blurring everything behind an element, usually a header, to create the “frosted glass” effect I mentioned earlier:</p>
<!-- --><div class="o11h7o12 w1bd8cgi" style="overflow:hidden"><div class="hduekzx"><div class="t1xdhus0"><div class="t1vhg4x9"><div class="rptyea t8ndwpz"></div><div class="y1qao05t t8ndwpz"></div><div class="g1v2ob21 t8ndwpz"></div></div><div class="tj6926w"><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="lep27c6 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(0deg);transform-origin:center center"></path></svg><span>Some Example Website</span><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="r2mz821 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(90deg);transform-origin:center center"></path></svg></div></div><div class="a13tcou3"><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-left"><path d="m12 19-7-7 7-7"></path><path d="M19 12H5"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-right"><path d="M5 12h14"></path><path d="m12 5 7 7-7 7"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-rotate-cw"><path d="M21 12a9 9 0 1 1-9-9c2.52 0 4.93 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path></svg></div><div class="a1booqr"><span>https://www.joshwcomeau.com/example-website</span></div></div></div><div style="overflow:clip"><div class="wi9xlz4 s1hfctpl d16kq653"><div class="hpv2kfm"><div class="b1dh7eg0"></div><div class="bi95upp"></div><p class="l19j3i1d">Some Website</p></div><div class="cufzfud"><img src="https://www.joshwcomeau.com/images/backdrop-filter/cupcake.png" style="animation-play-state:paused" class="c19tvtvc" referrerpolicy="no-referrer"><p>This is an example website showing how I typically use<!-- --> <code class="i165vvr1">backdrop-filter</code> to create glassy headers.</p><p class="o1fnc03u">Notice that as the cupcake moves behind the header, it appears blurry, as it would if it was passing behind frosted glass.</p><div class="atopy0n"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1t96eke w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><span style="width:20px;height:20px;--duration:1422.2222222222222ms;--size:20px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div></div></div>
<p class="prg881n">Alright. Let’s talk about the thing most developers miss.</p>
<h2 id="the-issue" data-element-type="ContentHeading" class="hrk0sy8"><a id="the-issue-2" href="https://www.joshwcomeau.com/css/backdrop-filter/#the-issue-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>The Issue</h2>
<p class="prg881n"><strong>Here’s the problem:</strong> The <code class="i165vvr1">backdrop-filter</code> algorithm only considers the pixels that are <i>directly behind</i> the element.</p>
<p class="prg881n">For filters like <code class="i165vvr1">brightness</code> or <code class="i165vvr1">hue-rotate</code>, that makes perfect sense. With blur, though, we actually want to consider pixels that are <em style="color:inherit" class="w10oesj0">near</em> the element too.</p>
<p class="prg881n">This is one of those things where a demo is worth a thousand words. Check out the difference:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="i1df5wci s360ott"><div id=":Rhneiavfelb:_mode" class="w73h7tw"><div class="r1cct0ab"><button value="default" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">Default</span></button><button value="fixed" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">Tweaked</span></button></div></div></div></div><div class="wr5us8n s1hfctpl d16kq653"><div class="hhqru8"><div class="g8qz4zc"></div></div><div style="animation-play-state:running" class="b1uj7lhw"></div></div><div class="awqze7l"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1rsl36b w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-pause"><rect x="14" y="4" width="4" height="16" rx="1"></rect><rect x="6" y="4" width="4" height="16" rx="1"></rect></svg></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div>
<p class="prg881n">By default, the gaussian blur algorithm is applied to all of the pixels behind the element. This means that if a big colorful element is <em style="color:inherit" class="w10oesj0">near</em> the element, it won’t have any effect.</p>
<p class="prg881n">That’s not really how frosted glass works in real life though. Light bounces off of objects and then goes through the glass. It looks <em style="color:inherit" class="w10oesj0">so much better</em> when the blurring algorithm includes nearby content.</p>
<p class="prg881n">Unfortunately, this isn’t something we can configure directly. Instead, we need to be a bit crafty.</p>
<p class="prg881n">Here’s the code:</p>
<!-- --><div data-stacked="false" class="il1gxx9"><div style="--color-text:hsl(210deg 10% 90%);--color-background:hsl(210deg 15% 6%);--color-blurred-background:hsl(210deg 15% 6% / 0.75);--color-muted-background:hsl(210deg 38% 15% / 0.85);--color-action:hsl(240deg 95% 62%);--color-primary:hsl(225deg 100% 75%);--color-secondary:hsl(333deg 100% 55%);--color-tertiary:hsl(280deg 100% 85%);--color-decorative:hsl(200deg 50% 60%);--color-info-100:hsl(214deg 40% 15%);--color-info-200:hsl(218deg 32% 20%);--color-info-300:hsl(218deg 30% 25%);--color-info-400:hsl(218deg 45% 40%);--color-info-500:hsl(225deg 100% 60%);--color-info-700:hsl(215deg 100% 72%);--color-warning-100:hsl(30deg 25% 11%);--color-warning-200:hsl(32deg 20% 15%);--color-warning-300:hsl(32deg 25% 20%);--color-warning-400:hsl(35deg 45% 35%);--color-warning-500:hsl(40deg 100% 50%);--color-warning-700:hsl(43deg 100% 72%);--color-success-100:hsl(176deg 35% 10%);--color-success-200:hsl(176deg 26% 14%);--color-success-300:hsl(176deg 28% 20%);--color-success-400:hsl(176deg 45% 30%);--color-success-500:hsl(160deg 100% 40%);--color-success-700:hsl(160deg 80% 65%);--color-cloud-100:hsl(210deg 15% 6%);--color-cloud-300:hsl(212deg 40% 9%);--color-cloud-400:hsl(213deg 40% 10%);--color-cloud-500:hsl(213deg 40% 12%);--color-sky-from:hsl(214deg 40% 11%);--color-sky-to:hsl(200deg 50% 30%);--color-sky-subtle:hsl(210deg 40% 16%);--color-gray-50:hsl(210deg 19% 9%);--color-gray-100:hsl(210deg 15% 12%);--color-gray-200:hsl(210deg 15% 18%);--color-gray-300:hsl(210deg 10% 30%);--color-gray-400:hsl(210deg 9% 40%);--color-gray-500:hsl(210deg 8% 50%);--color-gray-600:hsl(210deg 12% 55%);--color-gray-700:hsl(210deg 14% 66%);--color-gray-800:hsl(210deg 20% 77%);--color-gray-900:hsl(210deg 25% 88%);--color-gray-1000:hsl(210deg 25% 96%);--color-adaptive-white:hsl(210deg 25% 92%);--syntax-bg:hsl(210deg 15% 6%);--syntax-highlight:hsl(210deg 30% 18%);--syntax-txt:hsl(0deg 0% 100%);--syntax-comment:hsl(200deg 18% 51%);--syntax-prop:hsl(326deg 100% 61%);--syntax-bool:hsl(50deg 100% 50%);--syntax-val:hsl(210deg 12% 65%);--syntax-str:hsl(259deg 100% 71%);--syntax-name:hsl(280deg 100% 66%);--syntax-del:hsl(0deg 100% 67%);--syntax-regex:hsl(50deg 100% 50%);--syntax-fn:hsl(195deg 100% 50%);--color-info:hsl(225deg 100% 80%);--color-info-background:hsl(225deg 100% 80% / 0.1);--color-error:hsl(340deg 95% 60%);--color-error-background:hsl(340deg 95% 43% / 0.1);--color-success:hsl(160deg 100% 40%);--color-success-background:hsl(160deg 100% 40% / 0.1);--color-alert:hsl(40deg 100% 50%);--color-alert-background:hsl(40deg 100% 50% / 0.1);--color-page-background:hsl(210deg 15% 6%);--color-page-primary:hsl(225deg 100% 75%);--kbd-background-color:hsl(210deg 9% 40%);--kbd-border-color:hsl(210deg 10% 30%);--color-code-bg:hsl(210deg 15% 12%);--color-content-outline:hsl(210deg 10% 30%)" class="i1j6zvew s1hfctpl d16kq653"><header class="w14j10eb"><p class="tkoi9ui">Code Playground</p><div class="au0cxa1"><button class="wsn2i55 a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7;overflow:visible" class="s1gvsyfu"><g style="transform:rotate(-45deg)" class="w15rf7yg"><rect x="0" y="8" width="24" height="6" rx="2" fill="var(--color-gray-100)"></rect><line x1="18" y1="8" x2="18" y2="14"></line></g></svg><span class="wlgbkjj">Format code using Prettier</span></button><abbr style="display:inline-block"><div class="wy3w6yt"><button title="Reset code" class="a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7" class="s1gvsyfu"><mask id="skip-icon-mask-:R39i1eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="black"></rect><rect x="9" y="0" width="16" height="24" fill="white"></rect></mask><line x1="5" y1="19" x2="5" y2="5"></line><polygon points="19 20 9 12 19 4 19 20" mask="url(#skip-icon-mask-:R39i1eiavfelb:)"></polygon></svg><span class="wlgbkjj">Reset Code</span></button><div class="pszdhfz"><span style="gap:1.5px" class="w15sqd9b"><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span></span></div></div></abbr></div></header><div></div><div class="w198elmk"><div class="c1jztmaw"><div style="flex:516.12" class="f15xiyga"><div class="w16o0eiw"><div class="tt2ystl"><button style="--weight:var(--font-weight-bold);--color:var(--color-text)" class="ts2dciv">HTML</button><button style="--weight:var(--font-weight-normal);--color:var(--color-gray-500)" class="ts2dciv">CSS</button></div><div in="false" style="--max-height:80vh" class="w1bgwmtf"><button class="f9ibg9h"><span class="wlgbkjj">Focus the editor. This will trap focus until you press Escape.</span></button><label><span class="wlgbkjj">Code editor:</span><div translate="no" class="c18okpq0" style="position:relative;text-align:left;box-sizing:border-box;padding:0;overflow:hidden"><pre aria-hidden="true" style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:relative;pointer-events:none;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0"><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">style</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css selector" style="color:var(--syntax-name)">header</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">{</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">position</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> relative</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">}</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css selector class" style="color:var(--syntax-name)">.backdrop</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">{</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">position</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> absolute</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">inset</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">0</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">height</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">200</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">backdrop-filter</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css function" style="color:var(--syntax-fn)">blur</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">(</span><span class="token style language-css number" style="color:var(--syntax-bool)">16</span><span class="token style language-css unit" style="color:var(--syntax-str)">px</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">)</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">mask-image</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css function" style="color:var(--syntax-fn)">linear-gradient</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">(</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> to bottom</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">,</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css color" style="color:var(--syntax-str)">black</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">0</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">50</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">,</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css color" style="color:var(--syntax-str)">transparent</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">50</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">100</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">)</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">}</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">style</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain" style="display:inline-block">
</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">header</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"> </span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">class</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">backdrop</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">header</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">class</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">ball</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"&g ... |
http://localhost:1200/joshwcomeau/popular/false - Failed ❌
... |
http://localhost:1200/joshwcomeau/popular - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>Popular Content | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com</link>
<atom:link href="http://localhost:1200/joshwcomeau/popular" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on React, CSS, Animation, and more! - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Sat, 18 Jan 2025 04:57:42 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>How To Center a Div</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/css/center-a-div/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with auto margins</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering within the viewport</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-elements-with-unknown-sizes-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering elements with unknown sizes</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-css-grid-5" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with CSS Grid</a><a href="https://www.joshwcomeau.com/css/center-a-div/#differences-from-flexbox-6" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Differences from Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-a-stack-of-elements-7" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering a stack of elements</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-text-8" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering text</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-in-the-future-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering in the future</a><a href="https://www.joshwcomeau.com/css/center-a-div/#going-beyond-the-patterns-10" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Going beyond the patterns</a><a href="https://www.joshwcomeau.com/css/center-a-div/#when-to-use-which-method-11" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">When to use which method</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">For a long time, centering an element within its parent was a surprisingly tricky thing to do. As CSS has evolved, we've been granted more and more tools we can use to solve this problem. These days, we're spoiled for choice!</p>
<p class="prg881n">I decided to create this tutorial to help you understand the trade-offs between different approaches, and to give you an arsenal of strategies you can use, to handle centering in all sorts of scenarios.</p>
<p class="prg881n">Honestly, this turned out to be <em style="color:inherit" class="w10oesj0">way more interesting</em> than I initially thought&nbsp;😅. Even if you've been using CSS for a while, I bet you'll learn at least 1 new strategy!</p>
<h2 id="centering-with-auto-margins" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-auto-margins-1" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with auto margins</h2>
<p class="prg881n">The first strategy we'll look at is one of the oldest. If we want to center an element horizontally, we can do so using margins set to the special value <code class="i165vvr1">auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgbeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgbeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-right</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-left:auto;margin-right:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgbeiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgbeiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">First, we need to constrain the element's width; by default, elements in Flow layout will expand horizontally to fill the available space, and we can't really center something that is full-width.</p>
<p class="prg881n">I <em style="color:inherit" class="w10oesj0">could</em> constrain the width with a fixed value (eg. <code class="i165vvr1">200px</code>), but really what I want in this case is for the element to shrinkwrap around its content. <code class="i165vvr1">fit-content</code> is a magical value that does exactly this. Essentially, it makes “width” behave like “height”, so that the element’s size is determined by its contents.</p>
<p class="prg881n"><strong>Why am I setting <code class="i165vvr1">max-width</code> instead of <code class="i165vvr1">width</code>?</strong> Well, my goal is to stop the element from expanding horizontally. I want to clamp its maximum size. If I used <code class="i165vvr1">width</code> instead, it would lock it to that size, and the element would overflow when the container is really narrow. If you drag that “Container Width” slider all the way to the left, you can see that the element shrinks with its container.</p>
<p class="prg881n">Now that our element is constrained, we can center it with <em style="color:inherit" class="w10oesj0">auto margins</em>.</p>
<p class="prg881n">I like to think of auto margins like <i>Hungry Hungry Hippos</i>. Each auto margin will try to gobble up as much space as possible. For example, check out what happens if we <em style="color:inherit" class="w10oesj0">only</em> set <code class="i165vvr1">margin-left: auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgneiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgneiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div></div><div style="margin-left:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgneiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgneiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">When <code class="i165vvr1">margin-left</code> is the only side with auto margins, <i>all</i> of the extra space gets applied as margin to that side. When we set both <code class="i165vvr1">margin-left: auto</code> <i>and</i> <code class="i165vvr1">margin-right: auto</code>, the two hippos each gobble up an equal amount of space. This forces the element to the center.</p>
<p class="prg881n"><em style="color:inherit" class="w10oesj0">Also:</em> I've been using <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> because they're familiar, but there's a better, more-modern way to do this:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line highlighted"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-inline</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-inline:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgteiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgteiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n"><code class="i165vvr1">margin-inline</code> will set both <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> to the same value (<code class="i165vvr1">auto</code>). It has <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://caniuse.com/mdn-css_properties_margin-inline">very good browser<!-- --> <span class="p7a6phg">support<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R1e8veiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R1e8veiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a>, having landed in all major browsers several years ago.</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-logical-properties" class="t1xqw7pl">Logical properties</strong><div class="c690lmu"><p class="prg881n"><code class="i165vvr1">margin-inline</code> is more than just a convenient shorthand for <code class="i165vvr1">margin-left</code> + <code class="i165vvr1">margin-right</code>. It's part of a collection of <em style="color:inherit" class="w10oesj0">logical properties</em>, designed to make it easier to internationalize the web.</p><p class="prg881n">In English, characters are written in a horizontal line from left to right. Those characters are composed into words and sentences, and assembled into “blocks” (paragraphs, headings, lists, etc). Blocks are stacked vertically, from top to bottom. We can think of this as the <i>orientation</i> of English-language websites.</p><p class="prg881n">This isn't universal, though! Some languages, like Arabic and Hebrew, are written from right to left. Other languages, like Chinese, have historically been written vertically, with characters running from top to bottom, and <i>blocks</i> running side to side.<button class="t1cz8cv8"><span class="a1ni5s6f">*</span><span class="e2cu4f7"></span></button><span data-use-adaptive-colors="true" id=":Rhm91eiavfelb:" data-direction="n" role="tooltip" style="flex-direction:column" class="p1lgzcrp l17lpp8k"><span class="p833mxt"><span class="pj0hw36"><span class="cffclov">Interestingly, this has been changing over the past few decades, and most East Asian languages these days are written horizontally from left to right, especially on the web.</span></span></span><svg xmlns="http://www.w3.org/2000/svg" width="24" height="6" fill="none" viewBox="0 0 24 6" style="--origin-y:0%" class="s1vq0yeh s1k9z937"><path d="
M 0 0
C 6 0
7.199999999999999 6
12 6
C 16.8 6
18 0
24 0
Z
"></path></svg></span></p><p class="prg881n">The primary goal of logical properties is to create an abstraction that sits above these differences. Instead of setting <code class="i165vvr1">margin-left</code> for left-to-right languages and flipping it to <code class="i165vvr1">margin-right</code> for right-to-left languages, we can instead use <code class="i165vvr1">margin-inline-start</code>. The margin will automatically be applied to the correct side, depending on the page's language.</p></div></aside>
<p class="prg881n">Even though this centering method has been around forever, I still find myself reaching for it on a regular basis! It's particularly useful when we want to center a single child, without affecting any of its siblings (for example, an image in-between paragraphs in a blog post).</p>
<p class="prg881n">Let's continue on our centering journey.</p>
<h2 id="centering-with-flexbox" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-flexbox-2" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with Flexbox</h2>
<p class="prg881n">Flexbox is designed to give us a <em style="color:inherit" class="w10oesj0">ton</em> of control when it comes to distributing a group of items along a primary axis. It offers some <em style="color:inherit" class="w10oesj0">really</em> powerful tools for centering!</p>
<p class="prg881n">Let's start by centering a single element, both horizontally and vertically:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div class="rcwtqfe"><div class="c9vkmf2"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="djxxsdm c9vkmf2"><div data-use-adaptive-colors="true" style="width:100%;height:200px" class="c17eovmj l17lpp8k"><div class="f1vkqrq wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div></div></div>
<p class="prg881n">The really cool thing about Flexbox centering is that it works <i>even when the children don’t fit in their container!</i> Try shrinking the width/height, and notice that the element overflows symmetrically.</p>
<p class="prg881n">It also works for <em style="color:inherit" class="w10oesj0">multiple</em> children. We can control how they stack with the <code class="i165vvr1">flex-direction</code> property:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Flex Direction<!-- -->:</span><div id=":Rhjeiavfelb:_flexDirection" class="ray0b30 w73h7tw"><div class="r1cct0ab"><button value="row" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">row</span></button><button value="column" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column</span></button><button value="row-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">row-reverse</span></button><button value="column-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column-reverse</span></button></div></div></div></div><div class="r1xmilea"><div class="cgnjx9w c2tip9q"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> flex-direction</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> row</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> gap</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 4</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d1iw6ryr c2tip9q"><div data-use-adaptive-colors="true" style="width:100%;min-height:212px;flex-direction:row" class="c1oa8tf1 l17lpp8k"><div class="bt1ddka wythvuy"><span class="t16m3o3k">1</span></div><div style="z-index:2" class="bt1ddka wythvuy"><span class="t16m3o3k">2</span></div><div class="bt1ddka wythvuy"><span class="t16m3o3k">3</span></div></div></div></div></div></div>
<p class="prg881n">Out of all the centering patterns we'll explore in this tutorial, this is probably the one I use the most. It's a great jack-of-all-trades, a great default option.</p>
<h2 id="centering-within-the-viewport" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-within-the-viewport-3" href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering within the viewport</h2>
<p class="prg881n">So far, we've been looking at how to center an element within its parent container. But what if we want to center an element in a different context? Certain elements like dialogs, prompts, and GDPR banners need to be centered within the viewport.</p>
<p class="prg881n">This is the domain of <em style="color:inherit" class="w10oesj0">positioned layout,</em> a layout mode used when we want to take something out of flow and anchor it to something else.</p>
<p class="prg881n">Here's what this looks like:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerWidth" min="30" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="rd1irhw"><div class="c1fq67g4"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> inset</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 0</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 5</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">vw</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">dvh</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dougf27 c1fq67g4"><div data-use-adaptive-colors="true" style="width:100%;height:17.625rem" class="c8q6jp9 l17lpp8k"><div class="bfhwaun"><div class="hiqwc5b"><span style="width:3rem" class="hktm3y9 bx3nfyo"></span><span style="width:5rem" class="hktm3y9 bx3nfyo"></span><span style="width:1rem" class="hktm3y9 bx3nfyo"></span><span style="width:4rem" class="hktm3y9 bx3nfyo"></span><span style="width:7rem" class="hktm3y9 bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:1rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2rem" class="bx3nfyo"></span><span style="width:4.75rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:7rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:5rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div></div><div class="hpafenu m148wj37"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="h1b4sgtf m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="h1b4sgtf m101ijsn"></div></div><div class="vm87swv m148wj37"><div style="clip-path:polygon(0% 0%, 100% 0%, 100% 0%, 0% 0%)" class="v15iwzns m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(0% 100%, 100% 100%, 100% 100%, 0% 100%)" class="v15iwzns m101ijsn"></div></div><div class="m1b7mso0 p1j9axfr"></div></div></div></div></div></div></div>
<p class="prg881n">Of all the strategies we'll discuss, this one is probably the most complex. Let's break it down.</p>
<p class="prg881n">We're using <code class="i165vvr1">position: fixed</code>, which anchors this element to the viewport. I like to think of the viewport like a pane of glass that sits in front of the website, like the window of a train that shows the landscape scrolling by. An element with <code class="i165vvr1">position: fixed</code> is like a ladybug that lands on the window.</p>
<p class="prg881n">Next, we're setting <code class="i165vvr1">inset: 0px</code>, which is a shorthand that sets <code class="i165vvr1">top</code>, <code class="i165vvr1">left</code>, <code class="i165vvr1">right</code>, and <code class="i165vvr1">bottom</code> all to the same value, <code class="i165vvr1">0px</code>.</p>
<p class="prg881n">With only these two properties, the element would stretch to fill the entire viewport, growing so that it's 0px from each edge. This can be useful in some contexts, but it's not what we're going for here. We need to constrain it.</p>
<p class="prg881n">The exact values we pick will vary on the specifics of each situation, but in general we want to set default values (with <code class="i165vvr1">width</code> and <code class="i165vvr1">height</code>), as well as max values (<code class="i165vvr1">max-width</code> and <code class="i165vvr1">max-height</code>), so that the element doesn't overflow on smaller viewports.</p>
<p class="prg881n"><strong>There's something interesting here:</strong> we've set up an impossible condition. Our element can't be 0px from the left <i>and</i> 0px from the right <i>and</i> only 12rem wide (assuming the viewport is wider than 12rem). We can only pick 2:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Pick two<!-- -->:</span><div id=":Rideiavfelb:_values" class="w73h7tw"><div class="r1cct0ab"><button value="left: 0px" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">left: 0px</span></button><button value="right: 0px" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">right: 0px</span></button><button value="width: 12rem" tabindex="-1" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">width: 12rem</span></button></div></div></div></div><div class="r3vmhao"><div class="c7zul9b"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d19iwc1r c7zul9b"><div data-use-adaptive-colors="true" class="ctnf89z l17lpp8k"><div class="b14ie1by"><div class="fujgo4n"><span style="width:3rem" class="h1vx44g7 byjv7xx"></span><span style="width:5rem" class="h1vx44g7 byjv7xx"></span><span style="width:1rem" class="h1vx44g7 byjv7xx"></span><span style="width:4rem" class="h1vx44g7 byjv7xx"></span><span style="width:7rem" class="h1vx44g7 byjv7xx"></span></div><div class="f1p79fid"><span style="width:1rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2rem" class="byjv7xx"></span><span style="width:4.75rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:7rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:5rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:2.1rem" class="byjv7xx"></span><span style="width:0.785rem" class="byjv7xx"></span><span style="width:2.5rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:3.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span></div><div class="f1p79fid"><span style="width:0.785rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span styl |
Successfully generated as following: http://localhost:1200/joshwcomeau/latest/javascript - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>JavaScript | Articles and Tutorials | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com/javascript</link>
<atom:link href="http://localhost:1200/joshwcomeau/latest/javascript" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on JavaScript | - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Sat, 18 Jan 2025 06:18:13 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>Promises From The Ground Up</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/javascript/promises/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/javascript/promises/#why-would-they-design-it-this-way-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Why would they design it this way??</a><a href="https://www.joshwcomeau.com/javascript/promises/#callbacks-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Callbacks</a><a href="https://www.joshwcomeau.com/javascript/promises/#introducing-promises-3" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Introducing Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#working-with-promises-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Working with Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#creating-our-own-promises-5" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Creating our own Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#chaining-promises-6" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Chaining Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#passing-data-7" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Passing data</a><a href="https://www.joshwcomeau.com/javascript/promises/#rejected-promises-8" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Rejected Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#async--await-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Async / Await</a><a href="https://www.joshwcomeau.com/javascript/promises/#more-to-come-10" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">More to come!</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">There are a lot of speed bumps and potholes on the road to JavaScript proficiency. One of the biggest and most daunting is <em style="color:inherit" class="w10oesj0">Promises.</em></p>
<p class="prg881n">In order to understand Promises, we need a surprisingly deep understanding of how JavaScript works and what its limitations are. Without that context, Promises won’t really make much sense.</p>
<p class="prg881n">It can be frustrating because the Promises API is <i>so important</i> nowadays. It’s become the de facto way of working with asynchronous code. Modern web APIs are built on top of Promises. There’s no getting around it: if we want to be productive with JavaScript, it really helps to understand Promises.</p>
<p class="prg881n">So, in this tutorial, we’re going to learn about Promises, but we’ll start at the beginning. I’ll share all of the critical bits of context that took me years to understand. And by the end, hopefully, you’ll have a much deeper understanding of what Promises are and how to use them effectively. ✨</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-intended-audience" class="t1xqw7pl">Intended audience</strong><div class="c690lmu"><p class="prg881n">This blog post is intended for beginner-to-intermediate JavaScript developers. Some knowledge of basic JavaScript syntax is assumed.</p></div></aside>
<h2 id="why-would-they-design-it-this-way" data-element-type="ContentHeading" class="hrk0sy8"><a id="why-would-they-design-it-this-way-1" href="https://www.joshwcomeau.com/javascript/promises/#why-would-they-design-it-this-way-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Why would they design it this way??</h2>
<p class="prg881n">Suppose we wanted to build a <em style="color:inherit" class="w10oesj0">Happy New Year!</em> countdown, something like this:</p>
<!-- --><div class="w52b1qq"><p class="v793fuv">––</p><button data-disabled="false" class="b1b5r9ns dj4vswd">Start Countdown</button></div>
<p class="prg881n">If JavaScript was like most other programming languages, we could solve the problem like this:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA">function</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> newYearsCountdown</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">()</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">3</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">2</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">1</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">Happy New Year! 🎉</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">}</span></span></code></pre></div></div>
<p class="prg881n">In this hypothetical code snippet, the program would pause when it hits a <code class="i165vvr1">sleep()</code> call, and then resume after the specified amount of time has passed.</p>
<p class="prg881n">Unfortunately, there is no <code class="i165vvr1">sleep</code> function in JavaScript, because it’s a <em style="color:inherit" class="w10oesj0">single-threaded</em> language.<button class="t1cz8cv8"><span class="a1ni5s6f">*</span><span class="e2cu4f7"></span></button><span data-use-adaptive-colors="true" id=":R1gneiavfelb:" data-direction="n" role="tooltip" style="flex-direction:column" class="p1lgzcrp l17lpp8k"><span class="p833mxt"><span class="pj0hw36"><span class="cffclov">Technically, modern JavaScript has access to multiple threads via Web Workers, but those extra threads don't have access to the DOM, so they can’t really be used in most situations.</span></span></span><svg xmlns="http://www.w3.org/2000/svg" width="24" height="6" fill="none" viewBox="0 0 24 6" style="--origin-y:0%" class="s1vq0yeh s1k9z937"><path d="
M 0 0
C 6 0
7.199999999999999 6
12 6
C 16.8 6
18 0
24 0
Z
"></path></svg></span> A “thread” is a long-running process that executes code. JavaScript only has one thread, and so it can only do one thing at a time. It can’t multitask. This is a problem because if our lone JavaScript thread is busy managing this countdown timer, it can’t do anything <i>else</i>.</p>
<p class="prg881n">When I was first learning about this stuff, it wasn’t immediately obvious to me why this was a problem. If the countdown timer is the only thing happening right now, isn’t it fine if the JS thread was fully occupied during that time??</p>
<p class="prg881n">Well, even though JavaScript doesn’t have a <code class="i165vvr1">sleep</code> function, it <i>does</i> have some other functions that occupy the main thread for an extended amount of time. We can use those other methods to get a glimpse into what it would be like if JavaScript had a <code class="i165vvr1">sleep</code> function.</p>
<p class="prg881n">For example, <code class="i165vvr1">window.prompt()</code>. This function is used to gather information from the user, and it halts execution of our code much like our hypothetical <code class="i165vvr1">sleep()</code> function would.</p>
<p class="prg881n">Click the button in this playground, and then <strong>try to interact with the page</strong> while the prompt is open:</p>
<!-- --><div data-stacked="false" class="il1gxx9"><div style="--color-text:hsl(210deg 10% 90%);--color-background:hsl(210deg 15% 6%);--color-blurred-background:hsl(210deg 15% 6% / 0.75);--color-muted-background:hsl(210deg 38% 15% / 0.85);--color-action:hsl(240deg 95% 62%);--color-primary:hsl(225deg 100% 75%);--color-secondary:hsl(333deg 100% 55%);--color-tertiary:hsl(280deg 100% 85%);--color-decorative:hsl(200deg 50% 60%);--color-info-100:hsl(214deg 40% 15%);--color-info-200:hsl(218deg 32% 20%);--color-info-300:hsl(218deg 30% 25%);--color-info-400:hsl(218deg 45% 40%);--color-info-500:hsl(225deg 100% 60%);--color-info-700:hsl(215deg 100% 72%);--color-warning-100:hsl(30deg 25% 11%);--color-warning-200:hsl(32deg 20% 15%);--color-warning-300:hsl(32deg 25% 20%);--color-warning-400:hsl(35deg 45% 35%);--color-warning-500:hsl(40deg 100% 50%);--color-warning-700:hsl(43deg 100% 72%);--color-success-100:hsl(176deg 35% 10%);--color-success-200:hsl(176deg 26% 14%);--color-success-300:hsl(176deg 28% 20%);--color-success-400:hsl(176deg 45% 30%);--color-success-500:hsl(160deg 100% 40%);--color-success-700:hsl(160deg 80% 65%);--color-cloud-100:hsl(210deg 15% 6%);--color-cloud-300:hsl(212deg 40% 9%);--color-cloud-400:hsl(213deg 40% 10%);--color-cloud-500:hsl(213deg 40% 12%);--color-sky-from:hsl(214deg 40% 11%);--color-sky-to:hsl(200deg 50% 30%);--color-sky-subtle:hsl(210deg 40% 16%);--color-gray-50:hsl(210deg 19% 9%);--color-gray-100:hsl(210deg 15% 12%);--color-gray-200:hsl(210deg 15% 18%);--color-gray-300:hsl(210deg 10% 30%);--color-gray-400:hsl(210deg 9% 40%);--color-gray-500:hsl(210deg 8% 50%);--color-gray-600:hsl(210deg 12% 55%);--color-gray-700:hsl(210deg 14% 66%);--color-gray-800:hsl(210deg 20% 77%);--color-gray-900:hsl(210deg 25% 88%);--color-gray-1000:hsl(210deg 25% 96%);--color-adaptive-white:hsl(210deg 25% 92%);--syntax-bg:hsl(210deg 15% 6%);--syntax-highlight:hsl(210deg 30% 18%);--syntax-txt:hsl(0deg 0% 100%);--syntax-comment:hsl(200deg 18% 51%);--syntax-prop:hsl(326deg 100% 61%);--syntax-bool:hsl(50deg 100% 50%);--syntax-val:hsl(210deg 12% 65%);--syntax-str:hsl(259deg 100% 71%);--syntax-name:hsl(280deg 100% 66%);--syntax-del:hsl(0deg 100% 67%);--syntax-regex:hsl(50deg 100% 50%);--syntax-fn:hsl(195deg 100% 50%);--color-info:hsl(225deg 100% 80%);--color-info-background:hsl(225deg 100% 80% / 0.1);--color-error:hsl(340deg 95% 60%);--color-error-background:hsl(340deg 95% 43% / 0.1);--color-success:hsl(160deg 100% 40%);--color-success-background:hsl(160deg 100% 40% / 0.1);--color-alert:hsl(40deg 100% 50%);--color-alert-background:hsl(40deg 100% 50% / 0.1);--color-page-background:hsl(210deg 15% 6%);--color-page-primary:hsl(225deg 100% 75%);--kbd-background-color:hsl(210deg 9% 40%);--kbd-border-color:hsl(210deg 10% 30%);--color-code-bg:hsl(210deg 15% 12%);--color-content-outline:hsl(210deg 10% 30%)" class="i1j6zvew s1hfctpl d16kq653"><header class="w14j10eb"><p class="tkoi9ui">Code Playground</p><div class="au0cxa1"><button class="wsn2i55 a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7;overflow:visible" class="s1gvsyfu"><g style="transform:rotate(-45deg)" class="w15rf7yg"><rect x="0" y="8" width="24" height="6" rx="2" fill="var(--color-gray-100)"></rect><line x1="18" y1="8" x2="18" y2="14"></line></g></svg><span class="wlgbkjj">Format code using Prettier</span></button><abbr style="display:inline-block"><div class="wy3w6yt"><button title="Reset code" class="a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7" class="s1gvsyfu"><mask id="skip-icon-mask-:R39h1eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="black"></rect><rect x="9" y="0" width="16" height="24" fill="white"></rect></mask><line x1="5" y1="19" x2="5" y2="5"></line><polygon points="19 20 9 12 19 4 19 20" mask="url(#skip-icon-mask-:R39h1eiavfelb:)"></polygon></svg><span class="wlgbkjj">Reset Code</span></button><div class="pszdhfz"><span style="gap:1.5px" class="w15sqd9b"><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span></span></div></div></abbr></div></header><div></div><div class="w198elmk"><div class="c1jztmaw"><div style="flex:547.4" class="f15xiyga"><div style="min-height:100%" class="whrzwqb"><div class="h1nlnw7n"><p class="tbdao9h">HTML</p></div><div in="false" style="--max-height:80vh" class="w1bgwmtf"><button class="f9ibg9h"><span class="wlgbkjj">Focus the editor. This will trap focus until you press Escape.</span></button><label><span class="wlgbkjj">Code editor:</span><div translate="no" class="c18okpq0" style="position:relative;text-align:left;box-sizing:border-box;padding:0;overflow:hidden"><pre aria-hidden="true" style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:relative;pointer-events:none;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0"><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">script</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">function</span><span class="token script language-javascript"> </span><span class="token script language-javascript function" style="color:var(--syntax-fn)">askForName</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript"> </span><span class="token script language-javascript punctuation">{</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">const</span><span class="token script language-javascript"> name </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript dom variable" style="color:var(--syntax-str)">window</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript method function property-access" style="color:var(--syntax-fn)">prompt</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">'What is your name?'</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript punctuation">;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">const</span><span class="token script language-javascript"> elem </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript dom variable" style="color:var(--syntax-str)">document</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript method function property-access" style="color:var(--syntax-fn)">querySelector</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">'#greeting'</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript punctuation">;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> elem</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript property-access">innerText</span><span class="token script language-javascript"> </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript template-string template-punctuation string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">`</span><span class="token script language-javascript template-string string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">Hi </span><span class="token script language-javascript template-string interpolation interpolation-punctuation punctuation">${</span><span class="token script language-javascript template-string interpolation">name</span><span class="token script language-javascript template-string interpolation interpolation-punctuation punctuation">}</span><span class="token script language-javascript template-string string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">!</span><span class="token script language-javascript template-string template-punctuation string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">`</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript punctuation">}</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">script</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain" style="display:inline-block">
</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">button</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag special-attr attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">onclick</span><span class="token tag special-attr attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag special-attr attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag special-attr attr-value value javascript language-javascript function" style="color:var(--syntax-fn);font-weight:var(--font-weight-medium)">askForName</span><span class="token tag special-attr attr-value value javascript language-javascript punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">(</span><span class="token tag special-attr attr-value value javascript language-javascript punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">)</span><span class="token tag special-attr attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"> Click me</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">button</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">id</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">greeting</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span></div></pre><textarea style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:absolute;top:0;left:0;height:100%;width:100%;resize:none;color:inherit;overflow:hidden;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;-webkit-text-fill-color:transparent;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0" class="npm__react-simple-code-editor__textarea" autocapitalize="off" autocomplete="off" autocorrect="off" spellcheck="false" data-gramm="false">&lt;script&gt;
function askForName() {
const name = window.prompt('What is your name?');
const elem = document.querySelector('#greeting');
elem.innerText = `Hi ${name}!`
}
&lt;/script&gt;
&lt;button onclick="askForName()"&gt;
Click me
&lt;/button&gt;
&lt;div id="greeting"&gt;&lt;/div&gt;</textarea><style>
/**
* Reset the text fill color so that placeholder is visible
*/
.npm__react-simple-code-editor__textarea:empty {
-webkit-text-fill-color: inherit !important;
}
/**
* Hack to apply on some CSS on IE10 and IE11
*/
@media all and (-ms-high-contrast: none), (-ms-high-contrast: active) {
/**
* IE doesn't support '-webkit-text-fill-color'
* So we use 'color: transparent' to make the text transparent on IE
* Unlike other browsers, it doesn't affect caret color in IE
*/
.npm__react-simple-code-editor__textarea {
color: transparent !important;
}
.npm__react-simple-code-editor__textarea::selection {
background-color: #accef7 !important;
color: transparent !important;
}
}
</style></div></label></div></div></div><button class="dijulrw"><span class="wlgbkjj">Resize editor. Use left/right arrows.</span></button><div style="flex:234.60000000000002" class="s1c0ceq0"><div style="height:100%" class="whrzwqb"><div class="h1nlnw7n"><p class="tbdao9h">Result</p><abbr title="Refresh pane"><button style="transform:rotate(0deg);opacity:0.7" class="w11udlg3 ah084m5"><svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-refresh-cw"><path d="M3 12a9 9 0 0 1 9-9 9.75 9.75 0 0 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path><path d="M21 12a9 9 0 0 1-9 9 9.75 9.75 0 0 1-6.74-2.74L3 16"></path><path d="M8 16H3v5"></path></svg><span class="wlgbkjj">Refresh results pane</span></button></abbr></div><div style="--height:100%;--flex:1" class="wqqdb70"><iframe height="100%" width="100%" title="example" frameborder="0" loading="lazy" class="fxdlgbu" referrerpolicy="no-referrer"></iframe></div></div></div></div></div></div></div>
<p class="prg881n"><strong>Notice that while the prompt is open, the page is totally unresponsive?</strong> You can't scroll, click any links, or select any text! The JavaScript thread is busy waiting for us to provide a value so that it can finish running that code. While it’s waiting, it can’t do anything <i>else</i>, and so the browser locks down the UI.</p>
<p class="prg881n">Other languages have multiple threads, and so it's no big deal if one of them gets preoccupied for a while. In JavaScript, though, we only have the one, and it’s used for everything: handling events, managing network requests, updating the UI, etc.</p>
<p class="prg881n">If we want to create a countdown, we need to find a way to do it without blocking the thread.</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-but-why-is-the-entire-ui-frozen" class="t1xqw7pl">But why is the entire UI frozen??</strong><div class="c690lmu"><p class="prg881n">In the example above with <code class="i165vvr1">window.prompt()</code>, the entire UI becomes unresponsive while the browser waits for us to provide a value.</p><p class="prg881n">This is kinda strange… the browser doesn’t rely on JavaScript to scroll the page, or to select text. So why can’t we do any of those things?</p><p class="prg881n">I think browsers work this way to prevent bugs. Scrolling the page, for example, triggers “scroll” events which can be caught and handled with JavaScript. If the JS thread is occupied while the scroll event happens, that code never runs, which could lead to bugs if the developer assumed that scroll events would always be handled.</p><p class="prg881n">It could also be a UX thing; maybe the browser disables the UI so that the user can’t ignore the prompt. Either way, though, I suspect a native <code class="i165vvr1">sleep</code> function would need to work the same way to prevent bugs.</p></div></aside>
<h2 id="callbacks" data-element-type="ContentHeading" class="hrk0sy8"><a id="callbacks-2" href="https://www.joshwcomeau.com/javascript/promises/#callbacks-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Callbacks</h2>
<p class="prg881n">The main tool in our toolbox for solving these sorts of problems is <code class="i165vvr1">setTimeout</code>. <code class="i165vvr1">setTimeout</code> is a function which accepts two arguments:</p>
<ol class="oc96fnv">
<li class="wp18jjy"><div class="l1n7gg9f">A chunk of work to do, at some point in the future.</div></li>
<li class="wp18jjy"><div class="l1n7gg9f">The amount of time to wait for.</div></li>
</ol>
<p class="prg881n">Here's an example:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">Start</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">setTimeout</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> ()</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA"> =&gt;</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF"> console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">After one second</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> },</span></span>
<span class="line"><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF"> 1000</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span></code></pre></div></div>
<p class="prg881n">The chunk of work is passed in through a function. This pattern is known as a <em style="color:inherit" class="w10oesj0">callback.</em></p>
<p class="prg881n">The hypothetical <code class="i165vvr1">sleep()</code> function we saw before is like calling a company and waiting on hold for the next available representative. <code class="i165vvr1">setTimeout()</code> is like pressing <i>1</i> to have them <em style="color:inherit" class="w10oesj0">call you back</em> when the representative is available. You can hang up the phone and get on with your life.</p>
<p class="prg881n"><code class="i165vvr1">setTimeout()</code> is known as an <em style="color:inherit" class="w10oesj0">asynchronous</em> function. This means that it doesn’t block the thread. By contrast, <code class="i165vvr1">window.prompt()</code> is <em style="color:inherit" class="w10oesj0">synchronous</em>, because the JavaScript thread can't do anything else while it’s waiting.</p>
<p class="prg881n">The big downside with asynchronous code is that it means our code won't always run in a linear order. Consider the following setup:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">1. Before setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">setTimeout</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">()</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA"> =&gt;</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF"> console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">2. Inside setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">},</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF"> 500</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">3. After setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span></code></pre></div></div>
<p class="prg881n">You might expect these logs to fire in order from top to bottom: <code class="i165vvr1">1</code> &gt; <code class="i165vvr1">2</code> &gt; <code class="i165vvr1">3</code>. <strong>But remember, the whole idea with callbacks is that we’re scheduling a call back.</strong> The JavaScript thread doesn’t sit around and wait, it ke ... |
http://localhost:1200/joshwcomeau/latest - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>Articles and Tutorials | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com</link>
<atom:link href="http://localhost:1200/joshwcomeau/latest" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on React, CSS, Animation, and more! - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Sat, 18 Jan 2025 06:18:14 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>Next-level frosted glass with backdrop-filter</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/css/backdrop-filter/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#css-filters-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">CSS filters</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#the-issue-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">The Issue</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#pointer-events-3" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Pointer events</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#flickering-top-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Flickering top</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#thicker-glass-5" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Thicker glass</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#browser-support-6" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Browser support</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#glassy-edge-7" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Glassy edge</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#the-final-code-8" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">The final code</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#continue-learning-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Continue learning</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">One of my all-time favourite CSS tricks is using <code class="i165vvr1">backdrop-filter: blur()</code> to create a frosted glass effect. I use it in just about every project I work on, including this blog!</p>
<p class="prg881n">Here’s a quick demo, to show what I’m talking about:</p>
<!-- --><div class="o11h7o12 w1bd8cgi" style="overflow:hidden"><div class="hduekzx"><div class="t1xdhus0"><div class="t1vhg4x9"><div class="rptyea t8ndwpz"></div><div class="y1qao05t t8ndwpz"></div><div class="g1v2ob21 t8ndwpz"></div></div><div class="tj6926w"><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="lep27c6 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(0deg);transform-origin:center center"></path></svg><span>Some Example Website</span><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="r2mz821 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(90deg);transform-origin:center center"></path></svg></div></div><div class="a13tcou3"><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-left"><path d="m12 19-7-7 7-7"></path><path d="M19 12H5"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-right"><path d="M5 12h14"></path><path d="m12 5 7 7-7 7"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-rotate-cw"><path d="M21 12a9 9 0 1 1-9-9c2.52 0 4.93 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path></svg></div><div class="a1booqr"><span>https://www.joshwcomeau.com/example-website</span></div></div></div><div style="overflow:clip"><div class="wi9xlz4 s1hfctpl d16kq653"><div class="hpv2kfm"><div class="b1dh7eg0"></div><div class="bi95upp"></div><p class="l19j3i1d">Some Website</p></div><div class="cufzfud"><img src="https://www.joshwcomeau.com/images/backdrop-filter/cupcake.png" style="animation-play-state:paused" class="c19tvtvc" referrerpolicy="no-referrer"><p>This is an example website showing how I typically use<!-- --> <code class="i165vvr1">backdrop-filter</code> to create glassy headers.</p><p class="o1fnc03u">Notice that as the cupcake moves behind the header, it appears blurry, as it would if it was passing behind frosted glass.</p><div class="atopy0n"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1t96eke w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><span style="width:20px;height:20px;--duration:1422.2222222222222ms;--size:20px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div></div></div>
<p class="prg881n">This effect helps us add depth and realism to our projects. It’s lovely.</p>
<p class="prg881n"><strong>But when I see this effect in the wild, it’s almost always missing some crucial optimizations.</strong> A couple of small changes can make our frosted glass <em style="color:inherit" class="w10oesj0">so</em> much more lush and realistic!</p>
<p class="prg881n">In this post, you’ll learn how to make the slickest frosted glass ever ✨. We’ll also learn quite a bit about CSS filters along the way!</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-credit-where-credit-is-due" class="t1xqw7pl">Credit where credit is due</strong><div class="c690lmu"><p class="prg881n">I learned about this effect from <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://expensive.toys/">Artur<!-- --> <span class="p7a6phg">Bien’s<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R2qm8deiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R2qm8deiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a> incredible demos. He credits <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://www.jamiegray.net/">Jamie<!-- --> <span class="p7a6phg">Gray<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R2sm8deiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R2sm8deiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a> for the original concept.</p></div></aside>
<h2 id="css-filters" data-element-type="ContentHeading" class="hrk0sy8"><a id="css-filters-1" href="https://www.joshwcomeau.com/css/backdrop-filter/#css-filters-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>CSS filters</h2>
<p class="prg881n">To briefly explain the underlying concept: CSS gives us quick and easy access to SVG filters via the <code class="i165vvr1">filter</code> property.</p>
<p class="prg881n">For example, we can give elements a Gaussian blur with <code class="i165vvr1">filter: blur()</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="wskxkqa"><img alt="My 3D mascot smiling with his tongue out. It’s very blurry, but dragging the slider makes it clearer" src="https://www.joshwcomeau.com/images/josh-originals/josh-tongue-dark.png" loading="lazy" style="filter:blur(16px)" class="j5sqeuc" referrerpolicy="no-referrer"><div class="c162apj7"></div></div><div style="--template-cols:1fr 1fr;grid-template-columns:1fr" class="c1j2qstn"><div style="max-width:400px;width:100%;margin-inline:auto" class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgleiavfelb:_radius" class="lpsyt9t">Blur radius<!-- -->:</label><span class="w1194iyg">16px</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgleiavfelb:_radius" min="0" max="16" class="i1i8zjy7" value="16"></div></div></div></div></div>
<p class="prg881n">There are lots of fun filter options, the sorts of things you’d find in image-editing software. Like, rotating the hue of all the colors:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1d4uhch"><img alt="My 3D mascot making a puzzled face. Dragging the slider causes him to become unnatural shades of green and pink." src="https://www.joshwcomeau.com/images/newsletter/joy-of-react-mascot.png" loading="lazy" style="filter:hue-rotate(0deg)" class="m1wtl9k0" referrerpolicy="no-referrer"><div class="c1kud80y"></div></div><div style="--template-cols:1fr 1fr;grid-template-columns:1fr" class="c1j2qstn"><div style="max-width:400px;width:100%;margin-inline:auto" class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgpeiavfelb:_rotation" class="lpsyt9t">Hue rotation<!-- -->:</label><span class="w1194iyg">0deg</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgpeiavfelb:_rotation" min="0" max="360" class="i1i8zjy7" value="0"></div></div></div></div></div>
<p class="prg881n">In these examples, I’m applying the filters to an <code class="i165vvr1">&lt;img&gt;</code> tag, but we can apply them to standard DOM nodes as well:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1icc7dx"><div style="filter:blur(0px) sepia(0%)" class="c1l5p6hh"><p>Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged.</p></div><div class="c1koe0ch"></div></div><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_radius" class="lpsyt9t">Blur radius<!-- -->:</label><span class="w1194iyg">0px</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_radius" min="0" max="16" step="0.1" class="i1i8zjy7" value="0"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_sepia" class="lpsyt9t">Sepia percentage<!-- -->:</label><span class="w1194iyg">0%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_sepia" min="0" max="100" class="i1i8zjy7" value="0"></div></div></div></div></div>
<p class="prg881n">Pretty neat, right?</p>
<p class="prg881n">Things get even cooler with <code class="i165vvr1">backdrop-filter</code>. This property lets us apply these same filters to the stuff <em style="color:inherit" class="w10oesj0">behind</em> a given element.</p>
<p class="prg881n">For example:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1qszusq"><div class="isvxvbh"><img src="https://www.joshwcomeau.com/images/backdrop-filter/tokyo.jpg" loading="lazy" class="tvcyp0e" referrerpolicy="no-referrer"><div style="top:80%;left:50%;--scale:2;-webkit-backdrop-filter:sepia(50%) brightness(130%);backdrop-filter:sepia(50%) brightness(130%)" class="ctfyewy"></div></div><div class="ck6v2m7"></div></div><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rh5eiavfelb:_sepia" class="lpsyt9t">Sepia<!-- -->:</label><span class="w1194iyg">50%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rh5eiavfelb:_sepia" min="0" max="100" class="i1i8zjy7" value="50"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rh5eiavfelb:_brightness" class="lpsyt9t">Brightness<!-- -->:</label><span class="w1194iyg">130%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rh5eiavfelb:_brightness" min="50" max="300" class="i1i8zjy7" value="130"></div></div></div></div></div>
<p class="prg881n">In this demo, the <code class="i165vvr1">.magic-ring</code> element sits in front of a photo (<a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://unsplash.com/photos/assorted-signages-on-buildings-nTBW1cOY1qI"> <span class="p7a6phg">source<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:Rn17eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:Rn17eiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a>). It uses the <code class="i165vvr1">backdrop-filter</code> property to apply some filtering to everything behind it, which can be used for some pretty artistic effects.</p>
<p class="prg881n">In practice, I pretty much only use <code class="i165vvr1">backdrop-filter</code> for one use case: blurring everything behind an element, usually a header, to create the “frosted glass” effect I mentioned earlier:</p>
<!-- --><div class="o11h7o12 w1bd8cgi" style="overflow:hidden"><div class="hduekzx"><div class="t1xdhus0"><div class="t1vhg4x9"><div class="rptyea t8ndwpz"></div><div class="y1qao05t t8ndwpz"></div><div class="g1v2ob21 t8ndwpz"></div></div><div class="tj6926w"><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="lep27c6 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(0deg);transform-origin:center center"></path></svg><span>Some Example Website</span><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="r2mz821 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(90deg);transform-origin:center center"></path></svg></div></div><div class="a13tcou3"><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-left"><path d="m12 19-7-7 7-7"></path><path d="M19 12H5"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-right"><path d="M5 12h14"></path><path d="m12 5 7 7-7 7"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-rotate-cw"><path d="M21 12a9 9 0 1 1-9-9c2.52 0 4.93 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path></svg></div><div class="a1booqr"><span>https://www.joshwcomeau.com/example-website</span></div></div></div><div style="overflow:clip"><div class="wi9xlz4 s1hfctpl d16kq653"><div class="hpv2kfm"><div class="b1dh7eg0"></div><div class="bi95upp"></div><p class="l19j3i1d">Some Website</p></div><div class="cufzfud"><img src="https://www.joshwcomeau.com/images/backdrop-filter/cupcake.png" style="animation-play-state:paused" class="c19tvtvc" referrerpolicy="no-referrer"><p>This is an example website showing how I typically use<!-- --> <code class="i165vvr1">backdrop-filter</code> to create glassy headers.</p><p class="o1fnc03u">Notice that as the cupcake moves behind the header, it appears blurry, as it would if it was passing behind frosted glass.</p><div class="atopy0n"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1t96eke w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><span style="width:20px;height:20px;--duration:1422.2222222222222ms;--size:20px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div></div></div>
<p class="prg881n">Alright. Let’s talk about the thing most developers miss.</p>
<h2 id="the-issue" data-element-type="ContentHeading" class="hrk0sy8"><a id="the-issue-2" href="https://www.joshwcomeau.com/css/backdrop-filter/#the-issue-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>The Issue</h2>
<p class="prg881n"><strong>Here’s the problem:</strong> The <code class="i165vvr1">backdrop-filter</code> algorithm only considers the pixels that are <i>directly behind</i> the element.</p>
<p class="prg881n">For filters like <code class="i165vvr1">brightness</code> or <code class="i165vvr1">hue-rotate</code>, that makes perfect sense. With blur, though, we actually want to consider pixels that are <em style="color:inherit" class="w10oesj0">near</em> the element too.</p>
<p class="prg881n">This is one of those things where a demo is worth a thousand words. Check out the difference:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="i1df5wci s360ott"><div id=":Rhneiavfelb:_mode" class="w73h7tw"><div class="r1cct0ab"><button value="default" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">Default</span></button><button value="fixed" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">Tweaked</span></button></div></div></div></div><div class="wr5us8n s1hfctpl d16kq653"><div class="hhqru8"><div class="g8qz4zc"></div></div><div style="animation-play-state:running" class="b1uj7lhw"></div></div><div class="awqze7l"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1rsl36b w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-pause"><rect x="14" y="4" width="4" height="16" rx="1"></rect><rect x="6" y="4" width="4" height="16" rx="1"></rect></svg></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div>
<p class="prg881n">By default, the gaussian blur algorithm is applied to all of the pixels behind the element. This means that if a big colorful element is <em style="color:inherit" class="w10oesj0">near</em> the element, it won’t have any effect.</p>
<p class="prg881n">That’s not really how frosted glass works in real life though. Light bounces off of objects and then goes through the glass. It looks <em style="color:inherit" class="w10oesj0">so much better</em> when the blurring algorithm includes nearby content.</p>
<p class="prg881n">Unfortunately, this isn’t something we can configure directly. Instead, we need to be a bit crafty.</p>
<p class="prg881n">Here’s the code:</p>
<!-- --><div data-stacked="false" class="il1gxx9"><div style="--color-text:hsl(210deg 10% 90%);--color-background:hsl(210deg 15% 6%);--color-blurred-background:hsl(210deg 15% 6% / 0.75);--color-muted-background:hsl(210deg 38% 15% / 0.85);--color-action:hsl(240deg 95% 62%);--color-primary:hsl(225deg 100% 75%);--color-secondary:hsl(333deg 100% 55%);--color-tertiary:hsl(280deg 100% 85%);--color-decorative:hsl(200deg 50% 60%);--color-info-100:hsl(214deg 40% 15%);--color-info-200:hsl(218deg 32% 20%);--color-info-300:hsl(218deg 30% 25%);--color-info-400:hsl(218deg 45% 40%);--color-info-500:hsl(225deg 100% 60%);--color-info-700:hsl(215deg 100% 72%);--color-warning-100:hsl(30deg 25% 11%);--color-warning-200:hsl(32deg 20% 15%);--color-warning-300:hsl(32deg 25% 20%);--color-warning-400:hsl(35deg 45% 35%);--color-warning-500:hsl(40deg 100% 50%);--color-warning-700:hsl(43deg 100% 72%);--color-success-100:hsl(176deg 35% 10%);--color-success-200:hsl(176deg 26% 14%);--color-success-300:hsl(176deg 28% 20%);--color-success-400:hsl(176deg 45% 30%);--color-success-500:hsl(160deg 100% 40%);--color-success-700:hsl(160deg 80% 65%);--color-cloud-100:hsl(210deg 15% 6%);--color-cloud-300:hsl(212deg 40% 9%);--color-cloud-400:hsl(213deg 40% 10%);--color-cloud-500:hsl(213deg 40% 12%);--color-sky-from:hsl(214deg 40% 11%);--color-sky-to:hsl(200deg 50% 30%);--color-sky-subtle:hsl(210deg 40% 16%);--color-gray-50:hsl(210deg 19% 9%);--color-gray-100:hsl(210deg 15% 12%);--color-gray-200:hsl(210deg 15% 18%);--color-gray-300:hsl(210deg 10% 30%);--color-gray-400:hsl(210deg 9% 40%);--color-gray-500:hsl(210deg 8% 50%);--color-gray-600:hsl(210deg 12% 55%);--color-gray-700:hsl(210deg 14% 66%);--color-gray-800:hsl(210deg 20% 77%);--color-gray-900:hsl(210deg 25% 88%);--color-gray-1000:hsl(210deg 25% 96%);--color-adaptive-white:hsl(210deg 25% 92%);--syntax-bg:hsl(210deg 15% 6%);--syntax-highlight:hsl(210deg 30% 18%);--syntax-txt:hsl(0deg 0% 100%);--syntax-comment:hsl(200deg 18% 51%);--syntax-prop:hsl(326deg 100% 61%);--syntax-bool:hsl(50deg 100% 50%);--syntax-val:hsl(210deg 12% 65%);--syntax-str:hsl(259deg 100% 71%);--syntax-name:hsl(280deg 100% 66%);--syntax-del:hsl(0deg 100% 67%);--syntax-regex:hsl(50deg 100% 50%);--syntax-fn:hsl(195deg 100% 50%);--color-info:hsl(225deg 100% 80%);--color-info-background:hsl(225deg 100% 80% / 0.1);--color-error:hsl(340deg 95% 60%);--color-error-background:hsl(340deg 95% 43% / 0.1);--color-success:hsl(160deg 100% 40%);--color-success-background:hsl(160deg 100% 40% / 0.1);--color-alert:hsl(40deg 100% 50%);--color-alert-background:hsl(40deg 100% 50% / 0.1);--color-page-background:hsl(210deg 15% 6%);--color-page-primary:hsl(225deg 100% 75%);--kbd-background-color:hsl(210deg 9% 40%);--kbd-border-color:hsl(210deg 10% 30%);--color-code-bg:hsl(210deg 15% 12%);--color-content-outline:hsl(210deg 10% 30%)" class="i1j6zvew s1hfctpl d16kq653"><header class="w14j10eb"><p class="tkoi9ui">Code Playground</p><div class="au0cxa1"><button class="wsn2i55 a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7;overflow:visible" class="s1gvsyfu"><g style="transform:rotate(-45deg)" class="w15rf7yg"><rect x="0" y="8" width="24" height="6" rx="2" fill="var(--color-gray-100)"></rect><line x1="18" y1="8" x2="18" y2="14"></line></g></svg><span class="wlgbkjj">Format code using Prettier</span></button><abbr style="display:inline-block"><div class="wy3w6yt"><button title="Reset code" class="a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7" class="s1gvsyfu"><mask id="skip-icon-mask-:R39i1eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="black"></rect><rect x="9" y="0" width="16" height="24" fill="white"></rect></mask><line x1="5" y1="19" x2="5" y2="5"></line><polygon points="19 20 9 12 19 4 19 20" mask="url(#skip-icon-mask-:R39i1eiavfelb:)"></polygon></svg><span class="wlgbkjj">Reset Code</span></button><div class="pszdhfz"><span style="gap:1.5px" class="w15sqd9b"><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span></span></div></div></abbr></div></header><div></div><div class="w198elmk"><div class="c1jztmaw"><div style="flex:516.12" class="f15xiyga"><div class="w16o0eiw"><div class="tt2ystl"><button style="--weight:var(--font-weight-bold);--color:var(--color-text)" class="ts2dciv">HTML</button><button style="--weight:var(--font-weight-normal);--color:var(--color-gray-500)" class="ts2dciv">CSS</button></div><div in="false" style="--max-height:80vh" class="w1bgwmtf"><button class="f9ibg9h"><span class="wlgbkjj">Focus the editor. This will trap focus until you press Escape.</span></button><label><span class="wlgbkjj">Code editor:</span><div translate="no" class="c18okpq0" style="position:relative;text-align:left;box-sizing:border-box;padding:0;overflow:hidden"><pre aria-hidden="true" style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:relative;pointer-events:none;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0"><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">style</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css selector" style="color:var(--syntax-name)">header</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">{</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">position</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> relative</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">}</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css selector class" style="color:var(--syntax-name)">.backdrop</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">{</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">position</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> absolute</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">inset</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">0</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">height</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">200</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">backdrop-filter</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css function" style="color:var(--syntax-fn)">blur</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">(</span><span class="token style language-css number" style="color:var(--syntax-bool)">16</span><span class="token style language-css unit" style="color:var(--syntax-str)">px</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">)</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">mask-image</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css function" style="color:var(--syntax-fn)">linear-gradient</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">(</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> to bottom</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">,</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css color" style="color:var(--syntax-str)">black</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">0</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">50</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">,</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css color" style="color:var(--syntax-str)">transparent</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">50</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">100</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">)</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">}</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">style</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain" style="display:inline-block">
</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">header</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"> </span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">class</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">backdrop</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">header</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">class</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">ball</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"&g ... |
http://localhost:1200/joshwcomeau/popular - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>Popular Content | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com</link>
<atom:link href="http://localhost:1200/joshwcomeau/popular" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on React, CSS, Animation, and more! - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Sat, 18 Jan 2025 06:18:15 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>How To Center a Div</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/css/center-a-div/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with auto margins</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering within the viewport</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-elements-with-unknown-sizes-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering elements with unknown sizes</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-css-grid-5" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with CSS Grid</a><a href="https://www.joshwcomeau.com/css/center-a-div/#differences-from-flexbox-6" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Differences from Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-a-stack-of-elements-7" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering a stack of elements</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-text-8" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering text</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-in-the-future-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering in the future</a><a href="https://www.joshwcomeau.com/css/center-a-div/#going-beyond-the-patterns-10" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Going beyond the patterns</a><a href="https://www.joshwcomeau.com/css/center-a-div/#when-to-use-which-method-11" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">When to use which method</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">For a long time, centering an element within its parent was a surprisingly tricky thing to do. As CSS has evolved, we've been granted more and more tools we can use to solve this problem. These days, we're spoiled for choice!</p>
<p class="prg881n">I decided to create this tutorial to help you understand the trade-offs between different approaches, and to give you an arsenal of strategies you can use, to handle centering in all sorts of scenarios.</p>
<p class="prg881n">Honestly, this turned out to be <em style="color:inherit" class="w10oesj0">way more interesting</em> than I initially thought&nbsp;😅. Even if you've been using CSS for a while, I bet you'll learn at least 1 new strategy!</p>
<h2 id="centering-with-auto-margins" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-auto-margins-1" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with auto margins</h2>
<p class="prg881n">The first strategy we'll look at is one of the oldest. If we want to center an element horizontally, we can do so using margins set to the special value <code class="i165vvr1">auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgbeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgbeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-right</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-left:auto;margin-right:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgbeiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgbeiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">First, we need to constrain the element's width; by default, elements in Flow layout will expand horizontally to fill the available space, and we can't really center something that is full-width.</p>
<p class="prg881n">I <em style="color:inherit" class="w10oesj0">could</em> constrain the width with a fixed value (eg. <code class="i165vvr1">200px</code>), but really what I want in this case is for the element to shrinkwrap around its content. <code class="i165vvr1">fit-content</code> is a magical value that does exactly this. Essentially, it makes “width” behave like “height”, so that the element’s size is determined by its contents.</p>
<p class="prg881n"><strong>Why am I setting <code class="i165vvr1">max-width</code> instead of <code class="i165vvr1">width</code>?</strong> Well, my goal is to stop the element from expanding horizontally. I want to clamp its maximum size. If I used <code class="i165vvr1">width</code> instead, it would lock it to that size, and the element would overflow when the container is really narrow. If you drag that “Container Width” slider all the way to the left, you can see that the element shrinks with its container.</p>
<p class="prg881n">Now that our element is constrained, we can center it with <em style="color:inherit" class="w10oesj0">auto margins</em>.</p>
<p class="prg881n">I like to think of auto margins like <i>Hungry Hungry Hippos</i>. Each auto margin will try to gobble up as much space as possible. For example, check out what happens if we <em style="color:inherit" class="w10oesj0">only</em> set <code class="i165vvr1">margin-left: auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgneiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgneiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div></div><div style="margin-left:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgneiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgneiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">When <code class="i165vvr1">margin-left</code> is the only side with auto margins, <i>all</i> of the extra space gets applied as margin to that side. When we set both <code class="i165vvr1">margin-left: auto</code> <i>and</i> <code class="i165vvr1">margin-right: auto</code>, the two hippos each gobble up an equal amount of space. This forces the element to the center.</p>
<p class="prg881n"><em style="color:inherit" class="w10oesj0">Also:</em> I've been using <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> because they're familiar, but there's a better, more-modern way to do this:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line highlighted"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-inline</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-inline:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgteiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgteiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n"><code class="i165vvr1">margin-inline</code> will set both <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> to the same value (<code class="i165vvr1">auto</code>). It has <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://caniuse.com/mdn-css_properties_margin-inline">very good browser<!-- --> <span class="p7a6phg">support<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R1e8veiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R1e8veiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a>, having landed in all major browsers several years ago.</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-logical-properties" class="t1xqw7pl">Logical properties</strong><div class="c690lmu"><p class="prg881n"><code class="i165vvr1">margin-inline</code> is more than just a convenient shorthand for <code class="i165vvr1">margin-left</code> + <code class="i165vvr1">margin-right</code>. It's part of a collection of <em style="color:inherit" class="w10oesj0">logical properties</em>, designed to make it easier to internationalize the web.</p><p class="prg881n">In English, characters are written in a horizontal line from left to right. Those characters are composed into words and sentences, and assembled into “blocks” (paragraphs, headings, lists, etc). Blocks are stacked vertically, from top to bottom. We can think of this as the <i>orientation</i> of English-language websites.</p><p class="prg881n">This isn't universal, though! Some languages, like Arabic and Hebrew, are written from right to left. Other languages, like Chinese, have historically been written vertically, with characters running from top to bottom, and <i>blocks</i> running side to side.<button class="t1cz8cv8"><span class="a1ni5s6f">*</span><span class="e2cu4f7"></span></button><span data-use-adaptive-colors="true" id=":Rhm91eiavfelb:" data-direction="n" role="tooltip" style="flex-direction:column" class="p1lgzcrp l17lpp8k"><span class="p833mxt"><span class="pj0hw36"><span class="cffclov">Interestingly, this has been changing over the past few decades, and most East Asian languages these days are written horizontally from left to right, especially on the web.</span></span></span><svg xmlns="http://www.w3.org/2000/svg" width="24" height="6" fill="none" viewBox="0 0 24 6" style="--origin-y:0%" class="s1vq0yeh s1k9z937"><path d="
M 0 0
C 6 0
7.199999999999999 6
12 6
C 16.8 6
18 0
24 0
Z
"></path></svg></span></p><p class="prg881n">The primary goal of logical properties is to create an abstraction that sits above these differences. Instead of setting <code class="i165vvr1">margin-left</code> for left-to-right languages and flipping it to <code class="i165vvr1">margin-right</code> for right-to-left languages, we can instead use <code class="i165vvr1">margin-inline-start</code>. The margin will automatically be applied to the correct side, depending on the page's language.</p></div></aside>
<p class="prg881n">Even though this centering method has been around forever, I still find myself reaching for it on a regular basis! It's particularly useful when we want to center a single child, without affecting any of its siblings (for example, an image in-between paragraphs in a blog post).</p>
<p class="prg881n">Let's continue on our centering journey.</p>
<h2 id="centering-with-flexbox" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-flexbox-2" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with Flexbox</h2>
<p class="prg881n">Flexbox is designed to give us a <em style="color:inherit" class="w10oesj0">ton</em> of control when it comes to distributing a group of items along a primary axis. It offers some <em style="color:inherit" class="w10oesj0">really</em> powerful tools for centering!</p>
<p class="prg881n">Let's start by centering a single element, both horizontally and vertically:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div class="rcwtqfe"><div class="c9vkmf2"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="djxxsdm c9vkmf2"><div data-use-adaptive-colors="true" style="width:100%;height:200px" class="c17eovmj l17lpp8k"><div class="f1vkqrq wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div></div></div>
<p class="prg881n">The really cool thing about Flexbox centering is that it works <i>even when the children don’t fit in their container!</i> Try shrinking the width/height, and notice that the element overflows symmetrically.</p>
<p class="prg881n">It also works for <em style="color:inherit" class="w10oesj0">multiple</em> children. We can control how they stack with the <code class="i165vvr1">flex-direction</code> property:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Flex Direction<!-- -->:</span><div id=":Rhjeiavfelb:_flexDirection" class="ray0b30 w73h7tw"><div class="r1cct0ab"><button value="row" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">row</span></button><button value="column" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column</span></button><button value="row-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">row-reverse</span></button><button value="column-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column-reverse</span></button></div></div></div></div><div class="r1xmilea"><div class="cgnjx9w c2tip9q"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> flex-direction</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> row</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> gap</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 4</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d1iw6ryr c2tip9q"><div data-use-adaptive-colors="true" style="width:100%;min-height:212px;flex-direction:row" class="c1oa8tf1 l17lpp8k"><div class="bt1ddka wythvuy"><span class="t16m3o3k">1</span></div><div style="z-index:2" class="bt1ddka wythvuy"><span class="t16m3o3k">2</span></div><div class="bt1ddka wythvuy"><span class="t16m3o3k">3</span></div></div></div></div></div></div>
<p class="prg881n">Out of all the centering patterns we'll explore in this tutorial, this is probably the one I use the most. It's a great jack-of-all-trades, a great default option.</p>
<h2 id="centering-within-the-viewport" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-within-the-viewport-3" href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering within the viewport</h2>
<p class="prg881n">So far, we've been looking at how to center an element within its parent container. But what if we want to center an element in a different context? Certain elements like dialogs, prompts, and GDPR banners need to be centered within the viewport.</p>
<p class="prg881n">This is the domain of <em style="color:inherit" class="w10oesj0">positioned layout,</em> a layout mode used when we want to take something out of flow and anchor it to something else.</p>
<p class="prg881n">Here's what this looks like:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerWidth" min="30" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="rd1irhw"><div class="c1fq67g4"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> inset</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 0</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 5</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">vw</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">dvh</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dougf27 c1fq67g4"><div data-use-adaptive-colors="true" style="width:100%;height:17.625rem" class="c8q6jp9 l17lpp8k"><div class="bfhwaun"><div class="hiqwc5b"><span style="width:3rem" class="hktm3y9 bx3nfyo"></span><span style="width:5rem" class="hktm3y9 bx3nfyo"></span><span style="width:1rem" class="hktm3y9 bx3nfyo"></span><span style="width:4rem" class="hktm3y9 bx3nfyo"></span><span style="width:7rem" class="hktm3y9 bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:1rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2rem" class="bx3nfyo"></span><span style="width:4.75rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:7rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:5rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div></div><div class="hpafenu m148wj37"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="h1b4sgtf m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="h1b4sgtf m101ijsn"></div></div><div class="vm87swv m148wj37"><div style="clip-path:polygon(0% 0%, 100% 0%, 100% 0%, 0% 0%)" class="v15iwzns m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(0% 100%, 100% 100%, 100% 100%, 0% 100%)" class="v15iwzns m101ijsn"></div></div><div class="m1b7mso0 p1j9axfr"></div></div></div></div></div></div></div>
<p class="prg881n">Of all the strategies we'll discuss, this one is probably the most complex. Let's break it down.</p>
<p class="prg881n">We're using <code class="i165vvr1">position: fixed</code>, which anchors this element to the viewport. I like to think of the viewport like a pane of glass that sits in front of the website, like the window of a train that shows the landscape scrolling by. An element with <code class="i165vvr1">position: fixed</code> is like a ladybug that lands on the window.</p>
<p class="prg881n">Next, we're setting <code class="i165vvr1">inset: 0px</code>, which is a shorthand that sets <code class="i165vvr1">top</code>, <code class="i165vvr1">left</code>, <code class="i165vvr1">right</code>, and <code class="i165vvr1">bottom</code> all to the same value, <code class="i165vvr1">0px</code>.</p>
<p class="prg881n">With only these two properties, the element would stretch to fill the entire viewport, growing so that it's 0px from each edge. This can be useful in some contexts, but it's not what we're going for here. We need to constrain it.</p>
<p class="prg881n">The exact values we pick will vary on the specifics of each situation, but in general we want to set default values (with <code class="i165vvr1">width</code> and <code class="i165vvr1">height</code>), as well as max values (<code class="i165vvr1">max-width</code> and <code class="i165vvr1">max-height</code>), so that the element doesn't overflow on smaller viewports.</p>
<p class="prg881n"><strong>There's something interesting here:</strong> we've set up an impossible condition. Our element can't be 0px from the left <i>and</i> 0px from the right <i>and</i> only 12rem wide (assuming the viewport is wider than 12rem). We can only pick 2:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Pick two<!-- -->:</span><div id=":Rideiavfelb:_values" class="w73h7tw"><div class="r1cct0ab"><button value="left: 0px" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">left: 0px</span></button><button value="right: 0px" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">right: 0px</span></button><button value="width: 12rem" tabindex="-1" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">width: 12rem</span></button></div></div></div></div><div class="r3vmhao"><div class="c7zul9b"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d19iwc1r c7zul9b"><div data-use-adaptive-colors="true" class="ctnf89z l17lpp8k"><div class="b14ie1by"><div class="fujgo4n"><span style="width:3rem" class="h1vx44g7 byjv7xx"></span><span style="width:5rem" class="h1vx44g7 byjv7xx"></span><span style="width:1rem" class="h1vx44g7 byjv7xx"></span><span style="width:4rem" class="h1vx44g7 byjv7xx"></span><span style="width:7rem" class="h1vx44g7 byjv7xx"></span></div><div class="f1p79fid"><span style="width:1rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2rem" class="byjv7xx"></span><span style="width:4.75rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:7rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:5rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:2.1rem" class="byjv7xx"></span><span style="width:0.785rem" class="byjv7xx"></span><span style="width:2.5rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:3.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span></div><div class="f1p79fid"><span style="width:0.785rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span styl |
Successfully generated as following: http://localhost:1200/joshwcomeau/latest/javascript - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>JavaScript | Articles and Tutorials | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com/javascript</link>
<atom:link href="http://localhost:1200/joshwcomeau/latest/javascript" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on JavaScript | - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Sat, 18 Jan 2025 18:17:07 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>Promises From The Ground Up</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/javascript/promises/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/javascript/promises/#why-would-they-design-it-this-way-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Why would they design it this way??</a><a href="https://www.joshwcomeau.com/javascript/promises/#callbacks-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Callbacks</a><a href="https://www.joshwcomeau.com/javascript/promises/#introducing-promises-3" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Introducing Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#working-with-promises-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Working with Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#creating-our-own-promises-5" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Creating our own Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#chaining-promises-6" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Chaining Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#passing-data-7" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Passing data</a><a href="https://www.joshwcomeau.com/javascript/promises/#rejected-promises-8" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Rejected Promises</a><a href="https://www.joshwcomeau.com/javascript/promises/#async--await-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Async / Await</a><a href="https://www.joshwcomeau.com/javascript/promises/#more-to-come-10" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">More to come!</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">There are a lot of speed bumps and potholes on the road to JavaScript proficiency. One of the biggest and most daunting is <em style="color:inherit" class="w10oesj0">Promises.</em></p>
<p class="prg881n">In order to understand Promises, we need a surprisingly deep understanding of how JavaScript works and what its limitations are. Without that context, Promises won’t really make much sense.</p>
<p class="prg881n">It can be frustrating because the Promises API is <i>so important</i> nowadays. It’s become the de facto way of working with asynchronous code. Modern web APIs are built on top of Promises. There’s no getting around it: if we want to be productive with JavaScript, it really helps to understand Promises.</p>
<p class="prg881n">So, in this tutorial, we’re going to learn about Promises, but we’ll start at the beginning. I’ll share all of the critical bits of context that took me years to understand. And by the end, hopefully, you’ll have a much deeper understanding of what Promises are and how to use them effectively. ✨</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-intended-audience" class="t1xqw7pl">Intended audience</strong><div class="c690lmu"><p class="prg881n">This blog post is intended for beginner-to-intermediate JavaScript developers. Some knowledge of basic JavaScript syntax is assumed.</p></div></aside>
<h2 id="why-would-they-design-it-this-way" data-element-type="ContentHeading" class="hrk0sy8"><a id="why-would-they-design-it-this-way-1" href="https://www.joshwcomeau.com/javascript/promises/#why-would-they-design-it-this-way-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Why would they design it this way??</h2>
<p class="prg881n">Suppose we wanted to build a <em style="color:inherit" class="w10oesj0">Happy New Year!</em> countdown, something like this:</p>
<!-- --><div class="w52b1qq"><p class="v793fuv">––</p><button data-disabled="false" class="b1b5r9ns dj4vswd">Start Countdown</button></div>
<p class="prg881n">If JavaScript was like most other programming languages, we could solve the problem like this:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA">function</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> newYearsCountdown</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">()</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">3</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">2</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">1</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> sleep</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF">1000</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic"> print</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">Happy New Year! 🎉</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">"</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">}</span></span></code></pre></div></div>
<p class="prg881n">In this hypothetical code snippet, the program would pause when it hits a <code class="i165vvr1">sleep()</code> call, and then resume after the specified amount of time has passed.</p>
<p class="prg881n">Unfortunately, there is no <code class="i165vvr1">sleep</code> function in JavaScript, because it’s a <em style="color:inherit" class="w10oesj0">single-threaded</em> language.<button class="t1cz8cv8"><span class="a1ni5s6f">*</span><span class="e2cu4f7"></span></button><span data-use-adaptive-colors="true" id=":R1gneiavfelb:" data-direction="n" role="tooltip" style="flex-direction:column" class="p1lgzcrp l17lpp8k"><span class="p833mxt"><span class="pj0hw36"><span class="cffclov">Technically, modern JavaScript has access to multiple threads via Web Workers, but those extra threads don't have access to the DOM, so they can’t really be used in most situations.</span></span></span><svg xmlns="http://www.w3.org/2000/svg" width="24" height="6" fill="none" viewBox="0 0 24 6" style="--origin-y:0%" class="s1vq0yeh s1k9z937"><path d="
M 0 0
C 6 0
7.199999999999999 6
12 6
C 16.8 6
18 0
24 0
Z
"></path></svg></span> A “thread” is a long-running process that executes code. JavaScript only has one thread, and so it can only do one thing at a time. It can’t multitask. This is a problem because if our lone JavaScript thread is busy managing this countdown timer, it can’t do anything <i>else</i>.</p>
<p class="prg881n">When I was first learning about this stuff, it wasn’t immediately obvious to me why this was a problem. If the countdown timer is the only thing happening right now, isn’t it fine if the JS thread was fully occupied during that time??</p>
<p class="prg881n">Well, even though JavaScript doesn’t have a <code class="i165vvr1">sleep</code> function, it <i>does</i> have some other functions that occupy the main thread for an extended amount of time. We can use those other methods to get a glimpse into what it would be like if JavaScript had a <code class="i165vvr1">sleep</code> function.</p>
<p class="prg881n">For example, <code class="i165vvr1">window.prompt()</code>. This function is used to gather information from the user, and it halts execution of our code much like our hypothetical <code class="i165vvr1">sleep()</code> function would.</p>
<p class="prg881n">Click the button in this playground, and then <strong>try to interact with the page</strong> while the prompt is open:</p>
<!-- --><div data-stacked="false" class="il1gxx9"><div style="--color-text:hsl(210deg 10% 90%);--color-background:hsl(210deg 15% 6%);--color-blurred-background:hsl(210deg 15% 6% / 0.75);--color-muted-background:hsl(210deg 38% 15% / 0.85);--color-action:hsl(240deg 95% 62%);--color-primary:hsl(225deg 100% 75%);--color-secondary:hsl(333deg 100% 55%);--color-tertiary:hsl(280deg 100% 85%);--color-decorative:hsl(200deg 50% 60%);--color-info-100:hsl(214deg 40% 15%);--color-info-200:hsl(218deg 32% 20%);--color-info-300:hsl(218deg 30% 25%);--color-info-400:hsl(218deg 45% 40%);--color-info-500:hsl(225deg 100% 60%);--color-info-700:hsl(215deg 100% 72%);--color-warning-100:hsl(30deg 25% 11%);--color-warning-200:hsl(32deg 20% 15%);--color-warning-300:hsl(32deg 25% 20%);--color-warning-400:hsl(35deg 45% 35%);--color-warning-500:hsl(40deg 100% 50%);--color-warning-700:hsl(43deg 100% 72%);--color-success-100:hsl(176deg 35% 10%);--color-success-200:hsl(176deg 26% 14%);--color-success-300:hsl(176deg 28% 20%);--color-success-400:hsl(176deg 45% 30%);--color-success-500:hsl(160deg 100% 40%);--color-success-700:hsl(160deg 80% 65%);--color-cloud-100:hsl(210deg 15% 6%);--color-cloud-300:hsl(212deg 40% 9%);--color-cloud-400:hsl(213deg 40% 10%);--color-cloud-500:hsl(213deg 40% 12%);--color-sky-from:hsl(214deg 40% 11%);--color-sky-to:hsl(200deg 50% 30%);--color-sky-subtle:hsl(210deg 40% 16%);--color-gray-50:hsl(210deg 19% 9%);--color-gray-100:hsl(210deg 15% 12%);--color-gray-200:hsl(210deg 15% 18%);--color-gray-300:hsl(210deg 10% 30%);--color-gray-400:hsl(210deg 9% 40%);--color-gray-500:hsl(210deg 8% 50%);--color-gray-600:hsl(210deg 12% 55%);--color-gray-700:hsl(210deg 14% 66%);--color-gray-800:hsl(210deg 20% 77%);--color-gray-900:hsl(210deg 25% 88%);--color-gray-1000:hsl(210deg 25% 96%);--color-adaptive-white:hsl(210deg 25% 92%);--syntax-bg:hsl(210deg 15% 6%);--syntax-highlight:hsl(210deg 30% 18%);--syntax-txt:hsl(0deg 0% 100%);--syntax-comment:hsl(200deg 18% 51%);--syntax-prop:hsl(326deg 100% 61%);--syntax-bool:hsl(50deg 100% 50%);--syntax-val:hsl(210deg 12% 65%);--syntax-str:hsl(259deg 100% 71%);--syntax-name:hsl(280deg 100% 66%);--syntax-del:hsl(0deg 100% 67%);--syntax-regex:hsl(50deg 100% 50%);--syntax-fn:hsl(195deg 100% 50%);--color-info:hsl(225deg 100% 80%);--color-info-background:hsl(225deg 100% 80% / 0.1);--color-error:hsl(340deg 95% 60%);--color-error-background:hsl(340deg 95% 43% / 0.1);--color-success:hsl(160deg 100% 40%);--color-success-background:hsl(160deg 100% 40% / 0.1);--color-alert:hsl(40deg 100% 50%);--color-alert-background:hsl(40deg 100% 50% / 0.1);--color-page-background:hsl(210deg 15% 6%);--color-page-primary:hsl(225deg 100% 75%);--kbd-background-color:hsl(210deg 9% 40%);--kbd-border-color:hsl(210deg 10% 30%);--color-code-bg:hsl(210deg 15% 12%);--color-content-outline:hsl(210deg 10% 30%)" class="i1j6zvew s1hfctpl d16kq653"><header class="w14j10eb"><p class="tkoi9ui">Code Playground</p><div class="au0cxa1"><button class="wsn2i55 a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7;overflow:visible" class="s1gvsyfu"><g style="transform:rotate(-45deg)" class="w15rf7yg"><rect x="0" y="8" width="24" height="6" rx="2" fill="var(--color-gray-100)"></rect><line x1="18" y1="8" x2="18" y2="14"></line></g></svg><span class="wlgbkjj">Format code using Prettier</span></button><abbr style="display:inline-block"><div class="wy3w6yt"><button title="Reset code" class="a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7" class="s1gvsyfu"><mask id="skip-icon-mask-:R39h1eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="black"></rect><rect x="9" y="0" width="16" height="24" fill="white"></rect></mask><line x1="5" y1="19" x2="5" y2="5"></line><polygon points="19 20 9 12 19 4 19 20" mask="url(#skip-icon-mask-:R39h1eiavfelb:)"></polygon></svg><span class="wlgbkjj">Reset Code</span></button><div class="pszdhfz"><span style="gap:1.5px" class="w15sqd9b"><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span></span></div></div></abbr></div></header><div></div><div class="w198elmk"><div class="c1jztmaw"><div style="flex:547.4" class="f15xiyga"><div style="min-height:100%" class="whrzwqb"><div class="h1nlnw7n"><p class="tbdao9h">HTML</p></div><div in="false" style="--max-height:80vh" class="w1bgwmtf"><button class="f9ibg9h"><span class="wlgbkjj">Focus the editor. This will trap focus until you press Escape.</span></button><label><span class="wlgbkjj">Code editor:</span><div translate="no" class="c18okpq0" style="position:relative;text-align:left;box-sizing:border-box;padding:0;overflow:hidden"><pre aria-hidden="true" style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:relative;pointer-events:none;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0"><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">script</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">function</span><span class="token script language-javascript"> </span><span class="token script language-javascript function" style="color:var(--syntax-fn)">askForName</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript"> </span><span class="token script language-javascript punctuation">{</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">const</span><span class="token script language-javascript"> name </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript dom variable" style="color:var(--syntax-str)">window</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript method function property-access" style="color:var(--syntax-fn)">prompt</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">'What is your name?'</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript punctuation">;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript keyword" style="color:var(--syntax-str)">const</span><span class="token script language-javascript"> elem </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript dom variable" style="color:var(--syntax-str)">document</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript method function property-access" style="color:var(--syntax-fn)">querySelector</span><span class="token script language-javascript punctuation">(</span><span class="token script language-javascript string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">'#greeting'</span><span class="token script language-javascript punctuation">)</span><span class="token script language-javascript punctuation">;</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> elem</span><span class="token script language-javascript punctuation">.</span><span class="token script language-javascript property-access">innerText</span><span class="token script language-javascript"> </span><span class="token script language-javascript operator" style="color:var(--syntax-str)">=</span><span class="token script language-javascript"> </span><span class="token script language-javascript template-string template-punctuation string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">`</span><span class="token script language-javascript template-string string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">Hi </span><span class="token script language-javascript template-string interpolation interpolation-punctuation punctuation">${</span><span class="token script language-javascript template-string interpolation">name</span><span class="token script language-javascript template-string interpolation interpolation-punctuation punctuation">}</span><span class="token script language-javascript template-string string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">!</span><span class="token script language-javascript template-string template-punctuation string" style="color:var(--syntax-str);font-weight:var(--font-weight-medium)">`</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"> </span><span class="token script language-javascript punctuation">}</span><span class="token script language-javascript"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token script language-javascript"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">script</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain" style="display:inline-block">
</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">button</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag special-attr attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">onclick</span><span class="token tag special-attr attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag special-attr attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag special-attr attr-value value javascript language-javascript function" style="color:var(--syntax-fn);font-weight:var(--font-weight-medium)">askForName</span><span class="token tag special-attr attr-value value javascript language-javascript punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">(</span><span class="token tag special-attr attr-value value javascript language-javascript punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">)</span><span class="token tag special-attr attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"> Click me</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">button</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">id</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">greeting</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span></div></pre><textarea style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:absolute;top:0;left:0;height:100%;width:100%;resize:none;color:inherit;overflow:hidden;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;-webkit-text-fill-color:transparent;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0" class="npm__react-simple-code-editor__textarea" autocapitalize="off" autocomplete="off" autocorrect="off" spellcheck="false" data-gramm="false">&lt;script&gt;
function askForName() {
const name = window.prompt('What is your name?');
const elem = document.querySelector('#greeting');
elem.innerText = `Hi ${name}!`
}
&lt;/script&gt;
&lt;button onclick="askForName()"&gt;
Click me
&lt;/button&gt;
&lt;div id="greeting"&gt;&lt;/div&gt;</textarea><style>
/**
* Reset the text fill color so that placeholder is visible
*/
.npm__react-simple-code-editor__textarea:empty {
-webkit-text-fill-color: inherit !important;
}
/**
* Hack to apply on some CSS on IE10 and IE11
*/
@media all and (-ms-high-contrast: none), (-ms-high-contrast: active) {
/**
* IE doesn't support '-webkit-text-fill-color'
* So we use 'color: transparent' to make the text transparent on IE
* Unlike other browsers, it doesn't affect caret color in IE
*/
.npm__react-simple-code-editor__textarea {
color: transparent !important;
}
.npm__react-simple-code-editor__textarea::selection {
background-color: #accef7 !important;
color: transparent !important;
}
}
</style></div></label></div></div></div><button class="dijulrw"><span class="wlgbkjj">Resize editor. Use left/right arrows.</span></button><div style="flex:234.60000000000002" class="s1c0ceq0"><div style="height:100%" class="whrzwqb"><div class="h1nlnw7n"><p class="tbdao9h">Result</p><abbr title="Refresh pane"><button style="transform:rotate(0deg);opacity:0.7" class="w11udlg3 ah084m5"><svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-refresh-cw"><path d="M3 12a9 9 0 0 1 9-9 9.75 9.75 0 0 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path><path d="M21 12a9 9 0 0 1-9 9 9.75 9.75 0 0 1-6.74-2.74L3 16"></path><path d="M8 16H3v5"></path></svg><span class="wlgbkjj">Refresh results pane</span></button></abbr></div><div style="--height:100%;--flex:1" class="wqqdb70"><iframe height="100%" width="100%" title="example" frameborder="0" loading="lazy" class="fxdlgbu" referrerpolicy="no-referrer"></iframe></div></div></div></div></div></div></div>
<p class="prg881n"><strong>Notice that while the prompt is open, the page is totally unresponsive?</strong> You can't scroll, click any links, or select any text! The JavaScript thread is busy waiting for us to provide a value so that it can finish running that code. While it’s waiting, it can’t do anything <i>else</i>, and so the browser locks down the UI.</p>
<p class="prg881n">Other languages have multiple threads, and so it's no big deal if one of them gets preoccupied for a while. In JavaScript, though, we only have the one, and it’s used for everything: handling events, managing network requests, updating the UI, etc.</p>
<p class="prg881n">If we want to create a countdown, we need to find a way to do it without blocking the thread.</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-but-why-is-the-entire-ui-frozen" class="t1xqw7pl">But why is the entire UI frozen??</strong><div class="c690lmu"><p class="prg881n">In the example above with <code class="i165vvr1">window.prompt()</code>, the entire UI becomes unresponsive while the browser waits for us to provide a value.</p><p class="prg881n">This is kinda strange… the browser doesn’t rely on JavaScript to scroll the page, or to select text. So why can’t we do any of those things?</p><p class="prg881n">I think browsers work this way to prevent bugs. Scrolling the page, for example, triggers “scroll” events which can be caught and handled with JavaScript. If the JS thread is occupied while the scroll event happens, that code never runs, which could lead to bugs if the developer assumed that scroll events would always be handled.</p><p class="prg881n">It could also be a UX thing; maybe the browser disables the UI so that the user can’t ignore the prompt. Either way, though, I suspect a native <code class="i165vvr1">sleep</code> function would need to work the same way to prevent bugs.</p></div></aside>
<h2 id="callbacks" data-element-type="ContentHeading" class="hrk0sy8"><a id="callbacks-2" href="https://www.joshwcomeau.com/javascript/promises/#callbacks-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Callbacks</h2>
<p class="prg881n">The main tool in our toolbox for solving these sorts of problems is <code class="i165vvr1">setTimeout</code>. <code class="i165vvr1">setTimeout</code> is a function which accepts two arguments:</p>
<ol class="oc96fnv">
<li class="wp18jjy"><div class="l1n7gg9f">A chunk of work to do, at some point in the future.</div></li>
<li class="wp18jjy"><div class="l1n7gg9f">The amount of time to wait for.</div></li>
</ol>
<p class="prg881n">Here's an example:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">Start</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">setTimeout</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> ()</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA"> =&gt;</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF"> console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">After one second</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> },</span></span>
<span class="line"><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF"> 1000</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span></code></pre></div></div>
<p class="prg881n">The chunk of work is passed in through a function. This pattern is known as a <em style="color:inherit" class="w10oesj0">callback.</em></p>
<p class="prg881n">The hypothetical <code class="i165vvr1">sleep()</code> function we saw before is like calling a company and waiting on hold for the next available representative. <code class="i165vvr1">setTimeout()</code> is like pressing <i>1</i> to have them <em style="color:inherit" class="w10oesj0">call you back</em> when the representative is available. You can hang up the phone and get on with your life.</p>
<p class="prg881n"><code class="i165vvr1">setTimeout()</code> is known as an <em style="color:inherit" class="w10oesj0">asynchronous</em> function. This means that it doesn’t block the thread. By contrast, <code class="i165vvr1">window.prompt()</code> is <em style="color:inherit" class="w10oesj0">synchronous</em>, because the JavaScript thread can't do anything else while it’s waiting.</p>
<p class="prg881n">The big downside with asynchronous code is that it means our code won't always run in a linear order. Consider the following setup:</p>
<div translate="no" data-less-bottom-margin="true" style="--max-height:auto" class="coaxrjn c1qvfy1x ae97emi"><button class="wqxmqd2"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1htlkn0"><span style="transform:rotate(0deg)" class="i14jtb2i"><span style="width:24px;height:24px;--duration:1422.2222222222222ms;--size:24px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span><span style="opacity:0;transform:scale(0)" class="c6am5ut"><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check"><path d="M20 6 9 17l-5-5"></path></svg></span><span class="haso6lc wlgbkjj">Copy to clipboard</span></span></span></button><div class="w19zxykk w1ey322x"><pre class="shiki shiki-themes josh-theme-light josh-theme-dark" style="--shiki-light:#000000;--shiki-dark:#ffffffff;--shiki-light-bg:#f5f6f900;--shiki-dark-bg:#0d0f1200"><code><span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">1. Before setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">setTimeout</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">()</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#C792EA"> =&gt;</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF"> {</span></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF"> console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">2. Inside setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">},</span><span style="--shiki-light:#BF00B8FF;--shiki-dark:#B88CF2FF"> 500</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span>
<span class="line"></span>
<span class="line"><span style="--shiki-light:#2A2A2AFF;--shiki-dark:#FFFFFFFF">console</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">.</span><span style="--shiki-light:#3D5AFEFF;--shiki-dark:#FF38A9FF;--shiki-light-font-style:italic;--shiki-dark-font-style:italic">log</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">(</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#FFD500FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">3. After setTimeout</span><span style="--shiki-light:#651FFFFF;--shiki-dark:#B9C4D0FF;--shiki-light-font-weight:bold;--shiki-dark-font-weight:inherit">'</span><span style="--shiki-light:#000000;--shiki-dark:#FFFFFFFF">)</span><span style="--shiki-light:#000000;--shiki-dark:#B9C4D0FF">;</span></span></code></pre></div></div>
<p class="prg881n">You might expect these logs to fire in order from top to bottom: <code class="i165vvr1">1</code> &gt; <code class="i165vvr1">2</code> &gt; <code class="i165vvr1">3</code>. <strong>But remember, the whole idea with callbacks is that we’re scheduling a call back.</strong> The JavaScript thread doesn’t sit around and wait, it ke ... |
http://localhost:1200/joshwcomeau/latest - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>Articles and Tutorials | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com</link>
<atom:link href="http://localhost:1200/joshwcomeau/latest" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on React, CSS, Animation, and more! - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Sat, 18 Jan 2025 18:17:09 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>Next-level frosted glass with backdrop-filter</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/css/backdrop-filter/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#css-filters-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">CSS filters</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#the-issue-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">The Issue</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#pointer-events-3" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Pointer events</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#flickering-top-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Flickering top</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#thicker-glass-5" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Thicker glass</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#browser-support-6" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Browser support</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#glassy-edge-7" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Glassy edge</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#the-final-code-8" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">The final code</a><a href="https://www.joshwcomeau.com/css/backdrop-filter/#continue-learning-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Continue learning</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">One of my all-time favourite CSS tricks is using <code class="i165vvr1">backdrop-filter: blur()</code> to create a frosted glass effect. I use it in just about every project I work on, including this blog!</p>
<p class="prg881n">Here’s a quick demo, to show what I’m talking about:</p>
<!-- --><div class="o11h7o12 w1bd8cgi" style="overflow:hidden"><div class="hduekzx"><div class="t1xdhus0"><div class="t1vhg4x9"><div class="rptyea t8ndwpz"></div><div class="y1qao05t t8ndwpz"></div><div class="g1v2ob21 t8ndwpz"></div></div><div class="tj6926w"><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="lep27c6 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(0deg);transform-origin:center center"></path></svg><span>Some Example Website</span><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="r2mz821 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(90deg);transform-origin:center center"></path></svg></div></div><div class="a13tcou3"><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-left"><path d="m12 19-7-7 7-7"></path><path d="M19 12H5"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-right"><path d="M5 12h14"></path><path d="m12 5 7 7-7 7"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-rotate-cw"><path d="M21 12a9 9 0 1 1-9-9c2.52 0 4.93 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path></svg></div><div class="a1booqr"><span>https://www.joshwcomeau.com/example-website</span></div></div></div><div style="overflow:clip"><div class="wi9xlz4 s1hfctpl d16kq653"><div class="hpv2kfm"><div class="b1dh7eg0"></div><div class="bi95upp"></div><p class="l19j3i1d">Some Website</p></div><div class="cufzfud"><img src="https://www.joshwcomeau.com/images/backdrop-filter/cupcake.png" style="animation-play-state:paused" class="c19tvtvc" referrerpolicy="no-referrer"><p>This is an example website showing how I typically use<!-- --> <code class="i165vvr1">backdrop-filter</code> to create glassy headers.</p><p class="o1fnc03u">Notice that as the cupcake moves behind the header, it appears blurry, as it would if it was passing behind frosted glass.</p><div class="atopy0n"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1t96eke w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><span style="width:20px;height:20px;--duration:1422.2222222222222ms;--size:20px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div></div></div>
<p class="prg881n">This effect helps us add depth and realism to our projects. It’s lovely.</p>
<p class="prg881n"><strong>But when I see this effect in the wild, it’s almost always missing some crucial optimizations.</strong> A couple of small changes can make our frosted glass <em style="color:inherit" class="w10oesj0">so</em> much more lush and realistic!</p>
<p class="prg881n">In this post, you’ll learn how to make the slickest frosted glass ever ✨. We’ll also learn quite a bit about CSS filters along the way!</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-credit-where-credit-is-due" class="t1xqw7pl">Credit where credit is due</strong><div class="c690lmu"><p class="prg881n">I learned about this effect from <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://expensive.toys/">Artur<!-- --> <span class="p7a6phg">Bien’s<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R2qm8deiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R2qm8deiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a> incredible demos. He credits <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://www.jamiegray.net/">Jamie<!-- --> <span class="p7a6phg">Gray<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R2sm8deiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R2sm8deiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a> for the original concept.</p></div></aside>
<h2 id="css-filters" data-element-type="ContentHeading" class="hrk0sy8"><a id="css-filters-1" href="https://www.joshwcomeau.com/css/backdrop-filter/#css-filters-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>CSS filters</h2>
<p class="prg881n">To briefly explain the underlying concept: CSS gives us quick and easy access to SVG filters via the <code class="i165vvr1">filter</code> property.</p>
<p class="prg881n">For example, we can give elements a Gaussian blur with <code class="i165vvr1">filter: blur()</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="wskxkqa"><img alt="My 3D mascot smiling with his tongue out. It’s very blurry, but dragging the slider makes it clearer" src="https://www.joshwcomeau.com/images/josh-originals/josh-tongue-dark.png" loading="lazy" style="filter:blur(16px)" class="j5sqeuc" referrerpolicy="no-referrer"><div class="c162apj7"></div></div><div style="--template-cols:1fr 1fr;grid-template-columns:1fr" class="c1j2qstn"><div style="max-width:400px;width:100%;margin-inline:auto" class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgleiavfelb:_radius" class="lpsyt9t">Blur radius<!-- -->:</label><span class="w1194iyg">16px</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgleiavfelb:_radius" min="0" max="16" class="i1i8zjy7" value="16"></div></div></div></div></div>
<p class="prg881n">There are lots of fun filter options, the sorts of things you’d find in image-editing software. Like, rotating the hue of all the colors:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1d4uhch"><img alt="My 3D mascot making a puzzled face. Dragging the slider causes him to become unnatural shades of green and pink." src="https://www.joshwcomeau.com/images/newsletter/joy-of-react-mascot.png" loading="lazy" style="filter:hue-rotate(0deg)" class="m1wtl9k0" referrerpolicy="no-referrer"><div class="c1kud80y"></div></div><div style="--template-cols:1fr 1fr;grid-template-columns:1fr" class="c1j2qstn"><div style="max-width:400px;width:100%;margin-inline:auto" class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgpeiavfelb:_rotation" class="lpsyt9t">Hue rotation<!-- -->:</label><span class="w1194iyg">0deg</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgpeiavfelb:_rotation" min="0" max="360" class="i1i8zjy7" value="0"></div></div></div></div></div>
<p class="prg881n">In these examples, I’m applying the filters to an <code class="i165vvr1">&lt;img&gt;</code> tag, but we can apply them to standard DOM nodes as well:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1icc7dx"><div style="filter:blur(0px) sepia(0%)" class="c1l5p6hh"><p>Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged.</p></div><div class="c1koe0ch"></div></div><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_radius" class="lpsyt9t">Blur radius<!-- -->:</label><span class="w1194iyg">0px</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_radius" min="0" max="16" step="0.1" class="i1i8zjy7" value="0"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_sepia" class="lpsyt9t">Sepia percentage<!-- -->:</label><span class="w1194iyg">0%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_sepia" min="0" max="100" class="i1i8zjy7" value="0"></div></div></div></div></div>
<p class="prg881n">Pretty neat, right?</p>
<p class="prg881n">Things get even cooler with <code class="i165vvr1">backdrop-filter</code>. This property lets us apply these same filters to the stuff <em style="color:inherit" class="w10oesj0">behind</em> a given element.</p>
<p class="prg881n">For example:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="bottom" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div class="w1qszusq"><div class="isvxvbh"><img src="https://www.joshwcomeau.com/images/backdrop-filter/tokyo.jpg" loading="lazy" class="tvcyp0e" referrerpolicy="no-referrer"><div style="top:80%;left:50%;--scale:2;-webkit-backdrop-filter:sepia(50%) brightness(130%);backdrop-filter:sepia(50%) brightness(130%)" class="ctfyewy"></div></div><div class="ck6v2m7"></div></div><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rh5eiavfelb:_sepia" class="lpsyt9t">Sepia<!-- -->:</label><span class="w1194iyg">50%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rh5eiavfelb:_sepia" min="0" max="100" class="i1i8zjy7" value="50"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rh5eiavfelb:_brightness" class="lpsyt9t">Brightness<!-- -->:</label><span class="w1194iyg">130%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rh5eiavfelb:_brightness" min="50" max="300" class="i1i8zjy7" value="130"></div></div></div></div></div>
<p class="prg881n">In this demo, the <code class="i165vvr1">.magic-ring</code> element sits in front of a photo (<a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://unsplash.com/photos/assorted-signages-on-buildings-nTBW1cOY1qI"> <span class="p7a6phg">source<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:Rn17eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:Rn17eiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a>). It uses the <code class="i165vvr1">backdrop-filter</code> property to apply some filtering to everything behind it, which can be used for some pretty artistic effects.</p>
<p class="prg881n">In practice, I pretty much only use <code class="i165vvr1">backdrop-filter</code> for one use case: blurring everything behind an element, usually a header, to create the “frosted glass” effect I mentioned earlier:</p>
<!-- --><div class="o11h7o12 w1bd8cgi" style="overflow:hidden"><div class="hduekzx"><div class="t1xdhus0"><div class="t1vhg4x9"><div class="rptyea t8ndwpz"></div><div class="y1qao05t t8ndwpz"></div><div class="g1v2ob21 t8ndwpz"></div></div><div class="tj6926w"><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="lep27c6 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(0deg);transform-origin:center center"></path></svg><span>Some Example Website</span><svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 16 16" preserveAspectRatio="none" class="r2mz821 choiq24"><path fill="currentColor" d="M0 16c8.837 0 16-7.163 16-16v16z" style="transform:rotate(90deg);transform-origin:center center"></path></svg></div></div><div class="a13tcou3"><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-left"><path d="m12 19-7-7 7-7"></path><path d="M19 12H5"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-arrow-right"><path d="M5 12h14"></path><path d="m12 5 7 7-7 7"></path></svg></div><div aria-hidden="true" class="fn6azp"><svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-rotate-cw"><path d="M21 12a9 9 0 1 1-9-9c2.52 0 4.93 1 6.74 2.74L21 8"></path><path d="M21 3v5h-5"></path></svg></div><div class="a1booqr"><span>https://www.joshwcomeau.com/example-website</span></div></div></div><div style="overflow:clip"><div class="wi9xlz4 s1hfctpl d16kq653"><div class="hpv2kfm"><div class="b1dh7eg0"></div><div class="bi95upp"></div><p class="l19j3i1d">Some Website</p></div><div class="cufzfud"><img src="https://www.joshwcomeau.com/images/backdrop-filter/cupcake.png" style="animation-play-state:paused" class="c19tvtvc" referrerpolicy="no-referrer"><p>This is an example website showing how I typically use<!-- --> <code class="i165vvr1">backdrop-filter</code> to create glassy headers.</p><p class="o1fnc03u">Notice that as the cupcake moves behind the header, it appears blurry, as it would if it was passing behind frosted glass.</p><div class="atopy0n"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1t96eke w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><span style="width:20px;height:20px;--duration:1422.2222222222222ms;--size:20px" class="w1gpdsbs"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-loader"><path d="M12 2v4"></path><path d="m16.2 7.8 2.9-2.9"></path><path d="M18 12h4"></path><path d="m16.2 16.2 2.9 2.9"></path><path d="M12 18v4"></path><path d="m4.9 19.1 2.9-2.9"></path><path d="M2 12h4"></path><path d="m4.9 4.9 2.9 2.9"></path></svg></span></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div></div></div>
<p class="prg881n">Alright. Let’s talk about the thing most developers miss.</p>
<h2 id="the-issue" data-element-type="ContentHeading" class="hrk0sy8"><a id="the-issue-2" href="https://www.joshwcomeau.com/css/backdrop-filter/#the-issue-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>The Issue</h2>
<p class="prg881n"><strong>Here’s the problem:</strong> The <code class="i165vvr1">backdrop-filter</code> algorithm only considers the pixels that are <i>directly behind</i> the element.</p>
<p class="prg881n">For filters like <code class="i165vvr1">brightness</code> or <code class="i165vvr1">hue-rotate</code>, that makes perfect sense. With blur, though, we actually want to consider pixels that are <em style="color:inherit" class="w10oesj0">near</em> the element too.</p>
<p class="prg881n">This is one of those things where a demo is worth a thousand words. Check out the difference:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="i1df5wci s360ott"><div id=":Rhneiavfelb:_mode" class="w73h7tw"><div class="r1cct0ab"><button value="default" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">Default</span></button><button value="fixed" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">Tweaked</span></button></div></div></div></div><div class="wr5us8n s1hfctpl d16kq653"><div class="hhqru8"><div class="g8qz4zc"></div></div><div style="animation-play-state:running" class="b1uj7lhw"></div></div><div class="awqze7l"><button style="--background-color:hsl(175deg 60% 25%)" data-discouraged="false" class="b1rsl36b w19mrn5v"><span style="order:1" class="iwv8n0j c1rpf4ut"><span style="--background-color:hsl(168deg 70% 35%)" class="i1qpbbb9"><span style="transform:translate(0px, 0px)
rotate(0deg)
scale(1);backface-visibility:hidden" class="i1m1cbkp"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-pause"><rect x="14" y="4" width="4" height="16" rx="1"></rect><rect x="6" y="4" width="4" height="16" rx="1"></rect></svg></span></span></span><span style="order:2" class="c7sw782">Play/Pause Animation</span></button></div></div></div>
<p class="prg881n">By default, the gaussian blur algorithm is applied to all of the pixels behind the element. This means that if a big colorful element is <em style="color:inherit" class="w10oesj0">near</em> the element, it won’t have any effect.</p>
<p class="prg881n">That’s not really how frosted glass works in real life though. Light bounces off of objects and then goes through the glass. It looks <em style="color:inherit" class="w10oesj0">so much better</em> when the blurring algorithm includes nearby content.</p>
<p class="prg881n">Unfortunately, this isn’t something we can configure directly. Instead, we need to be a bit crafty.</p>
<p class="prg881n">Here’s the code:</p>
<!-- --><div data-stacked="false" class="il1gxx9"><div style="--color-text:hsl(210deg 10% 90%);--color-background:hsl(210deg 15% 6%);--color-blurred-background:hsl(210deg 15% 6% / 0.75);--color-muted-background:hsl(210deg 38% 15% / 0.85);--color-action:hsl(240deg 95% 62%);--color-primary:hsl(225deg 100% 75%);--color-secondary:hsl(333deg 100% 55%);--color-tertiary:hsl(280deg 100% 85%);--color-decorative:hsl(200deg 50% 60%);--color-info-100:hsl(214deg 40% 15%);--color-info-200:hsl(218deg 32% 20%);--color-info-300:hsl(218deg 30% 25%);--color-info-400:hsl(218deg 45% 40%);--color-info-500:hsl(225deg 100% 60%);--color-info-700:hsl(215deg 100% 72%);--color-warning-100:hsl(30deg 25% 11%);--color-warning-200:hsl(32deg 20% 15%);--color-warning-300:hsl(32deg 25% 20%);--color-warning-400:hsl(35deg 45% 35%);--color-warning-500:hsl(40deg 100% 50%);--color-warning-700:hsl(43deg 100% 72%);--color-success-100:hsl(176deg 35% 10%);--color-success-200:hsl(176deg 26% 14%);--color-success-300:hsl(176deg 28% 20%);--color-success-400:hsl(176deg 45% 30%);--color-success-500:hsl(160deg 100% 40%);--color-success-700:hsl(160deg 80% 65%);--color-cloud-100:hsl(210deg 15% 6%);--color-cloud-300:hsl(212deg 40% 9%);--color-cloud-400:hsl(213deg 40% 10%);--color-cloud-500:hsl(213deg 40% 12%);--color-sky-from:hsl(214deg 40% 11%);--color-sky-to:hsl(200deg 50% 30%);--color-sky-subtle:hsl(210deg 40% 16%);--color-gray-50:hsl(210deg 19% 9%);--color-gray-100:hsl(210deg 15% 12%);--color-gray-200:hsl(210deg 15% 18%);--color-gray-300:hsl(210deg 10% 30%);--color-gray-400:hsl(210deg 9% 40%);--color-gray-500:hsl(210deg 8% 50%);--color-gray-600:hsl(210deg 12% 55%);--color-gray-700:hsl(210deg 14% 66%);--color-gray-800:hsl(210deg 20% 77%);--color-gray-900:hsl(210deg 25% 88%);--color-gray-1000:hsl(210deg 25% 96%);--color-adaptive-white:hsl(210deg 25% 92%);--syntax-bg:hsl(210deg 15% 6%);--syntax-highlight:hsl(210deg 30% 18%);--syntax-txt:hsl(0deg 0% 100%);--syntax-comment:hsl(200deg 18% 51%);--syntax-prop:hsl(326deg 100% 61%);--syntax-bool:hsl(50deg 100% 50%);--syntax-val:hsl(210deg 12% 65%);--syntax-str:hsl(259deg 100% 71%);--syntax-name:hsl(280deg 100% 66%);--syntax-del:hsl(0deg 100% 67%);--syntax-regex:hsl(50deg 100% 50%);--syntax-fn:hsl(195deg 100% 50%);--color-info:hsl(225deg 100% 80%);--color-info-background:hsl(225deg 100% 80% / 0.1);--color-error:hsl(340deg 95% 60%);--color-error-background:hsl(340deg 95% 43% / 0.1);--color-success:hsl(160deg 100% 40%);--color-success-background:hsl(160deg 100% 40% / 0.1);--color-alert:hsl(40deg 100% 50%);--color-alert-background:hsl(40deg 100% 50% / 0.1);--color-page-background:hsl(210deg 15% 6%);--color-page-primary:hsl(225deg 100% 75%);--kbd-background-color:hsl(210deg 9% 40%);--kbd-border-color:hsl(210deg 10% 30%);--color-code-bg:hsl(210deg 15% 12%);--color-content-outline:hsl(210deg 10% 30%)" class="i1j6zvew s1hfctpl d16kq653"><header class="w14j10eb"><p class="tkoi9ui">Code Playground</p><div class="au0cxa1"><button class="wsn2i55 a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7;overflow:visible" class="s1gvsyfu"><g style="transform:rotate(-45deg)" class="w15rf7yg"><rect x="0" y="8" width="24" height="6" rx="2" fill="var(--color-gray-100)"></rect><line x1="18" y1="8" x2="18" y2="14"></line></g></svg><span class="wlgbkjj">Format code using Prettier</span></button><abbr style="display:inline-block"><div class="wy3w6yt"><button title="Reset code" class="a6j6o3o"><svg viewBox="0 0 24 24" fill="none" style="width:16px;height:16px;opacity:0.7" class="s1gvsyfu"><mask id="skip-icon-mask-:R39i1eiavfelb:"><rect x="0" y="0" width="24" height="24" fill="black"></rect><rect x="9" y="0" width="16" height="24" fill="white"></rect></mask><line x1="5" y1="19" x2="5" y2="5"></line><polygon points="19 20 9 12 19 4 19 20" mask="url(#skip-icon-mask-:R39i1eiavfelb:)"></polygon></svg><span class="wlgbkjj">Reset Code</span></button><div class="pszdhfz"><span style="gap:1.5px" class="w15sqd9b"><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span><span data-is-filled="false" style="width:3px;height:3px;background-color:var(--color-primary)" class="dhkb22d"></span></span></div></div></abbr></div></header><div></div><div class="w198elmk"><div class="c1jztmaw"><div style="flex:516.12" class="f15xiyga"><div class="w16o0eiw"><div class="tt2ystl"><button style="--weight:var(--font-weight-bold);--color:var(--color-text)" class="ts2dciv">HTML</button><button style="--weight:var(--font-weight-normal);--color:var(--color-gray-500)" class="ts2dciv">CSS</button></div><div in="false" style="--max-height:80vh" class="w1bgwmtf"><button class="f9ibg9h"><span class="wlgbkjj">Focus the editor. This will trap focus until you press Escape.</span></button><label><span class="wlgbkjj">Code editor:</span><div translate="no" class="c18okpq0" style="position:relative;text-align:left;box-sizing:border-box;padding:0;overflow:hidden"><pre aria-hidden="true" style="margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-break:keep-all;overflow-wrap:break-word;position:relative;pointer-events:none;padding-top:0;padding-right:0;padding-bottom:0;padding-left:0"><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">style</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css selector" style="color:var(--syntax-name)">header</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">{</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">position</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> relative</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">}</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css selector class" style="color:var(--syntax-name)">.backdrop</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">{</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">position</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> absolute</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">inset</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">0</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">height</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">200</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">backdrop-filter</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css function" style="color:var(--syntax-fn)">blur</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">(</span><span class="token style language-css number" style="color:var(--syntax-bool)">16</span><span class="token style language-css unit" style="color:var(--syntax-str)">px</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">)</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css property" style="color:var(--syntax-prop)">mask-image</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">:</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css function" style="color:var(--syntax-fn)">linear-gradient</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">(</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> to bottom</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">,</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css color" style="color:var(--syntax-str)">black</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">0</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">50</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">,</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css color" style="color:var(--syntax-str)">transparent</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">50</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css number" style="color:var(--syntax-bool)">100</span><span class="token style language-css unit" style="color:var(--syntax-str)">%</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">)</span><span class="token style language-css punctuation" style="color:var(--syntax-str)">;</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"> </span><span class="token style language-css punctuation" style="color:var(--syntax-str)">}</span><span class="token style language-css" style="color:var(--syntax-str)"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token style language-css" style="color:var(--syntax-str)"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">style</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain" style="display:inline-block">
</span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">header</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"> </span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">class</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">backdrop</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;/</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">header</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token plain"></span></div><div class="token-line" style="color:var(--syntax-txt);padding:0;font-family:var(--font-family-mono)"><span class="token plain"></span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&lt;</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">div</span><span class="token tag" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"> </span><span class="token tag attr-name" style="color:var(--syntax-name);font-weight:var(--font-weight-medium)">class</span><span class="token tag attr-value punctuation attr-equals" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">=</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag attr-value" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">ball</span><span class="token tag attr-value punctuation" style="color:var(--syntax-val);font-weight:var(--font-weight-medium)">"</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)">&gt;</span><span class="token tag punctuation" style="color:var(--syntax-prop);font-weight:var(--font-weight-medium)"&g ... |
http://localhost:1200/joshwcomeau/popular - Success ✔️<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0">
<channel>
<title>Popular Content | Josh W. Comeau</title>
<link>https://www.joshwcomeau.com</link>
<atom:link href="http://localhost:1200/joshwcomeau/popular" rel="self" type="application/rss+xml"></atom:link>
<description>Friendly tutorials for developers. Focus on React, CSS, Animation, and more! - Powered by RSSHub</description>
<generator>RSSHub</generator>
<webMaster>contact@rsshub.app (RSSHub)</webMaster>
<language>en</language>
<lastBuildDate>Sat, 18 Jan 2025 18:17:10 GMT</lastBuildDate>
<ttl>5</ttl>
<item>
<title>How To Center a Div</title>
<description><div class="t1ma5bj5"><nav style="opacity:0;--x:20px" class="wt7fu6i"><h2 class="th0e8y">Table of Contents</h2><a href="https://www.joshwcomeau.com/css/center-a-div/#introduction" style="color:var(--color-primary);opacity:1;margin-top:12px;--font-size-px:16" class="c1kp06zm">Introduction</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with auto margins</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering within the viewport</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-elements-with-unknown-sizes-4" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering elements with unknown sizes</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-css-grid-5" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering with CSS Grid</a><a href="https://www.joshwcomeau.com/css/center-a-div/#differences-from-flexbox-6" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Differences from Flexbox</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-a-stack-of-elements-7" style="margin-top:4px;--font-size-px:14;padding-left:12px" class="c1kp06zm">Centering a stack of elements</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-text-8" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering text</a><a href="https://www.joshwcomeau.com/css/center-a-div/#centering-in-the-future-9" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Centering in the future</a><a href="https://www.joshwcomeau.com/css/center-a-div/#going-beyond-the-patterns-10" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">Going beyond the patterns</a><a href="https://www.joshwcomeau.com/css/center-a-div/#when-to-use-which-method-11" style="margin-top:12px;--font-size-px:16" class="c1kp06zm">When to use which method</a></nav><div class="l1mdfuoz"></div></div><a id="introduction" class="wcmb8ol"><span class="wlgbkjj">Introduction</span></a><p class="prg881n">For a long time, centering an element within its parent was a surprisingly tricky thing to do. As CSS has evolved, we've been granted more and more tools we can use to solve this problem. These days, we're spoiled for choice!</p>
<p class="prg881n">I decided to create this tutorial to help you understand the trade-offs between different approaches, and to give you an arsenal of strategies you can use, to handle centering in all sorts of scenarios.</p>
<p class="prg881n">Honestly, this turned out to be <em style="color:inherit" class="w10oesj0">way more interesting</em> than I initially thought&nbsp;😅. Even if you've been using CSS for a while, I bet you'll learn at least 1 new strategy!</p>
<h2 id="centering-with-auto-margins" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-auto-margins-1" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-auto-margins-1" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with auto margins</h2>
<p class="prg881n">The first strategy we'll look at is one of the oldest. If we want to center an element horizontally, we can do so using margins set to the special value <code class="i165vvr1">auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgbeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgbeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-right</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-left:auto;margin-right:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgbeiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgbeiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">First, we need to constrain the element's width; by default, elements in Flow layout will expand horizontally to fill the available space, and we can't really center something that is full-width.</p>
<p class="prg881n">I <em style="color:inherit" class="w10oesj0">could</em> constrain the width with a fixed value (eg. <code class="i165vvr1">200px</code>), but really what I want in this case is for the element to shrinkwrap around its content. <code class="i165vvr1">fit-content</code> is a magical value that does exactly this. Essentially, it makes “width” behave like “height”, so that the element’s size is determined by its contents.</p>
<p class="prg881n"><strong>Why am I setting <code class="i165vvr1">max-width</code> instead of <code class="i165vvr1">width</code>?</strong> Well, my goal is to stop the element from expanding horizontally. I want to clamp its maximum size. If I used <code class="i165vvr1">width</code> instead, it would lock it to that size, and the element would overflow when the container is really narrow. If you drag that “Container Width” slider all the way to the left, you can see that the element shrinks with its container.</p>
<p class="prg881n">Now that our element is constrained, we can center it with <em style="color:inherit" class="w10oesj0">auto margins</em>.</p>
<p class="prg881n">I like to think of auto margins like <i>Hungry Hungry Hippos</i>. Each auto margin will try to gobble up as much space as possible. For example, check out what happens if we <em style="color:inherit" class="w10oesj0">only</em> set <code class="i165vvr1">margin-left: auto</code>:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgneiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgneiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-left</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div></div><div style="margin-left:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgneiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgneiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n">When <code class="i165vvr1">margin-left</code> is the only side with auto margins, <i>all</i> of the extra space gets applied as margin to that side. When we set both <code class="i165vvr1">margin-left: auto</code> <i>and</i> <code class="i165vvr1">margin-right: auto</code>, the two hippos each gobble up an equal amount of space. This forces the element to the center.</p>
<p class="prg881n"><em style="color:inherit" class="w10oesj0">Also:</em> I've been using <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> because they're familiar, but there's a better, more-modern way to do this:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rgteiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rgteiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="r1xunlpt"><div class="cl7u8o0"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fit-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line highlighted"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin-inline</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dtehhfl cl7u8o0"><div data-use-adaptive-colors="true" style="width:100%" class="c1ndpdak l17lpp8k"><div class="m10k3fo8"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="m8pozar"></div><div style="opacity:0" class="wythvuy"><span class="t16m3o3k">.element</span></div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="m8pozar"></div></div><div style="margin-inline:auto" class="m1nbct2d wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div><div class="f1h8awvy"><label for=":R4rgteiavfelb:" class="wfao701"><button type="button" aria-pressed="false" style="width:44px;padding:2px;--radius:5px" id=":R4rgteiavfelb:" class="w1o3yz26"><div style="width:20px;height:20px;transform:translateX(0%);--handle-color:var(--color-gray-400);--handle-focus-color:var(--color-gray-400)" class="bl5hz2w"></div></button><div style="min-width:16px;min-height:16px" class="w1fjy7al"></div>Reveal Margin</label></div></div></div></div>
<p class="prg881n"><code class="i165vvr1">margin-inline</code> will set both <code class="i165vvr1">margin-left</code> and <code class="i165vvr1">margin-right</code> to the same value (<code class="i165vvr1">auto</code>). It has <a rel="noopener noreferrer" target="_blank" class="a1tdgj4y s1j91s21" href="https://caniuse.com/mdn-css_properties_margin-inline">very good browser<!-- --> <span class="p7a6phg">support<svg viewBox="0 0 24 24" fill="none" style="width:0.7em;height:0.7em;opacity:1" class="s14qet46 s1gvsyfu"><mask id="external-icon-mask-:R1e8veiavfelb:"><rect x="0" y="0" width="24" height="24" fill="white"></rect><rect x="10" y="0" width="16" height="14" fill="black"></rect></mask><rect x="3" y="6" width="15" height="15" rx="2" mask="url(#external-icon-mask-:R1e8veiavfelb:)"></rect><path d="M 10 14 L 20 4 h -6 h 6 v 6" stroke="currentColor" class="amgfren"></path></svg><span class="wlgbkjj">(opens in new tab)</span></span></a>, having landed in all major browsers several years ago.</p>
<aside class="i6el1g6 b1btlj5u"><div class="dwkrqh8"><svg xmlns="http://www.w3.org/2000/svg" width="28.5" height="34.5" fill="none" viewBox="0 0 57 69" preserveAspectRatio="none" class="sl0r50n"><path fill="var(--color-page-background)" d="M54 0V0.716804C54 25.9434 35.0653 47.1517 10 50L0 57V0H54Z" style="transition:fill var(--color-swap-duration) var(--color-swap-timing-function)"></path><path fill="var(--color-primary)" d="M56.9961 4.15364C57.0809 2.49896 55.8083 1.08879 54.1536 1.00394C52.499 0.919082 51.0888 2.19168 51.0039 3.84636L56.9961 4.15364ZM9.09704 51.7557L8.49716 48.8163L9.09704 51.7557ZM6 69V59.2227H0V69H6ZM9.69692 54.6951L14.3373 53.7481L13.1375 47.8693L8.49716 48.8163L9.69692 54.6951ZM14.3373 53.7481C38.202 48.8777 55.7486 28.4783 56.9961 4.15364L51.0039 3.84636C49.8967 25.4384 34.3213 43.5461 13.1375 47.8693L14.3373 53.7481ZM6 59.2227C6 57.0268 7.54537 55.1342 9.69692 54.6951L8.49716 48.8163C3.55195 49.8255 0 54.1756 0 59.2227H6Z"></path></svg><div class="f1dj028m"></div></div><div class="m1m3o2ch"></div><div class="iohjz9i"><svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-info"><circle cx="12" cy="12" r="10"></circle><path d="M12 16v-4"></path><path d="M12 8h.01"></path></svg></div><strong id="aside-logical-properties" class="t1xqw7pl">Logical properties</strong><div class="c690lmu"><p class="prg881n"><code class="i165vvr1">margin-inline</code> is more than just a convenient shorthand for <code class="i165vvr1">margin-left</code> + <code class="i165vvr1">margin-right</code>. It's part of a collection of <em style="color:inherit" class="w10oesj0">logical properties</em>, designed to make it easier to internationalize the web.</p><p class="prg881n">In English, characters are written in a horizontal line from left to right. Those characters are composed into words and sentences, and assembled into “blocks” (paragraphs, headings, lists, etc). Blocks are stacked vertically, from top to bottom. We can think of this as the <i>orientation</i> of English-language websites.</p><p class="prg881n">This isn't universal, though! Some languages, like Arabic and Hebrew, are written from right to left. Other languages, like Chinese, have historically been written vertically, with characters running from top to bottom, and <i>blocks</i> running side to side.<button class="t1cz8cv8"><span class="a1ni5s6f">*</span><span class="e2cu4f7"></span></button><span data-use-adaptive-colors="true" id=":Rhm91eiavfelb:" data-direction="n" role="tooltip" style="flex-direction:column" class="p1lgzcrp l17lpp8k"><span class="p833mxt"><span class="pj0hw36"><span class="cffclov">Interestingly, this has been changing over the past few decades, and most East Asian languages these days are written horizontally from left to right, especially on the web.</span></span></span><svg xmlns="http://www.w3.org/2000/svg" width="24" height="6" fill="none" viewBox="0 0 24 6" style="--origin-y:0%" class="s1vq0yeh s1k9z937"><path d="
M 0 0
C 6 0
7.199999999999999 6
12 6
C 16.8 6
18 0
24 0
Z
"></path></svg></span></p><p class="prg881n">The primary goal of logical properties is to create an abstraction that sits above these differences. Instead of setting <code class="i165vvr1">margin-left</code> for left-to-right languages and flipping it to <code class="i165vvr1">margin-right</code> for right-to-left languages, we can instead use <code class="i165vvr1">margin-inline-start</code>. The margin will automatically be applied to the correct side, depending on the page's language.</p></div></aside>
<p class="prg881n">Even though this centering method has been around forever, I still find myself reaching for it on a regular basis! It's particularly useful when we want to center a single child, without affecting any of its siblings (for example, an image in-between paragraphs in a blog post).</p>
<p class="prg881n">Let's continue on our centering journey.</p>
<h2 id="centering-with-flexbox" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-with-flexbox-2" href="https://www.joshwcomeau.com/css/center-a-div/#centering-with-flexbox-2" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering with Flexbox</h2>
<p class="prg881n">Flexbox is designed to give us a <em style="color:inherit" class="w10oesj0">ton</em> of control when it comes to distributing a group of items along a primary axis. It offers some <em style="color:inherit" class="w10oesj0">really</em> powerful tools for centering!</p>
<p class="prg881n">Let's start by centering a single element, both horizontally and vertically:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerWidth" min="25" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhdeiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhdeiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div class="rcwtqfe"><div class="c9vkmf2"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="djxxsdm c9vkmf2"><div data-use-adaptive-colors="true" style="width:100%;height:200px" class="c17eovmj l17lpp8k"><div class="f1vkqrq wythvuy"><span class="t16m3o3k">.element</span></div></div></div></div></div></div>
<p class="prg881n">The really cool thing about Flexbox centering is that it works <i>even when the children don’t fit in their container!</i> Try shrinking the width/height, and notice that the element overflows symmetrically.</p>
<p class="prg881n">It also works for <em style="color:inherit" class="w10oesj0">multiple</em> children. We can control how they stack with the <code class="i165vvr1">flex-direction</code> property:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Flex Direction<!-- -->:</span><div id=":Rhjeiavfelb:_flexDirection" class="ray0b30 w73h7tw"><div class="r1cct0ab"><button value="row" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">row</span></button><button value="column" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column</span></button><button value="row-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">row-reverse</span></button><button value="column-reverse" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">column-reverse</span></button></div></div></div></div><div class="r1xmilea"><div class="cgnjx9w c2tip9q"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">container</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> display</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> flex</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> flex-direction</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> row</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> justify-content</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> align-items</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> center</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> gap</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 4</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d1iw6ryr c2tip9q"><div data-use-adaptive-colors="true" style="width:100%;min-height:212px;flex-direction:row" class="c1oa8tf1 l17lpp8k"><div class="bt1ddka wythvuy"><span class="t16m3o3k">1</span></div><div style="z-index:2" class="bt1ddka wythvuy"><span class="t16m3o3k">2</span></div><div class="bt1ddka wythvuy"><span class="t16m3o3k">3</span></div></div></div></div></div></div>
<p class="prg881n">Out of all the centering patterns we'll explore in this tutorial, this is probably the one I use the most. It's a great jack-of-all-trades, a great default option.</p>
<h2 id="centering-within-the-viewport" data-element-type="ContentHeading" class="hrk0sy8"><a id="centering-within-the-viewport-3" href="https://www.joshwcomeau.com/css/center-a-div/#centering-within-the-viewport-3" class="aax5dne"><svg viewBox="0 0 24 24" fill="none" style="width:21px;height:21px;opacity:1" class="s1gvsyfu"><line x1="2" y1="8" x2="22" y2="8" stroke="currentColor" class="l1etwsvd"></line><line x1="2" y1="16" x2="22" y2="16" stroke="currentColor" class="l1etwsvd"></line><line x1="10" y1="2" x2="6" y2="22" stroke="currentColor" class="l1etwsvd"></line><line x1="18" y1="2" x2="14" y2="22" stroke="currentColor" class="l1etwsvd"></line></svg><span class="wlgbkjj">Link to this heading</span></a>Centering within the viewport</h2>
<p class="prg881n">So far, we've been looking at how to center an element within its parent container. But what if we want to center an element in a different context? Certain elements like dialogs, prompts, and GDPR banners need to be centered within the viewport.</p>
<p class="prg881n">This is the domain of <em style="color:inherit" class="w10oesj0">positioned layout,</em> a layout mode used when we want to take something out of flow and anchor it to something else.</p>
<p class="prg881n">Here's what this looks like:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr 1fr" class="c1j2qstn"><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerWidth" class="lpsyt9t">Container Width<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerWidth" min="30" max="100" class="i1i8zjy7" value="100"></div></div><div class="s4h798d"><div style="opacity:1" class="h1pypve8"><label for=":Rhveiavfelb:_containerHeight" class="lpsyt9t">Container Height<!-- -->:</label><span class="w1194iyg">100%</span></div><div class="cg5mgxb"><input style="--handle-color:var(--color-gray-900);--handle-focus-color:radial-gradient(
hsl(48deg 100% 75% / 1),
hsl(48deg 100% 75% / 0) 40%
),
conic-gradient(
from 90deg at 50% 50%,
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 75%),
hsl(45deg 100% 55%),
hsl(52deg 100% 95%),
hsl(45deg 100% 55%)
);--background-color:var(--color-background);--base-handle-size:16px;--track-size:3px" type="range" id=":Rhveiavfelb:_containerHeight" min="30" max="100" class="i1i8zjy7" value="100"></div></div></div><div><div class="rd1irhw"><div class="c1fq67g4"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> inset</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 0</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">px</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 5</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">vw</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> max-height</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 100</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">dvh</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> margin</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> auto</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div class="dougf27 c1fq67g4"><div data-use-adaptive-colors="true" style="width:100%;height:17.625rem" class="c8q6jp9 l17lpp8k"><div class="bfhwaun"><div class="hiqwc5b"><span style="width:3rem" class="hktm3y9 bx3nfyo"></span><span style="width:5rem" class="hktm3y9 bx3nfyo"></span><span style="width:1rem" class="hktm3y9 bx3nfyo"></span><span style="width:4rem" class="hktm3y9 bx3nfyo"></span><span style="width:7rem" class="hktm3y9 bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:1rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2rem" class="bx3nfyo"></span><span style="width:4.75rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:7rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:5rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div><div class="p1pvsarm"><span style="width:0.785rem" class="bx3nfyo"></span><span style="width:4.2rem" class="bx3nfyo"></span><span style="width:1.25rem" class="bx3nfyo"></span><span style="width:4rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span><span style="width:2.25rem" class="bx3nfyo"></span><span style="width:0.5rem" class="bx3nfyo"></span><span style="width:2.1rem" class="bx3nfyo"></span><span style="width:2.5rem" class="bx3nfyo"></span><span style="width:3.5rem" class="bx3nfyo"></span><span style="width:1rem" class="bx3nfyo"></span></div></div><div class="hpafenu m148wj37"><div style="clip-path:polygon(0% 0%, 0% 0%, 0% 100%, 0% 100%)" class="h1b4sgtf m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(100% 0%, 100% 0%, 100% 100%, 100% 100%)" class="h1b4sgtf m101ijsn"></div></div><div class="vm87swv m148wj37"><div style="clip-path:polygon(0% 0%, 100% 0%, 100% 0%, 0% 0%)" class="v15iwzns m101ijsn"></div><div style="opacity:0" class="p1j9axfr">.element</div><div style="clip-path:polygon(0% 100%, 100% 100%, 100% 100%, 0% 100%)" class="v15iwzns m101ijsn"></div></div><div class="m1b7mso0 p1j9axfr"></div></div></div></div></div></div></div>
<p class="prg881n">Of all the strategies we'll discuss, this one is probably the most complex. Let's break it down.</p>
<p class="prg881n">We're using <code class="i165vvr1">position: fixed</code>, which anchors this element to the viewport. I like to think of the viewport like a pane of glass that sits in front of the website, like the window of a train that shows the landscape scrolling by. An element with <code class="i165vvr1">position: fixed</code> is like a ladybug that lands on the window.</p>
<p class="prg881n">Next, we're setting <code class="i165vvr1">inset: 0px</code>, which is a shorthand that sets <code class="i165vvr1">top</code>, <code class="i165vvr1">left</code>, <code class="i165vvr1">right</code>, and <code class="i165vvr1">bottom</code> all to the same value, <code class="i165vvr1">0px</code>.</p>
<p class="prg881n">With only these two properties, the element would stretch to fill the entire viewport, growing so that it's 0px from each edge. This can be useful in some contexts, but it's not what we're going for here. We need to constrain it.</p>
<p class="prg881n">The exact values we pick will vary on the specifics of each situation, but in general we want to set default values (with <code class="i165vvr1">width</code> and <code class="i165vvr1">height</code>), as well as max values (<code class="i165vvr1">max-width</code> and <code class="i165vvr1">max-height</code>), so that the element doesn't overflow on smaller viewports.</p>
<p class="prg881n"><strong>There's something interesting here:</strong> we've set up an impossible condition. Our element can't be 0px from the left <i>and</i> 0px from the right <i>and</i> only 12rem wide (assuming the viewport is wider than 12rem). We can only pick 2:</p>
<!-- --><div data-use-adaptive-colors="true" data-demo-unit="true" data-control-position="top" class="o1m9cdug d16kq653"><div class="w1v90hn6"><div style="--template-cols:1fr" class="c1j2qstn"><div class="s360ott"><span class="l1fyq1q5">Pick two<!-- -->:</span><div id=":Rideiavfelb:_values" class="w73h7tw"><div class="r1cct0ab"><button value="left: 0px" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">left: 0px</span></button><button value="right: 0px" tabindex="-1" style="opacity:0.65" class="t6358bx"><span style="height:2px;filter:saturate(0%);opacity:0.4" class="b10n1zbt"></span><span class="tglc2q3">right: 0px</span></button><button value="width: 12rem" tabindex="-1" style="opacity:1" class="t6358bx"><span style="height:4px;filter:saturate(100%);opacity:1" class="b10n1zbt"></span><span class="tglc2q3">width: 12rem</span></button></div></div></div></div><div class="r3vmhao"><div class="c7zul9b"><div class="gvjoyup"><div class="c132w1ct"><div translate="no" data-less-bottom-margin="false" style="--max-height:auto" class="c1qvfy1x ae97emi"><div class="w1ey322x"><pre class="shiki shiki-themes josh-theme-dark josh-theme-light" style="background-color:#0d0f1200;--shiki-light-bg:#f5f6f900;color:#ffffffff;--shiki-light:#000000"><code><span class="line"><span style="color:#B9C4D0FF;--shiki-light:#AA00FFFF">.</span><span style="color:#FFCB6B;--shiki-light:#AA00FFFF">element</span><span style="color:#B9C4D0FF;--shiki-light:#000000"> {</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> position</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#2A2A2AFF"> fixed</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#A3FFEA;--shiki-light:#DA0079FF"> width</span><span style="color:#B9C4D0FF;--shiki-light:#000000">:</span><span style="color:#B88CF2FF;--shiki-light:#BF00B8FF"> 12</span><span style="color:#B88CF2FF;--shiki-light:#3D5AFEFF">rem</span><span style="color:#B9C4D0FF;--shiki-light:#000000">;</span></span>
<span class="line"><span style="color:#B9C4D0FF;--shiki-light:#000000">}</span></span></code></pre></div></div></div><div style="opacity:0;pointer-events:none" class="c132w1ct"><!--$!--><template data-dgst="BAILOUT_TO_CLIENT_SIDE_RENDERING"></template><!--/$--></div></div></div><div class="d19iwc1r c7zul9b"><div data-use-adaptive-colors="true" class="ctnf89z l17lpp8k"><div class="b14ie1by"><div class="fujgo4n"><span style="width:3rem" class="h1vx44g7 byjv7xx"></span><span style="width:5rem" class="h1vx44g7 byjv7xx"></span><span style="width:1rem" class="h1vx44g7 byjv7xx"></span><span style="width:4rem" class="h1vx44g7 byjv7xx"></span><span style="width:7rem" class="h1vx44g7 byjv7xx"></span></div><div class="f1p79fid"><span style="width:1rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2rem" class="byjv7xx"></span><span style="width:4.75rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:7rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:5rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span style="width:2.1rem" class="byjv7xx"></span><span style="width:0.785rem" class="byjv7xx"></span><span style="width:2.5rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:3.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span></div><div class="f1p79fid"><span style="width:0.785rem" class="byjv7xx"></span><span style="width:4.2rem" class="byjv7xx"></span><span style="width:1.25rem" class="byjv7xx"></span><span style="width:4rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1.5rem" class="byjv7xx"></span><span style="width:1rem" class="byjv7xx"></span><span style="width:2.25rem" class="byjv7xx"></span><span style="width:0.5rem" class="byjv7xx"></span><span styl |
Involved Issue / 该 PR 相关 Issue
Close #
Example for the Proposed Route(s) / 路由地址示例
New RSS Route Checklist / 新 RSS 路由检查表
Puppeteer
Note / 说明