1
- import { useEffect , useMemo , useState } from "react" ;
1
+ import { useEffect , useState } from "react" ;
2
2
import {
3
3
toValue ,
4
+ type ShadowValue ,
4
5
type InvalidValue ,
5
6
type LayersValue ,
6
- type RgbValue ,
7
7
type StyleValue ,
8
- type TupleValue ,
9
- type UnitValue ,
10
8
type VarValue ,
9
+ type CssProperty ,
10
+ type RgbValue ,
11
11
} 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" ;
18
13
import {
19
14
Flex ,
20
15
Grid ,
@@ -31,16 +26,18 @@ import {
31
26
ShadowInsetIcon ,
32
27
ShadowNormalIcon ,
33
28
} 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" ;
37
34
import {
38
35
CssFragmentEditor ,
39
36
CssFragmentEditorContent ,
40
37
parseCssFragment ,
41
38
} from "./css-fragment" ;
42
- import { PropertyInlineLabel } from "../property-label" ;
43
39
import { ColorPicker } from "./color-picker" ;
40
+ import { $availableColorVariables , $availableUnitVariables } from "./model" ;
44
41
45
42
/*
46
43
When it comes to checking and validating individual CSS properties for the box-shadow,
@@ -80,18 +77,6 @@ type ShadowContentProps = {
80
77
hideCodeEditor ?: boolean ;
81
78
} ;
82
79
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
-
95
80
const shadowPropertySyntaxes = {
96
81
"box-shadow" : {
97
82
x : propertySyntaxes . boxShadowOffsetX ,
@@ -115,6 +100,14 @@ const shadowPropertySyntaxes = {
115
100
} ,
116
101
} as const ;
117
102
103
+ const defaultColor : RgbValue = {
104
+ type : "rgb" ,
105
+ r : 0 ,
106
+ g : 0 ,
107
+ b : 0 ,
108
+ alpha : 1 ,
109
+ } ;
110
+
118
111
export const ShadowContent = ( {
119
112
layer,
120
113
computedLayer,
@@ -130,25 +123,25 @@ export const ShadowContent = ({
130
123
useEffect ( ( ) => {
131
124
setIntermediateValue ( { type : "intermediate" , value : propertyValue } ) ;
132
125
} , [ 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" } ,
151
131
} ;
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" ;
152
145
153
146
const handleChange = ( value : string ) => {
154
147
setIntermediateValue ( {
@@ -161,17 +154,21 @@ export const ShadowContent = ({
161
154
if ( intermediateValue === undefined ) {
162
155
return ;
163
156
}
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
+ }
164
163
// dropShadow is a function under the filter property.
165
164
// To parse the value correctly, we need to change the property to textShadow.
166
165
// https://developer.mozilla.org/en-US/docs/Web/CSS/filter-function/drop-shadow#formal_syntax
167
166
// https://developer.mozilla.org/en-US/docs/Web/CSS/text-shadow#formal_syntax
168
167
// Both share a similar syntax but the property name is different.
169
168
const parsed = parseCssFragment ( intermediateValue . value , [
170
- property === "drop-shadow" ? "text-shadow" : property ,
169
+ parsedShadowProperty ,
171
170
] ) ;
172
- const parsedValue = parsed . get (
173
- property === "drop-shadow" ? "text-shadow" : property
174
- ) ;
171
+ const parsedValue = parsed . get ( parsedShadowProperty ) ;
175
172
if ( parsedValue ?. type === "layers" || parsedValue ?. type === "var" ) {
176
173
onEditLayer ( index , parsedValue , { isEphemeral : false } ) ;
177
174
return ;
@@ -182,11 +179,11 @@ export const ShadowContent = ({
182
179
} ) ;
183
180
} ;
184
181
185
- const handlePropertyChange = (
186
- params : Partial < Record < keyof ExtractedShadowProperties , StyleValue > > ,
182
+ const updateShadow = (
183
+ params : Partial < ShadowValue > ,
187
184
options : StyleUpdateOptions = { isEphemeral : false }
188
185
) => {
189
- const newLayer = convertValuesToTupple ( { ...layerValues , ...params } ) ;
186
+ const newLayer : ShadowValue = { ...shadowValue , ...params } ;
190
187
setIntermediateValue ( {
191
188
type : "intermediate" ,
192
189
value : toValue ( newLayer ) ,
@@ -214,13 +211,16 @@ export const ShadowContent = ({
214
211
// outline-offset is a fake property for validating box-shadow's offsetX.
215
212
property = "outline-offset"
216
213
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
+ } }
222
222
onDelete = { ( options ) =>
223
- handlePropertyChange ( { offsetX : offsetX ?? undefined } , options )
223
+ updateShadow ( { offsetX : shadowValue . offsetX } , options )
224
224
}
225
225
/>
226
226
</ Flex >
@@ -235,13 +235,16 @@ export const ShadowContent = ({
235
235
// outline-offset is a fake property for validating box-shadow's offsetY.
236
236
property = "outline-offset"
237
237
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
+ } }
243
246
onDelete = { ( options ) =>
244
- handlePropertyChange ( { offsetY : offsetY ?? undefined } , options )
247
+ updateShadow ( { offsetY : shadowValue . offsetY } , options )
245
248
}
246
249
/>
247
250
</ Flex >
@@ -256,13 +259,16 @@ export const ShadowContent = ({
256
259
// border-top-width is a fake property for validating box-shadow's blur.
257
260
property = "border-top-width"
258
261
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
+ } }
264
270
onDelete = { ( options ) =>
265
- handlePropertyChange ( { blur : blur ?? undefined } , options )
271
+ updateShadow ( { blur : shadowValue . blur } , options )
266
272
}
267
273
/>
268
274
</ Flex >
@@ -278,13 +284,18 @@ export const ShadowContent = ({
278
284
// outline-offset is a fake property for validating box-shadow's spread.
279
285
property = "outline-offset"
280
286
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" }
285
291
}
292
+ onUpdate = { ( value , options ) => {
293
+ if ( value . type === "unit" || value . type === "var" ) {
294
+ updateShadow ( { spread : value } , options ) ;
295
+ }
296
+ } }
286
297
onDelete = { ( options ) =>
287
- handlePropertyChange ( { spread : spread ?? undefined } , options )
298
+ updateShadow ( { spread : shadowValue . spread } , options )
288
299
}
289
300
/>
290
301
</ Flex >
@@ -305,48 +316,45 @@ export const ShadowContent = ({
305
316
/>
306
317
< ColorPicker
307
318
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
+ }
326
333
} }
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 } ) }
327
341
/>
328
342
</ Flex >
329
343
330
344
{ property === "box-shadow" ? (
331
345
< Flex direction = "column" gap = "1" >
332
346
< PropertyInlineLabel
333
- label = "Inset"
347
+ label = { humanizeString ( shadowValue . position ) }
334
348
description = { shadowPropertySyntaxes [ "box-shadow" ] . position }
335
349
/>
336
350
< ToggleGroup
337
351
type = "single"
338
- disabled = { layer . type === "var" }
339
- value = { inset ?. value ?? "outset" }
352
+ disabled = { disabledControls }
353
+ value = { shadowValue . position }
340
354
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
+ }
350
358
>
351
359
< Tooltip content = "Outset" >
352
360
< ToggleGroupButton value = "outset" >
0 commit comments