Browse Source

Links

master
David Larlet 1 month ago
parent
commit
15fc6365fc
Signed by: David Larlet <david@larlet.fr> GPG Key ID: 3E2953A359E7E7BD

+ 907
- 0
cache/2024/02eaae467a3a88479393c9fe026f655a/index.html
File diff suppressed because it is too large
View File


+ 740
- 0
cache/2024/02eaae467a3a88479393c9fe026f655a/index.md
File diff suppressed because it is too large
View File


+ 252
- 0
cache/2024/529fce4c2b7c378f07aead94e62d3923/index.html View File

@@ -0,0 +1,252 @@
<!doctype html><!-- This is a valid HTML5 document. -->
<!-- Screen readers, SEO, extensions and so on. -->
<html lang="en">
<!-- Has to be within the first 1024 bytes, hence before the `title` element
See: https://www.w3.org/TR/2012/CR-html5-20121217/document-metadata.html#charset -->
<meta charset="utf-8">
<!-- Why no `X-UA-Compatible` meta: https://stackoverflow.com/a/6771584 -->
<!-- The viewport meta is quite crowded and we are responsible for that.
See: https://codepen.io/tigt/post/meta-viewport-for-2015 -->
<meta name="viewport" content="width=device-width,initial-scale=1">
<!-- Required to make a valid HTML5 document. -->
<title>Some little ways I’m using CSS :has() in the real world (archive) — David Larlet</title>
<meta name="description" content="Publication mise en cache pour en conserver une trace.">
<!-- That good ol' feed, subscribe :). -->
<link rel="alternate" type="application/atom+xml" title="Feed" href="/david/log/">
<!-- Generated from https://realfavicongenerator.net/ such a mess. -->
<link rel="apple-touch-icon" sizes="180x180" href="/static/david/icons2/apple-touch-icon.png">
<link rel="icon" type="image/png" sizes="32x32" href="/static/david/icons2/favicon-32x32.png">
<link rel="icon" type="image/png" sizes="16x16" href="/static/david/icons2/favicon-16x16.png">
<link rel="manifest" href="/static/david/icons2/site.webmanifest">
<link rel="mask-icon" href="/static/david/icons2/safari-pinned-tab.svg" color="#07486c">
<link rel="shortcut icon" href="/static/david/icons2/favicon.ico">
<meta name="msapplication-TileColor" content="#f7f7f7">
<meta name="msapplication-config" content="/static/david/icons2/browserconfig.xml">
<meta name="theme-color" content="#f7f7f7" media="(prefers-color-scheme: light)">
<meta name="theme-color" content="#272727" media="(prefers-color-scheme: dark)">
<!-- Is that even respected? Retrospectively? What a shAItshow…
https://neil-clarke.com/block-the-bots-that-feed-ai-models-by-scraping-your-website/ -->
<meta name="robots" content="noai, noimageai">
<!-- Documented, feel free to shoot an email. -->
<link rel="stylesheet" href="/static/david/css/style_2021-01-20.css">
<!-- See https://www.zachleat.com/web/comprehensive-webfonts/ for the trade-off. -->
<link rel="preload" href="/static/david/css/fonts/triplicate_t4_poly_regular.woff2" as="font" type="font/woff2" media="(prefers-color-scheme: light), (prefers-color-scheme: no-preference)" crossorigin>
<link rel="preload" href="/static/david/css/fonts/triplicate_t4_poly_bold.woff2" as="font" type="font/woff2" media="(prefers-color-scheme: light), (prefers-color-scheme: no-preference)" crossorigin>
<link rel="preload" href="/static/david/css/fonts/triplicate_t4_poly_italic.woff2" as="font" type="font/woff2" media="(prefers-color-scheme: light), (prefers-color-scheme: no-preference)" crossorigin>
<link rel="preload" href="/static/david/css/fonts/triplicate_t3_regular.woff2" as="font" type="font/woff2" media="(prefers-color-scheme: dark)" crossorigin>
<link rel="preload" href="/static/david/css/fonts/triplicate_t3_bold.woff2" as="font" type="font/woff2" media="(prefers-color-scheme: dark)" crossorigin>
<link rel="preload" href="/static/david/css/fonts/triplicate_t3_italic.woff2" as="font" type="font/woff2" media="(prefers-color-scheme: dark)" crossorigin>
<script>
function toggleTheme(themeName) {
document.documentElement.classList.toggle(
'forced-dark',
themeName === 'dark'
)
document.documentElement.classList.toggle(
'forced-light',
themeName === 'light'
)
}
const selectedTheme = localStorage.getItem('theme')
if (selectedTheme !== 'undefined') {
toggleTheme(selectedTheme)
}
</script>

<meta name="robots" content="noindex, nofollow">
<meta content="origin-when-cross-origin" name="referrer">
<!-- Canonical URL for SEO purposes -->
<link rel="canonical" href="https://piccalil.li/blog/some-little-ways-im-using-css-has-in-the-real-world/">

<body class="remarkdown h1-underline h2-underline h3-underline em-underscore hr-center ul-star pre-tick" data-instant-intensity="viewport-all">


<article>
<header>
<h1>Some little ways I’m using CSS :has() in the real world</h1>
</header>
<nav>
<p class="center">
<a href="/david/" title="Aller à l’accueil"><svg class="icon icon-home">
<use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-home"></use>
</svg> Accueil</a> •
<a href="https://piccalil.li/blog/some-little-ways-im-using-css-has-in-the-real-world/" title="Lien vers le contenu original">Source originale</a>
<br>
Mis en cache le 2024-03-07
</p>
</nav>
<hr>
<section class="[ post ] [ p-summary ]" aria-label="Quick summary"><p><em class="color-dark-glare">I’ve created some low fidelity demos of :has() snippets that I’ve been using in real-world client projects.</em></p></section>
<div class="[ post ] [ flow ]"><p>There’s a lot of chatter around the new(ish) <code>:has()</code> pseudo-class. It’s something we’ve been crying out for, for years: being able to select parent elements!</p><p>A useful mental model for <code>:has()</code> is that you are querying the parent’s children’s state and/or presence rather than selecting the parent from the children themselves. I like that. It makes a lot of sense.</p><p>I’m not 100% convinced <code>:has()</code> is the silver bullet others might claim it is though. I personally still utilise <a href="https://cube.fyi/exception.html">CUBE exceptions</a> more regularly, but I am also in the privileged position where projects I work on in <a href="https://set.studio/">the studio</a> don’t restrict access to the markup. I see <code>:has()</code> as being more useful for little tweaks more than anything, but if you don’t have access to markup, it really <em>is</em> a silver bullet.</p><p>With all that in mind, I thought I’d produce some low fidelity examples of how I’ve been using <code>:has()</code> lately on proper client projects to give you some real world stuff to look at. Let’s dig in.</p><h2 id="heading-banner-layout-adjustments">Banner layout adjustments<a href="#heading-banner-layout-adjustments" class="heading-permalink"><span class="visually-hidden"> permalink</span><svg fill="currentColor" aria-hidden="true" focusable="false" xmlns="http://www.w3.org/2000/svg" viewbox="0 0 24 24"><path d="M9.199 13.599a5.99 5.99 0 0 0 3.949 2.345 5.987 5.987 0 0 0 5.105-1.702l2.995-2.994a5.992 5.992 0 0 0 1.695-4.285 5.976 5.976 0 0 0-1.831-4.211 5.99 5.99 0 0 0-6.431-1.242 6.003 6.003 0 0 0-1.905 1.24l-1.731 1.721a.999.999 0 1 0 1.41 1.418l1.709-1.699a3.985 3.985 0 0 1 2.761-1.123 3.975 3.975 0 0 1 2.799 1.122 3.997 3.997 0 0 1 .111 5.644l-3.005 3.006a3.982 3.982 0 0 1-3.395 1.126 3.987 3.987 0 0 1-2.632-1.563A1 1 0 0 0 9.201 13.6zm5.602-3.198a5.99 5.99 0 0 0-3.949-2.345 5.987 5.987 0 0 0-5.105 1.702l-2.995 2.994a5.992 5.992 0 0 0-1.695 4.285 5.976 5.976 0 0 0 1.831 4.211 5.99 5.99 0 0 0 6.431 1.242 6.003 6.003 0 0 0 1.905-1.24l1.723-1.723a.999.999 0 1 0-1.414-1.414L9.836 19.81a3.985 3.985 0 0 1-2.761 1.123 3.975 3.975 0 0 1-2.799-1.122 3.997 3.997 0 0 1-.111-5.644l3.005-3.006a3.982 3.982 0 0 1 3.395-1.126 3.987 3.987 0 0 1 2.632 1.563 1 1 0 0 0 1.602-1.198z"></path></svg></a></h2><p>In a design system we work on for a client, there’s a pretty straightforward banner. This was recently updated to be dismissible, so we had to create a new variant to the pattern.</p><p>The only difference to the default pattern though is there’s a <code>&lt;button&gt;</code> element present, so a quick <code>:has()</code> query later, we could apply a flex layout in a jiffy.</p><div class="code-block" data-element="code-block">

<div class="wrapper">
<div class="code-block__code">
<pre id="code-block-0"><code class="language-css">.banner {
background: var(--color-primary);
color: var(--color-light);
font-weight: var(--font-bold);
text-align: center;
}

.banner:has(button) {
display: flex;
justify-content: space-between;
gap: var(--space-s);
text-align: revert;
}
</code></pre>
</div>
</div>
</div><p><code-pen id="PogPWRe" title="Banner layout demo"></code-pen></p><h2 id="heading-flex-labels-with-input-children">Flex labels with input children<a href="#heading-flex-labels-with-input-children" class="heading-permalink"><span class="visually-hidden"> permalink</span><svg fill="currentColor" aria-hidden="true" focusable="false" xmlns="http://www.w3.org/2000/svg" viewbox="0 0 24 24"><path d="M9.199 13.599a5.99 5.99 0 0 0 3.949 2.345 5.987 5.987 0 0 0 5.105-1.702l2.995-2.994a5.992 5.992 0 0 0 1.695-4.285 5.976 5.976 0 0 0-1.831-4.211 5.99 5.99 0 0 0-6.431-1.242 6.003 6.003 0 0 0-1.905 1.24l-1.731 1.721a.999.999 0 1 0 1.41 1.418l1.709-1.699a3.985 3.985 0 0 1 2.761-1.123 3.975 3.975 0 0 1 2.799 1.122 3.997 3.997 0 0 1 .111 5.644l-3.005 3.006a3.982 3.982 0 0 1-3.395 1.126 3.987 3.987 0 0 1-2.632-1.563A1 1 0 0 0 9.201 13.6zm5.602-3.198a5.99 5.99 0 0 0-3.949-2.345 5.987 5.987 0 0 0-5.105 1.702l-2.995 2.994a5.992 5.992 0 0 0-1.695 4.285 5.976 5.976 0 0 0 1.831 4.211 5.99 5.99 0 0 0 6.431 1.242 6.003 6.003 0 0 0 1.905-1.24l1.723-1.723a.999.999 0 1 0-1.414-1.414L9.836 19.81a3.985 3.985 0 0 1-2.761 1.123 3.975 3.975 0 0 1-2.799-1.122 3.997 3.997 0 0 1-.111-5.644l3.005-3.006a3.982 3.982 0 0 1 3.395-1.126 3.987 3.987 0 0 1 2.632 1.563 1 1 0 0 0 1.602-1.198z"></path></svg></a></h2><p>The context for this is that I like the following pattern for labels because I like to keep them as <code>inline</code> elements, but form fields that follow them should break on to a new line.</p><div class="code-block" data-element="code-block">

<div class="wrapper">
<div class="code-block__code">
<pre id="code-block-1"><code class="language-css">label::after {
content: "\A";
white-space: pre;
}
</code></pre>
</div>
</div>
</div><p>For labels that contain inputs like checkboxes and radios though, it’s useful to render those as flexbox layouts. Historically, that would require a class being added (or several in <abbr title="Atomic Style Sheets">ASS</abbr> codebases), but now, I’ve updated our global styles to this instead.</p><div class="code-block" data-element="code-block">

<div class="wrapper">
<div class="code-block__code">
<pre id="code-block-2"><code class="language-css">label:has(input) {
display: flex;
align-items: flex-start;
gap: var(--space-s);
}
</code></pre>
</div>
</div>
</div><fyi-unit><p>The reason I only look for <code>input</code> is because I never put text inputs inside labels. If you do that, you should update the selector to this instead: <code>label:has(:is(input[type="checkbox"], input[type="radio"]))</code>.</p></fyi-unit><p><code-pen id="bGJVqgX" title="A label with text input followed by a label containing a checkbox"></code-pen></p><h2 id="heading-highlight-parent-elements-when-their-children-are-targeted">Highlight parent elements when their children are targeted<a href="#heading-highlight-parent-elements-when-their-children-are-targeted" class="heading-permalink"><span class="visually-hidden"> permalink</span><svg fill="currentColor" aria-hidden="true" focusable="false" xmlns="http://www.w3.org/2000/svg" viewbox="0 0 24 24"><path d="M9.199 13.599a5.99 5.99 0 0 0 3.949 2.345 5.987 5.987 0 0 0 5.105-1.702l2.995-2.994a5.992 5.992 0 0 0 1.695-4.285 5.976 5.976 0 0 0-1.831-4.211 5.99 5.99 0 0 0-6.431-1.242 6.003 6.003 0 0 0-1.905 1.24l-1.731 1.721a.999.999 0 1 0 1.41 1.418l1.709-1.699a3.985 3.985 0 0 1 2.761-1.123 3.975 3.975 0 0 1 2.799 1.122 3.997 3.997 0 0 1 .111 5.644l-3.005 3.006a3.982 3.982 0 0 1-3.395 1.126 3.987 3.987 0 0 1-2.632-1.563A1 1 0 0 0 9.201 13.6zm5.602-3.198a5.99 5.99 0 0 0-3.949-2.345 5.987 5.987 0 0 0-5.105 1.702l-2.995 2.994a5.992 5.992 0 0 0-1.695 4.285 5.976 5.976 0 0 0 1.831 4.211 5.99 5.99 0 0 0 6.431 1.242 6.003 6.003 0 0 0 1.905-1.24l1.723-1.723a.999.999 0 1 0-1.414-1.414L9.836 19.81a3.985 3.985 0 0 1-2.761 1.123 3.975 3.975 0 0 1-2.799-1.122 3.997 3.997 0 0 1-.111-5.644l3.005-3.006a3.982 3.982 0 0 1 3.395-1.126 3.987 3.987 0 0 1 2.632 1.563 1 1 0 0 0 1.602-1.198z"></path></svg></a></h2><p>This one is super quick and super simple. If you’ve got an element with an <code>id</code>, you can trigger its <code>:target</code> state by appending it’s id to the URL with a <code>#</code>, like this: <code>https://example.com/#my-element</code>.</p><p>Historically, you couldn’t apply styles to an element’s parent when it’s targeted, but now you can with <code>:has()</code>.</p><div class="code-block" data-element="code-block">

<div class="wrapper">
<div class="code-block__code">
<pre id="code-block-3"><code class="language-css">section:has(:target) {
background: var(--color-light-shade);
border: 2px solid var(--color-primary);
}
</code></pre>
</div>
</div>
</div><p>Handy as heck.</p><p><code-pen id="eYopgwJ" title="A section is highlighted when its heading is targeted"></code-pen></p><h2 id="heading-dimmer-siblings-when-an-element-is-hovered">Dimmer siblings when an element is hovered<a href="#heading-dimmer-siblings-when-an-element-is-hovered" class="heading-permalink"><span class="visually-hidden"> permalink</span><svg fill="currentColor" aria-hidden="true" focusable="false" xmlns="http://www.w3.org/2000/svg" viewbox="0 0 24 24"><path d="M9.199 13.599a5.99 5.99 0 0 0 3.949 2.345 5.987 5.987 0 0 0 5.105-1.702l2.995-2.994a5.992 5.992 0 0 0 1.695-4.285 5.976 5.976 0 0 0-1.831-4.211 5.99 5.99 0 0 0-6.431-1.242 6.003 6.003 0 0 0-1.905 1.24l-1.731 1.721a.999.999 0 1 0 1.41 1.418l1.709-1.699a3.985 3.985 0 0 1 2.761-1.123 3.975 3.975 0 0 1 2.799 1.122 3.997 3.997 0 0 1 .111 5.644l-3.005 3.006a3.982 3.982 0 0 1-3.395 1.126 3.987 3.987 0 0 1-2.632-1.563A1 1 0 0 0 9.201 13.6zm5.602-3.198a5.99 5.99 0 0 0-3.949-2.345 5.987 5.987 0 0 0-5.105 1.702l-2.995 2.994a5.992 5.992 0 0 0-1.695 4.285 5.976 5.976 0 0 0 1.831 4.211 5.99 5.99 0 0 0 6.431 1.242 6.003 6.003 0 0 0 1.905-1.24l1.723-1.723a.999.999 0 1 0-1.414-1.414L9.836 19.81a3.985 3.985 0 0 1-2.761 1.123 3.975 3.975 0 0 1-2.799-1.122 3.997 3.997 0 0 1-.111-5.644l3.005-3.006a3.982 3.982 0 0 1 3.395-1.126 3.987 3.987 0 0 1 2.632 1.563 1 1 0 0 0 1.602-1.198z"></path></svg></a></h2><p>It’s a design pattern that’s been on the web forever. The idea is when you hover an element its siblings all dim, so the user’s visual focus is more targeted.</p><p>We’ve been able to do this with CSS forever too, but the selectors to achieve the effect were pretty gnarly. Quite a few approaches resulted in flickering or all items dimmed if your pointer accidentally found itself in gutters too.</p><p>It’s not the case any more with <code>:has()</code> though!</p><div class="code-block" data-element="code-block">

<div class="wrapper">
<div class="code-block__code">
<pre id="code-block-4"><code class="language-css">.tiles:has(:hover) .tile:not(:hover) {
opacity: 70%;
}
</code></pre>
</div>
</div>
</div><p>The beauty of this selector is it’s <em>really</em> clear what’s going on too.</p><p><code-pen id="ZEZbeGW" title="Other tiles dim when one of them is hovered"></code-pen></p><p>Yeh, there’s nothing complicated or fancy in this article, but I just wanted to show some handy real-world ways to use <code>:has()</code>. If you really want to get into <code>:has()</code>, I strongly recommend checking out <a href="https://ishadeed.com/article/css-has-guide">Ahmad’s interactive guide</a>. It’s fantastic!</p><p>P.S. one last little trick. On this site, paragraphs in the <code>.post</code> block are limited to <code>60ch</code> as a <code>max-width</code>. That's not ideal for demos though, so…</p><div class="code-block" data-element="code-block">

<div class="wrapper">
<div class="code-block__code">
<pre id="code-block-5"><code class="language-css">.post p:has(code-pen) {
max-width: unset;
}
</code></pre>
</div>
</div>
</div><p>Easy peasy 🙂</p></div>
</article>


<hr>

<footer>
<p>
<a href="/david/" title="Aller à l’accueil"><svg class="icon icon-home">
<use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-home"></use>
</svg> Accueil</a> •
<a href="/david/log/" title="Accès au flux RSS"><svg class="icon icon-rss2">
<use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-rss2"></use>
</svg> Suivre</a> •
<a href="http://larlet.com" title="Go to my English profile" data-instant><svg class="icon icon-user-tie">
<use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-user-tie"></use>
</svg> Pro</a> •
<a href="mailto:david%40larlet.fr" title="Envoyer un courriel"><svg class="icon icon-mail">
<use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-mail"></use>
</svg> Email</a> •
<abbr class="nowrap" title="Hébergeur : Alwaysdata, 62 rue Tiquetonne 75002 Paris, +33184162340"><svg class="icon icon-hammer2">
<use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-hammer2"></use>
</svg> Légal</abbr>
</p>
<template id="theme-selector">
<form>
<fieldset>
<legend><svg class="icon icon-brightness-contrast">
<use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-brightness-contrast"></use>
</svg> Thème</legend>
<label>
<input type="radio" value="auto" name="chosen-color-scheme" checked> Auto
</label>
<label>
<input type="radio" value="dark" name="chosen-color-scheme"> Foncé
</label>
<label>
<input type="radio" value="light" name="chosen-color-scheme"> Clair
</label>
</fieldset>
</form>
</template>
</footer>
<script src="/static/david/js/instantpage-5.1.0.min.js" type="module"></script>
<script>
function loadThemeForm(templateName) {
const themeSelectorTemplate = document.querySelector(templateName)
const form = themeSelectorTemplate.content.firstElementChild
themeSelectorTemplate.replaceWith(form)

form.addEventListener('change', (e) => {
const chosenColorScheme = e.target.value
localStorage.setItem('theme', chosenColorScheme)
toggleTheme(chosenColorScheme)
})

const selectedTheme = localStorage.getItem('theme')
if (selectedTheme && selectedTheme !== 'undefined') {
form.querySelector(`[value="${selectedTheme}"]`).checked = true
}
}

const prefersColorSchemeDark = '(prefers-color-scheme: dark)'
window.addEventListener('load', () => {
let hasDarkRules = false
for (const styleSheet of Array.from(document.styleSheets)) {
let mediaRules = []
for (const cssRule of styleSheet.cssRules) {
if (cssRule.type !== CSSRule.MEDIA_RULE) {
continue
}
// WARNING: Safari does not have/supports `conditionText`.
if (cssRule.conditionText) {
if (cssRule.conditionText !== prefersColorSchemeDark) {
continue
}
} else {
if (cssRule.cssText.startsWith(prefersColorSchemeDark)) {
continue
}
}
mediaRules = mediaRules.concat(Array.from(cssRule.cssRules))
}

// WARNING: do not try to insert a Rule to a styleSheet you are
// currently iterating on, otherwise the browser will be stuck
// in a infinite loop…
for (const mediaRule of mediaRules) {
styleSheet.insertRule(mediaRule.cssText)
hasDarkRules = true
}
}
if (hasDarkRules) {
loadThemeForm('#theme-selector')
}
})
</script>
</body>
</html>

+ 84
- 0
cache/2024/529fce4c2b7c378f07aead94e62d3923/index.md View File

@@ -0,0 +1,84 @@
title: Some little ways I’m using CSS :has() in the real world
url: https://piccalil.li/blog/some-little-ways-im-using-css-has-in-the-real-world/
hash_url: 529fce4c2b7c378f07aead94e62d3923
archive_date: 2024-03-07
og_image: https://api.urlbox.io/v1/ln9ptArKXobLRpDQ/png?width=1024&height=526&url=https%3A%2F%2Fpiccalil.li%2Fblog%2Fsome-little-ways-im-using-css-has-in-the-real-world%2F%2Fsocial-image&retina=true
description: I’ve created some low fidelity demos of :has() snippets that I’ve been using in real-world client projects.
favicon: https://piccalil.li/images/favicon-32x32.png
language: en_US

<section class="[ post ] [ p-summary ]" aria-label="Quick summary"><p><em class="color-dark-glare">I’ve created some low fidelity demos of :has() snippets that I’ve been using in real-world client projects.</em></p></section><div class="[ post ] [ flow ]"><p>There’s a lot of chatter around the new(ish) <code>:has()</code> pseudo-class. It’s something we’ve been crying out for, for years: being able to select parent elements!</p><p>A useful mental model for <code>:has()</code> is that you are querying the parent’s children’s state and/or presence rather than selecting the parent from the children themselves. I like that. It makes a lot of sense.</p><p>I’m not 100% convinced <code>:has()</code> is the silver bullet others might claim it is though. I personally still utilise <a href="https://cube.fyi/exception.html">CUBE exceptions</a> more regularly, but I am also in the privileged position where projects I work on in <a href="https://set.studio/">the studio</a> don’t restrict access to the markup. I see <code>:has()</code> as being more useful for little tweaks more than anything, but if you don’t have access to markup, it really <em>is</em> a silver bullet.</p><p>With all that in mind, I thought I’d produce some low fidelity examples of how I’ve been using <code>:has()</code> lately on proper client projects to give you some real world stuff to look at. Let’s dig in.</p><h2 id="heading-banner-layout-adjustments">Banner layout adjustments<a href="#heading-banner-layout-adjustments" class="heading-permalink"><span class="visually-hidden"> permalink</span><svg fill="currentColor" aria-hidden="true" focusable="false" xmlns="http://www.w3.org/2000/svg" viewbox="0 0 24 24"><path d="M9.199 13.599a5.99 5.99 0 0 0 3.949 2.345 5.987 5.987 0 0 0 5.105-1.702l2.995-2.994a5.992 5.992 0 0 0 1.695-4.285 5.976 5.976 0 0 0-1.831-4.211 5.99 5.99 0 0 0-6.431-1.242 6.003 6.003 0 0 0-1.905 1.24l-1.731 1.721a.999.999 0 1 0 1.41 1.418l1.709-1.699a3.985 3.985 0 0 1 2.761-1.123 3.975 3.975 0 0 1 2.799 1.122 3.997 3.997 0 0 1 .111 5.644l-3.005 3.006a3.982 3.982 0 0 1-3.395 1.126 3.987 3.987 0 0 1-2.632-1.563A1 1 0 0 0 9.201 13.6zm5.602-3.198a5.99 5.99 0 0 0-3.949-2.345 5.987 5.987 0 0 0-5.105 1.702l-2.995 2.994a5.992 5.992 0 0 0-1.695 4.285 5.976 5.976 0 0 0 1.831 4.211 5.99 5.99 0 0 0 6.431 1.242 6.003 6.003 0 0 0 1.905-1.24l1.723-1.723a.999.999 0 1 0-1.414-1.414L9.836 19.81a3.985 3.985 0 0 1-2.761 1.123 3.975 3.975 0 0 1-2.799-1.122 3.997 3.997 0 0 1-.111-5.644l3.005-3.006a3.982 3.982 0 0 1 3.395-1.126 3.987 3.987 0 0 1 2.632 1.563 1 1 0 0 0 1.602-1.198z"></path></svg></a></h2><p>In a design system we work on for a client, there’s a pretty straightforward banner. This was recently updated to be dismissible, so we had to create a new variant to the pattern.</p><p>The only difference to the default pattern though is there’s a <code>&lt;button&gt;</code> element present, so a quick <code>:has()</code> query later, we could apply a flex layout in a jiffy.</p><div class="code-block" data-element="code-block">

<div class="wrapper">
<div class="code-block__code">
<pre id="code-block-0"><code class="language-css">.banner {
background: var(--color-primary);
color: var(--color-light);
font-weight: var(--font-bold);
text-align: center;
}

.banner:has(button) {
display: flex;
justify-content: space-between;
gap: var(--space-s);
text-align: revert;
}
</code></pre>
</div>
</div>
</div><p><code-pen id="PogPWRe" title="Banner layout demo"></code-pen></p><h2 id="heading-flex-labels-with-input-children">Flex labels with input children<a href="#heading-flex-labels-with-input-children" class="heading-permalink"><span class="visually-hidden"> permalink</span><svg fill="currentColor" aria-hidden="true" focusable="false" xmlns="http://www.w3.org/2000/svg" viewbox="0 0 24 24"><path d="M9.199 13.599a5.99 5.99 0 0 0 3.949 2.345 5.987 5.987 0 0 0 5.105-1.702l2.995-2.994a5.992 5.992 0 0 0 1.695-4.285 5.976 5.976 0 0 0-1.831-4.211 5.99 5.99 0 0 0-6.431-1.242 6.003 6.003 0 0 0-1.905 1.24l-1.731 1.721a.999.999 0 1 0 1.41 1.418l1.709-1.699a3.985 3.985 0 0 1 2.761-1.123 3.975 3.975 0 0 1 2.799 1.122 3.997 3.997 0 0 1 .111 5.644l-3.005 3.006a3.982 3.982 0 0 1-3.395 1.126 3.987 3.987 0 0 1-2.632-1.563A1 1 0 0 0 9.201 13.6zm5.602-3.198a5.99 5.99 0 0 0-3.949-2.345 5.987 5.987 0 0 0-5.105 1.702l-2.995 2.994a5.992 5.992 0 0 0-1.695 4.285 5.976 5.976 0 0 0 1.831 4.211 5.99 5.99 0 0 0 6.431 1.242 6.003 6.003 0 0 0 1.905-1.24l1.723-1.723a.999.999 0 1 0-1.414-1.414L9.836 19.81a3.985 3.985 0 0 1-2.761 1.123 3.975 3.975 0 0 1-2.799-1.122 3.997 3.997 0 0 1-.111-5.644l3.005-3.006a3.982 3.982 0 0 1 3.395-1.126 3.987 3.987 0 0 1 2.632 1.563 1 1 0 0 0 1.602-1.198z"></path></svg></a></h2><p>The context for this is that I like the following pattern for labels because I like to keep them as <code>inline</code> elements, but form fields that follow them should break on to a new line.</p><div class="code-block" data-element="code-block">

<div class="wrapper">
<div class="code-block__code">
<pre id="code-block-1"><code class="language-css">label::after {
content: "\A";
white-space: pre;
}
</code></pre>
</div>
</div>
</div><p>For labels that contain inputs like checkboxes and radios though, it’s useful to render those as flexbox layouts. Historically, that would require a class being added (or several in <abbr title="Atomic Style Sheets">ASS</abbr> codebases), but now, I’ve updated our global styles to this instead.</p><div class="code-block" data-element="code-block">

<div class="wrapper">
<div class="code-block__code">
<pre id="code-block-2"><code class="language-css">label:has(input) {
display: flex;
align-items: flex-start;
gap: var(--space-s);
}
</code></pre>
</div>
</div>
</div><fyi-unit><p>The reason I only look for <code>input</code> is because I never put text inputs inside labels. If you do that, you should update the selector to this instead: <code>label:has(:is(input[type="checkbox"], input[type="radio"]))</code>.</p></fyi-unit><p><code-pen id="bGJVqgX" title="A label with text input followed by a label containing a checkbox"></code-pen></p><h2 id="heading-highlight-parent-elements-when-their-children-are-targeted">Highlight parent elements when their children are targeted<a href="#heading-highlight-parent-elements-when-their-children-are-targeted" class="heading-permalink"><span class="visually-hidden"> permalink</span><svg fill="currentColor" aria-hidden="true" focusable="false" xmlns="http://www.w3.org/2000/svg" viewbox="0 0 24 24"><path d="M9.199 13.599a5.99 5.99 0 0 0 3.949 2.345 5.987 5.987 0 0 0 5.105-1.702l2.995-2.994a5.992 5.992 0 0 0 1.695-4.285 5.976 5.976 0 0 0-1.831-4.211 5.99 5.99 0 0 0-6.431-1.242 6.003 6.003 0 0 0-1.905 1.24l-1.731 1.721a.999.999 0 1 0 1.41 1.418l1.709-1.699a3.985 3.985 0 0 1 2.761-1.123 3.975 3.975 0 0 1 2.799 1.122 3.997 3.997 0 0 1 .111 5.644l-3.005 3.006a3.982 3.982 0 0 1-3.395 1.126 3.987 3.987 0 0 1-2.632-1.563A1 1 0 0 0 9.201 13.6zm5.602-3.198a5.99 5.99 0 0 0-3.949-2.345 5.987 5.987 0 0 0-5.105 1.702l-2.995 2.994a5.992 5.992 0 0 0-1.695 4.285 5.976 5.976 0 0 0 1.831 4.211 5.99 5.99 0 0 0 6.431 1.242 6.003 6.003 0 0 0 1.905-1.24l1.723-1.723a.999.999 0 1 0-1.414-1.414L9.836 19.81a3.985 3.985 0 0 1-2.761 1.123 3.975 3.975 0 0 1-2.799-1.122 3.997 3.997 0 0 1-.111-5.644l3.005-3.006a3.982 3.982 0 0 1 3.395-1.126 3.987 3.987 0 0 1 2.632 1.563 1 1 0 0 0 1.602-1.198z"></path></svg></a></h2><p>This one is super quick and super simple. If you’ve got an element with an <code>id</code>, you can trigger its <code>:target</code> state by appending it’s id to the URL with a <code>#</code>, like this: <code>https://example.com/#my-element</code>.</p><p>Historically, you couldn’t apply styles to an element’s parent when it’s targeted, but now you can with <code>:has()</code>.</p><div class="code-block" data-element="code-block">

<div class="wrapper">
<div class="code-block__code">
<pre id="code-block-3"><code class="language-css">section:has(:target) {
background: var(--color-light-shade);
border: 2px solid var(--color-primary);
}
</code></pre>
</div>
</div>
</div><p>Handy as heck.</p><p><code-pen id="eYopgwJ" title="A section is highlighted when its heading is targeted"></code-pen></p><h2 id="heading-dimmer-siblings-when-an-element-is-hovered">Dimmer siblings when an element is hovered<a href="#heading-dimmer-siblings-when-an-element-is-hovered" class="heading-permalink"><span class="visually-hidden"> permalink</span><svg fill="currentColor" aria-hidden="true" focusable="false" xmlns="http://www.w3.org/2000/svg" viewbox="0 0 24 24"><path d="M9.199 13.599a5.99 5.99 0 0 0 3.949 2.345 5.987 5.987 0 0 0 5.105-1.702l2.995-2.994a5.992 5.992 0 0 0 1.695-4.285 5.976 5.976 0 0 0-1.831-4.211 5.99 5.99 0 0 0-6.431-1.242 6.003 6.003 0 0 0-1.905 1.24l-1.731 1.721a.999.999 0 1 0 1.41 1.418l1.709-1.699a3.985 3.985 0 0 1 2.761-1.123 3.975 3.975 0 0 1 2.799 1.122 3.997 3.997 0 0 1 .111 5.644l-3.005 3.006a3.982 3.982 0 0 1-3.395 1.126 3.987 3.987 0 0 1-2.632-1.563A1 1 0 0 0 9.201 13.6zm5.602-3.198a5.99 5.99 0 0 0-3.949-2.345 5.987 5.987 0 0 0-5.105 1.702l-2.995 2.994a5.992 5.992 0 0 0-1.695 4.285 5.976 5.976 0 0 0 1.831 4.211 5.99 5.99 0 0 0 6.431 1.242 6.003 6.003 0 0 0 1.905-1.24l1.723-1.723a.999.999 0 1 0-1.414-1.414L9.836 19.81a3.985 3.985 0 0 1-2.761 1.123 3.975 3.975 0 0 1-2.799-1.122 3.997 3.997 0 0 1-.111-5.644l3.005-3.006a3.982 3.982 0 0 1 3.395-1.126 3.987 3.987 0 0 1 2.632 1.563 1 1 0 0 0 1.602-1.198z"></path></svg></a></h2><p>It’s a design pattern that’s been on the web forever. The idea is when you hover an element its siblings all dim, so the user’s visual focus is more targeted.</p><p>We’ve been able to do this with CSS forever too, but the selectors to achieve the effect were pretty gnarly. Quite a few approaches resulted in flickering or all items dimmed if your pointer accidentally found itself in gutters too.</p><p>It’s not the case any more with <code>:has()</code> though!</p><div class="code-block" data-element="code-block">

<div class="wrapper">
<div class="code-block__code">
<pre id="code-block-4"><code class="language-css">.tiles:has(:hover) .tile:not(:hover) {
opacity: 70%;
}
</code></pre>
</div>
</div>
</div><p>The beauty of this selector is it’s <em>really</em> clear what’s going on too.</p><p><code-pen id="ZEZbeGW" title="Other tiles dim when one of them is hovered"></code-pen></p><p>Yeh, there’s nothing complicated or fancy in this article, but I just wanted to show some handy real-world ways to use <code>:has()</code>. If you really want to get into <code>:has()</code>, I strongly recommend checking out <a href="https://ishadeed.com/article/css-has-guide">Ahmad’s interactive guide</a>. It’s fantastic!</p><p>P.S. one last little trick. On this site, paragraphs in the <code>.post</code> block are limited to <code>60ch</code> as a <code>max-width</code>. That's not ideal for demos though, so…</p><div class="code-block" data-element="code-block">

<div class="wrapper">
<div class="code-block__code">
<pre id="code-block-5"><code class="language-css">.post p:has(code-pen) {
max-width: unset;
}
</code></pre>
</div>
</div>
</div><p>Easy peasy 🙂</p></div>

+ 4
- 0
cache/2024/index.html View File

@@ -128,6 +128,8 @@
<li><a href="/david/cache/2024/d0ffe1891c332b6fc6e7d7826d8489da/" title="Accès à l’article dans le cache local : Naming Variables In CSS">Naming Variables In CSS</a> (<a href="https://jwdallas.com/posts/namingcssvariables/" title="Accès à l’article original distant : Naming Variables In CSS">original</a>)</li>
<li><a href="/david/cache/2024/02eaae467a3a88479393c9fe026f655a/" title="Accès à l’article dans le cache local : CSS :has() Interactive Guide">CSS :has() Interactive Guide</a> (<a href="https://ishadeed.com/article/css-has-guide/" title="Accès à l’article original distant : CSS :has() Interactive Guide">original</a>)</li>
<li><a href="/david/cache/2024/40aada3cc8d6897fda5a277c4299c1fd/" title="Accès à l’article dans le cache local : We Need to Talk About the Front Web">We Need to Talk About the Front Web</a> (<a href="https://gericci.me/we-need-to-talk-about-the-front-web-5.html" title="Accès à l’article original distant : We Need to Talk About the Front Web">original</a>)</li>
<li><a href="/david/cache/2024/44c12c8fbb59c7239f0f3b04bae189b7/" title="Accès à l’article dans le cache local : Tailwind marketing and misinformation engine">Tailwind marketing and misinformation engine</a> (<a href="https://nuejs.org/blog/tailwind-misinformation-engine/" title="Accès à l’article original distant : Tailwind marketing and misinformation engine">original</a>)</li>
@@ -178,6 +180,8 @@
<li><a href="/david/cache/2024/1f4e359d100592aee6e46505a40b2a01/" title="Accès à l’article dans le cache local : Playing with Infinity in CSS">Playing with Infinity in CSS</a> (<a href="https://codersblock.com/blog/playing-with-infinity-in-css/" title="Accès à l’article original distant : Playing with Infinity in CSS">original</a>)</li>
<li><a href="/david/cache/2024/529fce4c2b7c378f07aead94e62d3923/" title="Accès à l’article dans le cache local : Some little ways I’m using CSS :has() in the real world">Some little ways I’m using CSS :has() in the real world</a> (<a href="https://piccalil.li/blog/some-little-ways-im-using-css-has-in-the-real-world/" title="Accès à l’article original distant : Some little ways I’m using CSS :has() in the real world">original</a>)</li>
<li><a href="/david/cache/2024/89dbef9daef24f311b6401cef62f5855/" title="Accès à l’article dans le cache local : Des Oloés">Des Oloés</a> (<a href="https://tw5.immateriel.fr/wiki/immateriel/b/YXGEDFB" title="Accès à l’article original distant : Des Oloés">original</a>)</li>
<li><a href="/david/cache/2024/1f40a33f9c57a16d420eb0868a129e96/" title="Accès à l’article dans le cache local : Notes from “An approach to computing and sustainability inspired from permaculture” by Devine LuLinvega">Notes from “An approach to computing and sustainability inspired from permaculture” by Devine LuLinvega</a> (<a href="https://blog.jim-nielsen.com/2024/notes-from-computing-sustainably/" title="Accès à l’article original distant : Notes from “An approach to computing and sustainability inspired from permaculture” by Devine LuLinvega">original</a>)</li>

Loading…
Cancel
Save