forked from render-rs/render.rs
-
Notifications
You must be signed in to change notification settings - Fork 2
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Peter John
committed
Sep 18, 2022
1 parent
3823550
commit a2a8cc9
Showing
1 changed file
with
40 additions
and
106 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,119 +1,53 @@ | ||
# tide-jsx | ||
# tide-jsx [data:image/s3,"s3://crabby-images/61f00/61f00f82e6bd56f5e4248e5af114e96d1cec7065" alt="Build Status"](https://github.com/pyrossh/tide-jsx/actions?query=workflow%3ATest) [data:image/s3,"s3://crabby-images/a6897/a689716cb68c355bca01bc190235e1bbbaf17071" alt="crates.io"](https://crates.io/crates/tide-jsx) | ||
|
||
In order to render a simple HTML fragment into a `String`, use the `rsx!` macro to generate a | ||
component tree, and call `render` on it: | ||
This crate allows to use jsx like templating engine (which is a slightly modified version of [render.rs](https://github.com/render-rs/render.rs)) as templates in your tide request handlers. | ||
|
||
```rust | ||
use tide_jsx::{rsx, Render}; | ||
|
||
let tree = rsx! { | ||
<div> | ||
<h1>{"Hello!"}</h1> | ||
<p>{"Hello world!"}</p> | ||
</div> | ||
}; | ||
|
||
assert_eq!(tree.render(), "<div><h1>Hello!</h1><p>Hello world!</p></div>"); | ||
``` | ||
|
||
Because this is so common, there's another macro called `html!` that calls `rsx!` to generate | ||
a component tree, and then calls `render` on it. Most of the time, you'll find yourself using | ||
the `rsx!` macro to compose arbitrary components, and only calling `html!` when you need a | ||
String output, when sending a response or generating a Markdown file. | ||
|
||
In Render, attributes and plain strings are escaped using the `render::html_escaping` module. In order to | ||
use un-escaped values so you can dangerously insert raw HTML, use the `raw!` macro around your | ||
string: | ||
## Example | ||
|
||
```rust | ||
use tide_jsx::{html, raw}; | ||
|
||
let tree = html! { | ||
<div> | ||
<p>{"<Hello />"}</p> | ||
<p>{raw!("<Hello />")}</p> | ||
</div> | ||
}; | ||
|
||
assert_eq!(tree, "<div><p><Hello /></p><p><Hello /></p></div>"); | ||
``` | ||
|
||
### Custom components | ||
|
||
Render's greatest ability is to provide type-safety along with custom renderable components. | ||
Introducing new components is as easy as defining a function that returns a `Render` value. | ||
|
||
In order to build up components from other components or HTML nodes, you can use the `rsx!` | ||
macro, which generates a `Render` component tree: | ||
|
||
```rust | ||
use tide_jsx::{component, rsx, html}; | ||
use tide::http::mime; | ||
use tide::utils::After; | ||
use tide::{log, Request, Response}; | ||
use tide_jsx::html::HTML5Doctype; | ||
use tide_jsx::{component, rsx, view}; | ||
|
||
#[component] | ||
fn Heading<'title>(title: &'title str) { | ||
rsx! { <h1 class={"title"}>{title}</h1> } | ||
rsx! { <h1 class={"title"}>{title}</h1> } | ||
} | ||
|
||
let rendered_html = html! { | ||
<Heading title={"Hello world!"} /> | ||
}; | ||
|
||
assert_eq!(rendered_html, r#"<h1 class="title">Hello world!</h1>"#); | ||
``` | ||
|
||
If you pay close attention, you see that the function `Heading` is: | ||
|
||
- declared with an uppercase. Underneath, it generates a struct with the same name, and | ||
implements the `Render` trait on it. | ||
- does not have a return type. This is because everything is written to a writer, for | ||
performance reasons. | ||
|
||
### Visibility & Component Libraries | ||
|
||
Often you're going to want to store your components somewhere else in your | ||
project tree other than the module you're working on (if not in a different | ||
module entirely!). In these cases, the visibility applied top the function that | ||
defines your component will flow down into all fields of that struct. | ||
|
||
For example, if we add "pub" to the front of our Heading component above: | ||
|
||
```rust | ||
#[component] | ||
pub fn Heading<'title>(title: &'title str) { | ||
rsx! { <h1 class={"title"}>{title}</h1> } | ||
} | ||
``` | ||
|
||
...the struct that is generated would look something like... | ||
|
||
```rust | ||
pub struct Heading { | ||
pub title: &'title str | ||
} | ||
``` | ||
|
||
This is important to understand from a safety point of view when structuring | ||
your libraries. | ||
|
||
#### Full example | ||
|
||
```rust | ||
use render::html::HTML5Doctype; | ||
use render::{component, rsx, html, Render}; | ||
|
||
#[component] | ||
fn Page<'a, Children: Render>(title: &'a str, children: Children) { | ||
rsx! { | ||
|
||
} | ||
} | ||
|
||
pub fn some_page(user_name: &str) -> String { | ||
html! { | ||
<Page title={"Home"}> | ||
{format!("Welcome, {}", user_name)} | ||
</Page> | ||
async fn index(_req: Request<()>) -> tide::Result { | ||
view! { | ||
<> | ||
<HTML5Doctype /> | ||
<html> | ||
<head><title>{"Tide JSX"}</title></head> | ||
<body> | ||
<div> | ||
<Heading title={"Hello world"} /> | ||
</div> | ||
</body> | ||
</html> | ||
</> | ||
} | ||
} | ||
|
||
#[async_std::main] | ||
async fn main() -> tide::Result<()> { | ||
log::start(); | ||
let mut app = tide::new(); | ||
app.with(tide::log::LogMiddleware::new()); | ||
app.with(After(|mut res: Response| async { | ||
if let Some(err) = res.error() { | ||
let msg = format!("<h1>Error: {:?}</h1>", err); | ||
res.set_status(err.status()); | ||
res.set_content_type(mime::HTML); | ||
res.set_body(msg); | ||
} | ||
Ok(res) | ||
})); | ||
app.at("/").get(index); | ||
app.listen("127.0.0.1:5000").await?; | ||
Ok(()) | ||
} | ||
``` |