Эта статья — перевод оригинальной статьи "Type safe CSS design systems with @property".
Также я веду телеграм канал “Frontend по-флотски”, где рассказываю про интересные вещи из мира разработки интерфейсов.
Вступление
Типы CSS - это достойное вложение в безопасность типов при работе с внешним интерфейсом. Мы все еще ожидаем кроссбраузерности, но мы к этому придем ? .
Если вы никогда не видели типизированную CSS-переменную с @property, то вот пример:
@property --focal-size {
syntax: '<length-percentage>';
initial-value: 100%;
inherits: false;
}Я использовал её, чтобы анимировать изображение с градиентной маской. Довольно мило.
Вот предварительный обзор того, что может сделать безопасность типов CSS, и того, что я буду объяснять:
Основы безопасности CSS типов
При изучении Rust или TypeScript лучше всего начать с примитивов типов. В CSS их несколько:

Больше типов на MDN и полный список грамматик и типов на csswg.org/indexes/#types.
Еще одно определение переменной:
@property --hue {
syntax: '<angle>';
initial-value: .5turn;
inherits: false;
}Используйте её так же, как и var(--hue), и она будет равна .5turn. Но попробуйте установить его в значение, не соответствующее её типу? Не получится, значение по-прежнему будет равно .5turn. Переменная не позволит присвоить себе значение, не соответствующее её типу, всегда возвращаясь к последнему подходящему значению.
.card {
--hue: 90deg; /* ✅ */
--hue: #f00; /* ❌ */
background: oklch(98% .01 var(--hue));
/* background will always resolve ?? */
/* --hue resolves 90deg *.
}Это безопасность CSS типов. Она не приводит к краху страницы, не блокирует поток и, к сожалению, не сообщает в консоли о том, что была попытка установить для свойства --hue значение <color>, а не <angle>. Но я думаю, что более качественный инструментарий мог бы помочь ?.
Уровень 2
Пока что я создал переменную как <angle> и использовал её в свойстве background. Никакой вложенности свойств.
Перейдем на более глубокий уровень, сделав переменную использующую другую переменную. Здесь --_bg - это <any> (потому что на данный момент это не типизированная переменная), с вложенной переменной --hue:
.card {
--_bg: oklch(98% .01 var(--hue));
background: var(--_bg);
@media (prefers-color-scheme: dark) {
--_bg: oklch(15% .1 var(--hue));
}
}Вы можете углубиться на много уровней, но не бесконечно. И можно типизировать некоторые или все переменные. Далее мы создадим несколько типизированных переменных с двухуровневой глубиной.
Актуальность систем проектирования
Давайте сделаем типизированный стартер адаптивной цветовой схемы для светлых и темных тем!
Во-первых, безопасный для типа значение hue. Я сделаю элемент <input type=text>, который будет записывать в это значение все, что мы в него введем. Поскольку он безопасен для типов, мы увидим, как другие пользовательские свойства, зависящие от него, не сломаются, если значение параметра --hue будет установлено в "poots" или что-то в этом роде.
@property --hue {
syntax: '<angle>';
initial-value: 5rad;
inherits: true;
}Для краткости я буду задавать только поверхностные слои адаптивной цветовой схемы, но это даст понимание процесса создания системы дизайна.
Вот 3 слоя, один из которых будет фоном страницы --surface, и два других, которые будут либо поверх bg страницы, либо под ним. Их начальное значение не вызывает восторга, но мы дойдем до этого в следующей части.
@property --surface {
syntax: '<color>';
initial-value: #333;
inherits: true;
}
@property --surface-over {
syntax: '<color>';
initial-value: #444;
inherits: true;
}
@property --surface-under {
syntax: '<color>';
initial-value: #222;
inherits: true;
}Важным здесь является то, что они относятся к цветовому типу.
Теперь мы можем присвоить цветам слоёв более понятные значения. При желании можно использовать @media (prefers-color-scheme), но здесь, поскольку я хотел показать светлое и темное с помощью переключателя, я использую :has():
@layer demo.theme {
html:has(#light:checked) {
color-scheme: light;
--surface: oklch(90% .05 var(--hue));
--surface-over: oklch(99% .02 var(--hue));
--surface-under: oklch(85% .075 var(--hue));
}
html:has(#dark:checked) {
color-scheme: dark;
--surface: oklch(20% .1 var(--hue));
--surface-over: oklch(30% .1 var(--hue));
--surface-under: oklch(15% .1 var(--hue));
}
}Вот, собственно, и вся настройка и оркестровка типизированных переменных. Остается только использовать их. Загляните в Codepen, чтобы увидеть все возможные способы их использования для создания адаптивной цветовой схемы: тени, фона и многое другое!
Последняя часть
Попробуйте вводить всякую ерунду, в текстовое поле "Theme tint" в CodePen демо-версии. Ни одна из цветовых систем не даст сбоя из-за опечатки или присвоенного значения, не соответствующего типу. Браузер точно знает, как сделать обратный ход и обработать ошибки.
На @property можно построить очень надежную и большую систему. Те же типы безопасности типов при разработке, что и в Typescript, но типы действительно передаются браузеру и соблюдаются. Rad.
Firefox уже почти закончил свою реализацию, что сделает @property кроссбраузерно стабильной ?.
Информацию о поддержке можно узнать на сайте caniuse.
Системы проектирования скоро станут намного умнее и стабильнее.