mirror of
https://github.com/DioxusLabs/dioxus
synced 2024-11-25 13:40:19 +00:00
7ec3453ca3
* wip: fix manganis import path * upgrade tauri bundler, convert todo!() to unimplemented!() * feat: mobile hotreloading + ios asset configuration * remove oid demo and its associated env vars * nuke all the things that cause cache thrashing * swap to Asset type * add some more logs * display impl for attrvalue * dont panic on collect failure * cut down web deps to 150 * clean up deps in a lot of places, simplify build scripts * clean up asset resolution and cli-dev profile * wire up absolute paths for manganis asset in a particular mode * move document related things around to shorten compile times * move most things to `document::Item` and then decompose the cli-config * switch link to stylesheet for clarity in examples * move manganis workspace example to examples folder * decompose manganis-cli-support * dont need mobile demo anymore * remove gloo dialogs * html doesnt need document * rename hotreload to devtools * really clean up the html crate * fix weird rsx spacing * clean up desktop to use callback * clean up document a bit * re-wire up devsocket * fix utf css * yeeeet that js out of here * synthetic web system works * web crate almost done being cleaned up * desktop mostly cleaned up too * clean up cli a bit * more cli cleanup * cli builds again * clean up cli, inline structs, cut down number of unique types where possible * hotreload works again for desktop and this time, mobile too * cfg out tungstenite * devserver err * more refactor to cli builder * switch to unbounded_send * new structure is much cleaner * add http serve subcommand * bundled hotreload * kick stylsheets * clean up a bit more, split up eventloop * amazingly serve seems done * change from pub to pub(crate) in cli * remove tools * bit more polish to cli * fix issue with join * gracefully handle fullstack without a server * fullstack mobile demo * fix launch function, move projects into example-projects folder * hoist examples * add "run" command * clean up launch * remove old manual websocket receiver * doctor command * allow desktop to scroll * cut apart router crate * dont put launch in prelude * use dioxus::launch where possible * rename rsx, cut out hotreload tests * remove liveview project * bump native * fix compile for renderers * move sync event response out of interpreter * move render in serve * rollback settings change * cli compiles, huzzah! * change uris for asset * fix asset * new tui screen * new cargo-like tui works * very very verty close * it works! very small bug with incorrect grapheme calc * Clean up devserver a bit * status sytem * tidy up debug filters * clean up logging situation * Fix a number of bugs with log printing * new printing system is more reliable * wire up more stuff * things working but fullstack is having issues * fullstack works again! * hotreloading bundled assets works again * bundled hotreload and beginnings of macos bundling * Hotreload desktop * combined server builds * add build handle * fix fullstack assets * make open async, add some hooks for ios * migrate filemap to runner * wip global crate system * fixup bundles + organize asset * fix asset location bug * all the bundled reloading! * open ios simulator! * full hotreload support for mobile + serverfn * basic cleanups * clean up dx * Move filemap * fix cutting of newlines * assets workibg, some android * hoist wry/tao * use sync locks and headers to fix issues with android * desktop -> mac/win/linux with alias * better logging * feat: workspace (entire computer!) hotreload * should rebuild toggle, ios simulator bootup * proper mobile support in launch * more robust handling of assets * fix cargo * bring back some of tauri bundle * make warnings go away, clippy happy on cli * some final clippy cleanups * fmt * move manganis to its own folder * upgrade bundle to stable * drastically slim down manganis, prepping for merge * typos, failing test, docsrs config * remove static gen test * nix static gen test * we use --platform web instead of --platform fullstack now * only bind dev urls in desktop/mobile * install gtk * nix static gen * split build dir by app name
259 lines
8.9 KiB
Rust
259 lines
8.9 KiB
Rust
//! The typical TodoMVC app, implemented in Dioxus.
|
|
|
|
use dioxus::prelude::*;
|
|
use std::collections::HashMap;
|
|
|
|
const STYLE: Asset = asset!("/examples/assets/todomvc.css");
|
|
|
|
fn main() {
|
|
dioxus::launch(app);
|
|
}
|
|
|
|
#[derive(PartialEq, Eq, Clone, Copy)]
|
|
enum FilterState {
|
|
All,
|
|
Active,
|
|
Completed,
|
|
}
|
|
|
|
struct TodoItem {
|
|
checked: bool,
|
|
contents: String,
|
|
}
|
|
|
|
fn app() -> Element {
|
|
// We store the todos in a HashMap in a Signal.
|
|
// Each key is the id of the todo, and the value is the todo itself.
|
|
let mut todos = use_signal(HashMap::<u32, TodoItem>::new);
|
|
|
|
let filter = use_signal(|| FilterState::All);
|
|
|
|
// We use a simple memoized signal to calculate the number of active todos.
|
|
// Whenever the todos change, the active_todo_count will be recalculated.
|
|
let active_todo_count =
|
|
use_memo(move || todos.read().values().filter(|item| !item.checked).count());
|
|
|
|
// We use a memoized signal to filter the todos based on the current filter state.
|
|
// Whenever the todos or filter change, the filtered_todos will be recalculated.
|
|
// Note that we're only storing the IDs of the todos, not the todos themselves.
|
|
let filtered_todos = use_memo(move || {
|
|
let mut filtered_todos = todos
|
|
.read()
|
|
.iter()
|
|
.filter(|(_, item)| match filter() {
|
|
FilterState::All => true,
|
|
FilterState::Active => !item.checked,
|
|
FilterState::Completed => item.checked,
|
|
})
|
|
.map(|f| *f.0)
|
|
.collect::<Vec<_>>();
|
|
|
|
filtered_todos.sort_unstable();
|
|
|
|
filtered_todos
|
|
});
|
|
|
|
// Toggle all the todos to the opposite of the current state.
|
|
// If all todos are checked, uncheck them all. If any are unchecked, check them all.
|
|
let toggle_all = move |_| {
|
|
let check = active_todo_count() != 0;
|
|
for (_, item) in todos.write().iter_mut() {
|
|
item.checked = check;
|
|
}
|
|
};
|
|
|
|
rsx! {
|
|
document::Link { rel: "stylesheet", href: STYLE }
|
|
section { class: "todoapp",
|
|
TodoHeader { todos }
|
|
section { class: "main",
|
|
if !todos.read().is_empty() {
|
|
input {
|
|
id: "toggle-all",
|
|
class: "toggle-all",
|
|
r#type: "checkbox",
|
|
onchange: toggle_all,
|
|
checked: active_todo_count() == 0
|
|
}
|
|
label { r#for: "toggle-all" }
|
|
}
|
|
|
|
// Render the todos using the filtered_todos signal
|
|
// We pass the ID into the TodoEntry component so it can access the todo from the todos signal.
|
|
// Since we store the todos in a signal too, we also need to send down the todo list
|
|
ul { class: "todo-list",
|
|
for id in filtered_todos() {
|
|
TodoEntry { key: "{id}", id, todos }
|
|
}
|
|
}
|
|
|
|
// We only show the footer if there are todos.
|
|
if !todos.read().is_empty() {
|
|
ListFooter { active_todo_count, todos, filter }
|
|
}
|
|
}
|
|
}
|
|
|
|
// A simple info footer
|
|
footer { class: "info",
|
|
p { "Double-click to edit a todo" }
|
|
p {
|
|
"Created by "
|
|
a { href: "http://github.com/jkelleyrtp/", "jkelleyrtp" }
|
|
}
|
|
p {
|
|
"Part of "
|
|
a { href: "http://todomvc.com", "TodoMVC" }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#[component]
|
|
fn TodoHeader(mut todos: Signal<HashMap<u32, TodoItem>>) -> Element {
|
|
let mut draft = use_signal(|| "".to_string());
|
|
let mut todo_id = use_signal(|| 0);
|
|
|
|
let onkeydown = move |evt: KeyboardEvent| {
|
|
if evt.key() == Key::Enter && !draft.read().is_empty() {
|
|
let id = todo_id();
|
|
let todo = TodoItem {
|
|
checked: false,
|
|
contents: draft.to_string(),
|
|
};
|
|
todos.write().insert(id, todo);
|
|
todo_id += 1;
|
|
draft.set("".to_string());
|
|
}
|
|
};
|
|
|
|
rsx! {
|
|
header { class: "header",
|
|
h1 { "todos" }
|
|
input {
|
|
class: "new-todo",
|
|
placeholder: "What needs to be done?",
|
|
value: "{draft}",
|
|
autofocus: "true",
|
|
oninput: move |evt| draft.set(evt.value()),
|
|
onkeydown
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// A single todo entry
|
|
/// This takes the ID of the todo and the todos signal as props
|
|
/// We can use these together to memoize the todo contents and checked state
|
|
#[component]
|
|
fn TodoEntry(mut todos: Signal<HashMap<u32, TodoItem>>, id: u32) -> Element {
|
|
let mut is_editing = use_signal(|| false);
|
|
|
|
// To avoid re-rendering this component when the todo list changes, we isolate our reads to memos
|
|
// This way, the component will only re-render when the contents of the todo change, or when the editing state changes.
|
|
// This does involve taking a local clone of the todo contents, but it allows us to prevent this component from re-rendering
|
|
let checked = use_memo(move || todos.read().get(&id).unwrap().checked);
|
|
let contents = use_memo(move || todos.read().get(&id).unwrap().contents.clone());
|
|
|
|
rsx! {
|
|
li {
|
|
// Dioxus lets you use if statements in rsx to conditionally render attributes
|
|
// These will get merged into a single class attribute
|
|
class: if checked() { "completed" },
|
|
class: if is_editing() { "editing" },
|
|
|
|
// Some basic controls for the todo
|
|
div { class: "view",
|
|
input {
|
|
class: "toggle",
|
|
r#type: "checkbox",
|
|
id: "cbg-{id}",
|
|
checked: "{checked}",
|
|
oninput: move |evt| todos.write().get_mut(&id).unwrap().checked = evt.checked()
|
|
}
|
|
label {
|
|
r#for: "cbg-{id}",
|
|
ondoubleclick: move |_| is_editing.set(true),
|
|
onclick: |evt| evt.prevent_default(),
|
|
"{contents}"
|
|
}
|
|
button {
|
|
class: "destroy",
|
|
onclick: move |evt| {
|
|
evt.prevent_default();
|
|
todos.write().remove(&id);
|
|
},
|
|
}
|
|
}
|
|
|
|
// Only render the actual input if we're editing
|
|
if is_editing() {
|
|
input {
|
|
class: "edit",
|
|
value: "{contents}",
|
|
oninput: move |evt| todos.write().get_mut(&id).unwrap().contents = evt.value(),
|
|
autofocus: "true",
|
|
onfocusout: move |_| is_editing.set(false),
|
|
onkeydown: move |evt| {
|
|
match evt.key() {
|
|
Key::Enter | Key::Escape | Key::Tab => is_editing.set(false),
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#[component]
|
|
fn ListFooter(
|
|
mut todos: Signal<HashMap<u32, TodoItem>>,
|
|
active_todo_count: ReadOnlySignal<usize>,
|
|
mut filter: Signal<FilterState>,
|
|
) -> Element {
|
|
// We use a memoized signal to calculate whether we should show the "Clear completed" button.
|
|
// This will recompute whenever the todos change, and if the value is true, the button will be shown.
|
|
let show_clear_completed = use_memo(move || todos.read().values().any(|todo| todo.checked));
|
|
|
|
rsx! {
|
|
footer { class: "footer",
|
|
span { class: "todo-count",
|
|
strong { "{active_todo_count} " }
|
|
span {
|
|
match active_todo_count() {
|
|
1 => "item",
|
|
_ => "items",
|
|
}
|
|
" left"
|
|
}
|
|
}
|
|
ul { class: "filters",
|
|
for (state , state_text , url) in [
|
|
(FilterState::All, "All", "#/"),
|
|
(FilterState::Active, "Active", "#/active"),
|
|
(FilterState::Completed, "Completed", "#/completed"),
|
|
] {
|
|
li {
|
|
a {
|
|
href: url,
|
|
class: if filter() == state { "selected" },
|
|
onclick: move |evt| {
|
|
evt.prevent_default();
|
|
filter.set(state)
|
|
},
|
|
{state_text}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if show_clear_completed() {
|
|
button {
|
|
class: "clear-completed",
|
|
onclick: move |_| todos.write().retain(|_, todo| !todo.checked),
|
|
"Clear completed"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|