memo.rs 3.6 KB

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