Skip to content

Commit 11bb528

Browse files
authored
feat: support css variables in shadow properties (#5032)
Closes #4939 Added css variables support with autocomplete in text-shadow and box-shadow properties. Extended our ast with new "shadow" value to properly represent variables in shadow properties. Though parser still cannot extract variables from value with variables and falls back to "unparsed" value. <img width="542" alt="Screenshot 2025-03-23 at 00 45 03" src="https://github.com/user-attachments/assets/2493d925-3199-4257-b9e8-dc69daa27324" />
1 parent f4318ed commit 11bb528

15 files changed

+506
-511
lines changed

apps/builder/app/builder/features/style-panel/shared/filter-content.tsx

+3-5
Original file line numberDiff line numberDiff line change
@@ -215,23 +215,21 @@ export const FilterSectionContent = ({
215215
) : undefined}
216216
</Flex>
217217

218-
{filterFunction === "drop-shadow" &&
219-
layer.type === "function" &&
220-
layer.args.type === "tuple" ? (
218+
{filterFunction === "drop-shadow" && layer.type === "function" && (
221219
<ShadowContent
222220
index={index}
223221
property="drop-shadow"
224222
layer={layer.args}
225223
propertyValue={toValue(layer.args)}
224+
hideCodeEditor={true}
226225
onEditLayer={(_, dropShadowLayers, options) => {
227226
handleComplete(
228227
`drop-shadow(${toValue(dropShadowLayers)})`,
229228
options
230229
);
231230
}}
232-
hideCodeEditor={true}
233231
/>
234-
) : undefined}
232+
)}
235233

236234
<Separator css={{ gridAutoColumns: "span 2" }} />
237235
<Flex

apps/builder/app/builder/features/style-panel/shared/shadow-content.tsx

+112-104
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,15 @@
1-
import { useEffect, useMemo, useState } from "react";
1+
import { useEffect, useState } from "react";
22
import {
33
toValue,
4+
type ShadowValue,
45
type InvalidValue,
56
type LayersValue,
6-
type RgbValue,
77
type StyleValue,
8-
type TupleValue,
9-
type UnitValue,
108
type VarValue,
9+
type CssProperty,
10+
type RgbValue,
1111
} from "@webstudio-is/css-engine";
12-
import {
13-
extractShadowProperties,
14-
keywordValues,
15-
propertySyntaxes,
16-
type ExtractedShadowProperties,
17-
} from "@webstudio-is/css-data";
12+
import { keywordValues, propertySyntaxes } from "@webstudio-is/css-data";
1813
import {
1914
Flex,
2015
Grid,
@@ -31,16 +26,18 @@ import {
3126
ShadowInsetIcon,
3227
ShadowNormalIcon,
3328
} from "@webstudio-is/icons";
34-
import type { IntermediateStyleValue } from "../shared/css-value-input";
35-
import { CssValueInputContainer } from "../shared/css-value-input";
36-
import type { StyleUpdateOptions } from "../shared/use-style-data";
29+
import { humanizeString } from "~/shared/string-utils";
30+
import { PropertyInlineLabel } from "../property-label";
31+
import type { IntermediateStyleValue } from "./css-value-input";
32+
import { CssValueInputContainer } from "./css-value-input";
33+
import type { StyleUpdateOptions } from "./use-style-data";
3734
import {
3835
CssFragmentEditor,
3936
CssFragmentEditorContent,
4037
parseCssFragment,
4138
} from "./css-fragment";
42-
import { PropertyInlineLabel } from "../property-label";
4339
import { ColorPicker } from "./color-picker";
40+
import { $availableColorVariables, $availableUnitVariables } from "./model";
4441

4542
/*
4643
When it comes to checking and validating individual CSS properties for the box-shadow,
@@ -80,18 +77,6 @@ type ShadowContentProps = {
8077
hideCodeEditor?: boolean;
8178
};
8279

83-
const convertValuesToTupple = (
84-
values: Partial<Record<keyof ExtractedShadowProperties, StyleValue>>
85-
): TupleValue => {
86-
return {
87-
type: "tuple",
88-
value: (Object.values(values) as Array<StyleValue>).filter(
89-
(item: StyleValue): item is UnitValue | RgbValue =>
90-
item !== null && item !== undefined
91-
),
92-
};
93-
};
94-
9580
const shadowPropertySyntaxes = {
9681
"box-shadow": {
9782
x: propertySyntaxes.boxShadowOffsetX,
@@ -115,6 +100,14 @@ const shadowPropertySyntaxes = {
115100
},
116101
} as const;
117102

103+
const defaultColor: RgbValue = {
104+
type: "rgb",
105+
r: 0,
106+
g: 0,
107+
b: 0,
108+
alpha: 1,
109+
};
110+
118111
export const ShadowContent = ({
119112
layer,
120113
computedLayer,
@@ -130,25 +123,25 @@ export const ShadowContent = ({
130123
useEffect(() => {
131124
setIntermediateValue({ type: "intermediate", value: propertyValue });
132125
}, [propertyValue]);
133-
const layerValues = useMemo<ExtractedShadowProperties>(() => {
134-
let value: TupleValue = { type: "tuple", value: [] };
135-
if (layer.type === "tuple") {
136-
value = layer;
137-
}
138-
if (layer.type === "var" && computedLayer?.type === "tuple") {
139-
value = computedLayer;
140-
}
141-
return extractShadowProperties(value);
142-
}, [layer, computedLayer]);
143-
144-
const { offsetX, offsetY, blur, spread, color, inset } = layerValues;
145-
const colorControlProp = color ?? {
146-
type: "rgb",
147-
r: 0,
148-
g: 0,
149-
b: 0,
150-
alpha: 1,
126+
let shadowValue: ShadowValue = {
127+
type: "shadow",
128+
position: "outset",
129+
offsetX: { type: "unit", value: 0, unit: "px" },
130+
offsetY: { type: "unit", value: 0, unit: "px" },
151131
};
132+
if (layer.type === "shadow") {
133+
shadowValue = layer;
134+
}
135+
if (layer.type === "var" && computedLayer?.type === "shadow") {
136+
shadowValue = computedLayer;
137+
}
138+
const computedShadow =
139+
computedLayer?.type === "shadow" ? computedLayer : shadowValue;
140+
141+
const parsedShadowProperty: CssProperty =
142+
property === "drop-shadow" ? "text-shadow" : property;
143+
144+
const disabledControls = layer.type === "var" || layer.type === "unparsed";
152145

153146
const handleChange = (value: string) => {
154147
setIntermediateValue({
@@ -161,17 +154,21 @@ export const ShadowContent = ({
161154
if (intermediateValue === undefined) {
162155
return;
163156
}
157+
// prevent reparsing value from string when not changed
158+
// because it may contain css variables
159+
// which cannot be safely parsed into ShadowValue
160+
if (intermediateValue.value === propertyValue) {
161+
return;
162+
}
164163
// dropShadow is a function under the filter property.
165164
// To parse the value correctly, we need to change the property to textShadow.
166165
// https://developer.mozilla.org/en-US/docs/Web/CSS/filter-function/drop-shadow#formal_syntax
167166
// https://developer.mozilla.org/en-US/docs/Web/CSS/text-shadow#formal_syntax
168167
// Both share a similar syntax but the property name is different.
169168
const parsed = parseCssFragment(intermediateValue.value, [
170-
property === "drop-shadow" ? "text-shadow" : property,
169+
parsedShadowProperty,
171170
]);
172-
const parsedValue = parsed.get(
173-
property === "drop-shadow" ? "text-shadow" : property
174-
);
171+
const parsedValue = parsed.get(parsedShadowProperty);
175172
if (parsedValue?.type === "layers" || parsedValue?.type === "var") {
176173
onEditLayer(index, parsedValue, { isEphemeral: false });
177174
return;
@@ -182,11 +179,11 @@ export const ShadowContent = ({
182179
});
183180
};
184181

185-
const handlePropertyChange = (
186-
params: Partial<Record<keyof ExtractedShadowProperties, StyleValue>>,
182+
const updateShadow = (
183+
params: Partial<ShadowValue>,
187184
options: StyleUpdateOptions = { isEphemeral: false }
188185
) => {
189-
const newLayer = convertValuesToTupple({ ...layerValues, ...params });
186+
const newLayer: ShadowValue = { ...shadowValue, ...params };
190187
setIntermediateValue({
191188
type: "intermediate",
192189
value: toValue(newLayer),
@@ -214,13 +211,16 @@ export const ShadowContent = ({
214211
// outline-offset is a fake property for validating box-shadow's offsetX.
215212
property="outline-offset"
216213
styleSource="local"
217-
disabled={layer.type === "var"}
218-
value={offsetX ?? { type: "unit", value: 0, unit: "px" }}
219-
onUpdate={(value, options) =>
220-
handlePropertyChange({ offsetX: value }, options)
221-
}
214+
disabled={disabledControls}
215+
getOptions={() => $availableUnitVariables.get()}
216+
value={shadowValue.offsetX}
217+
onUpdate={(value, options) => {
218+
if (value.type === "unit" || value.type === "var") {
219+
updateShadow({ offsetX: value }, options);
220+
}
221+
}}
222222
onDelete={(options) =>
223-
handlePropertyChange({ offsetX: offsetX ?? undefined }, options)
223+
updateShadow({ offsetX: shadowValue.offsetX }, options)
224224
}
225225
/>
226226
</Flex>
@@ -235,13 +235,16 @@ export const ShadowContent = ({
235235
// outline-offset is a fake property for validating box-shadow's offsetY.
236236
property="outline-offset"
237237
styleSource="local"
238-
disabled={layer.type === "var"}
239-
value={offsetY ?? { type: "unit", value: 0, unit: "px" }}
240-
onUpdate={(value, options) =>
241-
handlePropertyChange({ offsetY: value }, options)
242-
}
238+
disabled={disabledControls}
239+
getOptions={() => $availableUnitVariables.get()}
240+
value={shadowValue.offsetY}
241+
onUpdate={(value, options) => {
242+
if (value.type === "unit" || value.type === "var") {
243+
updateShadow({ offsetY: value }, options);
244+
}
245+
}}
243246
onDelete={(options) =>
244-
handlePropertyChange({ offsetY: offsetY ?? undefined }, options)
247+
updateShadow({ offsetY: shadowValue.offsetY }, options)
245248
}
246249
/>
247250
</Flex>
@@ -256,13 +259,16 @@ export const ShadowContent = ({
256259
// border-top-width is a fake property for validating box-shadow's blur.
257260
property="border-top-width"
258261
styleSource="local"
259-
disabled={layer.type === "var"}
260-
value={blur ?? { type: "unit", value: 0, unit: "px" }}
261-
onUpdate={(value, options) =>
262-
handlePropertyChange({ blur: value }, options)
263-
}
262+
disabled={disabledControls}
263+
getOptions={() => $availableUnitVariables.get()}
264+
value={shadowValue.blur ?? { type: "unit", value: 0, unit: "px" }}
265+
onUpdate={(value, options) => {
266+
if (value.type === "unit" || value.type === "var") {
267+
updateShadow({ blur: value }, options);
268+
}
269+
}}
264270
onDelete={(options) =>
265-
handlePropertyChange({ blur: blur ?? undefined }, options)
271+
updateShadow({ blur: shadowValue.blur }, options)
266272
}
267273
/>
268274
</Flex>
@@ -278,13 +284,18 @@ export const ShadowContent = ({
278284
// outline-offset is a fake property for validating box-shadow's spread.
279285
property="outline-offset"
280286
styleSource="local"
281-
disabled={layer.type === "var"}
282-
value={spread ?? { type: "unit", value: 0, unit: "px" }}
283-
onUpdate={(value, options) =>
284-
handlePropertyChange({ spread: value }, options)
287+
disabled={disabledControls}
288+
getOptions={() => $availableUnitVariables.get()}
289+
value={
290+
shadowValue.spread ?? { type: "unit", value: 0, unit: "px" }
285291
}
292+
onUpdate={(value, options) => {
293+
if (value.type === "unit" || value.type === "var") {
294+
updateShadow({ spread: value }, options);
295+
}
296+
}}
286297
onDelete={(options) =>
287-
handlePropertyChange({ spread: spread ?? undefined }, options)
298+
updateShadow({ spread: shadowValue.spread }, options)
288299
}
289300
/>
290301
</Flex>
@@ -305,48 +316,45 @@ export const ShadowContent = ({
305316
/>
306317
<ColorPicker
307318
property="color"
308-
disabled={layer.type === "var"}
309-
value={colorControlProp}
310-
currentColor={colorControlProp}
311-
getOptions={() =>
312-
keywordValues["color"].map((value) => ({
313-
type: "keyword",
314-
value,
315-
}))
316-
}
317-
onChange={(styleValue) =>
318-
handlePropertyChange({ color: styleValue }, { isEphemeral: true })
319-
}
320-
onChangeComplete={(styleValue) =>
321-
handlePropertyChange({ color: styleValue })
322-
}
323-
onAbort={() => handlePropertyChange({ color: colorControlProp })}
324-
onReset={() => {
325-
handlePropertyChange({ color: undefined });
319+
disabled={disabledControls}
320+
value={shadowValue.color ?? defaultColor}
321+
currentColor={computedShadow?.color ?? defaultColor}
322+
getOptions={() => [
323+
...(keywordValues.color ?? []).map((item) => ({
324+
type: "keyword" as const,
325+
value: item,
326+
})),
327+
...$availableColorVariables.get(),
328+
]}
329+
onChange={(value) => {
330+
if (value.type === "rgb" || value.type === "var") {
331+
updateShadow({ color: value }, { isEphemeral: true });
332+
}
326333
}}
334+
onChangeComplete={(value) => {
335+
if (value.type === "rgb" || value.type === "var") {
336+
updateShadow({ color: value });
337+
}
338+
}}
339+
onAbort={() => updateShadow({ color: shadowValue.color })}
340+
onReset={() => updateShadow({ color: undefined })}
327341
/>
328342
</Flex>
329343

330344
{property === "box-shadow" ? (
331345
<Flex direction="column" gap="1">
332346
<PropertyInlineLabel
333-
label="Inset"
347+
label={humanizeString(shadowValue.position)}
334348
description={shadowPropertySyntaxes["box-shadow"].position}
335349
/>
336350
<ToggleGroup
337351
type="single"
338-
disabled={layer.type === "var"}
339-
value={inset?.value ?? "outset"}
352+
disabled={disabledControls}
353+
value={shadowValue.position}
340354
defaultValue="inset"
341-
onValueChange={(value) => {
342-
if (value === "inset") {
343-
handlePropertyChange({
344-
inset: { type: "keyword", value: "inset" },
345-
});
346-
} else {
347-
handlePropertyChange({ inset: undefined });
348-
}
349-
}}
355+
onValueChange={(value) =>
356+
updateShadow({ position: value as ShadowValue["position"] })
357+
}
350358
>
351359
<Tooltip content="Outset">
352360
<ToggleGroupButton value="outset">

apps/builder/app/shared/instance-utils.ts

+14
Original file line numberDiff line numberDiff line change
@@ -627,6 +627,20 @@ const traverseStyleValue = (
627627
}
628628
return;
629629
}
630+
if (value.type === "shadow") {
631+
traverseStyleValue(value.offsetX, callback);
632+
traverseStyleValue(value.offsetY, callback);
633+
if (value.blur) {
634+
traverseStyleValue(value.blur, callback);
635+
}
636+
if (value.spread) {
637+
traverseStyleValue(value.spread, callback);
638+
}
639+
if (value.color) {
640+
traverseStyleValue(value.color, callback);
641+
}
642+
return;
643+
}
630644
value satisfies never;
631645
};
632646

0 commit comments

Comments
 (0)