-## Examples:
-| File Navigator (Desktop) | WiFi scanner (Desktop) | TodoMVC (All platforms) | Ecommerce w/ Tailwind (Liveview) |
-| ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
-| [![File Explorer](https://github.com/DioxusLabs/example-projects/raw/master/file-explorer/image.png)](https://github.com/DioxusLabs/example-projects/blob/master/file-explorer) | [![Wifi Scanner Demo](https://github.com/DioxusLabs/example-projects/raw/master/wifi-scanner/demo_small.png)](https://github.com/DioxusLabs/example-projects/blob/master/wifi-scanner) | [![TodoMVC example](https://github.com/DioxusLabs/example-projects/raw/master/todomvc/example.png)](https://github.com/DioxusLabs/example-projects/blob/master/todomvc) | [![Ecommerce Example](https://github.com/DioxusLabs/example-projects/raw/master/ecommerce-site/demo.png)](https://github.com/DioxusLabs/example-projects/blob/master/ecommerce-site) |
+
+## Examples Projects:
+
+| File Navigator (Desktop) | WiFi scanner (Desktop) | TodoMVC (All platforms) | E-commerce w/ Tailwind (Liveview) |
+| ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| [![File Explorer](https://github.com/DioxusLabs/example-projects/raw/master/file-explorer/image.png)](https://github.com/DioxusLabs/example-projects/blob/master/file-explorer) | [![Wifi Scanner Demo](https://github.com/DioxusLabs/example-projects/raw/master/wifi-scanner/demo_small.png)](https://github.com/DioxusLabs/example-projects/blob/master/wifi-scanner) | [![TodoMVC example](https://github.com/DioxusLabs/example-projects/raw/master/todomvc/example.png)](https://github.com/DioxusLabs/example-projects/blob/master/todomvc) | [![E-commerce Example](https://github.com/DioxusLabs/example-projects/raw/master/ecommerce-site/demo.png)](https://github.com/DioxusLabs/example-projects/blob/master/ecommerce-site) |
See the awesome-dioxus page for a curated list of content in the Dioxus Ecosystem.
+## Running examples locally
+
+All local examples are built for the desktop renderer. This means you can simply clone this repo and call `cargo run --example EXAMPLE_NAME`. To run non-desktop examples, checkout the example projects shown above.
+
## Why Dioxus and why Rust?
TypeScript is a fantastic addition to JavaScript, but it's still fundamentally JavaScript. TS code runs slightly slower, has tons of configuration options, and not every package is properly typed.
diff --git a/docs/guide/src/README.md b/docs/guide/src/README.md
index ab2cc0403..c91f9f1bb 100644
--- a/docs/guide/src/README.md
+++ b/docs/guide/src/README.md
@@ -18,7 +18,7 @@ fn App(cx: Scope) -> Element {
In general, Dioxus and React share many functional similarities. If this guide is lacking in any general concept or an error message is confusing, React's documentation might be more helpful. We are dedicated to providing a *familiar* toolkit for UI in Rust, so we've chosen to follow in the footsteps of popular UI frameworks (React, Redux, etc). If you know React, then you already know Dioxus. If you don't know either, this guide will still help you!
-> This is introduction book! For advanced topics, check out the [Reference](https://dioxuslabs.com/reference) instead.
+> This is an introduction book! For advanced topics, check out the [Reference](https://dioxuslabs.com/reference) instead.
## Multiplatform
@@ -29,6 +29,7 @@ Right now, we have several 1st-party renderers:
- Tao/Tokio (for Desktop apps)
- Tao/Tokio (for Mobile apps)
- SSR (for generating static markup)
+- TUI/Rink (for terminal-based apps)
### Web Support
---
@@ -40,12 +41,10 @@ Because the web is a fairly mature platform, we expect there to be very little A
[Jump to the getting started guide for the web.]()
Examples:
-- [TodoMVC](https://github.com/dioxusLabs/todomvc/)
-- [ECommerce]()
-- [Photo Editor]()
-
-[![TODOMVC](https://github.com/DioxusLabs/todomvc/raw/master/example.png)](https://github.com/dioxusLabs/todomvc/)
+- [TodoMVC](https://github.com/DioxusLabs/example-projects/tree/master/todomvc)
+- [ECommerce](https://github.com/DioxusLabs/example-projects/tree/master/ecommerce-site)
+[![TodoMVC example](https://github.com/DioxusLabs/example-projects/raw/master/todomvc/example.png)](https://github.com/DioxusLabs/example-projects/blob/master/todomvc)
### SSR Support
---
Dioxus supports server-side rendering!
@@ -59,7 +58,7 @@ let contents = dioxus::ssr::render_vdom(&dom);
[Jump to the getting started guide for SSR.]()
Examples:
-- [Example DocSite]()
+- [Example DocSite](https://github.com/dioxusLabs/docsite)
- [Tide WebServer]()
- [Markdown to fancy HTML generator]()
@@ -72,9 +71,8 @@ Desktop APIs will likely be in flux as we figure out better patterns than our El
[Jump to the getting started guide for Desktop.]()
Examples:
-- [File explorer]()
-- [Bluetooth scanner]()
-- [Device Viewer]()
+- [File explorer](https://github.com/dioxusLabs/file-explorer/)
+- [WiFi scanner](https://github.com/DioxusLabs/example-projects/blob/master/wifi-scanner)
[![File ExplorerExample](https://github.com/DioxusLabs/file-explorer-example/raw/master/image.png)](https://github.com/dioxusLabs/file-explorer/)
@@ -87,8 +85,7 @@ Mobile support is currently best suited for CRUD-style apps, ideally for interna
[Jump to the getting started guide for Mobile.]()
Examples:
-- [Todo App]()
-- [Chat App]()
+- [Todo App](https://github.com/DioxusLabs/example-projects/blob/master/ios_demo)
### LiveView / Server Component Support
---
diff --git a/docs/guide/src/SUMMARY.md b/docs/guide/src/SUMMARY.md
index 4149008c8..8f9183dc2 100644
--- a/docs/guide/src/SUMMARY.md
+++ b/docs/guide/src/SUMMARY.md
@@ -3,96 +3,39 @@
- [Introduction](README.md)
- [Getting Setup](setup.md)
- [Hello, World!](hello_world.md)
-- [Describing the UI](concepts/00-index.md)
- - [Intro to Elements](concepts/vnodes.md)
- - [Intro to Components](concepts/components.md)
- - [Reusing, Importing, and Exporting Components](concepts/exporting_components.md)
- - [Passing children and attributes](concepts/component_children.md)
- - [Conditional Rendering](concepts/conditional_rendering.md)
- - [Lists](concepts/lists.md)
-- [Adding Interactivity](concepts/interactivity.md)
- - [Hooks and Internal State](concepts/hooks.md)
- - [Event handlers](concepts/event_handlers.md)
- - [User Input and Controlled Components](concepts/user_input.md)
- - [Lifecycle, updates, and effects](concepts/lifecycles.md)
-
-
-
-
-- [Managing State](concepts/managing_state.md)
- - [Global State](concepts/sharedstate.md)
- - [Error handling](concepts/errorhandling.md)
- - [Effects](concepts/effects.md)
-- [Working with Async](concepts/async.md)
- - [Tasks](concepts/asynctasks.md)
- - [Suspense](concepts/suspense.md)
- - [Async Callbacks](concepts/asynccallbacks.md)
-- [Putting it all together](tutorial/index.md)
+- [Describing the UI](elements/index.md)
+ - [Intro to Elements](elements/vnodes.md)
+ - [Intro to Components](elements/components.md)
+ - [Reusing, Importing, and Exporting Components](elements/exporting_components.md)
+ - [Passing children and attributes](elements/component_children.md)
+ - [Conditional Rendering](elements/conditional_rendering.md)
+ - [Lists](elements/lists.md)
+- [Adding Interactivity](interactivity/index.md)
+ - [Hooks and Internal State](interactivity/hooks.md)
+ - [Event handlers](interactivity/event_handlers.md)
+ - [User Input and Controlled Components](interactivity/user_input.md)
+ - [Lifecycle, updates, and effects](interactivity/lifecycles.md)
+- [Managing State](state/index.md)
+ - [Local State](state/localstate.md)
+ - [Lifting State](state/liftingstate.md)
+ - [Global State](state/sharedstate.md)
+ - [Error handling](state/errorhandling.md)
+- [Working with Async](async/index.md)
+ - [Tasks](async/asynctasks.md)
+- [Putting it all together: Dog Search Engine](tutorial/index.md)
- [New app](tutorial/new_app.md)
- [Structuring our app](tutorial/structure.md)
- [Defining State](tutorial/state.md)
- [Defining Components](tutorial/components.md)
- [Styling](tutorial/styling.md)
- - [Publishing](tutorial/publishing.md)
-- [Next Steps and Advanced Topics](final/index.md)
-
+ - [Bundling](tutorial/publishing.md)
+- [Next Steps and Advanced Topics](final.md)
+
-----------
[Contributors](misc/contributors.md)
+
+
+
+
diff --git a/docs/guide/src/concepts/00-index.md b/docs/guide/src/advanced-guides/00-index.md
similarity index 100%
rename from docs/guide/src/concepts/00-index.md
rename to docs/guide/src/advanced-guides/00-index.md
diff --git a/docs/guide/src/concepts/06-subscription-api.md b/docs/guide/src/advanced-guides/06-subscription-api.md
similarity index 100%
rename from docs/guide/src/concepts/06-subscription-api.md
rename to docs/guide/src/advanced-guides/06-subscription-api.md
diff --git a/docs/guide/src/concepts/10-concurrent-mode.md b/docs/guide/src/advanced-guides/10-concurrent-mode.md
similarity index 100%
rename from docs/guide/src/concepts/10-concurrent-mode.md
rename to docs/guide/src/advanced-guides/10-concurrent-mode.md
diff --git a/docs/guide/src/concepts/11-arena-memo.md b/docs/guide/src/advanced-guides/11-arena-memo.md
similarity index 100%
rename from docs/guide/src/concepts/11-arena-memo.md
rename to docs/guide/src/advanced-guides/11-arena-memo.md
diff --git a/docs/guide/src/concepts/12-signals.md b/docs/guide/src/advanced-guides/12-signals.md
similarity index 100%
rename from docs/guide/src/concepts/12-signals.md
rename to docs/guide/src/advanced-guides/12-signals.md
diff --git a/docs/guide/src/concepts/13-subtrees.md b/docs/guide/src/advanced-guides/13-subtrees.md
similarity index 100%
rename from docs/guide/src/concepts/13-subtrees.md
rename to docs/guide/src/advanced-guides/13-subtrees.md
diff --git a/docs/guide/src/concepts/rsx.md b/docs/guide/src/advanced-guides/rsx.md
similarity index 100%
rename from docs/guide/src/concepts/rsx.md
rename to docs/guide/src/advanced-guides/rsx.md
diff --git a/docs/guide/src/concepts/rsx_in_depth.md b/docs/guide/src/advanced-guides/rsx_in_depth.md
similarity index 100%
rename from docs/guide/src/concepts/rsx_in_depth.md
rename to docs/guide/src/advanced-guides/rsx_in_depth.md
diff --git a/docs/guide/src/concepts/testing.md b/docs/guide/src/advanced-guides/testing.md
similarity index 98%
rename from docs/guide/src/concepts/testing.md
rename to docs/guide/src/advanced-guides/testing.md
index 42f9af1d8..dc53feafd 100644
--- a/docs/guide/src/concepts/testing.md
+++ b/docs/guide/src/advanced-guides/testing.md
@@ -20,7 +20,7 @@ fn runs_in_browser() {
Then, when you run
-```
+```console
$ dioxus test --chrome
```
diff --git a/docs/guide/src/async/asynctasks.md b/docs/guide/src/async/asynctasks.md
new file mode 100644
index 000000000..bcb93633d
--- /dev/null
+++ b/docs/guide/src/async/asynctasks.md
@@ -0,0 +1,90 @@
+# Tasks
+
+All async code in Dioxus must be explicit and handled through Dioxus' task system.
+
+In this chapter, we'll learn how to spawn new tasks through our `Scope`.
+
+## Spawning a task
+
+You can push any `'static` future into the Dioxus future queue by simply calling `cx.spawn` to spawn a task. Pushing a future returns a `TaskId` which can then be used to cancel the
+
+```rust
+fn App(cx: Scope) -> Element {
+ cx.spawn(async {
+ let mut count = 0;
+ loop {
+ tokio::time::delay(std::instant::Duration::from_millis(500)).await;
+ count += 1;
+ println!("Current count is {}", count);
+ }
+ });
+
+ None
+}
+```
+
+The future must be `'static` - so any values captured by the task must not carry any references to `cx`. All the Dioxus hooks have a method called `for_async` which will create a slightly more limited handle to the hook for you to use in your async code.
+
+```rust
+fn App(cx: Scope) -> Element {
+ let mut count = use_state(&cx, || 0);
+
+ let taskid = cx.spawn({
+ let mut count = count.for_async();
+ async {
+ loop {
+ tokio::time::delay(std::instant::Duration::from_millis(500)).await;
+ count += 1;
+ println!("Current count is {}", count);
+ }
+ }
+ });
+}
+```
+
+The task will run in the background until it is completed.
+
+> Note: `spawn` will always spawn a *new* future. You probably want to call it from a hook initializer instead of the main body of your component.
+
+When bringing lots of values into your task, we provide the `for_async!` macro which will can `for_async` on all values passed in. For types that implement `ToOwned`, `for_async!` will simply call `ToOwned` for that value.
+
+```rust
+fn App(cx: Scope) -> Element {
+ let mut age = use_state(&cx, || 0);
+ let mut name = use_state(&cx, || "Bob");
+ let mut description = use_state(&cx, || "asd".to_string());
+
+ let taskid = cx.spawn({
+ for_async![count, name, description]
+ async { /* code that uses count/name/description */ }
+ });
+}
+```
+
+## Details of Tasks
+
+Calling `spawn` is *not* a hook and will *always* generate a new task. Make sure to only spawn tasks when you want to. You should *probably* not call `spawn` in the main body of your component, since a new task will be spawned on every render.
+
+## Spawning Tokio Tasks (for multithreaded use cases)
+
+Sometimes, you might want to spawn a background task that needs multiple threads or talk to hardware that might block your app code. In these cases, we can can directly spawn a `tokio task` from our future. For Dioxus-Desktop, your task will be spawned onto Tokio's Multithreaded runtime:
+
+```rust
+cx.spawn({
+ tokio::spawn(async {
+ // some multithreaded work
+ }).await;
+
+ tokio::spawn_blocking(|| {
+ // some extremely blocking work
+ }).await;
+
+ tokio::spawn_local(|| {
+ // some !Send work
+ }).await;
+})
+```
+
+> Note: Tokio tasks must be `Send`. Most hooks are `Send` compatible, but if they aren't, then you can use `spawn_local` to spawn onto Dioxus-Desktop's `localset`.
+
+
diff --git a/docs/guide/src/async/index.md b/docs/guide/src/async/index.md
new file mode 100644
index 000000000..41a8b9231
--- /dev/null
+++ b/docs/guide/src/async/index.md
@@ -0,0 +1,19 @@
+# Working with Async
+
+Not all apps you'll build can be self-contained with synchronous code. You'll often need to interact with file systems, network interfaces, hardware, or timers.
+
+So far, we've only talked about building apps with synchronous code, so this chapter will focus integrating asynchronous code into your app.
+
+
+## The Runtime
+
+By default, Dioxus-Desktop ships with the `Tokio` runtime and automatically sets everything up for you.
+
+
+
+## Send/Sync
+Writing apps that deal with Send/Sync can be frustrating at times. Under the hood, Dioxus is not currently thread-safe, so any async code you write does *not* need to be `Send/Sync`. That means Cell/Rc/RefCell are all fair game.
+
+
+
+All async code in your app is polled on a `LocalSet`, so any async code we w
diff --git a/docs/guide/src/concepts/async.md b/docs/guide/src/concepts/async.md
deleted file mode 100644
index f6c3d103d..000000000
--- a/docs/guide/src/concepts/async.md
+++ /dev/null
@@ -1 +0,0 @@
-# Working with Async
diff --git a/docs/guide/src/concepts/asynccallbacks.md b/docs/guide/src/concepts/asynccallbacks.md
deleted file mode 100644
index 6fcf65b66..000000000
--- a/docs/guide/src/concepts/asynccallbacks.md
+++ /dev/null
@@ -1 +0,0 @@
-# Async Callbacks
diff --git a/docs/guide/src/concepts/asynctasks.md b/docs/guide/src/concepts/asynctasks.md
deleted file mode 100644
index a26beca9a..000000000
--- a/docs/guide/src/concepts/asynctasks.md
+++ /dev/null
@@ -1 +0,0 @@
-# Tasks
diff --git a/docs/guide/src/concepts/bundline.md b/docs/guide/src/concepts/bundline.md
deleted file mode 100644
index 963cf5490..000000000
--- a/docs/guide/src/concepts/bundline.md
+++ /dev/null
@@ -1 +0,0 @@
-# Bundling and Distributing
diff --git a/docs/guide/src/concepts/custom_elements.md b/docs/guide/src/concepts/custom_elements.md
deleted file mode 100644
index 3b24ad9fe..000000000
--- a/docs/guide/src/concepts/custom_elements.md
+++ /dev/null
@@ -1 +0,0 @@
-# Custom Elements
diff --git a/docs/guide/src/concepts/custom_renderer.md b/docs/guide/src/concepts/custom_renderer.md
deleted file mode 100644
index 577b4f5d3..000000000
--- a/docs/guide/src/concepts/custom_renderer.md
+++ /dev/null
@@ -1 +0,0 @@
-# Custom Renderer
diff --git a/docs/guide/src/concepts/effects.md b/docs/guide/src/concepts/effects.md
deleted file mode 100644
index 7b8ba3e3f..000000000
--- a/docs/guide/src/concepts/effects.md
+++ /dev/null
@@ -1 +0,0 @@
-# Effects
diff --git a/docs/guide/src/concepts/event_handlers.md b/docs/guide/src/concepts/event_handlers.md
deleted file mode 100644
index d10fddd09..000000000
--- a/docs/guide/src/concepts/event_handlers.md
+++ /dev/null
@@ -1,11 +0,0 @@
-# Handling Events
-
-In the overview for this section, we mentioned how we can modify the state of our component by responding to user-generated events inside of event listeners.
-
-In this section, we'll talk more about event listeners:
-- What events are available
-- Handling event data
-- Mutability in event listeners
-
-## Event Listeners
-
diff --git a/docs/guide/src/concepts/managing_state.md b/docs/guide/src/concepts/managing_state.md
deleted file mode 100644
index 931d66c2f..000000000
--- a/docs/guide/src/concepts/managing_state.md
+++ /dev/null
@@ -1 +0,0 @@
-# Managing State
diff --git a/docs/guide/src/concepts/memoization.md b/docs/guide/src/concepts/memoization.md
deleted file mode 100644
index 3ed477f68..000000000
--- a/docs/guide/src/concepts/memoization.md
+++ /dev/null
@@ -1 +0,0 @@
-# Memoization
diff --git a/docs/guide/src/concepts/server_side_components.md b/docs/guide/src/concepts/server_side_components.md
deleted file mode 100644
index a8c4f6182..000000000
--- a/docs/guide/src/concepts/server_side_components.md
+++ /dev/null
@@ -1 +0,0 @@
-# Server-side components
diff --git a/docs/guide/src/concepts/suspense.md b/docs/guide/src/concepts/suspense.md
deleted file mode 100644
index 6c81fa992..000000000
--- a/docs/guide/src/concepts/suspense.md
+++ /dev/null
@@ -1,54 +0,0 @@
-# Suspense
-
-Suspense in Dioxus is enabled through placeholder nodes.
-
-just a component that renders nodes into the placeholder after the future is finished?
-
-in react, suspense is just completely pausing diffing while a
-
-```rust
-let n = use_suspense(cx || {
- cx.render(rsx!{
- Suspense {
- prom: fut,
- callback: || {}
- }
- })
-})
-
-suspense () {
- let value = use_state();
- if first_render {
- push_task({
- value.set(fut.await);
- });
- } else {
- callback(value)
- }
-}
-
-
-let name = fetch_name().await;
-
-
-function ProfileDetails() {
- // Try to read user info, although it might not have loaded yet
- const user = resource.read();
- return
{user.name}
;
-}
-
-
-fn ProfileDteails() {
- let user = resource.suspend(cx, |l| rsx!("{l}"));
-
- // waits for the resource to be ready and updates the placeholder with the tree
- let name = resource.suspend_with(cx, |val| rsx!( div { "hello" "{user}" } ));
-
- cx.render(rsx!(
- div {
- {user}
- {name}
- }
- ))
-}
-```
diff --git a/docs/guide/src/concepts/usestate.md b/docs/guide/src/concepts/usestate.md
deleted file mode 100644
index 86a6b2efc..000000000
--- a/docs/guide/src/concepts/usestate.md
+++ /dev/null
@@ -1 +0,0 @@
-# Fundamental Hooks and use_hook
diff --git a/docs/guide/src/concepts/component_children.md b/docs/guide/src/elements/component_children.md
similarity index 86%
rename from docs/guide/src/concepts/component_children.md
rename to docs/guide/src/elements/component_children.md
index ed482b007..d8469241a 100644
--- a/docs/guide/src/concepts/component_children.md
+++ b/docs/guide/src/elements/component_children.md
@@ -30,7 +30,7 @@ struct ClickableProps<'a> {
title: &'a str
}
-fn clickable(cx: Scope) -> Element {
+fn Clickable(cx: Scope) -> Element {
cx.render(rsx!(
a {
href: "{cx.props.href}"
@@ -44,10 +44,10 @@ And then use it in our code like so:
```rust
rsx!(
- clickable(
+ Clickable {
href: "https://google.com"
title: "Link to Google"
- )
+ }
)
```
@@ -64,11 +64,11 @@ struct ClickableProps<'a> {
body: Element<'a>
}
-fn clickable(cx: Scope) -> Element {
+fn Clickable(cx: Scope) -> Element {
cx.render(rsx!(
a {
- href: "{cx.props.href}"
- {&cx.props.body}
+ href: "{cx.props.href}",
+ &cx.props.body
}
))
}
@@ -78,15 +78,17 @@ Then, at the call site, we can render some nodes and pass them in:
```rust
rsx!(
- clickable(
+ Clickable {
href: "https://google.com"
body: cx.render(rsx!(
- img { src: "https://www.google.com/logos/doodles/2021/seasonal-holidays-2021-6753651837109324-6752733080595603-cst.gif" }
+ img { src: "https://www.google.com/logos/doodles/..." }
))
- )
+ }
)
```
+## Auto Conversion of the `Children` field
+
This pattern can become tedious in some instances, so Dioxus actually performs an implicit conversion of any `rsx` calls inside components into `Elements` at the `children` field. This means you must explicitly declare if a component can take children.
```rust
@@ -99,26 +101,26 @@ struct ClickableProps<'a> {
fn clickable(cx: Scope) -> Element {
cx.render(rsx!(
a {
- href: "{cx.props.href}"
- {&cx.props.children}
+ href: "{cx.props.href}",
+ &cx.props.children
}
))
}
```
-And to call `clickable`:
+Now, whenever we use `Clickable` in another component, we don't need to call `render` on child nodes - it will happen automatically!
```rust
rsx!(
- clickable(
+ Clickable {
href: "https://google.com"
- img { src: "https://www.google.com/logos/doodles/2021/seasonal-holidays-2021-6753651837109324-6752733080595603-cst.gif" }
- )
+ img { src: "https://www.google.com/logos/doodles/...." }
+ }
)
```
> Note: Passing children into components will break any memoization due to the associated lifetime.
-While technically allowed, it's an antipattern to pass children more than once in a component and will probably break your app significantly.
+While technically allowed, it's an antipattern to pass children more than once in a component and will probably cause your app to crash.
However, because the `Element` is transparently a `VNode`, we can actually match on it to extract the nodes themselves, in case we are expecting a specific format:
@@ -142,10 +144,10 @@ In the cases where you need to pass arbitrary element properties into a componen
```rust
rsx!(
- clickable(
+ Clickable {
"class": "blue-button",
"style": "background: red;"
- )
+ }
)
```
@@ -161,7 +163,7 @@ struct ClickableProps<'a> {
fn clickable(cx: Scope) -> Element {
cx.render(rsx!(
a {
- ..{cx.props.attributes},
+ ..cx.props.attributes,
"Any link, anywhere"
}
))
@@ -195,9 +197,9 @@ Then, we can attach a listener at the call site:
```rust
rsx!(
- clickable(
+ Clickable {
onclick: move |_| log::info!("Clicked"),
- )
+ }
)
```
diff --git a/docs/guide/src/concepts/components.md b/docs/guide/src/elements/components.md
similarity index 87%
rename from docs/guide/src/concepts/components.md
rename to docs/guide/src/elements/components.md
index a396d4b4b..8567e9270 100644
--- a/docs/guide/src/concepts/components.md
+++ b/docs/guide/src/elements/components.md
@@ -110,7 +110,7 @@ When declaring a component in `rsx!`, we can pass in properties using the tradit
Let's take a look at the `VoteButton` component. For now, we won't include any interactivity - just the rendering the score and buttons to the screen.
-Most of your Components will look exactly like this: a Props struct and a render function. Every component must take a tuple of `Scope` and `&Props` and return an `Element`.
+Most of your Components will look exactly like this: a Props struct and a render function. Every component must take a `Scope` generic over some `Props` and return an `Element`.
As covered before, we'll build our User Interface with the `rsx!` macro and HTML tags. However, with components, we must actually "render" our HTML markup. Calling `cx.render` converts our "lazy" `rsx!` structure into an `Element`.
@@ -145,7 +145,7 @@ struct TitleCardProps<'a> {
title: &'a str,
}
-fn TitleCard(cx: Scope) -> Element {
+fn TitleCard<'a>(cx: Scope<'a, TitleCardProps<'a>>) -> Element {
cx.render(rsx!{
h1 { "{cx.props.title}" }
})
@@ -156,6 +156,42 @@ For users of React: Dioxus knows *not* to memoize components that borrow propert
This means that during the render process, a newer version of `TitleCardProps` will never be compared with a previous version, saving some clock cycles.
+## The inline_props macro
+
+Yes - *another* macro! However, this one is entirely optional.
+
+For internal components, we provide the `inline_props` macro, which will let you embed your `Props` definition right into the function arguments of your component.
+
+Our title card above would be transformed from:
+
+```rust
+#[derive(Props, PartialEq)]
+struct TitleCardProps {
+ title: String,
+}
+
+fn TitleCard(cx: Scope) -> Element {
+ cx.render(rsx!{
+ h1 { "{cx.props.title}" }
+ })
+}
+```
+
+to:
+
+```rust
+#[inline_props]
+fn TitleCard(cx: Scope, title: String) -> Element {
+ cx.render(rsx!{
+ h1 { "{title}" }
+ })
+}
+```
+
+Again, this macro is optional and should not be used by library authors since you have less fine-grained control over documentation and optionality.
+
+However, it's great for quickly throwing together an app without dealing with *any* extra boilerplate.
+
## The `Scope` object
Though very similar to React, Dioxus is different in a few ways. Most notably, React components will not have a `Scope` parameter in the component declaration.
diff --git a/docs/guide/src/concepts/conditional_rendering.md b/docs/guide/src/elements/conditional_rendering.md
similarity index 88%
rename from docs/guide/src/concepts/conditional_rendering.md
rename to docs/guide/src/elements/conditional_rendering.md
index 5bb75c490..b23df6000 100644
--- a/docs/guide/src/concepts/conditional_rendering.md
+++ b/docs/guide/src/elements/conditional_rendering.md
@@ -101,14 +101,14 @@ fn App(cx: Scope)-> Element {
## Nesting RSX
-By looking at other examples, you might have noticed that it's possible to include `rsx!` calls inside other `rsx!` calls. With the curly-brace syntax, we can include anything in our `rsx!` that implements `IntoVnodeList`: a marker trait for iterators that produce Elements. `rsx!` itself implements this trait, so we can include it directly:
+By looking at other examples, you might have noticed that it's possible to include `rsx!` calls inside other `rsx!` calls. We can include anything in our `rsx!` that implements `IntoVnodeList`: a marker trait for iterators that produce Elements. `rsx!` itself implements this trait, so we can include it directly:
```rust
rsx!(
div {
- {rsx!(
+ rsx!(
"more rsx!"
- )}
+ )
}
)
```
@@ -120,7 +120,7 @@ let title = rsx!( "more rsx!" );
rsx!(
div {
- {title}
+ title
}
)
```
@@ -136,7 +136,7 @@ let screen = match logged_in {
cx.render(rsx!{
Navbar {}
- {screen}
+ screen,
Footer {}
})
```
@@ -152,9 +152,9 @@ By default, Rust lets you convert any `boolean` into any other type by calling `
let show_title = true;
rsx!(
div {
- {show_title.and_then(|| rsx!{
+ show_title.and_then(|| rsx!{
"This is the title"
- })}
+ })
}
)
```
@@ -164,20 +164,23 @@ We can use this pattern for many things, including options:
let user_name = Some("bob");
rsx!(
div {
- {user_name.map(|name| rsx!("Hello {name}"))}
+ user_name.map(|name| rsx!("Hello {name}"))
}
)
```
## Rendering Nothing
-Sometimes, you don't want your component to return anything at all. In these cases, we can pass "None" into our bracket. However, Rust is not able to infer that our `None` corresponds to `Element` so we need to cast it appropriately:
+Sometimes, you don't want your component to return anything at all. Under the hood, the `Element` type is just an alias for `Option`, so you can simply return `None`.
+
+This can be helpful in certain patterns where you need to perform some logical side-effects but don't want to render anything.
```rust
-rsx!(cx, {None as Element} )
+fn demo(cx: Scope) -> Element {
+ None
+}
```
-
## Moving Forward:
In this chapter, we learned how to render different Elements from a Component depending on a condition. This is a very powerful building block to assemble complex User Interfaces!
diff --git a/docs/guide/src/concepts/exporting_components.md b/docs/guide/src/elements/exporting_components.md
similarity index 86%
rename from docs/guide/src/concepts/exporting_components.md
rename to docs/guide/src/elements/exporting_components.md
index 59fea7046..16d6f1002 100644
--- a/docs/guide/src/concepts/exporting_components.md
+++ b/docs/guide/src/elements/exporting_components.md
@@ -25,27 +25,27 @@ fn main() {
dioxus::desktop::launch(App);
}
-fn App((cx, props): Component) -> Element {}
+fn App(Scope) -> Element {}
#[derive(PartialEq, Props)]
struct PostProps{}
-fn Post((cx, props): Component) -> Element {}
+fn Post(Scope) -> Element {}
#[derive(PartialEq, Props)]
struct VoteButtonsProps {}
-fn VoteButtons((cx, props): Component) -> Element {}
+fn VoteButtons(Scope) -> Element {}
#[derive(PartialEq, Props)]
struct TitleCardProps {}
-fn TitleCard((cx, props): Component) -> Element {}
+fn TitleCard(Scope) -> Element {}
#[derive(PartialEq, Props)]
struct MetaCardProps {}
-fn MetaCard((cx, props): Component) -> Element {}
+fn MetaCard(Scope) -> Element {}
#[derive(PartialEq, Props)]
struct ActionCardProps {}
-fn ActionCard((cx, props): Component) -> Element {}
+fn ActionCard(Scope) -> Element {}
```
That's a lot of components for one file! We've successfully refactored our app into components, but we should probably start breaking it up into a file for each component.
@@ -61,7 +61,7 @@ use dioxus::prelude::*;
#[derive(PartialEq, Props)]
struct ActionCardProps {}
-fn ActionCard((cx, props): Component) -> Element {}
+fn ActionCard(Scope) -> Element {}
```
We should also create a `mod.rs` file in the `post` folder so we can use it from our `main.rs`. Our `Post` component and its props will go into this file.
@@ -92,7 +92,7 @@ fn main() {
mod post;
-fn App((cx, props): Component) -> Element {
+fn App(Scope) -> Element {
cx.render(rsx!{
post::Post {
id: Uuid::new_v4(),
@@ -116,7 +116,7 @@ use dioxus::prelude::*;
#[derive(PartialEq, Props)]
pub struct PostProps {}
-pub fn Post((cx, props): Component) -> Element {}
+pub fn Post(Scope) -> Element {}
```
While we're here, we also need to make sure each of our subcomponents are included as modules and exported.
@@ -142,7 +142,7 @@ pub struct PostProps {
original_poster: String
}
-pub fn Post((cx, props): Component) -> Element {
+pub fn Post(Scope) -> Element {
cx.render(rsx!{
div { class: "post-container"
vote::VoteButtons {
@@ -191,7 +191,7 @@ fn main() {
mod post;
-fn App((cx, props): Component) -> Element {
+fn App(Scope) -> Element {
cx.render(rsx!{
post::Post {
id: Uuid::new_v4(),
@@ -227,7 +227,7 @@ pub struct PostProps {
original_poster: String
}
-pub fn Post((cx, props): Component) -> Element {
+pub fn Post(Scope) -> Element {
cx.render(rsx!{
div { class: "post-container"
vote::VoteButtons {
@@ -255,7 +255,7 @@ use dioxus::prelude::*;
#[derive(PartialEq, Props)]
pub struct VoteButtonsProps {}
-pub fn VoteButtons((cx, props): Component) -> Element {}
+pub fn VoteButtons(Scope) -> Element {}
```
```rust
@@ -264,7 +264,7 @@ use dioxus::prelude::*;
#[derive(PartialEq, Props)]
pub struct TitleCardProps {}
-pub fn TitleCard((cx, props): Component) -> Element {}
+pub fn TitleCard(Scope) -> Element {}
```
```rust
@@ -273,7 +273,7 @@ use dioxus::prelude::*;
#[derive(PartialEq, Props)]
pub struct MetaCardProps {}
-pub fn MetaCard((cx, props): Component) -> Element {}
+pub fn MetaCard(Scope) -> Element {}
```
```rust
@@ -282,7 +282,7 @@ use dioxus::prelude::*;
#[derive(PartialEq, Props)]
pub struct ActionCardProps {}
-pub fn ActionCard((cx, props): Component) -> Element {}
+pub fn ActionCard(Scope) -> Element {}
```
## Moving forward
diff --git a/docs/guide/src/elements/index.md b/docs/guide/src/elements/index.md
new file mode 100644
index 000000000..13a61a868
--- /dev/null
+++ b/docs/guide/src/elements/index.md
@@ -0,0 +1,73 @@
+# Core Topics
+
+In this chapter, we'll cover some core topics on how Dioxus works and how to best leverage the features to build a beautiful, reactive app.
+
+At a very high level, Dioxus is simply a Rust framework for _declaring_ user interfaces and _reacting_ to changes.
+
+1) We declare what we want our user interface to look like given a state using Rust-based logic and control flow.
+2) We declare how we want our state to change when the user triggers an event.
+
+## Declarative UI
+
+Dioxus is a *declarative* framework. This means that instead of manually writing calls to "create element" and "set element background to red," we simply *declare* what we want the element to look like and let Dioxus handle the differences.
+
+Let's pretend that we have a stoplight we need to control - it has a color state with red, yellow, and green as options.
+
+
+Using an imperative approach, we would have to manually declare each element and then handlers for advancing the stoplight.
+
+```rust
+let container = Container::new();
+
+let green_light = Light::new().color("green").enabled(true);
+let yellow_light = Light::new().color("yellow").enabled(false);
+let red_light = Light::new().color("red").enabled(false);
+container.push(green_light);
+container.push(yellow_light);
+container.push(red_light);
+
+container.set_onclick(move |_| {
+ if red_light.enabled() {
+ red_light.set_enabled(false);
+ green_light.set_enabled(true);
+ } else if yellow_light.enabled() {
+ yellow_light.set_enabled(false);
+ red_light.set_enabled(true);
+ } else if green_light.enabled() {
+ green_light.set_enabled(false);
+ yellow_light.set_enabled(true);
+ }
+});
+```
+
+As the UI grows in scale, our logic to keep each element in the proper state would grow exponentially. This can become very unwieldy and lead to out-of-sync UIs that harm user experience.
+
+Instead, with Dioxus, we *declare* what we want our UI to look like:
+
+```rust
+let mut state = use_state(&cx, || "red");
+
+cx.render(rsx!(
+ Container {
+ Light { color: "red", enabled: state == "red", }
+ Light { color: "yellow", enabled: state == "yellow", }
+ Light { color: "green", enabled: state == "green", }
+
+ onclick: move |_| {
+ state.set(match *state {
+ "green" => "yellow",
+ "yellow" => "red",
+ "red" => "green",
+ })
+ }
+ }
+))
+```
+
+Remember: this concept is not new! Many frameworks are declarative - with React being the most popular. Declarative frameworks tend to be much more enjoyable to work with than imperative frameworks.
+
+Here's some reading about declaring UI in React:
+
+- [https://stackoverflow.com/questions/33655534/difference-between-declarative-and-imperative-in-react-js](https://stackoverflow.com/questions/33655534/difference-between-declarative-and-imperative-in-react-js)
+
+- [https://medium.com/@myung.kim287/declarative-vs-imperative-251ce99c6c44](https://medium.com/@myung.kim287/declarative-vs-imperative-251ce99c6c44)
diff --git a/docs/guide/src/concepts/lists.md b/docs/guide/src/elements/lists.md
similarity index 100%
rename from docs/guide/src/concepts/lists.md
rename to docs/guide/src/elements/lists.md
diff --git a/docs/guide/src/concepts/vnodes.md b/docs/guide/src/elements/vnodes.md
similarity index 79%
rename from docs/guide/src/concepts/vnodes.md
rename to docs/guide/src/elements/vnodes.md
index 0a98c89d9..b407ed02f 100644
--- a/docs/guide/src/concepts/vnodes.md
+++ b/docs/guide/src/elements/vnodes.md
@@ -87,9 +87,15 @@ Unfortunately, you cannot drop in arbitrary expressions directly into the string
rsx!( {[format_args!("Hello {}", if enabled { "Jack" } else { "Bob" } )]} )
```
+Alternatively, `&str` can be included directly, though it must be inside of an array:
+
+```rust
+rsx!( "Hello ", [if enabled { "Jack" } else { "Bob" }] )
+```
+
This is different from React's way of generating arbitrary markup but fits within idiomatic Rust.
-Typically, with Dioxus, you'll just want to compute your substrings outside of the `rsx!` call:
+Typically, with Dioxus, you'll just want to compute your substrings outside of the `rsx!` call and leverage the f-string formatting:
```rust
let name = if enabled { "Jack" } else { "Bob" };
@@ -100,7 +106,7 @@ rsx! ( "hello {name}" )
Every Element in your User Interface will have some sort of properties that the renderer will use when drawing to the screen. These might inform the renderer if the component should be hidden, what its background color should be, or to give it a specific name or ID.
-To do this, we use the familiar struct-style syntax that Rust provides. Commas are optional:
+To do this, we use the familiar struct-style syntax that Rust provides:
```rust
rsx!(
@@ -127,19 +133,29 @@ rsx!(
)
```
-Note: the name of the custom attribute must match exactly what you want the renderer to output. All attributes defined as methods in `dioxus-html` follow the snake_case naming convention. However, they internally translate their snake_case convention to HTML's camelCase convention. When using custom attributes, make sure the name of the attribute exactly matches what the renderer is expecting.
+> Note: the name of the custom attribute must match exactly what you want the renderer to output. All attributes defined as methods in `dioxus-html` follow the snake_case naming convention. However, they internally translate their snake_case convention to HTML's camelCase convention. When using custom attributes, make sure the name of the attribute **exactly** matches what the renderer is expecting.
+
+All element attributes must occur *before* child elements. The `rsx!` macro will throw an error if your child elements come before any of your attributes. If you don't see the error, try editing your Rust-Analyzer IDE setting to ignore macro-errors. This is a temporary workaround because Rust-Analyzer currently throws *two* errors instead of just the one we care about.
+
+```rust
+// settings.json
+{
+ "rust-analyzer.diagnostics.disabled": [
+ "macro-error"
+ ],
+}
+```
## Listeners
Listeners are a special type of Attribute that only accept functions. Listeners let us attach functionality to our Elements by running a provided closure whenever the specified Listener is triggered.
-We'll cover listeners in more depth in the Listeners chapter, but for now, just know that every listener must start with the `on` keyword and can accept either a closure or an expression wrapped in curly braces.
+We'll cover listeners in more depth in the Listeners chapter, but for now, just know that every listener must start with the `on` keyword and accepts closures.
```rust
rsx!(
div {
- onclick: move |_| {}
- onmouseover: {handler},
+ onclick: move |_| log::debug!("div clicked!"),
}
)
```
diff --git a/docs/guide/src/final.md b/docs/guide/src/final.md
new file mode 100644
index 000000000..68706f1aa
--- /dev/null
+++ b/docs/guide/src/final.md
@@ -0,0 +1,49 @@
+# Congrats!
+
+Congrats! You've made it through the `learning Dioxus` book. Throughout this tutorial, you've learned a ton:
+
+- How to build User Interfaces with Elements
+- How to compose Element groups together as Components
+- How to handle user input with event listeners
+- How to manage local and global state
+- How to work with async using tasks, coroutines, and suspense
+- How to build custom hooks and handlers
+
+With any luck, you followed through the "Putting it All Together" mini guide and have your very own dog search engine app!
+
+# Next Steps and Advanced Topics
+
+Continuing on your journey with Dioxus, you can try a number of things:
+
+- Build a simple TUI app
+- Publish your search engine app
+- Deploy a WASM app to GitHub
+- Design a custom hook
+- Contribute to the ecosystem!
+
+There are a number of advanced topics we glossed over:
+
+- The underlying NodeFactory API
+- Static elements and templates
+- Anti-patterns
+- Bundling/distribution
+- Working with wasm apps
+
+# Contributing to the ecosystem
+
+Dioxus is still quite young and could use your help!
+
+The core team is actively working on:
+
+- Declarative window management (via Tauri) for Desktop apps
+- Portals for Dioxus Core
+- Mobile support
+- Integration with 3D renderers
+- Better async story (suspense, error handling)
+- Global state management
+- Web development server
+- LiveView
+- Broader platform support (iOS/Android/TV/embedded)
+
+If there's something specifically interesting to you, don't be afraid to jump in!
+
diff --git a/docs/guide/src/final/index.md b/docs/guide/src/final/index.md
deleted file mode 100644
index 32961beed..000000000
--- a/docs/guide/src/final/index.md
+++ /dev/null
@@ -1,10 +0,0 @@
-# Next Steps and Advanced Topics
-
-
-Congrats! You've made it through the `learning Dioxus` book. Throughout this tutorial, you've learned a ton:
-
-- How to build User Interfaces with Elements
-- How to compose Element groups together as Components
-- How to handle user input with event listeners
-- How to manage global and local state
-- How to work with async using tasks, coroutines, and suspense
diff --git a/docs/guide/src/hello_world.md b/docs/guide/src/hello_world.md
index fd5e90aa5..fbe6153f5 100644
--- a/docs/guide/src/hello_world.md
+++ b/docs/guide/src/hello_world.md
@@ -1,4 +1,4 @@
-# Hello, World desktop app
+# "Hello, World" desktop app
Let's put together a simple "hello world" desktop application to get acquainted with Dioxus.
@@ -138,11 +138,13 @@ Writing `fn App(cx: Scope) -> Element {` might become tedious. Rust will also le
static App: Component = |cx| cx.render(rsx!(div { "Hello, world!" }));
```
-### What is this `Context` object?
+### What is this `Scope` object?
-Coming from React, the `Context` object might be confusing. In React, you'll want to store data between renders with hooks. However, hooks rely on global variables which make them difficult to integrate in multi-tenant systems like server-rendering.
+Coming from React, the `Scope` object might be confusing. In React, you'll want to store data between renders with hooks. However, hooks rely on global variables which make them difficult to integrate in multi-tenant systems like server-rendering.
-In Dioxus, you are given an explicit `Context` object to control how the component renders and stores data. The `Context` object provides a handful of useful APIs for features like suspense, rendering, and more.
+In Dioxus, you are given an explicit `Scope` object to control how the component renders and stores data. The `Scope` object provides a handful of useful APIs for features like suspense, rendering, and more.
+
+For now, just know that `Scope` lets you store state with hooks and render elements with `cx.render`.
## Moving on
diff --git a/docs/guide/src/images/publish.png b/docs/guide/src/images/publish.png
new file mode 100644
index 000000000..b17d3bb06
Binary files /dev/null and b/docs/guide/src/images/publish.png differ
diff --git a/docs/guide/src/interactivity/event_handlers.md b/docs/guide/src/interactivity/event_handlers.md
new file mode 100644
index 000000000..ee4560b1e
--- /dev/null
+++ b/docs/guide/src/interactivity/event_handlers.md
@@ -0,0 +1 @@
+# Event handlers
diff --git a/docs/guide/src/concepts/hooks.md b/docs/guide/src/interactivity/hooks.md
similarity index 89%
rename from docs/guide/src/concepts/hooks.md
rename to docs/guide/src/interactivity/hooks.md
index 28f027db1..029e2db15 100644
--- a/docs/guide/src/concepts/hooks.md
+++ b/docs/guide/src/interactivity/hooks.md
@@ -19,16 +19,16 @@ Broadly, there are two types of GUI structures:
Typically, immediate-mode GUIs are simpler to write but can slow down as more features, like styling, are added.
-Many GUIs today - including Dioxus - are written in *Retained mode* - your code changes the data of the user interface but the renderer is responsible for actually drawing to the screen. In these cases, our GUI's state sticks around as the UI is rendered.
+Many GUIs today are written in *Retained mode* - your code changes the data of the user interface but the renderer is responsible for actually drawing to the screen. In these cases, our GUI's state sticks around as the UI is rendered. To help accommodate retained mode GUIs, like the web browser, Dioxus provides a mechanism to keep state around.
-Dioxus, following in the footsteps of React, provides a "Reactive" model for you to design your UI. This model emphasizes one-way data flow and encapsulation. Essentially, your UI code should be as predictable as possible.
+> Note: Even though hooks are accessible, you should still prefer to one-way data flow and encapsulation. Your UI code should be as predictable as possible. Dioxus is plenty fast, even for the largest apps.
## Mechanics of Hooks
In order to have state stick around between renders, Dioxus provides the `hook` through the `use_hook` API. This gives us a mutable reference to data returned from the initialization function.
```rust
fn example(cx: Scope) -> Element {
- let name: &mut String = cx.use_hook(|| "John Doe".to_string(), |hook| hook);
+ let name: &mut String = cx.use_hook(|| "John Doe".to_string());
//
}
@@ -38,7 +38,7 @@ We can even modify this value directly from an event handler:
```rust
fn example(cx: Scope) -> Element {
- let name: &mut String = cx.use_hook(|| "John Doe".to_string(), |hook| hook);
+ let name: &mut String = cx.use_hook(|| "John Doe".to_string());
cx.render(rsx!(
button {
@@ -52,9 +52,9 @@ Mechanically, each call to `use_hook` provides us with `&mut T` for a new value.
```rust
fn example(cx: Scope) -> Element {
- let name: &mut String = cx.use_hook(|| "John Doe".to_string(), |hook| hook);
- let age: &mut u32 = cx.use_hook(|| 10, |hook| hook);
- let friends: &mut Vec = cx.use_hook(|| vec!["Jane Doe".to_string()], |hook| hook);
+ let name: &mut String = cx.use_hook(|| "John Doe".to_string());
+ let age: &mut u32 = cx.use_hook(|| 10);
+ let friends: &mut Vec = cx.use_hook(|| vec!["Jane Doe".to_string()]);
//
}
@@ -84,7 +84,7 @@ Consider when we try to pass our `&mut String` into two different handlers:
```rust
fn example(cx: Scope) -> Element {
- let name: &mut String = cx.use_hook(|| "John Doe".to_string(), |hook| hook);
+ let name: &mut String = cx.use_hook(|| "John Doe".to_string());
cx.render(rsx!(
button { onclick: move |_| name.push_str("yes"), }
@@ -112,7 +112,7 @@ This example uses the `Cell` type to let us replace the value through interior m
```rust
fn example(cx: Scope) -> Element {
- let name: &Cell<&'static str> = cx.use_hook(|| "John Doe", |hook| hook);
+ let name: &Cell<&'static str> = cx.use_hook(|| Cell::new("John Doe"));
cx.render(rsx!(
button { onclick: move |_| name.set("John"), }
@@ -180,6 +180,8 @@ By default, we bundle a handful of hooks in the Dioxus-Hooks package. Feel free
- [use_callback](https://docs.rs/dioxus_hooks/use_callback) - store a callback that implements PartialEq for memoization
- [use_provide_context](https://docs.rs/dioxus_hooks/use_provide_context) - expose state to descendent components
- [use_context](https://docs.rs/dioxus_hooks/use_context) - consume state provided by `use_provide_context`
+
+For a more in-depth guide to building new hooks, checkout out the advanced hook building guide in the reference.
## Wrapping up
diff --git a/docs/guide/src/concepts/interactivity.md b/docs/guide/src/interactivity/index.md
similarity index 99%
rename from docs/guide/src/concepts/interactivity.md
rename to docs/guide/src/interactivity/index.md
index 26720cc1e..9e23862f1 100644
--- a/docs/guide/src/concepts/interactivity.md
+++ b/docs/guide/src/interactivity/index.md
@@ -139,7 +139,7 @@ fn App(cx: Scope)-> Element {
let mut sec_elapsed = use_state(&cx, || 0);
use_future(&cx, || {
- let mut sec_elapsed = sec_elapsed.to_owned();
+ let mut sec_elapsed = sec_elapsed.for_async();
async move {
loop {
TimeoutFuture::from_ms(1000).await;
diff --git a/docs/guide/src/concepts/lifecycles.md b/docs/guide/src/interactivity/lifecycles.md
similarity index 100%
rename from docs/guide/src/concepts/lifecycles.md
rename to docs/guide/src/interactivity/lifecycles.md
diff --git a/docs/guide/src/concepts/user_input.md b/docs/guide/src/interactivity/user_input.md
similarity index 100%
rename from docs/guide/src/concepts/user_input.md
rename to docs/guide/src/interactivity/user_input.md
diff --git a/docs/guide/src/setup.md b/docs/guide/src/setup.md
index c654179c9..11855d487 100644
--- a/docs/guide/src/setup.md
+++ b/docs/guide/src/setup.md
@@ -58,7 +58,7 @@ That's it! We won't need to touch NPM/WebPack/Babel/Parcel, etc. However, you _c
## Rust Knowledge
-With Rust, things like benchmarking, testing, and documentation are included in the language. We strongly recommend going through the official Rust book _completely_. However, our hope is that a Dioxus app can serve as a great first project. With Dioxus you'll learn about:
+With Rust, things like benchmarking, testing, and documentation are included in the language. We strongly recommend going through the official Rust book _completely_. However, our hope is that a Dioxus app can serve as a great first Rust project. With Dioxus you'll learn about:
- Error handling
- Structs, Functions, Enums
diff --git a/docs/guide/src/concepts/errorhandling.md b/docs/guide/src/state/errorhandling.md
similarity index 100%
rename from docs/guide/src/concepts/errorhandling.md
rename to docs/guide/src/state/errorhandling.md
diff --git a/docs/guide/src/state/index.md b/docs/guide/src/state/index.md
new file mode 100644
index 000000000..9b3e054fc
--- /dev/null
+++ b/docs/guide/src/state/index.md
@@ -0,0 +1,8 @@
+# Managing State
+
+Every app you'll build with Dioxus will have some sort of state that needs to be maintained and updated as your users interact with it. However, managing state can be particular challenging at times, and is frequently the source of bugs in many GUI frameworks.
+
+In this chapter, we'll cover the various ways to manage state, the appropriate terminology, various patterns, and then take an overview
+
+
+## Terminology
diff --git a/docs/guide/src/state/liftingstate.md b/docs/guide/src/state/liftingstate.md
new file mode 100644
index 000000000..3e906a0ca
--- /dev/null
+++ b/docs/guide/src/state/liftingstate.md
@@ -0,0 +1 @@
+# Lifting State
diff --git a/docs/guide/src/state/localstate.md b/docs/guide/src/state/localstate.md
new file mode 100644
index 000000000..29085ac45
--- /dev/null
+++ b/docs/guide/src/state/localstate.md
@@ -0,0 +1 @@
+# Local State
diff --git a/docs/guide/src/concepts/sharedstate.md b/docs/guide/src/state/sharedstate.md
similarity index 100%
rename from docs/guide/src/concepts/sharedstate.md
rename to docs/guide/src/state/sharedstate.md
diff --git a/docs/guide/src/tutorial/publishing.md b/docs/guide/src/tutorial/publishing.md
index 96670f1c8..89bf39edf 100644
--- a/docs/guide/src/tutorial/publishing.md
+++ b/docs/guide/src/tutorial/publishing.md
@@ -1 +1,65 @@
# Publishing
+
+Congrats! You've made your first Dioxus app that actually does some pretty cool stuff. This app uses your operating system's WebView library, so it's portable to be distributed for other platforms.
+
+In this section, we'll cover how to bundle your app for macOS, Windows, and Linux.
+
+
+
+## Install `cargo-bundle`
+
+
+The first thing we'll do is install [`cargo-bundle`](https://github.com/burtonageo/cargo-bundle). This extension to cargo will make it very easy to package our app for the various platforms.
+
+According to the `cargo-bundle` github page,
+
+
+
+*"cargo-bundle is a tool used to generate installers or app bundles for GUI executables built with cargo. It can create .app bundles for Mac OS X and iOS, .deb packages for Linux, and .msi installers for Windows (note however that iOS and Windows support is still experimental). Support for creating .rpm packages (for Linux) and .apk packages (for Android) is still pending."*
+
+
+To install, simply run
+
+
+`cargo install cargo-bundle`
+
+## Setting up your project
+
+
+To get a project setup for bundling, we need to add some flags to our `Cargo.toml` file.
+
+
+```toml
+[package]
+name = "example"
+# ...other fields...
+
+[package.metadata.bundle]
+name = "DogSearch"
+identifier = "com.dogs.dogsearch"
+version = "1.0.0"
+copyright = "Copyright (c) Jane Doe 2016. All rights reserved."
+category = "Developer Tool"
+short_description = "Easily search for Dog photos"
+long_description = """
+This app makes it quick and easy to browse photos of dogs from over 200 bree
+"""
+```
+
+
+## Building
+
+Following cargo-bundle's instructions, we simply `cargo-bundle --release` to produce a final app with all the optimizations and assets builtin.
+
+Once you've ran `cargo-bundle --release`, your app should be accessible in
+
+`target/release/bundle//`.
+
+For example, a macOS app would look like this:
+
+![Published App](../images/publish.png)
+
+Nice! And it's only 4.8 Mb - extremely lean!! Because Dioxus leverages your platform's native WebView, Dioxus apps are extremely memory efficient and won't waste your battery.
+
+> Note: not all CSS works the same on all platforms. Make sure to view your app's CSS on each platform - or web browser (Firefox, Chrome, Safari) before publishing.
+
diff --git a/docs/reference/src/SUMMARY.md b/docs/reference/src/SUMMARY.md
index e69de29bb..a83f3c476 100644
--- a/docs/reference/src/SUMMARY.md
+++ b/docs/reference/src/SUMMARY.md
@@ -0,0 +1,61 @@
+# Summary
+
+- [Introduction](README.md)
+
+- [Web](web/intro.md)
+ - [Getting Started](web/setup.md)
+ - [Events](concepts/errorhandling.md)
+
+- [Desktop](desktop/intro.md)
+ - [Getting Started](desktop/setup.md)
+ - [events](desktop/.md)
+
+- [Mobile](concepts/managing_state.md)
+ - [Getting Started](concepts/sharedstate.md)
+ - [Specifics](concepts/errorhandling.md)
+
+- [Working with Async](concepts/async.md)
+
+- [Code Reference]()
+ - [Code Reference]()
+ - [Code Reference]()
+ - [Code Reference]()
+ - [Code Reference]()
+
+- [Topics in Depth](depth/topics.md)
+ - [RSX](depth/rsx.md)
+ - [Components](depth/components.md)
+ - [Props](depth/props.md)
+ - [Memoization](depth/memoization.md)
+ - [Performance](depth/performance.md)
+ - [Testing](depth/testing.md)
+
+- [Advanced Guides](tutorial/advanced_guides.md)
+ - [Memoization](concepts/memoization.md)
+ - [RSX in Depth](concepts/rsx_in_depth.md)
+ - [Building Elements with NodeFactory](concepts/rsx.md)
+ - [Custom Elements](concepts/custom_elements.md)
+ - [Custom Renderer](concepts/custom_renderer.md)
+ - [Server-side components](concepts/server_side_components.md)
+ - [Bundling and Distributing](concepts/bundline.md)
+
+- [Reference Guide]()
+ - [Anti-patterns]()
+ - [Children]()
+ - [Conditional Rendering]()
+ - [Controlled Inputs]()
+ - [Custom Elements]()
+ - [Empty Components]()
+ - [Error Handling]()
+ - [Fragments]()
+ - [Global CSS]()
+ - [Inline Styles]()
+ - [Iterators]()
+ - [Listeners]()
+ - [Memoization]()
+ - [Node Refs]()
+ - [Spread Pattern]()
+ - [State Management]()
+ - [Suspense]()
+ - [task]()
+ - [Testing]()
diff --git a/examples/README.md b/examples/README.md
index eabb0f57d..2e4959e7f 100644
--- a/examples/README.md
+++ b/examples/README.md
@@ -42,14 +42,6 @@ These examples are not necessarily meant to be run, but rather serve as a refere
| [Complete rsx reference](./rsx_usage.rs) | A complete reference for all rsx! usage | ✅ |
| [Event Listeners](./listener.rs) | Attach closures to events on elements | ✅ |
-These web-specific examples must be run with `dioxus-cli` using `dioxus develop --example XYZ`
-
-| Example | What it does |
-| ------- | ------------ |
-| asd | this does |
-| asd | this does |
-
-
## Show me some examples!
@@ -64,7 +56,7 @@ Here's what a few common tasks look like in Dioxus:
Nested components with children and internal state:
```rust
-fn App(cx: Scope<()>) -> Element {
+fn App(cx: Scope) -> Element {
cx.render(rsx!( Toggle { "Toggle me" } ))
}
@@ -75,9 +67,9 @@ fn Toggle(cx: Scope) -> Element {
let mut toggled = use_state(&cx, || false);
cx.render(rsx!{
div {
- {&cx.props.children}
+ &cx.props.children
button { onclick: move |_| toggled.set(true),
- {toggled.and_then(|| "On").or_else(|| "Off")}
+ toggled.and_then(|| "On").or_else(|| "Off")
}
}
})
@@ -86,7 +78,7 @@ fn Toggle(cx: Scope) -> Element {
Controlled inputs:
```rust
-fn App(cx: Scope<()>) -> Element {
+fn App(cx: Scope) -> Element {
let value = use_state(&cx, String::new);
cx.render(rsx!(
input {
@@ -100,7 +92,7 @@ fn App(cx: Scope<()>) -> Element {
Lists and Conditional rendering:
```rust
-fn App(cx: Scope<()>) -> Element {
+fn App(cx: Scope) -> Element {
let list = (0..10).map(|i| {
rsx!(li { key: "{i}", "Value: {i}" })
});
@@ -112,8 +104,8 @@ fn App(cx: Scope<()>) -> Element {
if should_show {
cx.render(rsx!(
- {title}
- ul { {list} }
+ title,
+ ul { list }
))
} else {
None
@@ -128,7 +120,7 @@ static App: Component = |cx, _| rsx!(cx, div {"hello world!"});
Borrowed prop contents:
```rust
-fn App(cx: Scope<()>) -> Element {
+fn App(cx: Scope) -> Element {
let name = use_state(&cx, || String::from("example"));
rsx!(cx, Child { title: name.as_str() })
}
@@ -145,12 +137,12 @@ Global State
```rust
struct GlobalState { name: String }
-fn App(cx: Scope<()>) -> Element {
+fn App(cx: Scope) -> Element {
use_provide_shared_state(cx, || GlobalState { name: String::from("Toby") })
rsx!(cx, Leaf {})
}
-fn Leaf(cx: Scope<()>) -> Element {
+fn Leaf(cx: Scope) -> Element {
let state = use_consume_shared_state::(cx)?;
rsx!(cx, "Hello {state.name}")
}
@@ -166,20 +158,20 @@ enum Route {
Post(id)
}
-fn App(cx: Scope<()>) -> Element {
+fn App(cx: Scope) -> Element {
let route = use_router(cx, Route::parse);
cx.render(rsx!(div {
- {match route {
+ match route {
Route::Home => rsx!( Home {} ),
Route::Post(id) => rsx!( Post { id: id })
- }}
+ }
}))
}
```
Suspense
```rust
-fn App(cx: Scope<()>) -> Element {
+fn App(cx: Scope) -> Element {
let doggo = use_suspense(cx,
|| async { reqwest::get("https://dog.ceo/api/breeds/image/random").await.unwrap().json::().await.unwrap() },
|response| cx.render(rsx!( img { src: "{response.message}" }))
@@ -187,8 +179,8 @@ fn App(cx: Scope<()>) -> Element {
cx.render(rsx!{
div {
- "One doggo coming right up:"
- {doggo}
+ "One doggo coming right up:",
+ doggo
}
})
}
diff --git a/examples/async.rs b/examples/async.rs
index c473ead72..4d292779e 100644
--- a/examples/async.rs
+++ b/examples/async.rs
@@ -8,38 +8,36 @@ use gloo_timers::future::TimeoutFuture;
#[tokio::main]
async fn main() {
- dioxus::desktop::launch(App);
+ dioxus::desktop::launch(app);
}
-pub static App: Component = |cx| {
+fn app(cx: Scope) -> Element {
let count = use_state(&cx, || 0);
- let mut direction = use_state(&cx, || 1);
+ let direction = use_state(&cx, || 1);
let (async_count, dir) = (count.for_async(), *direction);
let task = use_coroutine(&cx, move || async move {
loop {
TimeoutFuture::new(250).await;
- *async_count.get_mut() += dir;
+ *async_count.modify() += dir;
}
});
rsx!(cx, div {
h1 {"count is {count}"}
- button {
+ button { onclick: move |_| task.stop(),
"Stop counting"
- onclick: move |_| task.stop()
}
- button {
+ button { onclick: move |_| task.resume(),
"Start counting"
- onclick: move |_| task.resume()
}
button {
- "Switch counting direcion"
onclick: move |_| {
- direction *= -1;
+ *direction.modify() *= -1;
task.restart();
- }
+ },
+ "Switch counting direcion"
}
})
-};
+}
diff --git a/examples/borrowed.rs b/examples/borrowed.rs
index 3b05b6f9f..82920029b 100644
--- a/examples/borrowed.rs
+++ b/examples/borrowed.rs
@@ -20,8 +20,8 @@ fn main() {
dioxus::desktop::launch(App);
}
-fn App(cx: Scope<()>) -> Element {
- let text: &mut Vec = cx.use_hook(|_| vec![String::from("abc=def")], |f| f);
+fn App(cx: Scope) -> Element {
+ let text = cx.use_hook(|_| vec![String::from("abc=def")]);
let first = text.get_mut(0).unwrap();
diff --git a/examples/calculator.rs b/examples/calculator.rs
index c7490a1b4..c09f1b2aa 100644
--- a/examples/calculator.rs
+++ b/examples/calculator.rs
@@ -1,3 +1,5 @@
+#![allow(non_snake_case)]
+
/*
This example is a simple iOS-style calculator. This particular example can run any platform - Web, Mobile, Desktop.
This calculator version uses React-style state management. All state is held as individual use_states.
@@ -10,10 +12,10 @@ use dioxus::prelude::*;
use separator::Separatable;
fn main() {
- dioxus::desktop::launch(APP);
+ // dioxus::desktop::launch(app);
}
-static APP: Component = |cx| {
+fn app(cx: Scope) -> Element {
let cur_val = use_state(&cx, || 0.0_f64);
let operator = use_state(&cx, || None as Option<&'static str>);
let display_value = use_state(&cx, || String::from(""));
@@ -21,109 +23,138 @@ static APP: Component = |cx| {
let toggle_percent = move |_| todo!();
let input_digit = move |num: u8| display_value.modify().push_str(num.to_string().as_str());
- rsx!(cx, div {
- class: "calculator",
- onkeydown: move |evt| match evt.key_code {
- KeyCode::Add => operator.set(Some("+")),
- KeyCode::Subtract => operator.set(Some("-")),
- KeyCode::Divide => operator.set(Some("/")),
- KeyCode::Multiply => operator.set(Some("*")),
- KeyCode::Num0 => input_digit(0),
- KeyCode::Num1 => input_digit(1),
- KeyCode::Num2 => input_digit(2),
- KeyCode::Num3 => input_digit(3),
- KeyCode::Num4 => input_digit(4),
- KeyCode::Num5 => input_digit(5),
- KeyCode::Num6 => input_digit(6),
- KeyCode::Num7 => input_digit(7),
- KeyCode::Num8 => input_digit(8),
- KeyCode::Num9 => input_digit(9),
- KeyCode::Backspace => {
- if !display_value.as_str().eq("0") {
- display_value.modify().pop();
- }
- }
- _ => {}
- }
- div { class: "calculator-display", {[format_args!("{}", cur_val.separated_string())]} }
- div { class: "input-keys"
- div { class: "function-keys"
- CalculatorKey {
- {[if display_value == "0" { "C" } else { "AC" }]}
- name: "key-clear",
- onclick: move |_| {
- display_value.set("0".to_string());
- if display_value != "0" {
- operator.set(None);
- cur_val.set(0.0);
- }
+ cx.render(rsx!(
+ style { [include_str!("./assets/calculator.css")] }
+ div {
+ class: "calculator",
+ onkeydown: move |evt| match evt.key_code {
+ KeyCode::Add => operator.set(Some("+")),
+ KeyCode::Subtract => operator.set(Some("-")),
+ KeyCode::Divide => operator.set(Some("/")),
+ KeyCode::Multiply => operator.set(Some("*")),
+ KeyCode::Num0 => input_digit(0),
+ KeyCode::Num1 => input_digit(1),
+ KeyCode::Num2 => input_digit(2),
+ KeyCode::Num3 => input_digit(3),
+ KeyCode::Num4 => input_digit(4),
+ KeyCode::Num5 => input_digit(5),
+ KeyCode::Num6 => input_digit(6),
+ KeyCode::Num7 => input_digit(7),
+ KeyCode::Num8 => input_digit(8),
+ KeyCode::Num9 => input_digit(9),
+ KeyCode::Backspace => {
+ if !display_value.as_str().eq("0") {
+ display_value.modify().pop();
}
}
- CalculatorKey {
- "±"
- name: "key-sign",
- onclick: move |_| {
- if display_value.starts_with("-") {
- display_value.set(display_value.trim_start_matches("-").to_string())
- } else {
- display_value.set(format!("-{}", *display_value))
+ _ => {}
+ },
+ div { class: "calculator-display", [cur_val.separated_string()] }
+ div { class: "input-keys",
+ div { class: "function-keys",
+ CalculatorKey {
+ name: "key-clear",
+ onclick: move |_| {
+ display_value.set("0".to_string());
+ if display_value != "0" {
+ operator.set(None);
+ cur_val.set(0.0);
+ }
+ },
+ [if display_value == "0" { "C" } else { "AC" }]
+ }
+ CalculatorKey {
+ name: "key-sign",
+ onclick: move |_| {
+ if display_value.starts_with("-") {
+ display_value.set(display_value.trim_start_matches("-").to_string())
+ } else {
+ display_value.set(format!("-{}", *display_value))
+ }
+ },
+ "±"
+ }
+ CalculatorKey {
+ onclick: toggle_percent,
+ name: "key-percent",
+ "%"
+ }
+ }
+ div { class: "digit-keys",
+ CalculatorKey { name: "key-0", onclick: move |_| input_digit(0),
+ "0"
+ }
+ CalculatorKey { name: "key-dot", onclick: move |_| display_value.modify().push_str("."),
+ "●"
+ }
+ (1..9).map(|k| rsx!{
+ CalculatorKey {
+ key: "{k}",
+ name: "key-{k}",
+ onclick: move |_| input_digit(k),
+ "{k}"
}
- },
+ }),
}
- CalculatorKey {
- "%"
- onclick: {toggle_percent}
- name: "key-percent",
- }
- }
- div { class: "digit-keys"
- CalculatorKey { name: "key-0", onclick: move |_| input_digit(0), "0" }
- CalculatorKey { name: "key-dot", onclick: move |_| display_value.modify().push_str("."), "●" }
- {(1..9).map(|k| rsx!{
- CalculatorKey { key: "{k}", name: "key-{k}", onclick: move |_| input_digit(k), "{k}" }
- })}
- }
- div { class: "operator-keys"
- CalculatorKey { name: "key-divide", onclick: move |_| operator.set(Some("/")) "÷" }
- CalculatorKey { name: "key-multiply", onclick: move |_| operator.set(Some("*")) "×" }
- CalculatorKey { name: "key-subtract", onclick: move |_| operator.set(Some("-")) "−" }
- CalculatorKey { name: "key-add", onclick: move |_| operator.set(Some("+")) "+" }
- CalculatorKey {
- "="
- name: "key-equals",
- onclick: move |_| {
- if let Some(op) = operator.as_ref() {
- let rhs = display_value.parse::().unwrap();
- let new_val = match *op {
- "+" => *cur_val + rhs,
- "-" => *cur_val - rhs,
- "*" => *cur_val * rhs,
- "/" => *cur_val / rhs,
- _ => unreachable!(),
- };
- cur_val.set(new_val);
- display_value.set(new_val.to_string());
- operator.set(None);
- }
- },
+ div { class: "operator-keys",
+ CalculatorKey {
+ name: "key-divide",
+ onclick: move |_| operator.set(Some("/")),
+ "÷"
+ }
+ CalculatorKey {
+ name: "key-multiply",
+ onclick: move |_| operator.set(Some("*")),
+ "×"
+ }
+ CalculatorKey {
+ name: "key-subtract",
+ onclick: move |_| operator.set(Some("-")),
+ "−"
+ }
+ CalculatorKey {
+ name: "key-add",
+ onclick: move |_| operator.set(Some("+")),
+ "+"
+ }
+ CalculatorKey {
+ name: "key-equals",
+ onclick: move |_| {
+ if let Some(op) = operator.as_ref() {
+ let rhs = display_value.parse::().unwrap();
+ let new_val = match *op {
+ "+" => *cur_val + rhs,
+ "-" => *cur_val - rhs,
+ "*" => *cur_val * rhs,
+ "/" => *cur_val / rhs,
+ _ => unreachable!(),
+ };
+ cur_val.set(new_val);
+ display_value.set(new_val.to_string());
+ operator.set(None);
+ }
+ },
+ "="
+ }
}
}
}
- })
-};
+ ))
+}
-#[derive(Props)]
-struct CalculatorKeyProps<'a> {
+#[inline_props]
+fn CalculatorKey<'a>(
+ cx: Scope,
name: &'static str,
onclick: &'a dyn Fn(Arc),
children: Element<'a>,
-}
-
-fn CalculatorKey<'a>(cx: Scope<'a, CalculatorKeyProps<'a>>) -> Element {
- rsx!(cx, button {
- class: "calculator-key {cx.props.name}"
- onclick: {cx.props.onclick}
- {&cx.props.children}
+) -> Element {
+ cx.render(rsx! {
+ button {
+ class: "calculator-key {name}",
+ onclick: onclick,
+ children
+ }
})
}
diff --git a/examples/core/jsframework.rs b/examples/core/jsframework.rs
index 15c25ec8d..f9fa9c431 100644
--- a/examples/core/jsframework.rs
+++ b/examples/core/jsframework.rs
@@ -15,7 +15,7 @@ fn main() {
assert!(g.edits.len() > 1);
}
-fn App((cx, props): Scope<()>) -> Element {
+fn App((cx, props): Scope) -> Element {
let mut rng = SmallRng::from_entropy();
let rows = (0..10_000_usize).map(|f| {
let label = Label::new(&mut rng);
diff --git a/examples/core_reference/iterators.rs b/examples/core_reference/iterators.rs
index 07452921c..ad376a1bd 100644
--- a/examples/core_reference/iterators.rs
+++ b/examples/core_reference/iterators.rs
@@ -20,11 +20,11 @@ pub static Example: Component = |cx| {
li { onclick: move |_| example_data.set(f)
"ID: {f}"
ul {
- {(0..10).map(|k| rsx!{
+ (0..10).map(|k| rsx!{
li {
"Sub iterator: {f}.{k}"
}
- })}
+ })
}
}
}
diff --git a/examples/crm.rs b/examples/crm.rs
index a8eeb456b..11acb4346 100644
--- a/examples/crm.rs
+++ b/examples/crm.rs
@@ -4,7 +4,7 @@ Tiny CRM: A port of the Yew CRM example to Dioxus.
use dioxus::prelude::*;
fn main() {
- dioxus::web::launch(App);
+ dioxus::desktop::launch(app);
}
enum Scene {
ClientsList,
@@ -19,88 +19,102 @@ pub struct Client {
pub description: String,
}
-static App: Component = |cx| {
- let mut clients = use_ref(&cx, || vec![] as Vec);
- let mut scene = use_state(&cx, || Scene::ClientsList);
+fn app(cx: Scope) -> Element {
+ let clients = use_ref(&cx, || vec![] as Vec);
- let mut firstname = use_state(&cx, || String::new());
- let mut lastname = use_state(&cx, || String::new());
- let mut description = use_state(&cx, || String::new());
-
- let scene = match *scene {
- Scene::ClientsList => {
- rsx!(cx, div { class: "crm"
- h2 { "List of clients" margin_bottom: "10px" }
- div { class: "clients" margin_left: "10px"
- {clients.read().iter().map(|client| rsx!(
- div { class: "client" style: "margin-bottom: 50px"
- p { "First Name: {client.first_name}" }
- p { "Last Name: {client.last_name}" }
- p {"Description: {client.description}"}
- })
- )}
- }
- button { class: "pure-button pure-button-primary" onclick: move |_| scene.set(Scene::NewClientForm), "Add New" }
- button { class: "pure-button" onclick: move |_| scene.set(Scene::Settings), "Settings" }
- })
- }
- Scene::NewClientForm => {
- let add_new = move |_| {
- clients.write().push(Client {
- description: (*description).clone(),
- first_name: (*firstname).clone(),
- last_name: (*lastname).clone(),
- });
- description.set(String::new());
- firstname.set(String::new());
- lastname.set(String::new());
- };
- rsx!(cx, div { class: "crm"
- h2 {"Add new client" margin_bottom: "10px" }
- form { class: "pure-form"
- input { class: "new-client firstname" placeholder: "First name" value: "{firstname}"
- oninput: move |e| firstname.set(e.value.clone())
- }
- input { class: "new-client lastname" placeholder: "Last name" value: "{lastname}"
- oninput: move |e| lastname.set(e.value.clone())
- }
- textarea { class: "new-client description" placeholder: "Description" value: "{description}"
- oninput: move |e| description.set(e.value.clone())
- }
- }
- button { class: "pure-button pure-button-primary", onclick: {add_new}, "Add New" }
- button { class: "pure-button", onclick: move |_| scene.set(Scene::ClientsList), "Go Back" }
- })
- }
- Scene::Settings => {
- rsx!(cx, div {
- h2 {"Settings" margin_bottom: "10px" }
- button {
- background: "rgb(202, 60, 60)"
- class: "pure-button pure-button-primary"
- onclick: move |_| clients.write().clear(),
- "Remove all clients"
- }
- button {
- class: "pure-button pure-button-primary"
- onclick: move |_| scene.set(Scene::ClientsList),
- "Go Back"
- }
- })
- }
- };
+ let scene = use_state(&cx, || Scene::ClientsList);
+ let firstname = use_state(&cx, String::new);
+ let lastname = use_state(&cx, String::new);
+ let description = use_state(&cx, String::new);
cx.render(rsx!(
- body {
- link {
- rel: "stylesheet"
- href: "https://unpkg.com/purecss@2.0.6/build/pure-min.css"
- integrity: "sha384-Uu6IeWbM+gzNVXJcM9XV3SohHtmWE+3VGi496jvgX1jyvDTXfdK+rfZc8C1Aehk5"
- crossorigin: "anonymous"
- }
- margin_left: "35%"
- h1 {"Dioxus CRM Example"}
- {scene}
- }
+ body {
+ margin_left: "35%",
+ link {
+ rel: "stylesheet",
+ href: "https://unpkg.com/purecss@2.0.6/build/pure-min.css",
+ integrity: "sha384-Uu6IeWbM+gzNVXJcM9XV3SohHtmWE+3VGi496jvgX1jyvDTXfdK+rfZc8C1Aehk5",
+ crossorigin: "anonymous",
+ }
+
+ h1 {"Dioxus CRM Example"}
+
+ match *scene {
+ Scene::ClientsList => rsx!(
+ div { class: "crm",
+ h2 { margin_bottom: "10px", "List of clients" }
+ div { class: "clients", margin_left: "10px",
+ clients.read().iter().map(|client| rsx!(
+ div { class: "client", style: "margin-bottom: 50px",
+ p { "First Name: {client.first_name}" }
+ p { "Last Name: {client.last_name}" }
+ p {"Description: {client.description}"}
+ })
+ )
+ }
+ button { class: "pure-button pure-button-primary", onclick: move |_| scene.set(Scene::NewClientForm), "Add New" }
+ button { class: "pure-button", onclick: move |_| scene.set(Scene::Settings), "Settings" }
+ }
+ ),
+ Scene::NewClientForm => rsx!(
+ div { class: "crm",
+ h2 { margin_bottom: "10px", "Add new client" }
+ form { class: "pure-form",
+ input {
+ class: "new-client firstname",
+ placeholder: "First name",
+ value: "{firstname}",
+ oninput: move |e| firstname.set(e.value.clone())
+ }
+ input {
+ class: "new-client lastname",
+ placeholder: "Last name",
+ value: "{lastname}",
+ oninput: move |e| lastname.set(e.value.clone())
+ }
+ textarea {
+ class: "new-client description",
+ placeholder: "Description",
+ value: "{description}",
+ oninput: move |e| description.set(e.value.clone())
+ }
+ }
+ button {
+ class: "pure-button pure-button-primary",
+ onclick: move |_| {
+ clients.write().push(Client {
+ description: (*description).clone(),
+ first_name: (*firstname).clone(),
+ last_name: (*lastname).clone(),
+ });
+ description.set(String::new());
+ firstname.set(String::new());
+ lastname.set(String::new());
+ },
+ "Add New"
+ }
+ button { class: "pure-button", onclick: move |_| scene.set(Scene::ClientsList),
+ "Go Back"
+ }
+ }
+ ),
+ Scene::Settings => rsx!(
+ div {
+ h2 { margin_bottom: "10px", "Settings" }
+ button {
+ background: "rgb(202, 60, 60)",
+ class: "pure-button pure-button-primary",
+ onclick: move |_| clients.write().clear(),
+ "Remove all clients"
+ }
+ button {
+ class: "pure-button pure-button-primary",
+ onclick: move |_| scene.set(Scene::ClientsList),
+ "Go Back"
+ }
+ }
+ )
+ }
+ }
))
-};
+}
diff --git a/examples/desktop/demo.rs b/examples/desktop/demo.rs
index 1782d8cb0..d8dc51697 100644
--- a/examples/desktop/demo.rs
+++ b/examples/desktop/demo.rs
@@ -13,6 +13,6 @@ static App: Component = |cx| {
div {
"hello world!"
}
- {(0..10).map(|f| rsx!( div {"abc {f}"}))}
+ (0..10).map(|f| rsx!( div {"abc {f}"}))
))
};
diff --git a/examples/desktop/kitchensink.rs b/examples/desktop/kitchensink.rs
deleted file mode 100644
index 5ec049ee9..000000000
--- a/examples/desktop/kitchensink.rs
+++ /dev/null
@@ -1,8 +0,0 @@
-use dioxus_core as dioxus;
-use dioxus_core::prelude::*;
-use dioxus_core_macro::*;
-use dioxus_hooks::*;
-
-use dioxus_html as dioxus_elements;
-
-fn main() {}
diff --git a/examples/desktop/tauri.rs b/examples/desktop/tauri.rs
deleted file mode 100644
index f328e4d9d..000000000
--- a/examples/desktop/tauri.rs
+++ /dev/null
@@ -1 +0,0 @@
-fn main() {}
diff --git a/examples/desktop/todomvc.rs b/examples/desktop/todomvc.rs
index ff982a585..64bfb2876 100644
--- a/examples/desktop/todomvc.rs
+++ b/examples/desktop/todomvc.rs
@@ -5,15 +5,10 @@ use dioxus_core::prelude::*;
use dioxus_core_macro::*;
use dioxus_hooks::*;
use dioxus_html as dioxus_elements;
-use simple_logger::SimpleLogger;
use std::collections::HashMap;
fn main() {
- if cfg!(debug_assertions) {
- SimpleLogger::new().init().unwrap();
- }
-
dioxus_desktop::launch(App)
}
@@ -103,24 +98,24 @@ pub static App: Component = |cx| {
}
}
}
- ul { class: "todo-list"
- {filtered_todos.iter().map(|id| rsx!(TodoEntry { key: "{id}", id: *id }))}
+ ul { class: "todo-list",
+ filtered_todos.iter().map(|id| rsx!(TodoEntry { key: "{id}", id: *id }))
}
- {(!todos.read().is_empty()).then(|| rsx!(
- footer { class: "footer"
+ (!todos.read().is_empty()).then(|| rsx!(
+ footer { class: "footer",
span { class: "todo-count" strong {"{items_left} "} span {"{item_text} left"} }
ul { class: "filters"
li { class: "All", a { onclick: move |_| filter.set(FilterState::All), "All" }}
li { class: "Active", a { onclick: move |_| filter.set(FilterState::Active), "Active" }}
li { class: "Completed", a { onclick: move |_| filter.set(FilterState::Completed), "Completed" }}
}
- {(show_clear_completed).then(|| rsx!(
+ show_clear_completed.then(|| rsx!(
button { class: "clear-completed", onclick: move |_| clear_completed(),
"Clear completed"
}
- ))}
+ ))
}
- ))}
+ ))
}
}
footer { class: "info"
diff --git a/examples/file_explorer.rs b/examples/file_explorer.rs
index a2f168d6b..f27e018a9 100644
--- a/examples/file_explorer.rs
+++ b/examples/file_explorer.rs
@@ -4,51 +4,45 @@
//! This is a fun little desktop application that lets you explore the file system.
//!
//! This example is interesting because it's mixing filesystem operations and GUI, which is typically hard for UI to do.
+//!
+//! It also uses `use_ref` to maintain a model, rather than `use_state`. That way,
+//! we dont need to clutter our code with `read` commands.
use dioxus::prelude::*;
fn main() {
- // simple_logger::init_with_level(log::Level::Debug);
- dioxus::desktop::launch_cfg(App, |c| {
+ dioxus::desktop::launch_cfg(app, |c| {
c.with_window(|w| {
- w.with_resizable(true).with_inner_size(
- dioxus::desktop::wry::application::dpi::LogicalSize::new(400.0, 800.0),
- )
+ w.with_resizable(true)
+ .with_inner_size(dioxus::desktop::tao::dpi::LogicalSize::new(400.0, 800.0))
})
});
}
-static App: Component = |cx| {
- let file_manager = use_ref(&cx, Files::new);
- let files = file_manager.read();
-
- let file_list = files.path_names.iter().enumerate().map(|(dir_id, path)| {
- rsx! (
- li { a {"{path}", onclick: move |_| file_manager.write().enter_dir(dir_id), href: "#"} }
- )
- });
-
- let err_disp = files.err.as_ref().map(|err| {
- rsx! (
- div {
- code {"{err}"}
- button {"x", onclick: move |_| file_manager.write().clear_err() }
- }
- )
- });
-
- let current_dir = files.current();
+fn app(cx: Scope) -> Element {
+ let files = use_ref(&cx, Files::new);
cx.render(rsx!(
- div {
- h1 {"Files: "}
- h3 {"Cur dir: {current_dir}"}
- button { "go up", onclick: move |_| file_manager.write().go_up() }
- ol { {file_list} }
- {err_disp}
+ h1 { "Files: " }
+ h3 { "Cur dir: " [files.read().current()] }
+ button { onclick: move |_| files.write().go_up(), "go up" }
+ ol {
+ files.read().path_names.iter().enumerate().map(|(dir_id, path)| rsx!(
+ li { key: "{path}",
+ a { href: "#", onclick: move |_| files.write().enter_dir(dir_id),
+ "{path}",
+ }
+ }
+ ))
}
+ files.read().err.as_ref().map(|err| rsx!(
+ div {
+ code { "{err}" }
+ button { onclick: move |_| files.write().clear_err(), "x" }
+ }
+ ))
))
-};
+}
struct Files {
path_stack: Vec,
@@ -70,29 +64,21 @@ impl Files {
}
fn reload_path_list(&mut self) {
- let cur_path = self.path_stack.last().unwrap();
- log::info!("Reloading path list for {:?}", cur_path);
- let paths = match std::fs::read_dir(cur_path) {
+ let paths = match std::fs::read_dir(self.path_stack.last().unwrap()) {
Ok(e) => e,
Err(err) => {
- let err = format!("An error occured: {:?}", err);
- self.err = Some(err);
+ self.err = Some(format!("An error occured: {:?}", err));
self.path_stack.pop();
return;
}
};
- let collected = paths.collect::>();
- log::info!("Path list reloaded {:#?}", collected);
// clear the current state
self.clear_err();
self.path_names.clear();
- for path in collected {
- self.path_names
- .push(path.unwrap().path().display().to_string());
- }
- log::info!("path namees are {:#?}", self.path_names);
+ self.path_names
+ .extend(paths.map(|path| path.unwrap().path().display().to_string()));
}
fn go_up(&mut self) {
diff --git a/examples/framework_benchmark.rs b/examples/framework_benchmark.rs
index 100169679..73a82ea2c 100644
--- a/examples/framework_benchmark.rs
+++ b/examples/framework_benchmark.rs
@@ -2,8 +2,7 @@ use dioxus::prelude::*;
use rand::prelude::*;
fn main() {
- dioxus::web::launch(App);
- // dioxus::desktop::launch(App);
+ dioxus::desktop::launch(App);
}
#[derive(Clone, PartialEq)]
@@ -31,16 +30,16 @@ impl Label {
}
static App: Component = |cx| {
- let mut items = use_ref(&cx, || vec![]);
- let mut selected = use_state(&cx, || None);
+ let items = use_ref(&cx, || vec![]);
+ let selected = use_state(&cx, || None);
cx.render(rsx! {
- div { class: "container"
- div { class: "jumbotron"
- div { class: "row"
+ div { class: "container",
+ div { class: "jumbotron",
+ div { class: "row",
div { class: "col-md-6", h1 { "Dioxus" } }
- div { class: "col-md-6"
- div { class: "row"
+ div { class: "col-md-6",
+ div { class: "row",
ActionButton { name: "Create 1,000 rows", id: "run",
onclick: move || items.set(Label::new_list(1_000)),
}
@@ -67,15 +66,15 @@ static App: Component = |cx| {
tbody {
{items.read().iter().enumerate().map(|(id, item)| {
let is_in_danger = if (*selected).map(|s| s == id).unwrap_or(false) {"danger"} else {""};
- rsx!(tr { class: "{is_in_danger}"
+ rsx!(tr { class: "{is_in_danger}",
td { class:"col-md-1" }
td { class:"col-md-1", "{item.key}" }
td { class:"col-md-1", onclick: move |_| selected.set(Some(id)),
- a { class: "lbl", {item.labels} }
+ a { class: "lbl", item.labels }
}
- td { class: "col-md-1"
+ td { class: "col-md-1",
a { class: "remove", onclick: move |_| { items.write().remove(id); },
- span { class: "glyphicon glyphicon-remove remove" aria_hidden: "true" }
+ span { class: "glyphicon glyphicon-remove remove", aria_hidden: "true" }
}
}
td { class: "col-md-6" }
@@ -83,7 +82,7 @@ static App: Component = |cx| {
})}
}
}
- // span { class: "preloadicon glyphicon glyphicon-remove" aria_hidden: "true" }
+ span { class: "preloadicon glyphicon glyphicon-remove", aria_hidden: "true" }
}
})
};
@@ -96,7 +95,7 @@ struct ActionButtonProps<'a> {
}
fn ActionButton<'a>(cx: Scope<'a, ActionButtonProps<'a>>) -> Element {
- rsx!(cx, div { class: "col-sm-6 smallpad"
+ rsx!(cx, div { class: "col-sm-6 smallpad",
button { class:"btn btn-primary btn-block", r#type: "button", id: "{cx.props.id}", onclick: move |_| (cx.props.onclick)(),
"{cx.props.name}"
}
diff --git a/examples/hello_world.rs b/examples/hello_world.rs
index 24f54883a..ab7bfffa3 100644
--- a/examples/hello_world.rs
+++ b/examples/hello_world.rs
@@ -4,7 +4,7 @@ fn main() {
dioxus::desktop::launch(app);
}
-fn app(cx: Scope<()>) -> Element {
+fn app(cx: Scope) -> Element {
cx.render(rsx! (
div { "Hello, world!" }
))
diff --git a/examples/hydration.rs b/examples/hydration.rs
index 3c4faa55d..e0a23b46b 100644
--- a/examples/hydration.rs
+++ b/examples/hydration.rs
@@ -15,19 +15,18 @@ use dioxus::ssr;
fn main() {
let vdom = VirtualDom::new(App);
let content = ssr::render_vdom_cfg(&vdom, |f| f.pre_render(true));
-
dioxus::desktop::launch_cfg(App, |c| c.with_prerendered(content));
}
static App: Component = |cx| {
- let mut val = use_state(&cx, || 0);
+ let val = use_state(&cx, || 0);
cx.render(rsx! {
div {
- h1 {"hello world. Count: {val}"}
+ h1 { "hello world. Count: {val}" }
button {
+ onclick: move |_| *val.modify() += 1,
"click to increment"
- onclick: move |_| val += 1
}
}
})
diff --git a/examples/inputs.rs b/examples/inputs.rs
new file mode 100644
index 000000000..18edc9e8f
--- /dev/null
+++ b/examples/inputs.rs
@@ -0,0 +1,150 @@
+//! This example roughly shows how events are serialized into Rust from JavaScript.
+//!
+//! There is some conversion happening when input types are checkbox/radio/select/textarea etc.
+
+use std::sync::Arc;
+
+use dioxus::{events::FormEvent, prelude::*};
+
+fn main() {
+ dioxus::desktop::launch(App);
+}
+
+const FIELDS: &[(&str, &str)] = &[
+ ("button", "Click me!"),
+ ("checkbox", "CHECKBOX"),
+ ("color", ""),
+ ("date", ""),
+ ("datetime-local", ""),
+ ("email", ""),
+ ("file", ""),
+ ("image", ""),
+ ("number", ""),
+ ("password", ""),
+ ("radio", ""),
+ ("range", ""),
+ ("reset", ""),
+ ("search", ""),
+ ("submit", ""),
+ ("tel", ""),
+ ("text", ""),
+ ("time", ""),
+ ("url", ""),
+ //
+ // less supported things
+ ("hidden", ""),
+ ("month", ""), // degrades to text most of the time, but works properly as "value'"
+ ("week", ""), // degrades to text most of the time
+];
+
+static App: Component = |cx| {
+ cx.render(rsx! {
+ div { margin_left: "30px",
+
+ // radio group
+
+
+
+
+ div {
+ // handling inputs on divs will catch all input events below
+ // so the value of our input event will be either huey, dewey, louie, or true/false (because of the checkboxe)
+ // be mindful in grouping inputs together, as they will all be handled by the same event handler
+ oninput: move |evt| {
+ println!("{:?}", evt);
+ },
+ div {
+ input {
+ id: "huey",
+ r#type: "radio",
+ value: "huey",
+ checked: "",
+ name: "drone",
+ }
+ label {
+ r#for: "huey",
+ "Huey"
+ }
+ }
+ div {
+ input {
+ id: "dewey",
+ r#type: "radio",
+ value: "dewey",
+ name: "drone",
+ }
+ label {
+ r#for: "dewey",
+ "Dewey"
+ }
+ }
+ div {
+ input {
+ id: "louie",
+ value: "louie",
+ r#type: "radio",
+ name: "drone",
+ }
+ label {
+ r#for: "louie",
+ "Louie"
+ }
+ }
+ div {
+ input {
+ id: "groovy",
+ value: "groovy",
+ r#type: "checkbox",
+ name: "drone",
+ }
+ label {
+ r#for: "groovy",
+ "groovy"
+ }
+ }
+ }
+
+ // elements with driven values will preventdefault automatically.
+ // you can disable this override with preventdefault: false
+
+ div {
+ input {
+ id: "pdf",
+ value: "pdf",
+ name: "pdf",
+ r#type: "checkbox",
+ oninput: move |evt| {
+ println!("{:?}", evt);
+ },
+ }
+ label {
+ r#for: "pdf",
+ "pdf"
+ }
+ }
+
+
+
+ FIELDS.iter().map(|(field, value)| rsx!(
+ div {
+ input {
+ id: "{field}",
+ name: "{field}",
+ r#type: "{field}",
+ value: "{value}",
+ // checked: "false",
+ oninput: move |evt: Arc| {
+ println!("{:?}", evt);
+ },
+ }
+ label {
+ r#for: "{field}",
+ "{field} element"
+ }
+ br {}
+ }
+ ))
+
+ }
+ })
+};
diff --git a/examples/pattern_model.rs b/examples/pattern_model.rs
index 252a000c9..d9d2eb47e 100644
--- a/examples/pattern_model.rs
+++ b/examples/pattern_model.rs
@@ -24,7 +24,7 @@ use dioxus::prelude::*;
const STYLE: &str = include_str!("./assets/calculator.css");
fn main() {
env_logger::init();
- dioxus::desktop::launch_cfg(App, |cfg| {
+ dioxus::desktop::launch_cfg(app, |cfg| {
cfg.with_window(|w| {
w.with_title("Calculator Demo")
.with_resizable(false)
@@ -33,33 +33,33 @@ fn main() {
});
}
-static App: Component = |cx| {
+fn app(cx: Scope) -> Element {
let state = use_ref(&cx, || Calculator::new());
let clear_display = state.read().display_value.eq("0");
let clear_text = if clear_display { "C" } else { "AC" };
let formatted = state.read().formatted_display();
- rsx!(cx, div { id: "wrapper"
+ rsx!(cx, div { id: "wrapper",
div { class: "app", style { "{STYLE}" }
div { class: "calculator", onkeypress: move |evt| state.write().handle_keydown(evt),
div { class: "calculator-display", "{formatted}"}
- div { class: "calculator-keypad"
- div { class: "input-keys"
- div { class: "function-keys"
+ div { class: "calculator-keypad",
+ div { class: "input-keys",
+ div { class: "function-keys",
CalculatorKey { name: "key-clear", onclick: move |_| state.write().clear_display(), "{clear_text}" }
CalculatorKey { name: "key-sign", onclick: move |_| state.write().toggle_sign(), "±"}
CalculatorKey { name: "key-percent", onclick: move |_| state.write().toggle_percent(), "%"}
}
- div { class: "digit-keys"
+ div { class: "digit-keys",
CalculatorKey { name: "key-0", onclick: move |_| state.write().input_digit(0), "0" }
CalculatorKey { name: "key-dot", onclick: move |_| state.write().input_dot(), "●" }
- {(1..10).map(move |k| rsx!{
+ (1..10).map(move |k| rsx!{
CalculatorKey { key: "{k}", name: "key-{k}", onclick: move |_| state.write().input_digit(k), "{k}" }
- })}
+ })
}
}
- div { class: "operator-keys"
+ div { class: "operator-keys",
CalculatorKey { name:"key-divide", onclick: move |_| state.write().set_operator(Operator::Div), "÷" }
CalculatorKey { name:"key-multiply", onclick: move |_| state.write().set_operator(Operator::Mul), "×" }
CalculatorKey { name:"key-subtract", onclick: move |_| state.write().set_operator(Operator::Sub), "−" }
@@ -70,7 +70,7 @@ static App: Component = |cx| {
}
}
})
-};
+}
#[derive(Props)]
struct CalculatorKeyProps<'a> {
@@ -82,9 +82,9 @@ struct CalculatorKeyProps<'a> {
fn CalculatorKey<'a>(cx: Scope<'a, CalculatorKeyProps<'a>>) -> Element {
cx.render(rsx! {
button {
- class: "calculator-key {cx.props.name}"
- onclick: move |e| (cx.props.onclick)(e)
- {&cx.props.children}
+ class: "calculator-key {cx.props.name}",
+ onclick: move |e| (cx.props.onclick)(e),
+ &cx.props.children
}
})
}
diff --git a/examples/pattern_reducer.rs b/examples/pattern_reducer.rs
index cd14460f6..26bf3acff 100644
--- a/examples/pattern_reducer.rs
+++ b/examples/pattern_reducer.rs
@@ -14,20 +14,18 @@ fn main() {
pub static App: Component = |cx| {
let state = use_state(&cx, PlayerState::new);
- let is_playing = state.is_playing();
-
- rsx!(cx, div {
- h1 {"Select an option"}
- h3 {"The radio is... {is_playing}!"}
- button {
- "Pause"
- onclick: move |_| state.modify().reduce(PlayerAction::Pause)
+ cx.render(rsx!(
+ div {
+ h1 {"Select an option"}
+ h3 { "The radio is... " [state.is_playing()], "!" }
+ button { onclick: move |_| state.modify().reduce(PlayerAction::Pause),
+ "Pause"
+ }
+ button { onclick: move |_| state.modify().reduce(PlayerAction::Play),
+ "Play"
+ }
}
- button {
- "Play"
- onclick: move |_| state.modify().reduce(PlayerAction::Play)
- }
- })
+ ))
};
enum PlayerAction {
diff --git a/examples/readme.rs b/examples/readme.rs
index c3f0a3de7..ae17c1cd9 100644
--- a/examples/readme.rs
+++ b/examples/readme.rs
@@ -3,6 +3,7 @@
//! The example from the README.md.
use dioxus::prelude::*;
+
fn main() {
dioxus::desktop::launch(App);
}
diff --git a/examples/router.rs b/examples/router.rs
index 47350ad16..c28853185 100644
--- a/examples/router.rs
+++ b/examples/router.rs
@@ -32,14 +32,14 @@ static App: Component = |cx| {
Link { to: Route::AllUsers { page: 0 }, "List all users" }
Link { to: Route::BlogList { page: 0 }, "Blog posts" }
}
- {match route {
+ match route {
Route::Home => rsx!("Home"),
Route::AllUsers { page } => rsx!("All users - page {page}"),
Route::User { id } => rsx!("User - id: {id}"),
Route::BlogList { page } => rsx!("Blog posts - page {page}"),
Route::BlogPost { post_id } => rsx!("Blog post - post {post_id}"),
Route::NotFound => rsx!("Not found"),
- }}
+ }
footer {}
})
};
diff --git a/examples/rsx_compile_fail.rs b/examples/rsx_compile_fail.rs
new file mode 100644
index 000000000..cdc69c21e
--- /dev/null
+++ b/examples/rsx_compile_fail.rs
@@ -0,0 +1,62 @@
+use dioxus::prelude::*;
+
+fn main() {
+ let mut vdom = VirtualDom::new(example);
+ vdom.rebuild();
+
+ let out = dioxus::ssr::render_vdom_cfg(&vdom, |c| c.newline(true).indent(true));
+ println!("{}", out);
+}
+
+fn example(cx: Scope) -> Element {
+ let items = use_state(&cx, || {
+ vec![Thing {
+ a: "asd".to_string(),
+ b: 10,
+ }]
+ });
+
+ let things = use_ref(&cx, || {
+ vec![Thing {
+ a: "asd".to_string(),
+ b: 10,
+ }]
+ });
+ let things_list = things.read();
+
+ let mything = use_ref(&cx, || Some(String::from("asd")));
+ let mything_read = mything.read();
+
+ cx.render(rsx!(
+ div {
+ div {
+ id: "asd",
+ "your neighborhood spiderman"
+
+ items.iter().cycle().take(5).map(|f| rsx!{
+ div {
+ "{f.a}"
+ }
+ })
+
+ things_list.iter().map(|f| rsx!{
+ div {
+ "{f.a}"
+ "{f.b}"
+ }
+ })
+
+ mything_read.as_ref().map(|f| rsx!{
+ div {
+ "{f}"
+ }
+ })
+ }
+ }
+ ))
+}
+
+struct Thing {
+ a: String,
+ b: u32,
+}
diff --git a/examples/rsx_usage.rs b/examples/rsx_usage.rs
index 195edac00..04afb7407 100644
--- a/examples/rsx_usage.rs
+++ b/examples/rsx_usage.rs
@@ -60,6 +60,7 @@ pub static EXAMPLE: Component = |cx| {
h1 {"Some text"}
h1 {"Some text with {formatting}"}
h1 {"Formatting basic expressions {formatting_tuple.0} and {formatting_tuple.1}"}
+ h1 {"Formatting without interpolation " [formatting_tuple.0] "and" [formatting_tuple.1] }
h2 {
"Multiple"
"Text"
@@ -72,7 +73,7 @@ pub static EXAMPLE: Component = |cx| {
h3 {"elements"}
}
div {
- class: "my special div"
+ class: "my special div",
h1 {"Headers and attributes!"}
}
div {
@@ -88,42 +89,51 @@ pub static EXAMPLE: Component = |cx| {
}
// Expressions can be used in element position too:
- {rsx!(p { "More templating!" })}
- // {html!(
"Even HTML templating!!"
)}
+ rsx!(p { "More templating!" }),
// Iterators
- {(0..10).map(|i| rsx!(li { "{i}" }))}
- {{
+ (0..10).map(|i| rsx!(li { "{i}" })),
+
+ // Iterators within expressions
+ {
let data = std::collections::HashMap::<&'static str, &'static str>::new();
// Iterators *should* have keys when you can provide them.
// Keys make your app run faster. Make sure your keys are stable, unique, and predictable.
// Using an "ID" associated with your data is a good idea.
- data.into_iter().map(|(k, v)| rsx!(li { key: "{k}" "{v}" }))
- }}
+ data.into_iter().map(|(k, v)| rsx!(li { key: "{k}", "{v}" }))
+ }
// Matching
- {match true {
+ match true {
true => rsx!( h1 {"Top text"}),
false => rsx!( h1 {"Bottom text"})
- }}
+ }
// Conditional rendering
// Dioxus conditional rendering is based around None/Some. We have no special syntax for conditionals.
// You can convert a bool condition to rsx! with .then and .or
- {true.then(|| rsx!(div {}))}
+ true.then(|| rsx!(div {})),
- // True conditions need to be rendered (same reasons as matching)
- {if true {
- rsx!(cx, h1 {"Top text"})
+ // Alternatively, you can use the "if" syntax - but both branches must be resolve to Element
+ if false {
+ rsx!(h1 {"Top text"})
} else {
- rsx!(cx, h1 {"Bottom text"})
- }}
+ rsx!(h1 {"Bottom text"})
+ }
- // returning "None" is a bit noisy... but rare in practice
- {None as Option<()>}
+ // Using optionals for diverging branches
+ if true {
+ Some(rsx!(h1 {"Top text"}))
+ } else {
+ None
+ }
+
+
+ // returning "None" without a diverging branch is a bit noisy... but rare in practice
+ None as Option<()>,
// Use the Dioxus type-alias for less noise
- {NONE_ELEMENT}
+ NONE_ELEMENT,
// can also just use empty fragments
Fragment {}
@@ -137,9 +147,8 @@ pub static EXAMPLE: Component = |cx| {
Fragment {
"D"
Fragment {
- "heavily nested fragments is an antipattern"
- "they cause Dioxus to do unnecessary work"
- "don't use them carelessly if you can help it"
+ "E"
+ "F"
}
}
}
@@ -158,22 +167,29 @@ pub static EXAMPLE: Component = |cx| {
Taller { a: "asd" }
// Can pass in props directly as an expression
- {{
+ {
let props = TallerProps {a: "hello", children: Default::default()};
rsx!(Taller { ..props })
- }}
+ }
// Spreading can also be overridden manually
Taller {
- ..TallerProps { a: "ballin!", children: Default::default() }
+ ..TallerProps { a: "ballin!", children: Default::default() },
a: "not ballin!"
}
// Can take children too!
Taller { a: "asd", div {"hello world!"} }
+ // Components can be used with the `call` syntax
+ // This component's props are defined *inline* with the `inline_props` macro
+ with_inline(
+ text: "using functionc all syntax"
+ )
+
// helper functions
- {helper(&cx, "hello world!")}
+ // Single values must be wrapped in braces or `Some` to satisfy `IntoIterator`
+ [helper(&cx, "hello world!")]
}
})
};
@@ -187,6 +203,7 @@ mod baller {
#[derive(Props, PartialEq)]
pub struct BallerProps {}
+ #[allow(non_snake_case)]
/// This component totally balls
pub fn Baller(_: Scope) -> Element {
todo!()
@@ -195,12 +212,20 @@ mod baller {
#[derive(Props)]
pub struct TallerProps<'a> {
+ /// Fields are documented and accessible in rsx!
a: &'static str,
children: Element<'a>,
}
-/// This component is taller than most :)
-pub fn Taller<'a>(_: Scope<'a, TallerProps<'a>>) -> Element {
- let b = true;
- todo!()
+/// Documention for this component is visible within the rsx macro
+#[allow(non_snake_case)]
+pub fn Taller<'a>(cx: Scope<'a, TallerProps<'a>>) -> Element {
+ cx.render(rsx! {
+ &cx.props.children
+ })
+}
+
+#[inline_props]
+fn with_inline<'a>(cx: Scope<'a>, text: &'a str) -> Element {
+ rsx!(cx, p { "{text}" })
}
diff --git a/examples/tailwind.rs b/examples/tailwind.rs
index 49da99ee8..9bd87c15e 100644
--- a/examples/tailwind.rs
+++ b/examples/tailwind.rs
@@ -16,9 +16,9 @@ const STYLE: &str = "body {overflow:hidden;}";
pub static App: Component = |cx| {
cx.render(rsx!(
- div { class: "overflow-hidden"
+ div { class: "overflow-hidden",
style { "{STYLE}" }
- link { href:"https://unpkg.com/tailwindcss@^2/dist/tailwind.min.css" rel:"stylesheet" }
+ link { href:"https://unpkg.com/tailwindcss@^2/dist/tailwind.min.css", rel:"stylesheet" }
Header {}
Entry {}
Hero {}
@@ -33,20 +33,20 @@ pub static App: Component = |cx| {
pub static Header: Component = |cx| {
cx.render(rsx! {
div {
- header { class: "text-gray-400 bg-gray-900 body-font"
- div { class: "container mx-auto flex flex-wrap p-5 flex-col md:flex-row items-center"
- a { class: "flex title-font font-medium items-center text-white mb-4 md:mb-0"
+ header { class: "text-gray-400 bg-gray-900 body-font",
+ div { class: "container mx-auto flex flex-wrap p-5 flex-col md:flex-row items-center",
+ a { class: "flex title-font font-medium items-center text-white mb-4 md:mb-0",
StacksIcon {}
- span { class: "ml-3 text-xl" "Hello Dioxus!"}
+ span { class: "ml-3 text-xl", "Hello Dioxus!"}
}
- nav { class: "md:ml-auto flex flex-wrap items-center text-base justify-center"
- a { class: "mr-5 hover:text-white" "First Link"}
- a { class: "mr-5 hover:text-white" "Second Link"}
- a { class: "mr-5 hover:text-white" "Third Link"}
- a { class: "mr-5 hover:text-white" "Fourth Link"}
+ nav { class: "md:ml-auto flex flex-wrap items-center text-base justify-center",
+ a { class: "mr-5 hover:text-white", "First Link"}
+ a { class: "mr-5 hover:text-white", "Second Link"}
+ a { class: "mr-5 hover:text-white", "Third Link"}
+ a { class: "mr-5 hover:text-white", "Fourth Link"}
}
button {
- class: "inline-flex items-center bg-gray-800 border-0 py-1 px-3 focus:outline-none hover:bg-gray-700 rounded text-base mt-4 md:mt-0"
+ class: "inline-flex items-center bg-gray-800 border-0 py-1 px-3 focus:outline-none hover:bg-gray-700 rounded text-base mt-4 md:mt-0",
"Button"
RightArrowIcon {}
}
@@ -59,34 +59,34 @@ pub static Header: Component = |cx| {
pub static Hero: Component = |cx| {
//
cx.render(rsx! {
- section{ class: "text-gray-400 bg-gray-900 body-font"
- div { class: "container mx-auto flex px-5 py-24 md:flex-row flex-col items-center"
- div { class: "lg:flex-grow md:w-1/2 lg:pr-24 md:pr-16 flex flex-col md:items-start md:text-left mb-16 md:mb-0 items-center text-center"
- h1 { class: "title-font sm:text-4xl text-3xl mb-4 font-medium text-white"
+ section{ class: "text-gray-400 bg-gray-900 body-font",
+ div { class: "container mx-auto flex px-5 py-24 md:flex-row flex-col items-center",
+ div { class: "lg:flex-grow md:w-1/2 lg:pr-24 md:pr-16 flex flex-col md:items-start md:text-left mb-16 md:mb-0 items-center text-center",
+ h1 { class: "title-font sm:text-4xl text-3xl mb-4 font-medium text-white",
br { class: "hidden lg:inline-block" }
"Dioxus Sneak Peek"
}
p {
- class: "mb-8 leading-relaxed"
+ class: "mb-8 leading-relaxed",
"Dioxus is a new UI framework that makes it easy and simple to write cross-platform apps using web
technologies! It is functional, fast, and portable. Dioxus can run on the web, on the desktop, and
on mobile and embedded platforms."
}
- div { class: "flex justify-center"
+ div { class: "flex justify-center",
button {
- class: "inline-flex text-white bg-indigo-500 border-0 py-2 px-6 focus:outline-none hover:bg-indigo-600 rounded text-lg"
+ class: "inline-flex text-white bg-indigo-500 border-0 py-2 px-6 focus:outline-none hover:bg-indigo-600 rounded text-lg",
"Learn more"
}
button {
- class: "ml-4 inline-flex text-gray-400 bg-gray-800 border-0 py-2 px-6 focus:outline-none hover:bg-gray-700 hover:text-white rounded text-lg"
+ class: "ml-4 inline-flex text-gray-400 bg-gray-800 border-0 py-2 px-6 focus:outline-none hover:bg-gray-700 hover:text-white rounded text-lg",
"Build an app"
}
}
}
- div { class: "lg:max-w-lg lg:w-full md:w-1/2 w-5/6"
- img { class: "object-cover object-center rounded" alt: "hero" src: "https://i.imgur.com/oK6BLtw.png"
+ div { class: "lg:max-w-lg lg:w-full md:w-1/2 w-5/6",
+ img { class: "object-cover object-center rounded", alt: "hero", src: "https://i.imgur.com/oK6BLtw.png",
referrerpolicy:"no-referrer"
}
}
@@ -97,8 +97,8 @@ pub static Hero: Component = |cx| {
pub static Entry: Component = |cx| {
//
cx.render(rsx! {
- section{ class: "text-gray-400 bg-gray-900 body-font"
- div { class: "container mx-auto flex px-5 py-24 md:flex-row flex-col items-center"
+ section{ class: "text-gray-400 bg-gray-900 body-font",
+ div { class: "container mx-auto flex px-5 py-24 md:flex-row flex-col items-center",
textarea {
}
@@ -111,13 +111,13 @@ pub static StacksIcon: Component = |cx| {
cx.render(rsx!(
svg {
// xmlns: "http://www.w3.org/2000/svg"
- fill: "none"
- stroke: "currentColor"
- stroke_linecap: "round"
- stroke_linejoin: "round"
- stroke_width: "2"
- class: "w-10 h-10 text-white p-2 bg-indigo-500 rounded-full"
- view_box: "0 0 24 24"
+ fill: "none",
+ stroke: "currentColor",
+ stroke_linecap: "round",
+ stroke_linejoin: "round",
+ stroke_width: "2",
+ class: "w-10 h-10 text-white p-2 bg-indigo-500 rounded-full",
+ view_box: "0 0 24 24",
path { d: "M12 2L2 7l10 5 10-5-10-5zM2 17l10 5 10-5M2 12l10 5 10-5"}
}
))
@@ -125,13 +125,13 @@ pub static StacksIcon: Component = |cx| {
pub static RightArrowIcon: Component = |cx| {
cx.render(rsx!(
svg {
- fill: "none"
- stroke: "currentColor"
- stroke_linecap: "round"
- stroke_linejoin: "round"
- stroke_width: "2"
- class: "w-4 h-4 ml-1"
- view_box: "0 0 24 24"
+ fill: "none",
+ stroke: "currentColor",
+ stroke_linecap: "round",
+ stroke_linejoin: "round",
+ stroke_width: "2",
+ class: "w-4 h-4 ml-1",
+ view_box: "0 0 24 24",
path { d: "M5 12h14M12 5l7 7-7 7"}
}
))
diff --git a/examples/tasks.rs b/examples/tasks.rs
index 0db459786..0838c1fa2 100644
--- a/examples/tasks.rs
+++ b/examples/tasks.rs
@@ -2,26 +2,31 @@
//!
//! The example from the README.md.
+use dioxus::prelude::*;
use std::time::Duration;
-use dioxus::prelude::*;
fn main() {
dioxus::desktop::launch(app);
}
-fn app(cx: Scope<()>) -> Element {
- let mut count = use_state(&cx, || 0);
+fn app(cx: Scope) -> Element {
+ let count = use_state(&cx, || 0);
- cx.push_future(|| async move {
- tokio::time::sleep(Duration::from_millis(100)).await;
- count += 1;
+ use_future(&cx, || {
+ for_async![count];
+ async move {
+ loop {
+ tokio::time::sleep(Duration::from_millis(1000)).await;
+ count += 1;
+ }
+ }
});
cx.render(rsx! {
div {
h1 { "High-Five counter: {count}" }
button {
- onclick: move |_| count +=1 ,
+ onclick: move |_| *count.modify() += 1,
"Click me!"
}
}
diff --git a/examples/todomvc.rs b/examples/todomvc.rs
index a41f78d5e..8a62e0ddd 100644
--- a/examples/todomvc.rs
+++ b/examples/todomvc.rs
@@ -29,7 +29,7 @@ const App: Component = |cx| {
let todolist = todos
.iter()
- .filter(|(id, item)| match *filter {
+ .filter(|(_id, item)| match *filter {
FilterState::All => true,
FilterState::Active => !item.checked,
FilterState::Completed => item.checked,
@@ -48,34 +48,37 @@ const App: Component = |cx| {
_ => "items",
};
- rsx!(cx, div { id: "app"
+ rsx!(cx, div { id: "app",
style {"{STYLE}"}
div {
- header { class: "header"
+ header { class: "header",
h1 {"todos"}
input {
- class: "new-todo"
- placeholder: "What needs to be done?"
- value: "{draft}"
- oninput: move |evt| draft.set(evt.value.clone())
+ class: "new-todo",
+ placeholder: "What needs to be done?",
+ value: "{draft}",
+ oninput: move |evt| draft.set(evt.value.clone()),
}
}
- {todolist}
- {(!todos.is_empty()).then(|| rsx!(
+ todolist,
+ (!todos.is_empty()).then(|| rsx!(
footer {
- span { strong {"{items_left}"} span {"{item_text} left"} }
- ul { class: "filters"
+ span {
+ strong {"{items_left}"}
+ span {"{item_text} left"}
+ }
+ ul { class: "filters",
li { class: "All", a { href: "", onclick: move |_| filter.set(FilterState::All), "All" }}
li { class: "Active", a { href: "active", onclick: move |_| filter.set(FilterState::Active), "Active" }}
li { class: "Completed", a { href: "completed", onclick: move |_| filter.set(FilterState::Completed), "Completed" }}
}
}
- ))}
+ ))
}
- footer { class: "info"
+ footer { class: "info",
p {"Double-click to edit a todo"}
- p { "Created by ", a { "jkelleyrtp", href: "http://github.com/jkelleyrtp/" }}
- p { "Part of ", a { "TodoMVC", href: "http://todomvc.com" }}
+ p { "Created by ", a { href: "http://github.com/jkelleyrtp/", "jkelleyrtp" }}
+ p { "Part of ", a { href: "http://todomvc.com", "TodoMVC" }}
}
})
};
@@ -93,13 +96,13 @@ pub fn TodoEntry(cx: Scope) -> Element {
rsx!(cx, li {
"{todo.id}"
input {
- class: "toggle"
- r#type: "checkbox"
+ class: "toggle",
+ r#type: "checkbox",
"{todo.checked}"
}
{is_editing.then(|| rsx!{
input {
- value: "{contents}"
+ value: "{contents}",
oninput: move |evt| contents.set(evt.value.clone())
}
})}
diff --git a/examples/weather_app.rs b/examples/weather_app.rs
index 3317a1774..66e90dbb7 100644
--- a/examples/weather_app.rs
+++ b/examples/weather_app.rs
@@ -43,21 +43,21 @@ struct WeatherProps {}
static WeatherDisplay: Component = |cx| {
//
cx.render(rsx!(
- div { class: "flex items-center justify-center flex-col"
- div { class: "flex items-center justify-center"
- div { class: "flex flex-col bg-white rounded p-4 w-full max-w-xs"
- div{ class: "font-bold text-xl"
+ div { class: "flex items-center justify-center flex-col",
+ div { class: "flex items-center justify-center",
+ div { class: "flex flex-col bg-white rounded p-4 w-full max-w-xs",
+ div{ class: "font-bold text-xl",
"Jon's awesome site!!"
}
- div{ class: "text-sm text-gray-500"
+ div{ class: "text-sm text-gray-500",
"He worked so hard on it :)"
}
- div { class: "flex flex-row items-center justify-center mt-6"
- div { class: "font-medium text-6xl"
+ div { class: "flex flex-row items-center justify-center mt-6",
+ div { class: "font-medium text-6xl",
"1337"
}
}
- div { class: "flex flex-row justify-between mt-6"
+ div { class: "flex flex-row justify-between mt-6",
"Legit made my own React"
}
}
diff --git a/examples/web_tick.rs b/examples/web_tick.rs
index 447bd1121..74cc1897f 100644
--- a/examples/web_tick.rs
+++ b/examples/web_tick.rs
@@ -13,28 +13,19 @@
use dioxus::prelude::*;
fn main() {
- #[cfg(target_arch = "wasm32")]
- intern_strings();
-
- dioxus::web::launch(App);
+ dioxus::desktop::launch(App);
}
static App: Component = |cx| {
let mut rng = SmallRng::from_entropy();
- let rows = (0..1_000).map(|f| {
- let label = Label::new(&mut rng);
- rsx! {
- Row {
- row_id: f,
- label: label
- }
- }
- });
cx.render(rsx! {
table {
tbody {
- {rows}
+ (0..1_000).map(|f| {
+ let label = Label::new(&mut rng);
+ rsx! (Row { row_id: f, label: label })
+ })
}
}
})
@@ -50,12 +41,12 @@ fn Row(cx: Scope) -> Element {
cx.render(rsx! {
tr {
td { class:"col-md-1", "{cx.props.row_id}" }
- td { class:"col-md-1", onclick: move |_| { /* run onselect */ }
+ td { class:"col-md-1", onclick: move |_| { /* run onselect */ },
a { class: "lbl", "{adj}" "{col}" "{noun}" }
}
- td { class: "col-md-1"
- a { class: "remove", onclick: move |_| {/* remove */}
- span { class: "glyphicon glyphicon-remove remove" aria_hidden: "true" }
+ td { class: "col-md-1",
+ a { class: "remove", onclick: move |_| {/* remove */},
+ span { class: "glyphicon glyphicon-remove remove", aria_hidden: "true" }
}
}
td { class: "col-md-6" }
diff --git a/examples/webview_web.rs b/examples/webview_web.rs
index 367094b2c..aa762cb88 100644
--- a/examples/webview_web.rs
+++ b/examples/webview_web.rs
@@ -1,4 +1,3 @@
-#![allow(non_upper_case_globals, non_snake_case)]
//! Example: Webview Renderer
//! -------------------------
//!
@@ -13,10 +12,10 @@
use dioxus::prelude::*;
fn main() {
- dioxus::web::launch(App);
+ dioxus::desktop::launch(app);
}
-static App: Component = |cx| {
+fn app(cx: Scope) -> Element {
let mut count = use_state(&cx, || 0);
cx.render(rsx! {
@@ -26,4 +25,4 @@ static App: Component = |cx| {
button { onclick: move |_| count -= 1, "Down low!" }
}
})
-};
+}
diff --git a/examples/xss_safety.rs b/examples/xss_safety.rs
new file mode 100644
index 000000000..c694d646f
--- /dev/null
+++ b/examples/xss_safety.rs
@@ -0,0 +1,28 @@
+use dioxus::prelude::*;
+
+fn main() {
+ dioxus::desktop::launch(app);
+}
+
+fn app(cx: Scope) -> Element {
+ let contents = use_state(&cx, || String::from(""));
+
+ cx.render(rsx! {
+ div {
+ "hello world!"
+
+ h1 { "{contents}" }
+
+ h3 { [contents.as_str()] }
+
+ input {
+ value: "{contents}",
+ oninput: move |e| {
+ contents.set(e.value.clone());
+ eprintln!("asd");
+ },
+ "type": "text",
+ }
+ }
+ })
+}
diff --git a/notes/SUSPENSE.md b/notes/SUSPENSE.md
index b377f1021..a3cb442cb 100644
--- a/notes/SUSPENSE.md
+++ b/notes/SUSPENSE.md
@@ -17,11 +17,7 @@ for more general tasks, we need some way of submitting a future or task into som
```rust
-let task = use_hook(
- || { /* */ },
- || { /* update the future if it needs to be updated */ },
- || {}
-);
+let task = use_hook(|| { /* */ });
cx.poll_future()
// let recoil_event_loop = cx.use_task(move |_| async move {
// loop {
diff --git a/packages/core-macro/Cargo.toml b/packages/core-macro/Cargo.toml
index dee8b0a36..9af9f3126 100644
--- a/packages/core-macro/Cargo.toml
+++ b/packages/core-macro/Cargo.toml
@@ -1,6 +1,6 @@
[package]
name = "dioxus-core-macro"
-version = "0.1.2"
+version = "0.1.3"
authors = ["Jonathan Kelley"]
edition = "2021"
description = "Core macro for Dioxus Virtual DOM"
@@ -16,6 +16,7 @@ proc-macro = true
[dependencies]
once_cell = "1.8"
+proc-macro-error = "1.0.4"
proc-macro2 = { version = "1.0.6" }
quote = "1.0"
syn = { version = "1.0.11", features = ["full", "extra-traits"] }
diff --git a/packages/core-macro/src/inlineprops.rs b/packages/core-macro/src/inlineprops.rs
index 64dab1ee3..acbf778d3 100644
--- a/packages/core-macro/src/inlineprops.rs
+++ b/packages/core-macro/src/inlineprops.rs
@@ -99,6 +99,7 @@ impl ToTokens for InlinePropsBody {
};
out_tokens.append_all(quote! {
+ #[allow(non_camel_case)]
#modifiers
#vis struct #struct_name #generics {
#(#fields),*
diff --git a/packages/core-macro/src/lib.rs b/packages/core-macro/src/lib.rs
index a196441b8..8f603bbb7 100644
--- a/packages/core-macro/src/lib.rs
+++ b/packages/core-macro/src/lib.rs
@@ -161,7 +161,7 @@ pub fn derive_typed_builder(input: proc_macro::TokenStream) -> proc_macro::Token
/// pub struct BallerProps {}
///
/// /// This component totally balls
-/// pub fn Baller(cx: Scope<()>) -> DomTree {
+/// pub fn Baller(cx: Scope) -> DomTree {
/// todo!()
/// }
/// }
@@ -177,11 +177,12 @@ pub fn derive_typed_builder(input: proc_macro::TokenStream) -> proc_macro::Token
/// todo!()
/// }
/// ```
+#[proc_macro_error::proc_macro_error]
#[proc_macro]
pub fn rsx(s: TokenStream) -> TokenStream {
match syn::parse::(s) {
- Err(e) => e.to_compile_error().into(),
- Ok(s) => s.to_token_stream().into(),
+ Err(err) => err.to_compile_error().into(),
+ Ok(stream) => stream.to_token_stream().into(),
}
}
diff --git a/packages/core-macro/src/rsx/ambiguous.rs b/packages/core-macro/src/rsx/ambiguous.rs
deleted file mode 100644
index 16cb2c4e5..000000000
--- a/packages/core-macro/src/rsx/ambiguous.rs
+++ /dev/null
@@ -1,64 +0,0 @@
-//! Parse anything that has a pattern of < Ident, Bracket >
-//! ========================================================
-//!
-//! Whenever a `name {}` pattern emerges, we need to parse it into an element, a component, or a fragment.
-//! This feature must support:
-//! - Namepsaced/pathed components
-//! - Differentiating between built-in and custom elements
-
-use super::*;
-
-use proc_macro2::TokenStream as TokenStream2;
-use quote::ToTokens;
-use syn::{
- parse::{Parse, ParseStream},
- Error, Ident, Result, Token,
-};
-
-#[allow(clippy::large_enum_variant)]
-pub enum AmbiguousElement {
- Element(Element),
- Component(Component),
-}
-
-impl Parse for AmbiguousElement {
- fn parse(input: ParseStream) -> Result {
- // Try to parse as an absolute path and immediately defer to the componetn
- if input.peek(Token![::]) {
- return input.parse::().map(AmbiguousElement::Component);
- }
-
- // If not an absolute path, then parse the ident and check if it's a valid tag
- if let Ok(pat) = input.fork().parse::() {
- if pat.segments.len() > 1 {
- return input.parse::().map(AmbiguousElement::Component);
- }
- }
-
- use syn::ext::IdentExt;
- if let Ok(name) = input.fork().call(Ident::parse_any) {
- let name_str = name.to_string();
-
- let first_char = name_str.chars().next().unwrap();
- if first_char.is_ascii_uppercase() {
- input.parse::().map(AmbiguousElement::Component)
- } else {
- if input.peek2(syn::token::Paren) {
- input.parse::().map(AmbiguousElement::Component)
- } else {
- input.parse::().map(AmbiguousElement::Element)
- }
- }
- } else {
- Err(Error::new(input.span(), "Not a valid Html tag"))
- }
- }
-}
-impl ToTokens for AmbiguousElement {
- fn to_tokens(&self, tokens: &mut TokenStream2) {
- match self {
- AmbiguousElement::Element(el) => el.to_tokens(tokens),
- AmbiguousElement::Component(comp) => comp.to_tokens(tokens),
- }
- }
-}
diff --git a/packages/core-macro/src/rsx/body.rs b/packages/core-macro/src/rsx/body.rs
deleted file mode 100644
index 83d2886e6..000000000
--- a/packages/core-macro/src/rsx/body.rs
+++ /dev/null
@@ -1,66 +0,0 @@
-use proc_macro2::TokenStream as TokenStream2;
-use quote::{quote, ToTokens, TokenStreamExt};
-use syn::{
- parse::{Parse, ParseStream},
- Ident, Result, Token,
-};
-
-use super::*;
-
-pub struct CallBody {
- custom_context: Option,
- roots: Vec,
-}
-
-/// The custom rusty variant of parsing rsx!
-impl Parse for CallBody {
- fn parse(input: ParseStream) -> Result {
- let custom_context = try_parse_custom_context(input)?;
- let (_, roots, _) = BodyConfig::new_call_body().parse_component_body(input)?;
- Ok(Self {
- custom_context,
- roots,
- })
- }
-}
-
-fn try_parse_custom_context(input: ParseStream) -> Result