rsx_usage.rs 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. #![allow(non_snake_case)]
  2. //! A tour of the rsx! macro
  3. //! ------------------------
  4. //!
  5. //! This example serves as an informal quick reference of all the things that the rsx! macro can do.
  6. //!
  7. //! A full in-depth reference guide is available at: https://www.notion.so/rsx-macro-basics-ef6e367dec124f4784e736d91b0d0b19
  8. //!
  9. //! ### Elements
  10. //! - Create any element from its tag
  11. //! - Accept compile-safe attributes for each tag
  12. //! - Display documentation for elements
  13. //! - Arguments instead of String
  14. //! - Text
  15. //! - Inline Styles
  16. //!
  17. //! ## General Concepts
  18. //! - Iterators
  19. //! - Keys
  20. //! - Match statements
  21. //! - Conditional Rendering
  22. //!
  23. //! ### Events
  24. //! - Handle events with the "onXYZ" syntax
  25. //! - Closures can capture their environment with the 'a lifetime
  26. //!
  27. //!
  28. //! ### Components
  29. //! - Components can be made by specifying the name
  30. //! - Components can be referenced by path
  31. //! - Components may have optional parameters
  32. //! - Components may have their properties specified by spread syntax
  33. //! - Components may accept child nodes
  34. //! - Components that accept "onXYZ" get those closures bump allocated
  35. //!
  36. //! ### Fragments
  37. //! - Allow fragments using the built-in `Fragment` component
  38. //! - Accept a list of vnodes as children for a Fragment component
  39. //! - Allow keyed fragments in iterators
  40. //! - Allow top-level fragments
  41. fn main() {
  42. dioxus_desktop::launch(app);
  43. }
  44. use core::{fmt, str::FromStr};
  45. use std::fmt::Display;
  46. use baller::Baller;
  47. use dioxus::prelude::*;
  48. fn app(cx: Scope) -> Element {
  49. let formatting = "formatting!";
  50. let formatting_tuple = ("a", "b");
  51. let lazy_fmt = format_args!("lazily formatted text");
  52. cx.render(rsx! {
  53. div {
  54. // Elements
  55. div {}
  56. h1 {"Some text"}
  57. h1 {"Some text with {formatting}"}
  58. h1 {"Formatting basic expressions {formatting_tuple.0} and {formatting_tuple.1}"}
  59. h1 {"Formatting without interpolation " formatting_tuple.0 "and" formatting_tuple.1 }
  60. h2 {
  61. "Multiple"
  62. "Text"
  63. "Blocks"
  64. "Use comments as separators in html"
  65. }
  66. div {
  67. h1 {"multiple"}
  68. h2 {"nested"}
  69. h3 {"elements"}
  70. }
  71. div {
  72. class: "my special div",
  73. h1 {"Headers and attributes!"}
  74. }
  75. div {
  76. // pass simple rust expressions in
  77. class: lazy_fmt,
  78. id: format_args!("attributes can be passed lazily with std::fmt::Arguments"),
  79. div {
  80. class: {
  81. const WORD: &str = "expressions";
  82. format_args!("Arguments can be passed in through curly braces for complex {WORD}")
  83. }
  84. }
  85. }
  86. // Expressions can be used in element position too:
  87. rsx!(p { "More templating!" }),
  88. // Iterators
  89. (0..10).map(|i| rsx!(li { "{i}" })),
  90. // Iterators within expressions
  91. {
  92. let data = std::collections::HashMap::<&'static str, &'static str>::new();
  93. // Iterators *should* have keys when you can provide them.
  94. // Keys make your app run faster. Make sure your keys are stable, unique, and predictable.
  95. // Using an "ID" associated with your data is a good idea.
  96. data.into_iter().map(|(k, v)| rsx!(li { key: "{k}", "{v}" }))
  97. }
  98. // Matching
  99. match true {
  100. true => rsx!( h1 {"Top text"}),
  101. false => rsx!( h1 {"Bottom text"})
  102. }
  103. // Conditional rendering
  104. // Dioxus conditional rendering is based around None/Some. We have no special syntax for conditionals.
  105. // You can convert a bool condition to rsx! with .then and .or
  106. true.then(|| rsx!(div {})),
  107. // Alternatively, you can use the "if" syntax - but both branches must be resolve to Element
  108. if false {
  109. rsx!(h1 {"Top text"})
  110. } else {
  111. rsx!(h1 {"Bottom text"})
  112. }
  113. // Using optionals for diverging branches
  114. if true {
  115. Some(rsx!(h1 {"Top text"}))
  116. } else {
  117. None
  118. }
  119. // returning "None" without a diverging branch is a bit noisy... but rare in practice
  120. None as Option<()>,
  121. // can also just use empty fragments
  122. Fragment {}
  123. // Fragments let you insert groups of nodes without a parent.
  124. // This lets you make components that insert elements as siblings without a container.
  125. div {"A"}
  126. Fragment {
  127. div {"B"}
  128. div {"C"}
  129. Fragment {
  130. "D"
  131. Fragment {
  132. "E"
  133. "F"
  134. }
  135. }
  136. }
  137. // Components
  138. // Can accept any paths
  139. // Notice how you still get syntax highlighting and IDE support :)
  140. Baller {}
  141. baller::Baller {}
  142. crate::baller::Baller {}
  143. // Can take properties
  144. Taller { a: "asd" }
  145. // Can take optional properties
  146. Taller { a: "asd" }
  147. // Can pass in props directly as an expression
  148. {
  149. let props = TallerProps {a: "hello", children: cx.render(rsx!(()))};
  150. rsx!(Taller { ..props })
  151. }
  152. // Spreading can also be overridden manually
  153. Taller {
  154. ..TallerProps { a: "ballin!", children: cx.render(rsx!(()) )},
  155. a: "not ballin!"
  156. }
  157. // Can take children too!
  158. Taller { a: "asd", div {"hello world!"} }
  159. // This component's props are defined *inline* with the `inline_props` macro
  160. WithInline { text: "using functionc all syntax" }
  161. // Components can be generic too
  162. // This component takes i32 type to give you typed input
  163. TypedInput::<i32> {}
  164. // Type inference can be used too
  165. TypedInput { initial: 10.0 }
  166. // geneircs with the `inline_props` macro
  167. Label { text: "hello geneirc world!" }
  168. Label { text: 99.9 }
  169. // Lowercase components work too, as long as they are access using a path
  170. baller::lowercase_component {}
  171. // For in-scope lowercase components, use the `self` keyword
  172. self::lowercase_helper {}
  173. // helper functions
  174. // Anything that implements IntoVnode can be dropped directly into Rsx
  175. helper(cx, "hello world!")
  176. // Strings can be supplied directly
  177. String::from("Hello world!")
  178. // So can format_args
  179. format_args!("Hello {}!", "world")
  180. // Or we can shell out to a helper function
  181. format_dollars(10, 50)
  182. }
  183. })
  184. }
  185. fn format_dollars(dollars: u32, cents: u32) -> String {
  186. format!("${dollars}.{cents:02}")
  187. }
  188. fn helper<'a>(cx: &'a ScopeState, text: &str) -> Element<'a> {
  189. cx.render(rsx! {
  190. p { "{text}" }
  191. })
  192. }
  193. fn lowercase_helper(cx: Scope) -> Element {
  194. cx.render(rsx! {
  195. "asd"
  196. })
  197. }
  198. mod baller {
  199. use super::*;
  200. #[derive(Props, PartialEq, Eq)]
  201. pub struct BallerProps {}
  202. #[allow(non_snake_case)]
  203. /// This component totally balls
  204. pub fn Baller(_: Scope<BallerProps>) -> Element {
  205. todo!()
  206. }
  207. pub fn lowercase_component(cx: Scope) -> Element {
  208. cx.render(rsx! { "look ma, no uppercase" })
  209. }
  210. }
  211. #[derive(Props)]
  212. pub struct TallerProps<'a> {
  213. /// Fields are documented and accessible in rsx!
  214. a: &'static str,
  215. children: Element<'a>,
  216. }
  217. /// Documention for this component is visible within the rsx macro
  218. #[allow(non_snake_case)]
  219. pub fn Taller<'a>(cx: Scope<'a, TallerProps<'a>>) -> Element {
  220. cx.render(rsx! {
  221. &cx.props.children
  222. })
  223. }
  224. #[derive(Props, PartialEq, Eq)]
  225. pub struct TypedInputProps<T> {
  226. #[props(optional, default)]
  227. initial: Option<T>,
  228. }
  229. #[allow(non_snake_case)]
  230. pub fn TypedInput<T>(_: Scope<TypedInputProps<T>>) -> Element
  231. where
  232. T: FromStr + fmt::Display,
  233. <T as FromStr>::Err: std::fmt::Display,
  234. {
  235. todo!()
  236. }
  237. #[inline_props]
  238. fn WithInline<'a>(cx: Scope<'a>, text: &'a str) -> Element {
  239. cx.render(rsx! {
  240. p { "{text}" }
  241. })
  242. }
  243. // generic component with inline_props too
  244. #[inline_props]
  245. fn Label<T>(cx: Scope, text: T) -> Element
  246. where
  247. T: Display,
  248. {
  249. cx.render(rsx! {
  250. p { "{text}" }
  251. })
  252. }