dioxus/examples/todomvc.rs

149 lines
5.4 KiB
Rust
Raw Normal View History

2021-07-16 20:11:25 +00:00
use dioxus::prelude::*;
2021-10-11 02:27:08 +00:00
fn main() {
dioxus::desktop::launch(app);
2021-07-16 20:11:25 +00:00
}
#[derive(PartialEq)]
pub enum FilterState {
All,
Active,
Completed,
}
#[derive(Debug, PartialEq, Clone)]
2021-07-16 20:11:25 +00:00
pub struct TodoItem {
pub id: u32,
pub checked: bool,
pub contents: String,
}
pub fn app(cx: Scope<()>) -> Element {
2022-01-26 02:41:40 +00:00
let (todos, set_todos) = use_state(&cx, im_rc::HashMap::<u32, TodoItem>::default);
let (filter, set_filter) = use_state(&cx, || FilterState::All);
let (draft, set_draft) = use_state(&cx, || "".to_string());
let (todo_id, set_todo_id) = use_state(&cx, || 0);
2021-07-16 20:11:25 +00:00
// Filter the todos based on the filter state
let mut filtered_todos = todos
2021-07-16 20:11:25 +00:00
.iter()
2022-01-26 02:41:40 +00:00
.filter(|(_, item)| match filter {
2021-07-16 20:11:25 +00:00
FilterState::All => true,
FilterState::Active => !item.checked,
FilterState::Completed => item.checked,
})
.map(|f| *f.0)
2021-07-16 20:11:25 +00:00
.collect::<Vec<_>>();
filtered_todos.sort_unstable();
2021-07-16 20:11:25 +00:00
let show_clear_completed = todos.values().any(|todo| todo.checked);
let items_left = filtered_todos.len();
2021-07-16 20:11:25 +00:00
let item_text = match items_left {
1 => "item",
_ => "items",
};
cx.render(rsx!{
section { class: "todoapp",
style { [include_str!("./assets/todomvc.css")] }
div {
header { class: "header",
h1 {"todos"}
input {
class: "new-todo",
placeholder: "What needs to be done?",
value: "{draft}",
autofocus: "true",
2022-01-26 02:41:40 +00:00
oninput: move |evt| set_draft(evt.value.clone()),
onkeydown: move |evt| {
2022-01-16 20:56:48 +00:00
if evt.key == "Enter" && !draft.is_empty() {
2022-01-26 02:41:40 +00:00
set_todos.make_mut().insert(
2022-01-16 20:56:48 +00:00
*todo_id,
TodoItem {
id: *todo_id,
checked: false,
2022-01-26 02:41:40 +00:00
contents: draft.clone(),
2022-01-16 20:56:48 +00:00
},
);
set_todo_id(todo_id + 1);
2022-01-26 02:41:40 +00:00
set_draft("".to_string());
}
}
2021-09-24 06:10:54 +00:00
}
}
ul { class: "todo-list",
2022-01-26 02:41:40 +00:00
filtered_todos.iter().map(|id| rsx!(todo_entry( key: "{id}", id: *id, set_todos: set_todos )))
}
(!todos.is_empty()).then(|| rsx!(
footer { class: "footer",
2022-01-03 06:20:05 +00:00
span { class: "todo-count",
strong {"{items_left} "}
span {"{item_text} left"}
}
ul { class: "filters",
2022-01-26 02:41:40 +00:00
li { class: "All", a { onclick: move |_| set_filter(FilterState::All), "All" }}
li { class: "Active", a { onclick: move |_| set_filter(FilterState::Active), "Active" }}
li { class: "Completed", a { onclick: move |_| set_filter(FilterState::Completed), "Completed" }}
}
(show_clear_completed).then(|| rsx!(
2022-01-03 06:20:05 +00:00
button {
class: "clear-completed",
2022-01-26 02:41:40 +00:00
onclick: move |_| set_todos.make_mut().retain(|_, todo| !todo.checked),
"Clear completed"
}
))
}
))
}
2021-07-16 20:11:25 +00:00
}
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" }}
}
})
}
2021-07-16 20:11:25 +00:00
#[derive(Props)]
pub struct TodoEntryProps<'a> {
2022-01-26 02:41:40 +00:00
set_todos: &'a UseState<im_rc::HashMap<u32, TodoItem>>,
id: u32,
2021-07-16 20:11:25 +00:00
}
pub fn todo_entry<'a>(cx: Scope<'a, TodoEntryProps<'a>>) -> Element {
2022-01-26 02:41:40 +00:00
let (is_editing, set_is_editing) = use_state(&cx, || false);
let todos = cx.props.set_todos.get();
let todo = &todos[&cx.props.id];
let completed = if todo.checked { "completed" } else { "" };
2022-01-26 02:41:40 +00:00
let editing = if *is_editing { "editing" } else { "" };
2021-07-16 20:11:25 +00:00
rsx!(cx, li {
class: "{completed} {editing}",
2022-01-26 02:41:40 +00:00
onclick: move |_| set_is_editing(true),
onfocusout: move |_| set_is_editing(false),
div { class: "view",
input { class: "toggle", r#type: "checkbox", id: "cbg-{todo.id}", checked: "{todo.checked}",
onchange: move |evt| {
2022-01-26 02:41:40 +00:00
cx.props.set_todos.make_mut()[&cx.props.id].checked = evt.value.parse().unwrap();
}
2021-07-16 20:11:25 +00:00
}
label { r#for: "cbg-{todo.id}", pointer_events: "none", "{todo.contents}" }
}
is_editing.then(|| rsx!{
input {
class: "edit",
value: "{todo.contents}",
2022-01-26 02:41:40 +00:00
oninput: move |evt| cx.props.set_todos.make_mut()[&cx.props.id].contents = evt.value.clone(),
autofocus: "true",
onkeydown: move |evt| {
match evt.key.as_str() {
2022-01-26 02:41:40 +00:00
"Enter" | "Escape" | "Tab" => set_is_editing(false),
_ => {}
}
},
}
})
2021-09-24 06:10:54 +00:00
})
2021-07-16 20:11:25 +00:00
}