vizia_core/modifiers/
style.rs

1use vizia_style::{ColorStop, CornerRadius, Rect};
2
3use super::internal;
4use crate::prelude::*;
5
6/// Modifiers for changing the style properties of a view.
7pub trait StyleModifiers: internal::Modifiable {
8    // Selectors
9
10    /// Sets the ID name of the view.
11    ///
12    /// A view can have only one ID name and it must be unique.
13    /// The ID name can be referenced by a CSS selector.
14    /// # Example
15    /// ```
16    /// # use vizia_core::prelude::*;
17    /// # let cx = &mut Context::default();
18    /// Element::new(cx).id("foo");
19    /// ```
20    /// css
21    /// ```css
22    /// #foo {
23    ///     background-color: red;
24    /// }
25    ///```
26    fn id(mut self, id: impl Into<String>) -> Self {
27        // TODO - What should happen if the id already exists?
28        let id = id.into();
29        let entity = self.entity();
30        self.context().style.ids.insert(entity, id.clone());
31        self.context().needs_restyle(entity);
32
33        self.context().entity_identifiers.insert(id, entity);
34
35        self
36    }
37
38    /// Adds a class name to the view.
39    ///
40    /// A view can have multiple classes.
41    /// The class name can be referenced by a CSS selector.
42    /// # Example
43    /// ```
44    /// # use vizia_core::prelude::*;
45    /// # let cx = &mut Context::default();
46    /// Element::new(cx).class("foo");
47    /// ```
48    /// css
49    /// ```css
50    /// .foo {
51    ///     background-color: red;
52    /// }
53    ///```
54    fn class(mut self, name: &str) -> Self {
55        let entity = self.entity();
56        if let Some(class_list) = self.context().style.classes.get_mut(entity) {
57            class_list.insert(name.to_string());
58        }
59
60        self.context().needs_restyle(entity);
61
62        self
63    }
64
65    /// Sets whether a view should have the given class name.
66    fn toggle_class(mut self, name: &str, applied: impl Res<bool>) -> Self {
67        let name = name.to_owned();
68        let entity = self.entity();
69        let current = self.current();
70        self.context().with_current(current, |cx| {
71            applied.set_or_bind(cx, entity, move |cx, applied| {
72                let applied = applied.get(cx);
73                if let Some(class_list) = cx.style.classes.get_mut(entity) {
74                    if applied {
75                        class_list.insert(name.clone());
76                    } else {
77                        class_list.remove(&name);
78                    }
79                }
80
81                cx.needs_restyle(entity);
82            });
83        });
84
85        self
86    }
87
88    // PseudoClassFlags
89    // TODO: Should these have their own modifiers trait?
90
91    /// Sets the checked state of the view.
92    fn checked<U: Into<bool>>(mut self, state: impl Res<U>) -> Self {
93        let entity = self.entity();
94        let current = self.current();
95        // Setting a checked state should make it checkable
96        if let Some(abilities) = self.context().style.abilities.get_mut(entity) {
97            abilities.set(Abilities::CHECKABLE, true);
98        }
99
100        self.context().with_current(current, move |cx| {
101            state.set_or_bind(cx, entity, move |cx, val| {
102                let val = val.get(cx).into();
103                if let Some(pseudo_classes) = cx.style.pseudo_classes.get_mut(entity) {
104                    pseudo_classes.set(PseudoClassFlags::CHECKED, val);
105                }
106                cx.needs_restyle(entity);
107            });
108        });
109
110        self
111    }
112
113    /// Sets the focused state of the view.
114    ///
115    /// Since only one view can have keyboard focus at a time, subsequent calls to this
116    /// function on other views will cause those views to gain focus and this view to lose it.
117    fn focused<U: Into<bool>>(mut self, state: impl Res<U>) -> Self {
118        let entity = self.entity();
119        let current = self.current();
120
121        self.context().with_current(current, |cx| {
122            state.set_or_bind(cx, entity, |cx, val| {
123                let val = val.get(cx).into();
124
125                if val {
126                    cx.focus();
127                    // cx.focus_with_visibility(true);
128                }
129
130                cx.needs_restyle(cx.current);
131            });
132        });
133
134        self
135    }
136
137    /// Sets the focused state of the view as well as the focus visibility.
138    fn focused_with_visibility<U: Into<bool>>(
139        mut self,
140        focus: impl Res<U> + Copy + 'static,
141        visibility: impl Res<U> + Copy + 'static,
142    ) -> Self {
143        let entity = self.entity();
144        let current = self.current();
145
146        self.context().with_current(current, move |cx| {
147            focus.set_or_bind(cx, entity, move |cx, f| {
148                visibility.set_or_bind(cx, entity, move |cx, v| {
149                    let focus = f.get(cx).into();
150                    let visibility = v.get(cx).into();
151                    if focus {
152                        //cx.focus();
153                        cx.focus_with_visibility(visibility);
154                        cx.needs_restyle(cx.current);
155                    }
156                });
157            });
158        });
159
160        self
161    }
162
163    /// Sets whether the view should be in a read-only state.
164    fn read_only<U: Into<bool>>(mut self, state: impl Res<U>) -> Self {
165        let entity = self.entity();
166        let current = self.current();
167        self.context().with_current(current, |cx| {
168            state.set_or_bind(cx, entity, move |cx, val| {
169                let val = val.get(cx).into();
170                if let Some(pseudo_classes) = cx.style.pseudo_classes.get_mut(cx.current) {
171                    pseudo_classes.set(PseudoClassFlags::READ_ONLY, val);
172                }
173
174                cx.needs_restyle(cx.current);
175            });
176        });
177
178        self
179    }
180
181    /// Sets whether the view should be in a read-write state.
182    fn read_write<U: Into<bool>>(mut self, state: impl Res<U>) -> Self {
183        let entity = self.entity();
184        let current = self.current();
185        self.context().with_current(current, |cx| {
186            state.set_or_bind(cx, entity, move |cx, val| {
187                let val = val.get(cx).into();
188                if let Some(pseudo_classes) = cx.style.pseudo_classes.get_mut(cx.current) {
189                    pseudo_classes.set(PseudoClassFlags::READ_WRITE, val);
190                }
191
192                cx.needs_restyle(cx.current);
193            });
194        });
195
196        self
197    }
198
199    /// Sets whether the view is showing a placeholder.
200    fn placeholder_shown<U: Into<bool>>(mut self, state: impl Res<U>) -> Self {
201        let entity = self.entity();
202        let current = self.current();
203        self.context().with_current(current, |cx| {
204            state.set_or_bind(cx, entity, move |cx, val| {
205                let val = val.get(cx).into();
206                if let Some(pseudo_classes) = cx.style.pseudo_classes.get_mut(cx.current) {
207                    pseudo_classes.set(PseudoClassFlags::PLACEHOLDER_SHOWN, val);
208                }
209
210                cx.needs_restyle(cx.current);
211            });
212        });
213
214        self
215    }
216
217    modifier!(
218        /// Sets the view to be disabled.
219        ///
220        /// This property is inherited by the descendants of the view.
221        disabled,
222        bool,
223        SystemFlags::RESTYLE
224    );
225
226    modifier!(
227        /// Sets whether the view should be positioned and rendered.
228        ///
229        /// A display value of `Display::None` causes the view to be ignored by both layout and rendering.
230        display,
231        Display,
232        SystemFlags::RELAYOUT | SystemFlags::REDRAW
233    );
234
235    modifier!(
236        /// Sets whether the view should be rendered.
237        ///
238        /// The layout system will still compute the size and position of an invisible (hidden) view.
239        visibility,
240        Visibility,
241        SystemFlags::REDRAW
242    );
243
244    modifier!(
245        /// Sets the opacity of the view.
246        ///
247        /// Exects a value between 0.0 (transparent) and 1.0 (opaque).
248        opacity,
249        Opacity,
250        SystemFlags::REDRAW
251    );
252
253    /// Sets the z-index of the view.
254    ///
255    /// Views with a higher z-index will be rendered on top of those with a lower z-order.
256    /// Views with the same z-index are rendered in tree order.
257    fn z_index<U: Into<i32>>(mut self, value: impl Res<U>) -> Self {
258        let entity = self.entity();
259        // value.set_or_bind(self.context(), entity, |cx, v| {
260        let cx = self.context();
261        let value = value.get(cx).into();
262        cx.style.z_index.insert(entity, value);
263        cx.needs_redraw(entity);
264        // });
265
266        self
267    }
268
269    /// Sets the clip path for the the view.
270    fn clip_path<U: Into<ClipPath>>(mut self, value: impl Res<U>) -> Self {
271        let entity = self.entity();
272        let current = self.current();
273        self.context().with_current(current, |cx| {
274            value.set_or_bind(cx, entity, move |cx, v| {
275                let value = v.get(cx).into();
276                cx.style.clip_path.insert(cx.current, value);
277
278                cx.needs_redraw(entity);
279            });
280        });
281
282        self
283    }
284
285    /// Sets the overflow behavior of the view in the horizontal and vertical directions simultaneously.
286    fn overflow<U: Into<Overflow>>(mut self, value: impl Res<U>) -> Self {
287        let entity = self.entity();
288        let current = self.current();
289        self.context().with_current(current, |cx| {
290            value.set_or_bind(cx, entity, move |cx, v| {
291                let value = v.get(cx).into();
292                cx.style.overflowx.insert(cx.current, value);
293                cx.style.overflowy.insert(cx.current, value);
294
295                cx.needs_redraw(entity);
296            });
297        });
298
299        self
300    }
301
302    modifier!(
303        /// Sets the overflow behavior of the view in the horizontal direction.
304        ///
305        /// The overflow behavior determines whether child views can render outside the bounds of their parent.
306        overflowx,
307        Overflow,
308        SystemFlags::REDRAW
309    );
310
311    modifier!(
312        /// Sets the overflow behavior of the view in the vertical direction.
313        ///
314        /// The overflow behavior determines whether child views can render outside the bounds of their parent.
315        overflowy,
316        Overflow,
317        SystemFlags::REDRAW
318    );
319
320    /// Sets the backdrop filter for the view.
321    fn backdrop_filter<U: Into<Filter>>(mut self, value: impl Res<U>) -> Self {
322        let entity = self.entity();
323        let current = self.current();
324        self.context().with_current(current, |cx| {
325            value.set_or_bind(cx, entity, move |cx, v| {
326                let value = v.get(cx).into();
327                cx.style.backdrop_filter.insert(cx.current, value);
328
329                cx.needs_redraw(entity);
330            });
331        });
332
333        self
334    }
335
336    /// Add a shadow to the view.
337    fn shadow<U: Into<Shadow>>(mut self, value: impl Res<U>) -> Self {
338        let entity = self.entity();
339        let current = self.current();
340        self.context().with_current(current, |cx| {
341            value.set_or_bind(cx, entity, move |cx, v| {
342                let value = v.get(cx).into();
343                if let Some(shadows) = cx.style.shadow.get_inline_mut(cx.current) {
344                    shadows.push(value);
345                } else {
346                    cx.style.shadow.insert(cx.current, vec![value]);
347                }
348
349                cx.needs_redraw(entity);
350            });
351        });
352
353        self
354    }
355
356    /// Set the shadows of the view.
357    fn shadows<U: Into<Vec<Shadow>>>(mut self, value: impl Res<U>) -> Self {
358        let entity = self.entity();
359        let current = self.current();
360        self.context().with_current(current, |cx| {
361            value.set_or_bind(cx, entity, move |cx, v| {
362                let value = v.get(cx).into();
363
364                cx.style.shadow.insert(cx.current, value);
365
366                cx.needs_redraw(entity);
367            });
368        });
369
370        self
371    }
372
373    /// Set the background gradient of the view.
374    fn background_gradient<U: Into<Gradient>>(mut self, value: impl Res<U>) -> Self {
375        let entity = self.entity();
376        let current = self.current();
377        self.context().with_current(current, |cx| {
378            value.set_or_bind(cx, entity, move |cx, v| {
379                let value = v.get(cx).into();
380                if let Some(background_images) =
381                    cx.style.background_image.get_inline_mut(cx.current)
382                {
383                    background_images.push(ImageOrGradient::Gradient(value));
384                } else {
385                    cx.style
386                        .background_image
387                        .insert(cx.current, vec![ImageOrGradient::Gradient(value)]);
388                }
389
390                cx.needs_redraw(entity);
391            });
392        });
393
394        self
395    }
396
397    // Background Properties
398    modifier!(
399        /// Sets the background color of the view.
400        background_color,
401        Color,
402        SystemFlags::REDRAW
403    );
404
405    /// Set the background image of the view.
406    fn background_image<'i, U: Into<BackgroundImage<'i>>>(mut self, value: impl Res<U>) -> Self {
407        let entity = self.entity();
408        let current = self.current();
409        self.context().with_current(current, |cx| {
410            value.set_or_bind(cx, entity, move |cx, val| {
411                let image = val.get(cx).into();
412                let image = match image {
413                    BackgroundImage::Gradient(gradient) => {
414                        Some(ImageOrGradient::Gradient(*gradient))
415                    }
416                    BackgroundImage::Url(url) => Some(ImageOrGradient::Image(url.url.to_string())),
417                    _ => None,
418                };
419
420                if let Some(image) = image {
421                    cx.style.background_image.insert(cx.current, vec![image]);
422                }
423
424                cx.needs_redraw(entity);
425            });
426        });
427
428        self
429    }
430
431    // Border Properties
432    fn border_width<U: Into<LengthOrPercentage>>(mut self, value: impl Res<U>) -> Self {
433        let entity = self.entity();
434        let current = self.current();
435        value.set_or_bind(self.context(), current, move |cx, v| {
436            cx.style.border_width.insert(entity, v.get(cx).into());
437            cx.cache.path.remove(entity);
438            cx.style.system_flags |= SystemFlags::RELAYOUT | SystemFlags::REDRAW;
439            cx.set_system_flags(entity, SystemFlags::RELAYOUT | SystemFlags::REDRAW);
440        });
441
442        self
443    }
444
445    modifier!(
446        /// Sets the border color of the view.
447        border_color,
448        Color,
449        SystemFlags::REDRAW
450    );
451
452    modifier!(
453        /// Sets the border color of the view.
454        border_style,
455        BorderStyleKeyword,
456        SystemFlags::REDRAW
457    );
458
459    modifier!(
460        /// Sets the corner radius for the top-left corner of the view.
461        corner_top_left_radius,
462        LengthOrPercentage,
463        SystemFlags::REDRAW
464    );
465
466    modifier!(
467        /// Sets the corner radius for the top-right corner of the view.
468        corner_top_right_radius,
469        LengthOrPercentage,
470        SystemFlags::REDRAW
471    );
472
473    modifier!(
474        /// Sets the corner radius for the bottom-left corner of the view.
475        corner_bottom_left_radius,
476        LengthOrPercentage,
477        SystemFlags::REDRAW
478    );
479
480    modifier!(
481        /// Sets the corner radius for the bottom-right corner of the view.
482        corner_bottom_right_radius,
483        LengthOrPercentage,
484        SystemFlags::REDRAW
485    );
486
487    /// Sets the corner radius for all four corners of the view.
488    fn corner_radius<U: std::fmt::Debug + Into<CornerRadius>>(
489        mut self,
490        value: impl Res<U>,
491    ) -> Self {
492        let entity = self.entity();
493        let current = self.current();
494        self.context().with_current(current, |cx| {
495            value.set_or_bind(cx, entity, move |cx, v| {
496                let value = v.get(cx).into();
497                cx.style.corner_top_left_radius.insert(cx.current, value.top_left);
498                cx.style.corner_top_right_radius.insert(cx.current, value.top_right);
499                cx.style.corner_bottom_left_radius.insert(cx.current, value.bottom_left);
500                cx.style.corner_bottom_right_radius.insert(cx.current, value.bottom_right);
501
502                cx.needs_redraw(entity);
503            });
504        });
505
506        self
507    }
508
509    modifier!(
510        /// Sets the corner corner shape for the top-left corner of the view.
511        corner_top_left_shape,
512        CornerShape,
513        SystemFlags::REDRAW
514    );
515
516    modifier!(
517        /// Sets the corner corner shape for the top-right corner of the view.
518        corner_top_right_shape,
519        CornerShape,
520        SystemFlags::REDRAW
521    );
522
523    modifier!(
524        /// Sets the corner corner shape for the bottom-left corner of the view.
525        corner_bottom_left_shape,
526        CornerShape,
527        SystemFlags::REDRAW
528    );
529
530    modifier!(
531        /// Sets the corner corner shape for the bottom-right corner of the view.
532        corner_bottom_right_shape,
533        CornerShape,
534        SystemFlags::REDRAW
535    );
536
537    /// Sets the corner shape for all four corners of the view.
538    fn corner_shape<U: std::fmt::Debug + Into<Rect<CornerShape>>>(
539        mut self,
540        value: impl Res<U>,
541    ) -> Self {
542        let entity = self.entity();
543        let current = self.current();
544        self.context().with_current(current, |cx| {
545            value.set_or_bind(cx, entity, move |cx, v| {
546                let value = v.get(cx).into();
547                cx.style.corner_top_left_shape.insert(cx.current, value.0);
548                cx.style.corner_top_right_shape.insert(cx.current, value.1);
549                cx.style.corner_bottom_right_shape.insert(cx.current, value.2);
550                cx.style.corner_bottom_left_shape.insert(cx.current, value.3);
551
552                cx.needs_redraw(entity);
553            });
554        });
555
556        self
557    }
558
559    modifier!(
560        /// Sets the corner smoothing for the top-left corner of the view.
561        corner_top_left_smoothing,
562        f32,
563        SystemFlags::REDRAW
564    );
565
566    modifier!(
567        /// Sets the corner smoothing for the top-right corner of the view.
568        corner_top_right_smoothing,
569        f32,
570        SystemFlags::REDRAW
571    );
572
573    modifier!(
574        /// Sets the corner smoothing for the bottom-left corner of the view.
575        corner_bottom_left_smoothing,
576        f32,
577        SystemFlags::REDRAW
578    );
579
580    modifier!(
581        /// Sets the corner smoothing for the bottom-right corner of the view.
582        corner_bottom_right_smoothing,
583        f32,
584        SystemFlags::REDRAW
585    );
586
587    /// Sets the corner smoothing for all four corners of the view.
588    fn corner_smoothing<U: std::fmt::Debug + Into<Rect<f32>>>(
589        mut self,
590        value: impl Res<U>,
591    ) -> Self {
592        let entity = self.entity();
593        let current = self.current();
594        self.context().with_current(current, |cx| {
595            value.set_or_bind(cx, entity, move |cx, v| {
596                let value = v.get(cx).into();
597                cx.style.corner_top_left_smoothing.insert(cx.current, value.0);
598                cx.style.corner_top_right_smoothing.insert(cx.current, value.1);
599                cx.style.corner_bottom_left_smoothing.insert(cx.current, value.2);
600                cx.style.corner_bottom_right_smoothing.insert(cx.current, value.3);
601
602                cx.needs_redraw(entity);
603            });
604        });
605
606        self
607    }
608
609    // Outline Properties
610    modifier!(
611        /// Sets the outline width of the view.
612        outline_width,
613        LengthOrPercentage,
614        SystemFlags::REDRAW
615    );
616
617    modifier!(
618        /// Sets the outline color of the view.
619        outline_color,
620        Color,
621        SystemFlags::REDRAW
622    );
623
624    modifier!(
625        /// Sets the outline offset of the view.
626        outline_offset,
627        LengthOrPercentage,
628        SystemFlags::REDRAW
629    );
630
631    // Cursor Icon
632    modifier!(
633        /// Sets the mouse cursor used when the view is hovered.
634        cursor,
635        CursorIcon,
636        SystemFlags::empty()
637    );
638
639    /// Sets whether the view can be become the target of pointer events.
640    fn pointer_events<U: Into<PointerEvents>>(mut self, value: impl Res<U>) -> Self {
641        let entity = self.entity();
642        let current = self.current();
643        self.context().with_current(current, |cx| {
644            value.set_or_bind(cx, entity, move |cx, v| {
645                let value = v.get(cx).into();
646                cx.style.pointer_events.insert(cx.current, value);
647            });
648        });
649
650        self
651    }
652
653    /// Sets the transform of the view with a list of transform functions.
654    fn transform<U: Into<Vec<Transform>>>(mut self, value: impl Res<U>) -> Self {
655        let entity = self.entity();
656        let current = self.current();
657        self.context().with_current(current, |cx| {
658            value.set_or_bind(cx, entity, move |cx, v| {
659                let value = v.get(cx).into();
660                cx.style.transform.insert(cx.current, value);
661                cx.needs_redraw(entity);
662            });
663        });
664
665        self
666    }
667
668    /// Sets the transform origin of the the view.
669    fn transform_origin<U: Into<Position>>(mut self, value: impl Res<U>) -> Self {
670        let entity = self.entity();
671        let current = self.current();
672        self.context().with_current(current, |cx| {
673            value.set_or_bind(cx, entity, move |cx, v| {
674                let value = v.get(cx).into();
675                let x = value.x.to_length_or_percentage();
676                let y = value.y.to_length_or_percentage();
677                cx.style.transform_origin.insert(cx.current, Translate { x, y });
678                cx.needs_redraw(entity);
679            });
680        });
681
682        self
683    }
684
685    // Translate
686    modifier!(
687        /// Sets the translation offset of the view.
688        ///
689        /// Translation applies to the rendered view and does not affect layout.
690        translate,
691        Translate,
692        SystemFlags::REDRAW
693    );
694
695    // Rotate
696    modifier!(
697        /// Sets the angle of rotation for the view.
698        ///
699        /// Rotation applies to the rendered view and does not affect layout.
700        rotate,
701        Angle,
702        SystemFlags::REDRAW
703    );
704
705    // Scale
706    modifier!(
707        /// Sets the scale of the view.
708        ///
709        /// Scale applies to the rendered view and does not affect layout.
710        scale,
711        Scale,
712        SystemFlags::REDRAW
713    );
714}
715
716impl<V: View> StyleModifiers for Handle<'_, V> {}
717
718/// A builder for constructing linear gradients.
719#[derive(Debug, Clone)]
720pub struct LinearGradientBuilder {
721    direction: LineDirection,
722    stops: Vec<ColorStop<LengthOrPercentage>>,
723}
724
725impl Default for LinearGradientBuilder {
726    fn default() -> Self {
727        Self::new()
728    }
729}
730
731impl LinearGradientBuilder {
732    /// Creates a new [LinearGradientBuilder].
733    pub fn new() -> Self {
734        LinearGradientBuilder { direction: LineDirection::default(), stops: Vec::new() }
735    }
736
737    /// Set the direction of the linear gradient.
738    pub fn with_direction(direction: impl Into<LineDirection>) -> Self {
739        LinearGradientBuilder { direction: direction.into(), stops: Vec::new() }
740    }
741
742    fn build(self) -> Gradient {
743        Gradient::Linear(LinearGradient { direction: self.direction, stops: self.stops })
744    }
745
746    /// Add a color stop to the linear gradient.
747    pub fn add_stop(mut self, stop: impl Into<ColorStop<LengthOrPercentage>>) -> Self {
748        self.stops.push(stop.into());
749
750        self
751    }
752}
753
754impl From<LinearGradientBuilder> for Gradient {
755    fn from(value: LinearGradientBuilder) -> Self {
756        value.build()
757    }
758}
759
760/// A builder for constructing a shadow.
761#[derive(Debug, Clone)]
762pub struct ShadowBuilder {
763    shadow: Shadow,
764}
765
766impl Default for ShadowBuilder {
767    fn default() -> Self {
768        Self::new()
769    }
770}
771
772impl ShadowBuilder {
773    /// Creates a new [ShadowBuilder].
774    pub fn new() -> Self {
775        Self { shadow: Shadow::default() }
776    }
777
778    fn build(self) -> Shadow {
779        self.shadow
780    }
781
782    /// Sets the horizontal offset of the shadow.
783    pub fn x_offset(mut self, offset: impl Into<Length>) -> Self {
784        self.shadow.x_offset = offset.into();
785
786        self
787    }
788
789    /// Set the vertical offset of the shadow.
790    pub fn y_offset(mut self, offset: impl Into<Length>) -> Self {
791        self.shadow.y_offset = offset.into();
792
793        self
794    }
795
796    /// Sets the blur radius of the shadow.
797    pub fn blur(mut self, radius: Length) -> Self {
798        self.shadow.blur_radius = Some(radius);
799
800        self
801    }
802
803    /// Sets the spread amount of the shadow.
804    pub fn spread(mut self, radius: Length) -> Self {
805        self.shadow.spread_radius = Some(radius);
806
807        self
808    }
809
810    /// Sets the color of the shadow.
811    pub fn color(mut self, color: Color) -> Self {
812        self.shadow.color = Some(color);
813
814        self
815    }
816
817    /// Sets whether the shadow should be inset.
818    pub fn inset(mut self) -> Self {
819        self.shadow.inset = true;
820
821        self
822    }
823}
824
825impl From<ShadowBuilder> for Shadow {
826    fn from(value: ShadowBuilder) -> Self {
827        value.build()
828    }
829}