自定义控件介绍
一个可点击的按钮
import { VerticalBox, Button } from "std-widgets.slint";export component Recipe inherits Window { in-out property <int> counter: 0; VerticalBox { button := Button { text: "Button, pressed " + root.counter + " times"; clicked => { root.counter += 1; } } }}在第一个示例中,你将看到 Slint 语言的基础知识:
- 我们使用
import语句从标准库导入VerticalBox布局和Button控件。 此语句可以导入不同文件中声明的控件或你自己的组件。你不需要导入Window或Rectangle等内置元素。 - 我们使用
component关键字声明Recipe组件。Recipe继承自Window并具有以下元素:一个布局(VerticalBox),其中包含一个按钮。 - 你可以通过元素的名称后跟一对花括号(带可选内容)来实例化元素。 你可以使用
:=为特定元素分配名称 - 元素具有属性。使用
:设置属性值。这里我们分配一个 通过连接一些字符串字面量和counter属性来计算字符串的绑定到Button的text属性。 - 你可以使用
property <...>为任何元素声明自定义属性。属性 需要具有类型,并且可以具有默认值和访问 说明符。private、in、out或in-out等访问说明符定义了 外部元素如何与该属性交互。Private是默认 值,阻止任何外部元素访问该属性。 本例中的counter属性是自定义属性。 - 元素还可以具有回调。在这种情况下,我们使用
=> { ... }将回调 处理程序分配给button的clicked回调。 - 如果绑定所依赖的任何属性发生变化,属性绑定将自动重新求值。只要
counter发生变化,按钮的text绑定就会自动重新计算。
在原生代码中响应按钮点击
此示例使用原生代码递增 counter:
import { VerticalBox, Button } from "std-widgets.slint";export component Recipe inherits Window { in-out property <int> counter: 0; callback button-pressed <=> button.clicked; VerticalBox { button := Button { text: "Button, pressed " + root.counter + " times"; } }}<=> 语法将两个回调绑定在一起。这里新的 button-pressed 回调绑定到 button.clicked。
主组件的根元素向原生代码公开所有非 private 属性和回调。
在 Slint 中,- 和 _ 在所有标识符中都是等价的且可互换的。 这在原生代码中是不同的:大多数编程语言禁止在标识符中使用 -,因此 - 会被替换为 _。
出于技术原因,本示例在 slint! 宏中使用 export {Recipe}。 在实际代码中,你可以将整个 Slint 代码放在 slint! 宏中,或者使用外部 .slint 文件以及构建脚本。
slint::slint!(export { Recipe } from "docs/reference/src/recipes/button_native.slint";);
fn main() { let recipe = Recipe::new().unwrap(); let recipe_weak = recipe.as_weak(); recipe.on_button_pressed(move || { let recipe = recipe_weak.upgrade().unwrap(); let mut value = recipe.get_counter(); value = value + 1; recipe.set_counter(value); }); recipe.run().unwrap();}Slint 编译器会生成一个 struct Recipe,其中包含根元素的每个可访问属性的 getter(get_counter)和 setter(set_counter)。它还会为每个可访问的回调生成一个函数, 就像本例中的 on_button_pressed。
Recipe 结构体实现了 slint::ComponentHandle trait。组件管理强引用计数和弱引用计数,类似于 Rc。 我们调用 as_weak 函数以获取组件的弱句柄,然后可以将其移动到回调中。
此处不能使用强句柄,因为那样会形成一个循环:组件句柄拥有回调的所有权,而回调又拥有闭包捕获的变量的所有权。
在 C++ 中你可以这样写
#include "button_native.h"
int main(int argc, char **argv){ auto recipe = Recipe::create(); recipe->on_button_pressed([&]() { auto value = recipe->get_counter(); value += 1; recipe->set_counter(value); }); recipe->run();}CMake 集成会根据需要处理 Slint 编译器的调用, 它将解析 .slint 文件并生成 button_native.h 头文件。
此头文件包含一个 Recipe 类,其中每个可访问属性都有 getter 和 setter,以及为 Recipe 中每个可访问回调设置回调的函数。在本例中,我们将拥有 get_counter、 set_counter 来访问 counter 属性,并使用 on_button_pressed 来设置回调。
在 Python 中,你可以这样写:
import slint
class App(slint.loader.recipe.Recipe): @slint.callback def button_pressed(self): value = self.counter value = value + 1 self.counter = value
app = App()app.run()Slint 自动加载器提供了一个源自 recipe.slint 的 Recipe 类,该类被子类化。 Recipe 类提供 counter 属性,@slint.callback 装饰器将 button_pressed 方法与 button-pressed 回调连接起来。
使用属性绑定同步控件
import { VerticalBox, Slider } from "std-widgets.slint";export component Recipe inherits Window { VerticalBox { slider := Slider { maximum: 100; } Text { text: "Value: \{round(slider.value)}"; } }}此示例介绍了 Slider 控件。
它还介绍了字符串字面量中的插值:使用 \{...} 将 花括号之间代码的结果作为字符串渲染。
动画示例
为元素的位置设置动画
import { CheckBox } from "std-widgets.slint";export component Recipe inherits Window { width: 200px; height: 100px;
rect := Rectangle { x:0; y: 5px; width: 40px; height: 40px; background: blue; animate x { duration: 500ms; easing: ease-in-out; } }
CheckBox { y: 25px; text: "Align rect to the right"; toggled => { if (self.checked) { rect.x = parent.width - rect.width; } else { rect.x = 0px; } } }}布局会自动定位元素。在此示例中,我们改为手动定位元素,使用 x、y、width、height 属性。
注意指定动画的 animate x 块。每当属性 发生变化时它就会运行:无论是回调设置了该属性,还是 其绑定值发生变化。
动画序列
import { CheckBox } from "std-widgets.slint";export component Recipe inherits Window { width: 200px; height: 100px;
rect := Rectangle { x:0; y: 5px; width: 40px; height: 40px; background: blue; animate x { duration: 500ms; easing: ease-in-out; } animate y { duration: 250ms; delay: 500ms; easing: ease-in; } }
CheckBox { y: 25px; text: "Align rect bottom right"; toggled => { if (self.checked) { rect.x = parent.width - rect.width; rect.y = parent.height - rect.height; } else { rect.x = 0px; rect.y = 0px; } } }}此示例使用 delay 属性来使一个动画在另一个动画之后运行。
状态示例
将属性值与状态关联
import { HorizontalBox, VerticalBox, Button } from "std-widgets.slint";
component Circle inherits Rectangle { width: 30px; height: 30px; border-radius: root.width / 2; animate x { duration: 250ms; easing: ease-in; } animate y { duration: 250ms; easing: ease-in-out; } animate background { duration: 250ms; }}
export component Recipe inherits Window { states [ left-aligned when b1.pressed: { circle1.x: 0px; circle1.y: 40px; circle1.background: green; circle2.x: 0px; circle2.y: 0px; circle2.background: blue; }
right-aligned when b2.pressed: { circle1.x: 170px; circle1.y: 70px; circle1.background: green; circle2.x: 170px; circle2.y: 00px; circle2.background: blue; }
]
VerticalBox { HorizontalBox { max-height: self.min-height;
b1 := Button { text: "State 1"; }
b2 := Button { text: "State 2"; }
}
Rectangle { background: root.background.darker(20%); width: 200px; height: 100px;
circle1 := Circle { y:0; background: green; x: 85px; } circle2 := Circle { background: green; x: 85px; y: 40px; } } }}过渡
import { HorizontalBox, VerticalBox, Button } from "std-widgets.slint";
component Circle inherits Rectangle { width: 30px; height: 30px; border-radius: root.width / 2;}
export component Recipe inherits Window { states [ left-aligned when b1.pressed: { circle1.x: 0px; circle1.y: 40px; circle2.x: 0px; circle2.y: 0px; in { animate circle1.x, circle2.x { duration: 250ms; } }
out { animate circle1.x, circle2.x { duration: 500ms; } }
}
right-aligned when !b1.pressed: { circle1.x: 170px; circle1.y: 70px; circle2.x: 170px; circle2.y: 00px; }
]
VerticalBox { HorizontalBox { max-height: self.min-height;
b1 := Button { text: "Press and hold to change state"; }
}
Rectangle { background: root.background.darker(20%); width: 250px; height: 100px;
circle1 := Circle { y:0; background: green; x: 85px; } circle2 := Circle { background: blue; x: 85px; y: 40px; } } }}布局示例
垂直
import { VerticalBox, Button } from "std-widgets.slint";export component Recipe inherits Window { VerticalBox { Button { text: "First"; } Button { text: "Second"; } Button { text: "Third"; } }}水平
import { HorizontalBox, Button } from "std-widgets.slint";export component Recipe inherits Window { HorizontalBox { Button { text: "First"; } Button { text: "Second"; } Button { text: "Third"; } }}网格
import { GridBox, Button, Slider } from "std-widgets.slint";export component Recipe inherits Window { GridBox { Row { Button { text: "First"; } Button { text: "Second"; } }
Row { Button { text: "Third"; } Button { text: "Fourth"; } }
Row { Slider { colspan: 2; }
} }}全局回调
从 Slint 调用全局注册的原生回调
此示例使用全局单例在原生代码中实现通用逻辑。 此单例还可以存储原生代码可访问的属性。
注意:预览仅可视化 Slint 代码。它未连接到原生代码。
import { HorizontalBox, VerticalBox, LineEdit } from "std-widgets.slint";
export global Logic { pure callback to-upper-case(string) -> string; // 你可以在此处收集其他全局属性}
export component Recipe inherits Window { VerticalBox { input := LineEdit { text: "Text to be transformed"; }
HorizontalBox { Text { text: "Transformed:"; } // 在绑定表达式中调用的回调 Text { text: { Logic.to-upper-case(input.text); }
}
} }}在 Rust 中你可以这样设置回调:
fn main() { let recipe = Recipe::new().unwrap(); recipe.global::<Logic>().on_to_upper_case(|string| { string.as_str().to_uppercase().into() }); // ...}C++ 代码 在 C++ 中你可以这样设置回调:
int main(int argc, char **argv){ auto recipe = Recipe::create(); recipe->global<Logic>().on_to_upper_case([](slint::SharedString str) -> slint::SharedString { std::string arg(str); std::transform(arg.begin(), arg.end(), arg.begin(), toupper); return slint::SharedString(arg); }); // ...}在 JavaScript 中你可以这样设置回调:
let slint = require("slint-ui");let file = slint.loadFile("recipe.slint");let recipe = new file.Recipe();recipe.Logic.to_upper_case = (str) => { return str.toUpperCase();};// ...在 Python 中,回调与 @slint.callback 装饰器的 global_name 参数相关联:
import slint
class App(slint.loader.recipe.Recipe): @slint.callback(global_name="Logic")
def to_upper_case(&self, value: str) -> str: return value.upper()
# ...自定义控件
自定义按钮
component Button inherits Rectangle { in-out property text <=> txt.text; callback clicked <=> touch.clicked; border-radius: root.height / 2; border-width: 1px; border-color: root.background.darker(25%); background: touch.pressed ? #6b8282 : touch.has-hover ? #6c616c : #456; height: txt.preferred-height * 1.33; min-width: txt.preferred-width + 20px; txt := Text { x: (parent.width - self.width)/2 + (touch.pressed ? 2px : 0); y: (parent.height - self.height)/2 + (touch.pressed ? 1px : 0); color: touch.pressed ? #fff : #eee; }
touch := TouchArea { }}
export component Recipe inherits Window { VerticalLayout { alignment: start; Button { text: "Button"; } }}开关控件
export component ToggleSwitch inherits Rectangle { callback toggled; in-out property <string> text; in-out property <bool> checked; in-out property<bool> enabled <=> touch-area.enabled; height: 20px; horizontal-stretch: 0; vertical-stretch: 0;
HorizontalLayout { spacing: 8px; indicator := Rectangle { width: 40px; border-width: 1px; border-radius: root.height / 2; border-color: self.background.darker(25%); background: root.enabled ? (root.checked ? blue: white) : white; animate background { duration: 100ms; }
bubble := Rectangle { width: root.height - 8px; height: bubble.width; border-radius: bubble.height / 2; y: 4px; x: 4px + self.a * (indicator.width - bubble.width - 8px); property <float> a: root.checked ? 1 : 0; background: root.checked ? white : (root.enabled ? blue : gray); animate a, background { duration: 200ms; easing: ease;}
}
}
Text { min-width: max(100px, self.preferred-width); text: root.text; vertical-alignment: center; color: root.enabled ? black : gray; }
}
touch-area := TouchArea { width: root.width; height: root.height; clicked => { if (root.enabled) { root.checked = !root.checked; root.toggled(); }
} }}
export component Recipe inherits Window { VerticalLayout { alignment: start; ToggleSwitch { text: "Toggle me"; } ToggleSwitch { text: "Disabled"; enabled: false; } }}自定义滑块
TouchArea 覆盖整个控件,因此你可以从自身内的任何点拖动此滑块。
import { VerticalBox } from "std-widgets.slint";
export component MySlider inherits Rectangle { in-out property<float> maximum: 100; in-out property<float> minimum: 0; in-out property<float> value;
min-height: 24px; min-width: 100px; horizontal-stretch: 1; vertical-stretch: 0;
border-radius: root.height/2; background: touch.pressed ? #eee: #ddd; border-width: 1px; border-color: root.background.darker(25%);
handle := Rectangle { width: self.height; height: parent.height; border-width: 3px; border-radius: self.height / 2; background: touch.pressed ? #f8f: touch.has-hover ? #66f : #0000ff; border-color: self.background.darker(15%); x: (root.width - handle.width) * (root.value - root.minimum)/(root.maximum - root.minimum); }
touch := TouchArea { property <float> pressed-value; pointer-event(event) => { if (event.button == PointerEventButton.left && event.kind == PointerEventKind.down) { self.pressed-value = root.value; }
}
moved => { if (self.enabled && self.pressed) { root.value = max(root.minimum, min(root.maximum, self.pressed-value + (touch.mouse-x - touch.pressed-x) * (root.maximum - root.minimum) / (root.width - handle.width)));
}
} }}
export component Recipe inherits Window { VerticalBox { alignment: start; slider := MySlider { maximum: 100; }
Text { text: "Value: \{round(slider.value)}"; } }}此示例展示了另一种具有可拖动手柄的实现: 手柄仅在我们单击该手柄时才会移动。 TouchArea 在手柄内并随其移动。
import { VerticalBox } from "std-widgets.slint";
export component MySlider inherits Rectangle { in-out property<float> maximum: 100; in-out property<float> minimum: 0; in-out property<float> value;
min-height: 24px; min-width: 100px; horizontal-stretch: 1; vertical-stretch: 0;
border-radius: root.height/2; background: touch.pressed ? #eee: #ddd; border-width: 1px; border-color: root.background.darker(25%);
handle := Rectangle { width: self.height; height: parent.height; border-width: 3px; border-radius: self.height / 2; background: touch.pressed ? #f8f: touch.has-hover ? #66f : #0000ff; border-color: self.background.darker(15%); x: (root.width - handle.width) * (root.value - root.minimum)/(root.maximum - root.minimum);
touch := TouchArea { moved => { if (self.enabled && self.pressed) { root.value = max(root.minimum, min(root.maximum, root.value + (self.mouse-x - self.pressed-x) * (root.maximum - root.minimum) / root.width)); }
}
} }}
export component Recipe inherits Window { VerticalBox { alignment: start; slider := MySlider { maximum: 100; }
Text { text: "Value: \{round(slider.value)}"; } }}自定义选项卡
当你想要创建自己的自定义选项卡控件时,可以以此配方为基础。
import { Button } from "std-widgets.slint";
export component Recipe inherits Window { preferred-height: 200px; in-out property <int> active-tab; VerticalLayout { tab_bar := HorizontalLayout { spacing: 3px; Button { text: "Red"; clicked => { root.active-tab = 0; } }
Button { text: "Blue"; clicked => { root.active-tab = 1; } }
Button { text: "Green"; clicked => { root.active-tab = 2; } }
}
Rectangle { clip: true; Rectangle { background: red; x: root.active-tab == 0 ? 0 : root.active-tab < 0 ? - self.width - 1px : parent.width + 1px; animate x { duration: 125ms; easing: ease; } }
Rectangle { background: blue; x: root.active-tab == 1 ? 0 : root.active-tab < 1 ? - self.width - 1px : parent.width + 1px; animate x { duration: 125ms; easing: ease; } }
Rectangle { background: green; x: root.active-tab == 2 ? 0 : root.active-tab < 2 ? - self.width - 1px : parent.width + 1px; animate x { duration: 125ms; easing: ease; } }
} }}自定义表格视图
Slint 提供了一个表格控件,但你也可以基于 ListView 进行自定义。
import { VerticalBox, ListView } from "std-widgets.slint";
component TableView inherits Rectangle { in property <[string]> columns; in property <[[string]]> values;
private property <length> e: self.width / root.columns.length; private property <[length]> column_sizes: [ root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, root.e, ];
VerticalBox { padding: 5px; HorizontalLayout { padding: 5px; spacing: 5px; vertical-stretch: 0; for title[idx] in root.columns : HorizontalLayout { width: root.column_sizes[idx]; Text { overflow: elide; text: title; } Rectangle { width: 1px; background: gray; TouchArea { width: 10px; x: (parent.width - self.width) / 2; property <length> cached; pointer-event(event) => { if (event.button == PointerEventButton.left && event.kind == PointerEventKind.down) { self.cached = root.column_sizes[idx]; }
}
moved => { if (self.pressed) { root.column_sizes[idx] += (self.mouse-x - self.pressed-x); if (root.column_sizes[idx] < 0) { root.column_sizes[idx] = 0; }
}
}
mouse-cursor: ew-resize; }
}
}
}
ListView { for r in root.values : HorizontalLayout { padding: 5px; spacing: 5px; for t[idx] in r : HorizontalLayout { width: root.column_sizes[idx]; Text { overflow: elide; text: t; } }
} } }}
export component Example inherits Window { TableView { columns: ["Device", "Mount Point", "Total", "Free"]; values: [ ["/dev/sda1", "/", "255GB", "82.2GB"] , ["/dev/sda2", "/tmp", "60.5GB", "44.5GB"] , ["/dev/sdb1", "/home", "255GB", "32.2GB"] , ]; }}响应式用户界面的断点
此配方实现了一个响应式 SideBar,当父级 宽度小于给定的断点时会折叠。单击按钮时, SideBar 会再次展开。使用蓝色 Splitter 调整容器大小并 测试响应式行为。
import { Button, Palette } from "std-widgets.slint";
export component SideBar inherits Rectangle { private property <bool> collapsed: root.reference-width < root.break-point;
/// 定义检查 `break-point` 的参考宽度。 in-out property <length> reference-width;
/// 如果 `reference-width` 小于 `break-point`,则 `SideBar` 会折叠。 in-out property <length> break-point: 600px;
/// 设置展开按钮的文本。 in-out property <string> expand-button-text;
width: 160px;
container := Rectangle { private property <bool> expanded;
width: parent.width; background: Palette.background.darker(0.2);
VerticalLayout { padding: 2px; alignment: start;
HorizontalLayout { alignment: start;
if (root.collapsed) : Button { checked: container.expanded; text: root.expand-button-text;
clicked => { container.expanded = !container.expanded; }
}
}
@children }
states [ expanded when container.expanded && root.collapsed : { width: 160px;
in { animate width { duration: 200ms; } }
out { animate width { duration: 200ms; } }
in { animate width { duration: 200ms; } }
out { animate width { duration: 200ms; } }
}
] }
states [ collapsed when root.collapsed : { width: 62px; } ]}
component Splitter inherits TouchArea { width: 4px; mouse-cursor: ew-resize;
Rectangle { width: 100%; height: 100%; background: blue; }}
export component SideBarTest inherits Window { preferred-width: 700px; min-height: 400px; background: gray;
GridLayout { x: 0; width: splitter.x;
Rectangle { height: 100%; col: 1; background: white;
HorizontalLayout { padding: 8px;
Text { color: black; text: "Content"; }
}
}
SideBar { col: 0; reference-width: parent.width; expand-button-text: "E"; } }
splitter := Splitter { x: root.width - self.width; height: 100%;
moved => { self.x = min(root.width - self.width, max(400px, self.x + self.mouse-x - self.pressed-x)); } }}