5 个版本

1.0.0-alpha.82022年3月2日
1.0.0-alpha.72022年3月1日
1.0.0-alpha.62022年2月28日
1.0.0-alpha.52022年2月27日
1.0.0-alpha.22022年1月28日

1497网页编程

MIT 许可证

15KB
156

frender

Crates.io docs.rs GitHub license GitHub stars

功能渲染: ReactRust

frender 仍处于 alpha 版本,其 API 可能会更改。目前建议在 Cargo.toml 中指定确切版本。在更新之前,请查看完整的 变更日志,以防出现重大更改。

一些示例应用程序位于 examples 文件夹中。您可以在 此网站 预览它们。

快速入门

  1. 创建新的 cargo 项目

    cargo new my-frender-app
    cd my-frender-app
    
  2. Cargo.toml 中将 frender 添加到依赖项。

    [dependencies]
    frender = "= 1.0.0-alpha.8"
    
  3. 在项目根目录中创建 index.html

    <!DOCTYPE html>
    <html>
      <head>
        <meta charset="UTF-8" />
        <title>My frender App</title>
        <script src="https://unpkg.com/react@17/umd/react.development.js"></script>
        <script src="https://unpkg.com/react-dom@17/umd/react-dom.development.js"></script>
        <link data-trunk rel="rust" href="Cargo.toml" />
      </head>
      <body>
        <div id="frender-root"></div>
      </body>
    </html>
    
  4. 修改 src/main.rs

    use frender::prelude::*;
    
    #[component(main(mount_element_id = "frender-root"))]
    fn Main() {
        rsx!(
            <div>
                "Hello, frender!"
            </div>
        )
    }
    
  5. 使用 trunk 运行

    安装 trunk 然后执行

    trunk serve
    

    然后您可以通过访问 https://127.0.0.1:8080 来查看您的 frender 应用程序。

rsx 语法

rsx 元素

use frender::prelude::*;

rsx! (
  <MyComp id="my-component">
    // Child node can be any literal strings or numbers
    "some string"
    1
    // Child node can be any rust expressions wrapped in braces
    { 1 + 6 }
    { value }
    // Child node can be an element
    <MyChild key="k" prop={any_expr} />

    // Prop without value means `true`, just like React
    <MyDialog show />

    // Fragment
    <>1 2 3</>
    // Fragment with key
    <# key="key">1 2 3</#>

    // you can also use `</_>` to enclose any element
    <path::to::Component></_>
    // the above is equivalent to:
    <path::to::Component></path::to::Component>
  </MyComp>
)

以小写字母开头的任何组件名称都将被解释为 内建组件。例如,rsx!( <div id="my-div" /> ) 将被解析为

use frender::prelude::*;
use self::intrinsic_components::div::prelude::*;

rsx! (
  <self::intrinsic_components::div::prelude::Component id="my-div" />
)

rsx 属性

为了使 rsx 更加简洁,frender 提供了 IntoPropValue 特性。在 <MyComponent prop={value} /> 中的 value 将映射到 IntoPropValue::into_prop_value(value)

假设属性接受 Option<i32>,你可以将 prop={Some(1)} 简化为 prop={1},因为 T 实现了 IntoPropValue<Option<T>>

如果你想直接传递值,可以使用 := 来设置属性。 prop:={value}

编写组件

无属性的组件

use frender::prelude::*;

#[component]
fn MyComponent() {
  //            ^
  //            the return type defaults to react::Element
  rsx!( <div /> )
}

// Or you can specify the return type explicitly
#[component]
fn MyAnotherComponent() -> Option<react::Element> {
  if check_something() {
    Some(rsx!( <MyComponent /> ))
  } else {
    None
  }
}

有属性的组件

首先定义 MyProps

use frender::prelude::*;

def_props! {
  pub struct MyProps {
    // Required prop
    name: String,

    // Optional prop which defaults to `Default::default()`
    // The following property `age` is optional, and defaults to `None`
    age?: Option<u8>,

    // The following property `tags` is optional, and defaults to `Vec::default()`
    tags?: Vec<String>,

    // If the prop type is not specified,
    // then frender will infer the type by prop name.
    // For example, `class_name` default has type `Option<String>`
    // The following property `class_name` is optional, has type Option<String>
    class_name?,

    // The following property `id` is required, has type Option<String>
    id,

    // Prop can also have type generics.
    // For example, the following is
    // the default definition for prop `children`,
    // which means it accepts any `Option<TNode>` where TNode implements react::Node,
    // and then map the value into `Option<react::Children>` and store it into MyProps.
    children<TNode: react::Node>(value: Option<TNode>) -> Option<react::Children> {
      value.and_then(react::Node::into_children)
    },
  }
}

然后使用上述属性编写组件

use frender::prelude::*;
#[component]
pub fn MyComponent(props: &MyProps) {
  rsx!(<div>{&props.children}</div>)
}

由于泛型,在非常罕见的情况下,你可能遇到诸如 type annotations needed cannot infer type for type parameter 的错误。你可以通过使用涡轮鱼语法 ::<> 来指定类型。例如

rsx! (
  // ERROR: type annotations needed
  <a children={None} />
)
rsx! (
  // it works!
  <a children={None::<()>} />
)

钩子

frender 也支持 React 钩子。

你可以查看示例来了解使用方法。

未来开发计划

  • 文档
  • 内联 SVG 组件
  • frender 组件导出到 js
  • 服务器端渲染
  • CssProperties 的类型检查
  • Css-in-rust(例如,与 emotion/react 集成)

贡献

frenderGitHub 上开源。欢迎提交 pull request 和 issues。

你也可以赞助我,我会非常感激 ❤️

Buy Me a Coffee at ko-fi.com

依赖

~10–13MB
~248K SLoC