1
0

todomvc.rs 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. #![allow(non_snake_case)]
  2. use dioxus::prelude::*;
  3. use dioxus_elements::input_data::keyboard_types::Key;
  4. use std::collections::HashMap;
  5. fn main() {
  6. launch(app);
  7. }
  8. #[derive(PartialEq, Eq, Clone, Copy)]
  9. enum FilterState {
  10. All,
  11. Active,
  12. Completed,
  13. }
  14. #[derive(Debug, PartialEq, Eq)]
  15. struct TodoItem {
  16. id: u32,
  17. checked: bool,
  18. contents: String,
  19. }
  20. const STYLE: &str = include_str!("./assets/todomvc.css");
  21. fn app() -> Element {
  22. let mut todos = use_signal(HashMap::<u32, TodoItem>::new);
  23. let filter = use_signal(|| FilterState::All);
  24. let active_todo_count =
  25. use_memo(move || todos.read().values().filter(|item| !item.checked).count());
  26. let filtered_todos = use_memo(move || {
  27. let mut filtered_todos = todos
  28. .read()
  29. .iter()
  30. .filter(|(_, item)| match filter() {
  31. FilterState::All => true,
  32. FilterState::Active => !item.checked,
  33. FilterState::Completed => item.checked,
  34. })
  35. .map(|f| *f.0)
  36. .collect::<Vec<_>>();
  37. filtered_todos.sort_unstable();
  38. filtered_todos
  39. });
  40. let toggle_all = move |_| {
  41. let check = active_todo_count() != 0;
  42. for (_, item) in todos.write().iter_mut() {
  43. item.checked = check;
  44. }
  45. };
  46. rsx! {
  47. section { class: "todoapp",
  48. style { {STYLE} }
  49. TodoHeader { todos }
  50. section { class: "main",
  51. if !todos.read().is_empty() {
  52. input {
  53. id: "toggle-all",
  54. class: "toggle-all",
  55. r#type: "checkbox",
  56. onchange: toggle_all,
  57. checked: active_todo_count() == 0,
  58. }
  59. label { r#for: "toggle-all" }
  60. }
  61. ul { class: "todo-list",
  62. for id in filtered_todos() {
  63. TodoEntry { key: "{id}", id, todos }
  64. }
  65. }
  66. if !todos.read().is_empty() {
  67. ListFooter { active_todo_count, todos, filter }
  68. }
  69. }
  70. }
  71. PageFooter {}
  72. }
  73. }
  74. #[component]
  75. fn TodoHeader(mut todos: Signal<HashMap<u32, TodoItem>>) -> Element {
  76. let mut draft = use_signal(|| "".to_string());
  77. let mut todo_id = use_signal(|| 0);
  78. let onkeydown = move |evt: KeyboardEvent| {
  79. if evt.key() == Key::Enter && !draft.read().is_empty() {
  80. let id = todo_id();
  81. let todo = TodoItem {
  82. id,
  83. checked: false,
  84. contents: draft.to_string(),
  85. };
  86. todos.write().insert(id, todo);
  87. todo_id += 1;
  88. draft.set("".to_string());
  89. }
  90. };
  91. rsx! {
  92. header { class: "header",
  93. h1 { "todos" }
  94. input {
  95. class: "new-todo",
  96. placeholder: "What needs to be done?",
  97. value: "{draft}",
  98. autofocus: "true",
  99. oninput: move |evt| draft.set(evt.value().clone()),
  100. onkeydown,
  101. }
  102. }
  103. }
  104. }
  105. #[component]
  106. fn TodoEntry(mut todos: Signal<HashMap<u32, TodoItem>>, id: u32) -> Element {
  107. let mut is_editing = use_signal(|| false);
  108. let checked = use_memo(move || todos.read().get(&id).unwrap().checked);
  109. let contents = use_memo(move || todos.read().get(&id).unwrap().contents.clone());
  110. rsx! {
  111. li { class: if checked() { "completed" }, class: if is_editing() { "editing" },
  112. div { class: "view",
  113. input {
  114. class: "toggle",
  115. r#type: "checkbox",
  116. id: "cbg-{id}",
  117. checked: "{checked}",
  118. oninput: move |evt| todos.write().get_mut(&id).unwrap().checked = evt.value().parse().unwrap(),
  119. }
  120. label {
  121. r#for: "cbg-{id}",
  122. ondoubleclick: move |_| is_editing.set(true),
  123. prevent_default: "onclick",
  124. "{contents}"
  125. }
  126. button {
  127. class: "destroy",
  128. onclick: move |_| { todos.write().remove(&id); },
  129. prevent_default: "onclick"
  130. }
  131. }
  132. if is_editing() {
  133. input {
  134. class: "edit",
  135. value: "{contents}",
  136. oninput: move |evt| todos.write().get_mut(&id).unwrap().contents = evt.value(),
  137. autofocus: "true",
  138. onfocusout: move |_| is_editing.set(false),
  139. onkeydown: move |evt| {
  140. match evt.key() {
  141. Key::Enter | Key::Escape | Key::Tab => is_editing.set(false),
  142. _ => {}
  143. }
  144. }
  145. }
  146. }
  147. }
  148. }
  149. }
  150. #[component]
  151. fn ListFooter(
  152. mut todos: Signal<HashMap<u32, TodoItem>>,
  153. active_todo_count: ReadOnlySignal<usize>,
  154. mut filter: Signal<FilterState>,
  155. ) -> Element {
  156. let show_clear_completed = use_memo(move || todos.read().values().any(|todo| todo.checked));
  157. rsx! {
  158. footer { class: "footer",
  159. span { class: "todo-count",
  160. strong { "{active_todo_count} " }
  161. span {
  162. match active_todo_count() {
  163. 1 => "item",
  164. _ => "items",
  165. }
  166. " left"
  167. }
  168. }
  169. ul { class: "filters",
  170. for (state , state_text , url) in [
  171. (FilterState::All, "All", "#/"),
  172. (FilterState::Active, "Active", "#/active"),
  173. (FilterState::Completed, "Completed", "#/completed"),
  174. ] {
  175. li {
  176. a {
  177. href: url,
  178. class: if filter() == state { "selected" },
  179. onclick: move |_| filter.set(state),
  180. prevent_default: "onclick",
  181. {state_text}
  182. }
  183. }
  184. }
  185. }
  186. if show_clear_completed() {
  187. button {
  188. class: "clear-completed",
  189. onclick: move |_| todos.write().retain(|_, todo| !todo.checked),
  190. "Clear completed"
  191. }
  192. }
  193. }
  194. }
  195. }
  196. fn PageFooter() -> Element {
  197. rsx! {
  198. footer { class: "info",
  199. p { "Double-click to edit a todo" }
  200. p {
  201. "Created by "
  202. a { href: "http://github.com/jkelleyrtp/", "jkelleyrtp" }
  203. }
  204. p {
  205. "Part of "
  206. a { href: "http://todomvc.com", "TodoMVC" }
  207. }
  208. }
  209. }
  210. }