2 个不稳定版本
0.6.0 | 2021年3月6日 |
---|---|
0.0.1 | 2020年5月9日 |
#5 in #spa
每月 23 次下载
在 2 crates 中使用
44KB
1K SLoC
Savory
Rust / Wasm 前端库,用于构建用户界面。
Savory 是基于 Seed 的用户界面构建库
特性
- 设计系统:元素完全使用
DesignSystem
样式化。 - 可重用性:元素高度可重用/可组合。
- 解耦开发:设计系统可以在单独的 crate 中开发,无需触摸元素代码,同样,元素的开发也可以独立于设计系统进行,这要归功于
DesignSystemImpl
trait。 - 干净的视图:以干净和声明性的方式构建视图,不再需要任何宏。
- 基于 trait:拥抱 Rust trait 系统,所有 savory 元素都实现了
Element
和/或View
trait。 - 类型化 HTML:使用类型化的 CSS 和 HTML 属性,Savory 尽量不依赖字符串来创建 CSS 和 HTML 属性,因为这些可能会产生难以调试的错误。
- UI 元素集合:Savory 随附一系列可重用和可主题化的 UI 元素。
- 增强 Seed API:对 Seed API 的增强,使得使用
Node
、Orders
变得有趣。
Savory 努力让编写 UI 元素变得有趣且无需样板代码。
截图
入门
快速入门的最佳方式是按照 Savory 入门 仓库的说明进行。
示例
在这里,我们将创建一个在 Elm 教程 中找到的相同计数器应用,然后我们将用样式化和可重用的元素编写相同的程序。
简单计数器
use savory::prelude::*;
// app element (the model)
pub struct Counter(i32);
// app message
pub enum Msg {
Increment,
Decrement,
}
impl Element for Counter {
type Message = Msg;
type Config = Url;
// initialize the app in this function
fn init(_: Url, _: &mut impl Orders<Msg>) -> Self {
Self(0)
}
// handle app messages
fn update(&mut self, msg: Msg, _: &mut impl Orders<Msg>) {
match msg {
Msg::Increment => self.0 += 1,
Msg::Decrement => self.0 -= 1,
}
}
}
impl View<Node<Msg>> for Counter {
// view the app
fn view(&self) -> Node<Msg> {
let inc_btn = html::button().push("Increment").on_click(|_| Msg::Increment);
let dec_btn = html::button().push("Decrement").on_click(|_| Msg::Decrement);
html::div()
.push(inc_btn)
.push(self.0.to_string())
.push(dec_btn)
}
}
#[wasm_bindgen(start)]
pub fn view() {
// mount and start the app at `app` element
Counter::start();
}
预览:
计数器作为元素
现在我们将创建计数器元素和应用程序元素,这展示了如何创建父子元素,以及如何创建可复用和可样式的元素。
use savory::prelude::*;
use savory_elements::prelude::*;
use savory_style::{
css::{unit::px, values as val, Color, St},
prelude::*,
};
#[derive(Element)]
pub struct Counter {
#[element(config(default = "10"))]
value: i32,
}
pub enum Msg {
Increment,
Decrement,
}
impl Element for Counter {
type Message = Msg;
type Config = Config;
fn init(config: Self::Config, _: &mut impl Orders<Msg>) -> Self {
Self {
value: config.value,
}
}
fn update(&mut self, msg: Msg, _: &mut impl Orders<Msg>) {
match msg {
Msg::Increment => self.value += 1,
Msg::Decrement => self.value -= 1,
}
}
}
impl View<Node<Msg>> for Counter {
fn view(&self) -> Node<Msg> {
// sharde style for buttons
let style_btns = |conf: css::Style| {
conf.push(St::Appearance, val::None)
.background(Color::SlateBlue)
.text(Color::White)
.and_border(|conf| conf.none().radius(px(4)))
.margin(px(4))
.padding(px(4))
};
// increment button node
let inc_btn = html::button()
.class("inc-btn")
.and_style(style_btns)
.on_click(|_| Msg::Increment)
.push("Increment");
// decrement button node
let dec_btn = html::button()
.class("dec-btn")
.and_style(style_btns)
.on_click(|_| Msg::Decrement)
.push("Decrement");
// contianer node
html::div()
.push(dec_btn)
.push(self.value.to_string())
.push(inc_btn)
}
}
// convenient way to convert Config into Counter
impl Config {
pub fn init(self, orders: &mut impl Orders<Msg>) -> Counter {
Counter::init(self, orders)
}
}
// App Element ---
pub enum AppMsg {
Counter(Msg),
}
pub struct MyApp {
counter_element: Counter,
}
impl Element for MyApp {
type Message = AppMsg;
type Config = Url;
fn init(_: Url, orders: &mut impl Orders<AppMsg>) -> Self {
Self {
counter_element: Counter::config()
// give it starting value. 10 will be used as default value if
// we didn't pass value
.value(100)
.init(&mut orders.proxy(AppMsg::Counter)),
}
}
fn update(&mut self, msg: AppMsg, orders: &mut impl Orders<AppMsg>) {
match msg {
AppMsg::Counter(msg) => self
.counter_element
.update(msg, &mut orders.proxy(AppMsg::Counter)),
}
}
}
impl View<Node<AppMsg>> for MyApp {
fn view(&self) -> Node<AppMsg> {
self.counter_element.view().map_msg(AppMsg::Counter)
}
}
#[wasm_bindgen(start)]
pub fn view() {
// mount and start the app at `app` element
MyApp::start();
}
预览:
这个例子中有许多事情发生,首先我们创建了一个元素结构 Counter
,并定义了它的属性、事件和样式类型,这些都是通过 Element
derive 宏完成的,我们将在稍后解释它是如何工作的,然后我们定义了一个包含计数器元素的应用程序元素,并在 init
函数中初始化它。最后,我们只是调用 start
方法来挂载和启动应用程序。
计数器使用 Savory 元素!
Savory 随带一系列元素,我们将使用它们来构建计数器应用程序并查看 Savory 元素为我们提供了哪些功能。
待办事项:添加示例
生态系统
savory
- 用于构建用户界面的核心库savory-router
- Savory 路由器,用于为您的应用程序生成路由savory-style
- Savory 的类型化 CSS 样式savory-elements
- 基于 Savory 的 UI 元素集合savory-elements-derive
- 提供元素 derive 的 crate
许可证
根据您的选择,在 Apache License, Version 2.0 或 MIT 许可证下授权。Apache 许可证,版本 2.0 或 MIT 许可证。
除非您明确声明,否则您提交给 Savory 的任何贡献,根据 Apache-2.0 许可证的定义,都应如上双重授权,无需任何额外条款或条件。
依赖项
~16MB
~289K SLoC