selector.rs 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. #![allow(unused, non_upper_case_globals, non_snake_case)]
  2. use std::collections::HashMap;
  3. use std::rc::Rc;
  4. use dioxus::html::p;
  5. use dioxus::prelude::*;
  6. use dioxus_core::ElementId;
  7. use dioxus_signals::*;
  8. #[test]
  9. fn memos_rerun() {
  10. let _ = simple_logger::SimpleLogger::new().init();
  11. #[derive(Default)]
  12. struct RunCounter {
  13. component: usize,
  14. effect: usize,
  15. }
  16. let counter = Rc::new(RefCell::new(RunCounter::default()));
  17. let mut dom = VirtualDom::new_with_props(
  18. |cx| {
  19. let counter = cx.props;
  20. counter.borrow_mut().component += 1;
  21. let mut signal = use_signal(|| 0);
  22. let memo = cx.use_hook(move || {
  23. to_owned![counter];
  24. selector(move || {
  25. counter.borrow_mut().effect += 1;
  26. println!("Signal: {:?}", signal);
  27. signal.cloned()
  28. })
  29. });
  30. let generation = use_signal(cx, || cx.generation());
  31. generation.set(cx.generation());
  32. dioxus_signals::use_effect(cx, move || {
  33. if generation == 1 {
  34. assert_eq!(memo.value(), 0);
  35. }
  36. if generation == 3 {
  37. assert_eq!(memo.value(), 1);
  38. }
  39. });
  40. signal += 1;
  41. rsx! { div {} }
  42. },
  43. counter.clone(),
  44. );
  45. let _ = dom.rebuild().santize();
  46. let _ = dom.render_immediate();
  47. let current_counter = counter.borrow();
  48. assert_eq!(current_counter.component, 1);
  49. assert_eq!(current_counter.effect, 2);
  50. }
  51. #[tokio::test]
  52. async fn memos_prevents_component_rerun() {
  53. let _ = simple_logger::SimpleLogger::new().init();
  54. #[derive(Default)]
  55. struct RunCounter {
  56. component: usize,
  57. effect: usize,
  58. }
  59. let counter = Rc::new(RefCell::new(RunCounter::default()));
  60. let mut dom = VirtualDom::new_with_props(
  61. |cx| {
  62. let mut signal = use_signal(|| 0);
  63. if generation() == 1 {
  64. *signal.write() = 0;
  65. }
  66. if generation() == 2 {
  67. println!("Writing to signal");
  68. *signal.write() = 1;
  69. }
  70. rsx! { Child { signal: signal, counter: cx.props.clone() } }
  71. },
  72. counter.clone(),
  73. );
  74. #[derive(Default, Props)]
  75. struct ChildProps {
  76. signal: Signal<usize>,
  77. counter: Rc<RefCell<RunCounter>>,
  78. }
  79. impl PartialEq for ChildProps {
  80. fn eq(&self, other: &Self) -> bool {
  81. self.signal == other.signal
  82. }
  83. }
  84. fn Child(cx: Scope<ChildProps>) -> Element {
  85. let counter = &cx.props.counter;
  86. let signal = cx.props.signal;
  87. counter.borrow_mut().component += 1;
  88. let memo = cx.use_hook(move || {
  89. to_owned![counter];
  90. selector(move || {
  91. counter.borrow_mut().effect += 1;
  92. println!("Signal: {:?}", signal);
  93. signal.value()
  94. })
  95. });
  96. match generation() {
  97. 0 => {
  98. assert_eq!(memo.value(), 0);
  99. }
  100. 1 => {
  101. assert_eq!(memo.value(), 1);
  102. }
  103. _ => panic!("Unexpected generation"),
  104. }
  105. rsx! { div {} }
  106. }
  107. let _ = dom.rebuild().santize();
  108. dom.mark_dirty(ScopeId::ROOT);
  109. dom.render_immediate();
  110. dom.render_immediate();
  111. {
  112. let current_counter = counter.borrow();
  113. assert_eq!(current_counter.component, 1);
  114. assert_eq!(current_counter.effect, 2);
  115. }
  116. dom.mark_dirty(ScopeId::ROOT);
  117. dom.render_immediate();
  118. dom.render_immediate();
  119. {
  120. let current_counter = counter.borrow();
  121. assert_eq!(current_counter.component, 2);
  122. assert_eq!(current_counter.effect, 3);
  123. }
  124. }