// Copyright © SixtyFPS GmbH // SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-Slint-Royalty-free-2.0 OR LicenseRef-Slint-Software-3.0 // cSpell: ignore nesw /*! This module contains the builtin items, either in this file or in sub-modules. When adding an item or a property, it needs to be kept in sync with different place. (This is less than ideal and maybe we can have some automation later) - It needs to be changed in this module - In the compiler: builtins.slint - In the interpreter (new item only): dynamic_item_tree.rs - For the C++ code (new item only): the cbindgen.rs to export the new item - Don't forget to update the documentation */ #![allow(unsafe_code)] #![allow(non_upper_case_globals)] #![allow(missing_docs)] // because documenting each property of items is redundant use crate::graphics::{Brush, Color, FontRequest}; use crate::input::{ FocusEvent, FocusEventResult, InputEventFilterResult, InputEventResult, KeyEventResult, KeyEventType, MouseEvent, }; use crate::item_rendering::{CachedRenderingData, RenderBorderRectangle, RenderRectangle}; pub use crate::item_tree::{ItemRc, ItemTreeVTable}; use crate::layout::LayoutInfo; use crate::lengths::{ LogicalBorderRadius, LogicalLength, LogicalRect, LogicalSize, LogicalVector, PointLengths, RectLengths, }; #[cfg(feature = "rtti")] use crate::rtti::*; use crate::window::{WindowAdapter, WindowAdapterRc, WindowInner}; use crate::{Callback, Coord, Property, SharedString}; use alloc::rc::Rc; use const_field_offset::FieldOffsets; use core::cell::Cell; use core::num::NonZeroU32; use core::pin::Pin; use i_slint_core_macros::*; use vtable::*; mod component_container; pub use self::component_container::*; mod flickable; pub use flickable::Flickable; mod text; pub use text::*; mod input_items; pub use input_items::*; mod image; pub use self::image::*; #[cfg(feature = "std")] mod path; pub use crate::menus::MenuItem; #[cfg(feature = "std")] pub use path::*; /// Alias for `&mut dyn ItemRenderer`. Required so cbindgen generates the ItemVTable /// despite the presence of trait object type ItemRendererRef<'a> = &'a mut dyn crate::item_rendering::ItemRenderer; /// Workarounds for cbindgen pub type VoidArg = (); pub type KeyEventArg = (KeyEvent,); type FocusReasonArg = (FocusReason,); type PointerEventArg = (PointerEvent,); type PointerScrollEventArg = (PointerScrollEvent,); type PointArg = (crate::api::LogicalPosition,); type MenuEntryArg = (MenuEntry,); type MenuEntryModel = crate::model::ModelRc; #[cfg(all(feature = "ffi", windows))] #[macro_export] macro_rules! declare_item_vtable { (fn $getter:ident() -> $item_vtable_ty:ident for $item_ty:ty) => { ItemVTable_static! { #[unsafe(no_mangle)] pub static $item_vtable_ty for $item_ty } #[unsafe(no_mangle)] pub extern "C" fn $getter() -> *const ItemVTable { use vtable::HasStaticVTable; <$item_ty>::static_vtable() } }; } #[cfg(not(all(feature = "ffi", windows)))] #[macro_export] macro_rules! declare_item_vtable { (fn $getter:ident() -> $item_vtable_ty:ident for $item_ty:ty) => { ItemVTable_static! { #[unsafe(no_mangle)] pub static $item_vtable_ty for $item_ty } }; } /// Returned by the `render()` function on items to indicate whether the rendering of /// children should be handled by the caller, of if the item took care of that (for example /// through layer indirection) #[repr(C)] #[derive(Default)] pub enum RenderingResult { #[default] ContinueRenderingChildren, ContinueRenderingWithoutChildren, } /// Items are the nodes in the render tree. #[cfg_attr(not(feature = "ffi"), i_slint_core_macros::remove_extern)] #[vtable] #[repr(C)] pub struct ItemVTable { /// This function is called by the run-time after the memory for the item /// has been allocated and initialized. It will be called before any user specified /// bindings are set. pub init: extern "C-unwind" fn(core::pin::Pin>, my_item: &ItemRc), /// offset in bytes from the *const ItemImpl. /// isize::MAX means None #[allow(non_upper_case_globals)] #[field_offset(CachedRenderingData)] pub cached_rendering_data_offset: usize, /// We would need max/min/preferred size, and all layout info pub layout_info: extern "C-unwind" fn( core::pin::Pin>, orientation: Orientation, window_adapter: &WindowAdapterRc, self_rc: &ItemRc, ) -> LayoutInfo, /// Event handler for mouse and touch event. This function is called before being called on children. /// Then, depending on the return value, it is called for the children, and their children, then /// [`Self::input_event`] is called on the children, and finally [`Self::input_event`] is called /// on this item again. pub input_event_filter_before_children: extern "C-unwind" fn( core::pin::Pin>, MouseEvent, window_adapter: &WindowAdapterRc, self_rc: &ItemRc, ) -> InputEventFilterResult, /// Handle input event for mouse and touch event pub input_event: extern "C-unwind" fn( core::pin::Pin>, MouseEvent, window_adapter: &WindowAdapterRc, self_rc: &ItemRc, ) -> InputEventResult, pub focus_event: extern "C-unwind" fn( core::pin::Pin>, &FocusEvent, window_adapter: &WindowAdapterRc, self_rc: &ItemRc, ) -> FocusEventResult, pub key_event: extern "C-unwind" fn( core::pin::Pin>, &KeyEvent, window_adapter: &WindowAdapterRc, self_rc: &ItemRc, ) -> KeyEventResult, pub render: extern "C-unwind" fn( core::pin::Pin>, backend: &mut ItemRendererRef, self_rc: &ItemRc, size: LogicalSize, ) -> RenderingResult, pub bounding_rect: extern "C" fn( core::pin::Pin>, window_adapter: &WindowAdapterRc, self_rc: &ItemRc, geometry: LogicalRect, ) -> LogicalRect, pub clips_children: extern "C" fn(core::pin::Pin>) -> bool, } /// Alias for `vtable::VRef` which represent a pointer to a `dyn Item` with /// the associated vtable pub type ItemRef<'a> = vtable::VRef<'a, ItemVTable>; #[repr(C)] #[derive(FieldOffsets, Default, SlintElement)] #[pin] /// The implementation of an empty items that does nothing pub struct Empty { pub cached_rendering_data: CachedRenderingData, } impl Item for Empty { fn init(self: Pin<&Self>, _self_rc: &ItemRc) {} fn layout_info( self: Pin<&Self>, _orientation: Orientation, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> LayoutInfo { LayoutInfo { stretch: 1., ..LayoutInfo::default() } } fn input_event_filter_before_children( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventFilterResult { InputEventFilterResult::ForwardAndIgnore } fn input_event( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventResult { InputEventResult::EventIgnored } fn key_event( self: Pin<&Self>, _: &KeyEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> KeyEventResult { KeyEventResult::EventIgnored } fn focus_event( self: Pin<&Self>, _: &FocusEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> FocusEventResult { FocusEventResult::FocusIgnored } fn render( self: Pin<&Self>, _backend: &mut ItemRendererRef, _self_rc: &ItemRc, _size: LogicalSize, ) -> RenderingResult { RenderingResult::ContinueRenderingChildren } fn bounding_rect( self: core::pin::Pin<&Self>, _window_adapter: &Rc, _self_rc: &ItemRc, mut geometry: LogicalRect, ) -> LogicalRect { geometry.size = LogicalSize::zero(); geometry } fn clips_children(self: core::pin::Pin<&Self>) -> bool { false } } impl ItemConsts for Empty { const cached_rendering_data_offset: const_field_offset::FieldOffset< Empty, CachedRenderingData, > = Empty::FIELD_OFFSETS.cached_rendering_data.as_unpinned_projection(); } declare_item_vtable! { fn slint_get_EmptyVTable() -> EmptyVTable for Empty } #[repr(C)] #[derive(FieldOffsets, Default, SlintElement)] #[pin] /// The implementation of the `Rectangle` element pub struct Rectangle { pub background: Property, pub cached_rendering_data: CachedRenderingData, } impl Item for Rectangle { fn init(self: Pin<&Self>, _self_rc: &ItemRc) {} fn layout_info( self: Pin<&Self>, _orientation: Orientation, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> LayoutInfo { LayoutInfo { stretch: 1., ..LayoutInfo::default() } } fn input_event_filter_before_children( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventFilterResult { InputEventFilterResult::ForwardAndIgnore } fn input_event( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventResult { InputEventResult::EventIgnored } fn key_event( self: Pin<&Self>, _: &KeyEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> KeyEventResult { KeyEventResult::EventIgnored } fn focus_event( self: Pin<&Self>, _: &FocusEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> FocusEventResult { FocusEventResult::FocusIgnored } fn render( self: Pin<&Self>, backend: &mut ItemRendererRef, self_rc: &ItemRc, size: LogicalSize, ) -> RenderingResult { (*backend).draw_rectangle(self, self_rc, size, &self.cached_rendering_data); RenderingResult::ContinueRenderingChildren } fn bounding_rect( self: core::pin::Pin<&Self>, _window_adapter: &Rc, _self_rc: &ItemRc, geometry: LogicalRect, ) -> LogicalRect { geometry } fn clips_children(self: core::pin::Pin<&Self>) -> bool { false } } impl RenderRectangle for Rectangle { fn background(self: Pin<&Self>) -> Brush { self.background() } } impl ItemConsts for Rectangle { const cached_rendering_data_offset: const_field_offset::FieldOffset< Rectangle, CachedRenderingData, > = Rectangle::FIELD_OFFSETS.cached_rendering_data.as_unpinned_projection(); } declare_item_vtable! { fn slint_get_RectangleVTable() -> RectangleVTable for Rectangle } #[repr(C)] #[derive(FieldOffsets, Default, SlintElement)] #[pin] /// The implementation of the `BasicBorderRectangle` element pub struct BasicBorderRectangle { pub background: Property, pub border_width: Property, pub border_radius: Property, pub border_color: Property, pub cached_rendering_data: CachedRenderingData, } impl Item for BasicBorderRectangle { fn init(self: Pin<&Self>, _self_rc: &ItemRc) {} fn layout_info( self: Pin<&Self>, _orientation: Orientation, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> LayoutInfo { LayoutInfo { stretch: 1., ..LayoutInfo::default() } } fn input_event_filter_before_children( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventFilterResult { InputEventFilterResult::ForwardAndIgnore } fn input_event( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventResult { InputEventResult::EventIgnored } fn key_event( self: Pin<&Self>, _: &KeyEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> KeyEventResult { KeyEventResult::EventIgnored } fn focus_event( self: Pin<&Self>, _: &FocusEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> FocusEventResult { FocusEventResult::FocusIgnored } fn render( self: Pin<&Self>, backend: &mut ItemRendererRef, self_rc: &ItemRc, size: LogicalSize, ) -> RenderingResult { (*backend).draw_border_rectangle(self, self_rc, size, &self.cached_rendering_data); RenderingResult::ContinueRenderingChildren } fn bounding_rect( self: core::pin::Pin<&Self>, _window_adapter: &Rc, _self_rc: &ItemRc, geometry: LogicalRect, ) -> LogicalRect { geometry } fn clips_children(self: core::pin::Pin<&Self>) -> bool { false } } impl RenderBorderRectangle for BasicBorderRectangle { fn background(self: Pin<&Self>) -> Brush { self.background() } fn border_width(self: Pin<&Self>) -> LogicalLength { self.border_width() } fn border_radius(self: Pin<&Self>) -> LogicalBorderRadius { LogicalBorderRadius::from_length(self.border_radius()) } fn border_color(self: Pin<&Self>) -> Brush { self.border_color() } } impl ItemConsts for BasicBorderRectangle { const cached_rendering_data_offset: const_field_offset::FieldOffset< BasicBorderRectangle, CachedRenderingData, > = BasicBorderRectangle::FIELD_OFFSETS.cached_rendering_data.as_unpinned_projection(); } declare_item_vtable! { fn slint_get_BasicBorderRectangleVTable() -> BasicBorderRectangleVTable for BasicBorderRectangle } #[repr(C)] #[derive(FieldOffsets, Default, SlintElement)] #[pin] /// The implementation of the `BorderRectangle` element pub struct BorderRectangle { pub background: Property, pub border_width: Property, pub border_radius: Property, pub border_top_left_radius: Property, pub border_top_right_radius: Property, pub border_bottom_left_radius: Property, pub border_bottom_right_radius: Property, pub border_color: Property, pub cached_rendering_data: CachedRenderingData, } impl Item for BorderRectangle { fn init(self: Pin<&Self>, _self_rc: &ItemRc) {} fn layout_info( self: Pin<&Self>, _orientation: Orientation, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> LayoutInfo { LayoutInfo { stretch: 1., ..LayoutInfo::default() } } fn input_event_filter_before_children( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventFilterResult { InputEventFilterResult::ForwardAndIgnore } fn input_event( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventResult { InputEventResult::EventIgnored } fn key_event( self: Pin<&Self>, _: &KeyEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> KeyEventResult { KeyEventResult::EventIgnored } fn focus_event( self: Pin<&Self>, _: &FocusEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> FocusEventResult { FocusEventResult::FocusIgnored } fn render( self: Pin<&Self>, backend: &mut ItemRendererRef, self_rc: &ItemRc, size: LogicalSize, ) -> RenderingResult { (*backend).draw_border_rectangle(self, self_rc, size, &self.cached_rendering_data); RenderingResult::ContinueRenderingChildren } fn bounding_rect( self: core::pin::Pin<&Self>, _window_adapter: &Rc, _self_rc: &ItemRc, geometry: LogicalRect, ) -> LogicalRect { geometry } fn clips_children(self: core::pin::Pin<&Self>) -> bool { false } } impl RenderBorderRectangle for BorderRectangle { fn background(self: Pin<&Self>) -> Brush { self.background() } fn border_width(self: Pin<&Self>) -> LogicalLength { self.border_width() } fn border_radius(self: Pin<&Self>) -> LogicalBorderRadius { LogicalBorderRadius::from_lengths( self.border_top_left_radius(), self.border_top_right_radius(), self.border_bottom_right_radius(), self.border_bottom_left_radius(), ) } fn border_color(self: Pin<&Self>) -> Brush { self.border_color() } } impl ItemConsts for BorderRectangle { const cached_rendering_data_offset: const_field_offset::FieldOffset< BorderRectangle, CachedRenderingData, > = BorderRectangle::FIELD_OFFSETS.cached_rendering_data.as_unpinned_projection(); } declare_item_vtable! { fn slint_get_BorderRectangleVTable() -> BorderRectangleVTable for BorderRectangle } declare_item_vtable! { fn slint_get_TouchAreaVTable() -> TouchAreaVTable for TouchArea } declare_item_vtable! { fn slint_get_FocusScopeVTable() -> FocusScopeVTable for FocusScope } declare_item_vtable! { fn slint_get_SwipeGestureHandlerVTable() -> SwipeGestureHandlerVTable for SwipeGestureHandler } #[repr(C)] #[derive(FieldOffsets, Default, SlintElement)] #[pin] /// The implementation of the `Clip` element pub struct Clip { pub border_top_left_radius: Property, pub border_top_right_radius: Property, pub border_bottom_left_radius: Property, pub border_bottom_right_radius: Property, pub border_width: Property, pub cached_rendering_data: CachedRenderingData, pub clip: Property, } impl Item for Clip { fn init(self: Pin<&Self>, _self_rc: &ItemRc) {} fn layout_info( self: Pin<&Self>, _orientation: Orientation, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> LayoutInfo { LayoutInfo { stretch: 1., ..LayoutInfo::default() } } fn input_event_filter_before_children( self: Pin<&Self>, event: MouseEvent, _window_adapter: &Rc, self_rc: &ItemRc, ) -> InputEventFilterResult { if let Some(pos) = event.position() { let geometry = self_rc.geometry(); if self.clip() && (pos.x < 0 as Coord || pos.y < 0 as Coord || pos.x_length() > geometry.width_length() || pos.y_length() > geometry.height_length()) { return InputEventFilterResult::Intercept; } } InputEventFilterResult::ForwardAndIgnore } fn input_event( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventResult { InputEventResult::EventIgnored } fn key_event( self: Pin<&Self>, _: &KeyEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> KeyEventResult { KeyEventResult::EventIgnored } fn focus_event( self: Pin<&Self>, _: &FocusEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> FocusEventResult { FocusEventResult::FocusIgnored } fn render( self: Pin<&Self>, backend: &mut ItemRendererRef, self_rc: &ItemRc, size: LogicalSize, ) -> RenderingResult { (*backend).visit_clip(self, self_rc, size) } fn bounding_rect( self: core::pin::Pin<&Self>, _window_adapter: &Rc, _self_rc: &ItemRc, geometry: LogicalRect, ) -> LogicalRect { geometry } fn clips_children(self: core::pin::Pin<&Self>) -> bool { self.clip() } } impl Clip { pub fn logical_border_radius(self: Pin<&Self>) -> LogicalBorderRadius { LogicalBorderRadius::from_lengths( self.border_top_left_radius(), self.border_top_right_radius(), self.border_bottom_right_radius(), self.border_bottom_left_radius(), ) } } impl ItemConsts for Clip { const cached_rendering_data_offset: const_field_offset::FieldOffset = Clip::FIELD_OFFSETS.cached_rendering_data.as_unpinned_projection(); } declare_item_vtable! { fn slint_get_ClipVTable() -> ClipVTable for Clip } #[repr(C)] #[derive(FieldOffsets, Default, SlintElement)] #[pin] /// The Opacity Item is not meant to be used directly by the .slint code, instead, the `opacity: xxx` or `visible: false` should be used pub struct Opacity { // FIXME: this element shouldn't need these geometry property pub opacity: Property, pub cached_rendering_data: CachedRenderingData, } impl Item for Opacity { fn init(self: Pin<&Self>, _self_rc: &ItemRc) {} fn layout_info( self: Pin<&Self>, _orientation: Orientation, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> LayoutInfo { LayoutInfo { stretch: 1., ..LayoutInfo::default() } } fn input_event_filter_before_children( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventFilterResult { InputEventFilterResult::ForwardAndIgnore } fn input_event( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventResult { InputEventResult::EventIgnored } fn key_event( self: Pin<&Self>, _: &KeyEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> KeyEventResult { KeyEventResult::EventIgnored } fn focus_event( self: Pin<&Self>, _: &FocusEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> FocusEventResult { FocusEventResult::FocusIgnored } fn render( self: Pin<&Self>, backend: &mut ItemRendererRef, self_rc: &ItemRc, size: LogicalSize, ) -> RenderingResult { backend.visit_opacity(self, self_rc, size) } fn bounding_rect( self: core::pin::Pin<&Self>, _window_adapter: &Rc, _self_rc: &ItemRc, geometry: LogicalRect, ) -> LogicalRect { geometry } fn clips_children(self: core::pin::Pin<&Self>) -> bool { false } } impl Opacity { // This function determines the optimization opportunities for not having to render the // children of the Opacity element into a layer: // * The opacity item typically only one child (this is not guaranteed). If that item has // no children, then we can skip the layer and apply the opacity directly. This is not perfect though, // for example if the compiler inserts another synthetic element between the `Opacity` and the actual child, // then this check will apply a layer even though it might not actually be necessary. // * If the vale of the opacity is 1.0 then we don't need to do anything. pub fn need_layer(self_rc: &ItemRc, opacity: f32) -> bool { if opacity == 1.0 { return false; } let opacity_child = match self_rc.first_child() { Some(first_child) => first_child, None => return false, // No children? Don't need a layer then. }; if opacity_child.next_sibling().is_some() { return true; // If the opacity item has more than one child, then we need a layer } // If the target of the opacity has any children then we need a layer opacity_child.first_child().is_some() } } impl ItemConsts for Opacity { const cached_rendering_data_offset: const_field_offset::FieldOffset< Opacity, CachedRenderingData, > = Opacity::FIELD_OFFSETS.cached_rendering_data.as_unpinned_projection(); } declare_item_vtable! { fn slint_get_OpacityVTable() -> OpacityVTable for Opacity } #[repr(C)] #[derive(FieldOffsets, Default, SlintElement)] #[pin] /// The Layer Item is not meant to be used directly by the .slint code, instead, the `layer: xxx` property should be used pub struct Layer { pub cache_rendering_hint: Property, pub cached_rendering_data: CachedRenderingData, } impl Item for Layer { fn init(self: Pin<&Self>, _self_rc: &ItemRc) {} fn layout_info( self: Pin<&Self>, _orientation: Orientation, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> LayoutInfo { LayoutInfo { stretch: 1., ..LayoutInfo::default() } } fn input_event_filter_before_children( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventFilterResult { InputEventFilterResult::ForwardAndIgnore } fn input_event( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventResult { InputEventResult::EventIgnored } fn key_event( self: Pin<&Self>, _: &KeyEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> KeyEventResult { KeyEventResult::EventIgnored } fn focus_event( self: Pin<&Self>, _: &FocusEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> FocusEventResult { FocusEventResult::FocusIgnored } fn render( self: Pin<&Self>, backend: &mut ItemRendererRef, self_rc: &ItemRc, size: LogicalSize, ) -> RenderingResult { backend.visit_layer(self, self_rc, size) } fn bounding_rect( self: core::pin::Pin<&Self>, _window_adapter: &Rc, _self_rc: &ItemRc, geometry: LogicalRect, ) -> LogicalRect { geometry } fn clips_children(self: core::pin::Pin<&Self>) -> bool { false } } impl ItemConsts for Layer { const cached_rendering_data_offset: const_field_offset::FieldOffset< Layer, CachedRenderingData, > = Layer::FIELD_OFFSETS.cached_rendering_data.as_unpinned_projection(); } declare_item_vtable! { fn slint_get_LayerVTable() -> LayerVTable for Layer } #[repr(C)] #[derive(FieldOffsets, Default, SlintElement)] #[pin] /// The implementation of the `Rotate` element pub struct Rotate { pub rotation_angle: Property, pub rotation_origin_x: Property, pub rotation_origin_y: Property, pub cached_rendering_data: CachedRenderingData, } impl Item for Rotate { fn init(self: Pin<&Self>, _self_rc: &ItemRc) {} fn layout_info( self: Pin<&Self>, _orientation: Orientation, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> LayoutInfo { LayoutInfo { stretch: 1., ..LayoutInfo::default() } } fn input_event_filter_before_children( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventFilterResult { InputEventFilterResult::ForwardAndIgnore } fn input_event( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventResult { InputEventResult::EventIgnored } fn key_event( self: Pin<&Self>, _: &KeyEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> KeyEventResult { KeyEventResult::EventIgnored } fn focus_event( self: Pin<&Self>, _: &FocusEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> FocusEventResult { FocusEventResult::FocusIgnored } fn render( self: Pin<&Self>, backend: &mut ItemRendererRef, _self_rc: &ItemRc, _size: LogicalSize, ) -> RenderingResult { let origin = LogicalVector::from_lengths(self.rotation_origin_x(), self.rotation_origin_y()); (*backend).translate(origin); (*backend).rotate(self.rotation_angle()); (*backend).translate(-origin); RenderingResult::ContinueRenderingChildren } fn bounding_rect( self: core::pin::Pin<&Self>, _window_adapter: &Rc, _self_rc: &ItemRc, geometry: LogicalRect, ) -> LogicalRect { geometry } fn clips_children(self: core::pin::Pin<&Self>) -> bool { false } } impl ItemConsts for Rotate { const cached_rendering_data_offset: const_field_offset::FieldOffset< Rotate, CachedRenderingData, > = Rotate::FIELD_OFFSETS.cached_rendering_data.as_unpinned_projection(); } declare_item_vtable! { fn slint_get_RotateVTable() -> RotateVTable for Rotate } declare_item_vtable! { fn slint_get_FlickableVTable() -> FlickableVTable for Flickable } /// The implementation of the `PropertyAnimation` element #[repr(C)] #[derive(FieldOffsets, SlintElement, Clone, Debug)] #[pin] pub struct PropertyAnimation { #[rtti_field] pub delay: i32, #[rtti_field] pub duration: i32, #[rtti_field] pub iteration_count: f32, #[rtti_field] pub direction: AnimationDirection, #[rtti_field] pub easing: crate::animations::EasingCurve, } impl Default for PropertyAnimation { fn default() -> Self { // Defaults for PropertyAnimation are defined here (for internal Rust code doing programmatic animations) // as well as in `builtins.slint` (for generated C++ and Rust code) Self { delay: 0, duration: 0, iteration_count: 1., direction: Default::default(), easing: Default::default(), } } } /// The implementation of the `Window` element #[repr(C)] #[derive(FieldOffsets, Default, SlintElement)] #[pin] pub struct WindowItem { pub width: Property, pub height: Property, pub background: Property, pub title: Property, pub no_frame: Property, pub resize_border_width: Property, pub always_on_top: Property, pub full_screen: Property, pub icon: Property, pub default_font_family: Property, pub default_font_size: Property, pub default_font_weight: Property, pub cached_rendering_data: CachedRenderingData, } impl Item for WindowItem { fn init(self: Pin<&Self>, _self_rc: &ItemRc) { #[cfg(feature = "std")] self.full_screen.set(std::env::var("SLINT_FULLSCREEN").is_ok()); } fn layout_info( self: Pin<&Self>, _orientation: Orientation, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> LayoutInfo { LayoutInfo::default() } fn input_event_filter_before_children( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventFilterResult { InputEventFilterResult::ForwardAndIgnore } fn input_event( self: Pin<&Self>, _event: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventResult { InputEventResult::EventIgnored } fn key_event( self: Pin<&Self>, _: &KeyEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> KeyEventResult { KeyEventResult::EventIgnored } fn focus_event( self: Pin<&Self>, _: &FocusEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> FocusEventResult { FocusEventResult::FocusIgnored } fn render( self: Pin<&Self>, backend: &mut ItemRendererRef, self_rc: &ItemRc, size: LogicalSize, ) -> RenderingResult { backend.draw_window_background(self, self_rc, size, &self.cached_rendering_data); RenderingResult::ContinueRenderingChildren } fn bounding_rect( self: core::pin::Pin<&Self>, _window_adapter: &Rc, _self_rc: &ItemRc, geometry: LogicalRect, ) -> LogicalRect { geometry } fn clips_children(self: core::pin::Pin<&Self>) -> bool { false } } impl RenderRectangle for WindowItem { fn background(self: Pin<&Self>) -> Brush { self.background() } } fn next_window_item(item: &ItemRc) -> Option { let root_item_in_local_item_tree = ItemRc::new(item.item_tree().clone(), 0); if root_item_in_local_item_tree.downcast::().is_some() { Some(root_item_in_local_item_tree) } else { root_item_in_local_item_tree .parent_item(crate::item_tree::ParentItemTraversalMode::FindAllParents) .and_then(|parent| next_window_item(&parent)) } } impl WindowItem { pub fn font_family(self: Pin<&Self>) -> Option { let maybe_family = self.default_font_family(); if !maybe_family.is_empty() { Some(maybe_family) } else { None } } pub fn font_size(self: Pin<&Self>) -> Option { let font_size = self.default_font_size(); if font_size.get() <= 0 as Coord { None } else { Some(font_size) } } pub fn font_weight(self: Pin<&Self>) -> Option { let font_weight = self.default_font_weight(); if font_weight == 0 { None } else { Some(font_weight) } } pub fn resolve_font_property( self_rc: &ItemRc, property_fn: impl Fn(Pin<&Self>) -> Option, ) -> Option { let mut window_item_rc = self_rc.clone(); loop { let window_item = window_item_rc.downcast::()?; if let Some(result) = property_fn(window_item.as_pin_ref()) { return Some(result); } match window_item_rc .parent_item(crate::item_tree::ParentItemTraversalMode::FindAllParents) .and_then(|p| next_window_item(&p)) { Some(item) => window_item_rc = item, None => return None, } } } /// Creates a new FontRequest that uses the provide local font properties. If they're not set, i.e. /// the family is an empty string, or the weight is zero, the corresponding properties are fetched /// from the next parent WindowItem. pub fn resolved_font_request( self_rc: &crate::items::ItemRc, local_font_family: SharedString, local_font_weight: i32, local_font_size: LogicalLength, local_letter_spacing: LogicalLength, local_italic: bool, ) -> FontRequest { let Some(window_item_rc) = next_window_item(self_rc) else { return FontRequest::default(); }; FontRequest { family: { if !local_font_family.is_empty() { Some(local_font_family) } else { crate::items::WindowItem::resolve_font_property( &window_item_rc, crate::items::WindowItem::font_family, ) } }, weight: { if local_font_weight == 0 { crate::items::WindowItem::resolve_font_property( &window_item_rc, crate::items::WindowItem::font_weight, ) } else { Some(local_font_weight) } }, pixel_size: { if local_font_size.get() == 0 as Coord { crate::items::WindowItem::resolve_font_property( &window_item_rc, crate::items::WindowItem::font_size, ) } else { Some(local_font_size) } }, letter_spacing: Some(local_letter_spacing), italic: local_italic, } } } impl ItemConsts for WindowItem { const cached_rendering_data_offset: const_field_offset::FieldOffset = Self::FIELD_OFFSETS.cached_rendering_data.as_unpinned_projection(); } declare_item_vtable! { fn slint_get_WindowItemVTable() -> WindowItemVTable for WindowItem } /// The implementation used for `ContextMenuArea` and `ContextMenuInternal` elements #[repr(C)] #[derive(FieldOffsets, Default, SlintElement)] #[pin] pub struct ContextMenu { //pub entries: Property>, pub sub_menu: Callback, pub activated: Callback, pub show: Callback, pub cached_rendering_data: CachedRenderingData, pub popup_id: Cell>, pub enabled: Property, #[cfg(target_os = "android")] long_press_timer: Cell>, } impl Item for ContextMenu { fn init(self: Pin<&Self>, _self_rc: &ItemRc) {} fn layout_info( self: Pin<&Self>, _orientation: Orientation, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> LayoutInfo { LayoutInfo::default() } fn input_event_filter_before_children( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventFilterResult { InputEventFilterResult::ForwardEvent } fn input_event( self: Pin<&Self>, event: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventResult { if !self.enabled() { return InputEventResult::EventIgnored; } match event { MouseEvent::Pressed { position, button: PointerEventButton::Right, .. } => { self.show.call(&(crate::api::LogicalPosition::from_euclid(position),)); InputEventResult::EventAccepted } #[cfg(target_os = "android")] MouseEvent::Pressed { position, button: PointerEventButton::Left, .. } => { let timer = crate::timers::Timer::default(); let self_weak = _self_rc.downgrade(); timer.start( crate::timers::TimerMode::SingleShot, WindowInner::from_pub(_window_adapter.window()) .ctx .platform() .long_press_interval(crate::InternalToken), move || { let Some(self_rc) = self_weak.upgrade() else { return }; let Some(self_) = self_rc.downcast::() else { return }; self_.show.call(&(crate::api::LogicalPosition::from_euclid(position),)); }, ); self.long_press_timer.set(Some(timer)); InputEventResult::GrabMouse } #[cfg(target_os = "android")] MouseEvent::Released { .. } | MouseEvent::Exit => { if let Some(timer) = self.long_press_timer.take() { timer.stop(); } InputEventResult::EventIgnored } #[cfg(target_os = "android")] MouseEvent::Moved { .. } => InputEventResult::EventAccepted, _ => InputEventResult::EventIgnored, } } fn key_event( self: Pin<&Self>, event: &KeyEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> KeyEventResult { if !self.enabled() { return KeyEventResult::EventIgnored; } if event.event_type == KeyEventType::KeyPressed && event.text.starts_with(crate::input::key_codes::Menu) { self.show.call(&(Default::default(),)); KeyEventResult::EventAccepted } else { KeyEventResult::EventIgnored } } fn focus_event( self: Pin<&Self>, _: &FocusEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> FocusEventResult { FocusEventResult::FocusIgnored } fn render( self: Pin<&Self>, _backend: &mut ItemRendererRef, _self_rc: &ItemRc, _size: LogicalSize, ) -> RenderingResult { RenderingResult::ContinueRenderingChildren } fn bounding_rect( self: core::pin::Pin<&Self>, _window_adapter: &Rc, _self_rc: &ItemRc, geometry: LogicalRect, ) -> LogicalRect { geometry } fn clips_children(self: core::pin::Pin<&Self>) -> bool { false } } impl ContextMenu { pub fn close(self: Pin<&Self>, window_adapter: &Rc, _: &ItemRc) { if let Some(id) = self.popup_id.take() { WindowInner::from_pub(window_adapter.window()).close_popup(id); } } pub fn is_open(self: Pin<&Self>, window_adapter: &Rc, _: &ItemRc) -> bool { self.popup_id.get().is_some_and(|id| { WindowInner::from_pub(window_adapter.window()) .active_popups() .iter() .any(|p| p.popup_id == id) }) } } impl ItemConsts for ContextMenu { const cached_rendering_data_offset: const_field_offset::FieldOffset = Self::FIELD_OFFSETS.cached_rendering_data.as_unpinned_projection(); } declare_item_vtable! { fn slint_get_ContextMenuVTable() -> ContextMenuVTable for ContextMenu } #[cfg(feature = "ffi")] #[unsafe(no_mangle)] pub unsafe extern "C" fn slint_contextmenu_close( s: Pin<&ContextMenu>, window_adapter: *const crate::window::ffi::WindowAdapterRcOpaque, self_component: &vtable::VRc, self_index: u32, ) { let window_adapter = &*(window_adapter as *const Rc); let self_rc = ItemRc::new(self_component.clone(), self_index); s.close(window_adapter, &self_rc); } #[cfg(feature = "ffi")] #[unsafe(no_mangle)] pub unsafe extern "C" fn slint_contextmenu_is_open( s: Pin<&ContextMenu>, window_adapter: *const crate::window::ffi::WindowAdapterRcOpaque, self_component: &vtable::VRc, self_index: u32, ) -> bool { let window_adapter = &*(window_adapter as *const Rc); let self_rc = ItemRc::new(self_component.clone(), self_index); s.is_open(window_adapter, &self_rc) } /// The implementation of the `BoxShadow` element #[repr(C)] #[derive(FieldOffsets, Default, SlintElement)] #[pin] pub struct BoxShadow { pub border_radius: Property, // Shadow specific properties pub offset_x: Property, pub offset_y: Property, pub color: Property, pub blur: Property, pub cached_rendering_data: CachedRenderingData, } impl Item for BoxShadow { fn init(self: Pin<&Self>, _self_rc: &ItemRc) {} fn layout_info( self: Pin<&Self>, _orientation: Orientation, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> LayoutInfo { LayoutInfo { stretch: 1., ..LayoutInfo::default() } } fn input_event_filter_before_children( self: Pin<&Self>, _: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventFilterResult { InputEventFilterResult::ForwardAndIgnore } fn input_event( self: Pin<&Self>, _event: MouseEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> InputEventResult { InputEventResult::EventIgnored } fn key_event( self: Pin<&Self>, _: &KeyEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> KeyEventResult { KeyEventResult::EventIgnored } fn focus_event( self: Pin<&Self>, _: &FocusEvent, _window_adapter: &Rc, _self_rc: &ItemRc, ) -> FocusEventResult { FocusEventResult::FocusIgnored } fn render( self: Pin<&Self>, backend: &mut ItemRendererRef, self_rc: &ItemRc, size: LogicalSize, ) -> RenderingResult { (*backend).draw_box_shadow(self, self_rc, size); RenderingResult::ContinueRenderingChildren } fn bounding_rect( self: core::pin::Pin<&Self>, _window_adapter: &Rc, _self_rc: &ItemRc, geometry: LogicalRect, ) -> LogicalRect { geometry .outer_rect(euclid::SideOffsets2D::from_length_all_same(self.blur())) .translate(LogicalVector::from_lengths(self.offset_x(), self.offset_y())) } fn clips_children(self: core::pin::Pin<&Self>) -> bool { false } } impl ItemConsts for BoxShadow { const cached_rendering_data_offset: const_field_offset::FieldOffset = Self::FIELD_OFFSETS.cached_rendering_data.as_unpinned_projection(); } declare_item_vtable! { fn slint_get_BoxShadowVTable() -> BoxShadowVTable for BoxShadow } declare_item_vtable! { fn slint_get_ComponentContainerVTable() -> ComponentContainerVTable for ComponentContainer } declare_item_vtable! { fn slint_get_ComplexTextVTable() -> ComplexTextVTable for ComplexText } declare_item_vtable! { fn slint_get_SimpleTextVTable() -> SimpleTextVTable for SimpleText } declare_item_vtable! { fn slint_get_TextInputVTable() -> TextInputVTable for TextInput } declare_item_vtable! { fn slint_get_ImageItemVTable() -> ImageItemVTable for ImageItem } declare_item_vtable! { fn slint_get_ClippedImageVTable() -> ClippedImageVTable for ClippedImage } #[cfg(feature = "std")] declare_item_vtable! { fn slint_get_PathVTable() -> PathVTable for Path } declare_item_vtable! { fn slint_get_MenuItemVTable() -> MenuItemVTable for MenuItem } macro_rules! declare_enums { ($( $(#[$enum_doc:meta])* enum $Name:ident { $( $(#[$value_doc:meta])* $Value:ident,)* })*) => { $( #[derive(Copy, Clone, Debug, PartialEq, Eq, strum::EnumString, strum::Display, Hash)] #[repr(u32)] #[strum(serialize_all = "kebab-case")] $(#[$enum_doc])* pub enum $Name { $( $(#[$value_doc])* $Value),* } impl Default for $Name { fn default() -> Self { // Always return the first value ($(Self::$Value,)*).0 } } )* }; } i_slint_common::for_each_enums!(declare_enums); macro_rules! declare_builtin_structs { ($( $(#[$struct_attr:meta])* struct $Name:ident { @name = $inner_name:literal export { $( $(#[$pub_attr:meta])* $pub_field:ident : $pub_type:ty, )* } private { $( $(#[$pri_attr:meta])* $pri_field:ident : $pri_type:ty, )* } } )*) => { $( #[derive(Clone, Debug, Default, PartialEq)] #[repr(C)] $(#[$struct_attr])* pub struct $Name { $( $(#[$pub_attr])* pub $pub_field : $pub_type, )* $( $(#[$pri_attr])* pub $pri_field : $pri_type, )* } )* }; } i_slint_common::for_each_builtin_structs!(declare_builtin_structs); #[cfg(feature = "ffi")] #[unsafe(no_mangle)] pub unsafe extern "C" fn slint_item_absolute_position( self_component: &vtable::VRc, self_index: u32, ) -> crate::lengths::LogicalPoint { let self_rc = ItemRc::new(self_component.clone(), self_index); self_rc.map_to_window(Default::default()) }