diffing.rs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824
  1. //! Diffing Tests
  2. //!
  3. //! These tests only verify that the diffing algorithm works properly for single components.
  4. //!
  5. //! It does not validated that component lifecycles work properly. This is done in another test file.
  6. use dioxus::{nodes::VSuspended, prelude::*, DomEdit, TestDom};
  7. use dioxus_core as dioxus;
  8. use dioxus_core_macro::*;
  9. use dioxus_html as dioxus_elements;
  10. mod test_logging;
  11. fn new_dom() -> TestDom {
  12. const IS_LOGGING_ENABLED: bool = false;
  13. test_logging::set_up_logging(IS_LOGGING_ENABLED);
  14. TestDom::new()
  15. }
  16. use DomEdit::*;
  17. /// Should push the text node onto the stack and modify it
  18. #[test]
  19. fn html_and_rsx_generate_the_same_output() {
  20. let dom = new_dom();
  21. let (create, change) = dom.lazy_diff(
  22. rsx! ( div { "Hello world" } ),
  23. rsx! ( div { "Goodbye world" } ),
  24. );
  25. assert_eq!(
  26. create.edits,
  27. [
  28. CreateElement {
  29. root: 0,
  30. tag: "div"
  31. },
  32. CreateTextNode {
  33. root: 1,
  34. text: "Hello world"
  35. },
  36. AppendChildren { many: 1 },
  37. AppendChildren { many: 1 },
  38. ]
  39. );
  40. assert_eq!(
  41. change.edits,
  42. [
  43. PushRoot { root: 1 },
  44. SetText {
  45. text: "Goodbye world"
  46. },
  47. PopRoot
  48. ]
  49. );
  50. }
  51. /// Should result in 3 elements on the stack
  52. #[test]
  53. fn fragments_create_properly() {
  54. let dom = new_dom();
  55. let create = dom.create(rsx! {
  56. div { "Hello a" }
  57. div { "Hello b" }
  58. div { "Hello c" }
  59. });
  60. assert_eq!(
  61. create.edits,
  62. [
  63. CreateElement {
  64. root: 0,
  65. tag: "div"
  66. },
  67. CreateTextNode {
  68. root: 1,
  69. text: "Hello a"
  70. },
  71. AppendChildren { many: 1 },
  72. CreateElement {
  73. root: 2,
  74. tag: "div"
  75. },
  76. CreateTextNode {
  77. root: 3,
  78. text: "Hello b"
  79. },
  80. AppendChildren { many: 1 },
  81. CreateElement {
  82. root: 4,
  83. tag: "div"
  84. },
  85. CreateTextNode {
  86. root: 5,
  87. text: "Hello c"
  88. },
  89. AppendChildren { many: 1 },
  90. AppendChildren { many: 3 },
  91. ]
  92. );
  93. }
  94. /// Should result in the creation of an anchor (placeholder) and then a replacewith
  95. #[test]
  96. fn empty_fragments_create_anchors() {
  97. let dom = new_dom();
  98. let left = rsx!({ (0..0).map(|_f| rsx! { div {}}) });
  99. let right = rsx!({ (0..1).map(|_f| rsx! { div {}}) });
  100. let (create, change) = dom.lazy_diff(left, right);
  101. assert_eq!(
  102. create.edits,
  103. [CreatePlaceholder { root: 0 }, AppendChildren { many: 1 }]
  104. );
  105. assert_eq!(
  106. change.edits,
  107. [
  108. CreateElement {
  109. root: 1,
  110. tag: "div"
  111. },
  112. ReplaceWith { m: 1, root: 0 }
  113. ]
  114. );
  115. }
  116. /// Should result in the creation of an anchor (placeholder) and then a replacewith m=5
  117. #[test]
  118. fn empty_fragments_create_many_anchors() {
  119. let dom = new_dom();
  120. let left = rsx!({ (0..0).map(|_f| rsx! { div {}}) });
  121. let right = rsx!({ (0..5).map(|_f| rsx! { div {}}) });
  122. let (create, change) = dom.lazy_diff(left, right);
  123. assert_eq!(
  124. create.edits,
  125. [CreatePlaceholder { root: 0 }, AppendChildren { many: 1 }]
  126. );
  127. assert_eq!(
  128. change.edits,
  129. [
  130. CreateElement {
  131. root: 1,
  132. tag: "div"
  133. },
  134. CreateElement {
  135. root: 2,
  136. tag: "div"
  137. },
  138. CreateElement {
  139. root: 3,
  140. tag: "div"
  141. },
  142. CreateElement {
  143. root: 4,
  144. tag: "div"
  145. },
  146. CreateElement {
  147. root: 5,
  148. tag: "div"
  149. },
  150. ReplaceWith { m: 5, root: 0 }
  151. ]
  152. );
  153. }
  154. /// Should result in the creation of an anchor (placeholder) and then a replacewith
  155. /// Includes child nodes inside the fragment
  156. #[test]
  157. fn empty_fragments_create_anchors_with_many_children() {
  158. let dom = new_dom();
  159. let left = rsx!({ (0..0).map(|_| rsx! { div {} }) });
  160. let right = rsx!({
  161. (0..3).map(|f| {
  162. rsx! { div { "hello: {f}" }}
  163. })
  164. });
  165. let (create, change) = dom.lazy_diff(left, right);
  166. assert_eq!(
  167. create.edits,
  168. [CreatePlaceholder { root: 0 }, AppendChildren { many: 1 }]
  169. );
  170. assert_eq!(
  171. change.edits,
  172. [
  173. CreateElement {
  174. root: 1,
  175. tag: "div"
  176. },
  177. CreateTextNode {
  178. text: "hello: 0",
  179. root: 2
  180. },
  181. AppendChildren { many: 1 },
  182. CreateElement {
  183. root: 3,
  184. tag: "div"
  185. },
  186. CreateTextNode {
  187. text: "hello: 1",
  188. root: 4
  189. },
  190. AppendChildren { many: 1 },
  191. CreateElement {
  192. root: 5,
  193. tag: "div"
  194. },
  195. CreateTextNode {
  196. text: "hello: 2",
  197. root: 6
  198. },
  199. AppendChildren { many: 1 },
  200. ReplaceWith { m: 3, root: 0 }
  201. ]
  202. );
  203. }
  204. /// Should result in every node being pushed and then replaced with an anchor
  205. #[test]
  206. fn many_items_become_fragment() {
  207. let dom = new_dom();
  208. let left = rsx!({
  209. (0..2).map(|_| {
  210. rsx! { div { "hello" }}
  211. })
  212. });
  213. let right = rsx!({ (0..0).map(|_| rsx! { div {} }) });
  214. let (create, change) = dom.lazy_diff(left, right);
  215. assert_eq!(
  216. create.edits,
  217. [
  218. CreateElement {
  219. root: 0,
  220. tag: "div"
  221. },
  222. CreateTextNode {
  223. text: "hello",
  224. root: 1
  225. },
  226. AppendChildren { many: 1 },
  227. CreateElement {
  228. root: 2,
  229. tag: "div"
  230. },
  231. CreateTextNode {
  232. text: "hello",
  233. root: 3
  234. },
  235. AppendChildren { many: 1 },
  236. AppendChildren { many: 2 },
  237. ]
  238. );
  239. // hmmmmmmmmm worried about reusing IDs that we shouldnt be
  240. assert_eq!(
  241. change.edits,
  242. [
  243. Remove { root: 2 },
  244. CreatePlaceholder { root: 4 },
  245. ReplaceWith { root: 0, m: 1 },
  246. ]
  247. );
  248. }
  249. /// Should result in no edits
  250. #[test]
  251. fn two_equal_fragments_are_equal() {
  252. let dom = new_dom();
  253. let left = rsx!({
  254. (0..2).map(|_| {
  255. rsx! { div { "hello" }}
  256. })
  257. });
  258. let right = rsx!({
  259. (0..2).map(|_| {
  260. rsx! { div { "hello" }}
  261. })
  262. });
  263. let (_create, change) = dom.lazy_diff(left, right);
  264. assert!(change.edits.is_empty());
  265. }
  266. /// Should result the creation of more nodes appended after the old last node
  267. #[test]
  268. fn two_fragments_with_differrent_elements_are_differet() {
  269. let dom = new_dom();
  270. let left = rsx!(
  271. { (0..2).map(|_| rsx! { div { }} ) }
  272. p {}
  273. );
  274. let right = rsx!(
  275. { (0..5).map(|_| rsx! (h1 { }) ) }
  276. p {}
  277. );
  278. let (_create, changes) = dom.lazy_diff(left, right);
  279. log::debug!("{:#?}", &changes);
  280. assert_eq!(
  281. changes.edits,
  282. [
  283. // create the new h1s
  284. CreateElement { tag: "h1", root: 3 },
  285. CreateElement { tag: "h1", root: 4 },
  286. CreateElement { tag: "h1", root: 5 },
  287. InsertAfter { root: 1, n: 3 },
  288. // replace the divs with new h1s
  289. CreateElement { tag: "h1", root: 6 },
  290. ReplaceWith { root: 0, m: 1 },
  291. CreateElement { tag: "h1", root: 7 },
  292. ReplaceWith { root: 1, m: 1 },
  293. ]
  294. );
  295. }
  296. /// Should result in multiple nodes destroyed - with changes to the first nodes
  297. #[test]
  298. fn two_fragments_with_differrent_elements_are_differet_shorter() {
  299. let dom = new_dom();
  300. let left = rsx!(
  301. {(0..5).map(|f| {rsx! { div { }}})}
  302. p {}
  303. );
  304. let right = rsx!(
  305. {(0..2).map(|f| {rsx! { h1 { }}})}
  306. p {}
  307. );
  308. let (create, change) = dom.lazy_diff(left, right);
  309. assert_eq!(
  310. create.edits,
  311. [
  312. CreateElement {
  313. root: 0,
  314. tag: "div"
  315. },
  316. CreateElement {
  317. root: 1,
  318. tag: "div"
  319. },
  320. CreateElement {
  321. root: 2,
  322. tag: "div"
  323. },
  324. CreateElement {
  325. root: 3,
  326. tag: "div"
  327. },
  328. CreateElement {
  329. root: 4,
  330. tag: "div"
  331. },
  332. CreateElement { root: 5, tag: "p" },
  333. AppendChildren { many: 6 },
  334. ]
  335. );
  336. assert_eq!(
  337. change.edits,
  338. [
  339. Remove { root: 2 },
  340. Remove { root: 3 },
  341. Remove { root: 4 },
  342. CreateElement { root: 6, tag: "h1" },
  343. ReplaceWith { root: 0, m: 1 },
  344. CreateElement { root: 7, tag: "h1" },
  345. ReplaceWith { root: 1, m: 1 },
  346. ]
  347. );
  348. }
  349. /// Should result in multiple nodes destroyed - with no changes
  350. #[test]
  351. fn two_fragments_with_same_elements_are_differet() {
  352. let dom = new_dom();
  353. let left = rsx!(
  354. {(0..2).map(|f| {rsx! { div { }}})}
  355. p {}
  356. );
  357. let right = rsx!(
  358. {(0..5).map(|f| {rsx! { div { }}})}
  359. p {}
  360. );
  361. let (create, change) = dom.lazy_diff(left, right);
  362. assert_eq!(
  363. create.edits,
  364. [
  365. CreateElement {
  366. root: 0,
  367. tag: "div"
  368. },
  369. CreateElement {
  370. root: 1,
  371. tag: "div"
  372. },
  373. CreateElement { root: 2, tag: "p" },
  374. AppendChildren { many: 3 },
  375. ]
  376. );
  377. assert_eq!(
  378. change.edits,
  379. [
  380. CreateElement {
  381. root: 3,
  382. tag: "div"
  383. },
  384. CreateElement {
  385. root: 4,
  386. tag: "div"
  387. },
  388. CreateElement {
  389. root: 5,
  390. tag: "div"
  391. },
  392. InsertAfter { root: 1, n: 3 },
  393. ]
  394. );
  395. }
  396. /// should result in the removal of elements
  397. #[test]
  398. fn keyed_diffing_order() {
  399. let dom = new_dom();
  400. let left = rsx!(
  401. {(0..5).map(|f| {rsx! { div { key: "{f}" }}})}
  402. p {"e"}
  403. );
  404. let right = rsx!(
  405. {(0..2).map(|f| {rsx! { div { key: "{f}" }}})}
  406. p {"e"}
  407. );
  408. let (create, change) = dom.lazy_diff(left, right);
  409. assert_eq!(
  410. change.edits,
  411. [Remove { root: 2 }, Remove { root: 3 }, Remove { root: 4 },]
  412. );
  413. }
  414. /// Should result in moves, but not removals or additions
  415. #[test]
  416. fn keyed_diffing_out_of_order() {
  417. let dom = new_dom();
  418. let left = rsx!({
  419. [0, 1, 2, 3, /**/ 4, 5, 6, /**/ 7, 8, 9].iter().map(|f| {
  420. rsx! { div { key: "{f}" }}
  421. })
  422. });
  423. let right = rsx!({
  424. [0, 1, 2, 3, /**/ 6, 4, 5, /**/ 7, 8, 9].iter().map(|f| {
  425. rsx! { div { key: "{f}" }}
  426. })
  427. });
  428. let (_, changes) = dom.lazy_diff(left, right);
  429. log::debug!("{:?}", &changes);
  430. assert_eq!(
  431. changes.edits,
  432. [PushRoot { root: 6 }, InsertBefore { root: 4, n: 1 }]
  433. );
  434. }
  435. /// Should result in moves only
  436. #[test]
  437. fn keyed_diffing_out_of_order_adds() {
  438. let dom = new_dom();
  439. let left = rsx!({
  440. [/**/ 4, 5, 6, 7, 8 /**/].iter().map(|f| {
  441. rsx! { div { key: "{f}" }}
  442. })
  443. });
  444. let right = rsx!({
  445. [/**/ 8, 7, 4, 5, 6 /**/].iter().map(|f| {
  446. rsx! { div { key: "{f}" }}
  447. })
  448. });
  449. let (_, change) = dom.lazy_diff(left, right);
  450. assert_eq!(
  451. change.edits,
  452. [
  453. PushRoot { root: 4 },
  454. PushRoot { root: 3 },
  455. InsertBefore { n: 2, root: 0 }
  456. ]
  457. );
  458. }
  459. /// Should result in moves onl
  460. #[test]
  461. fn keyed_diffing_out_of_order_adds_2() {
  462. let dom = new_dom();
  463. let left = rsx!({
  464. [/**/ 4, 5, 6, 7, 8 /**/].iter().map(|f| {
  465. rsx! { div { key: "{f}" }}
  466. })
  467. });
  468. let right = rsx!({
  469. [/**/ 7, 8, 4, 5, 6 /**/].iter().map(|f| {
  470. rsx! { div { key: "{f}" }}
  471. })
  472. });
  473. let (_, change) = dom.lazy_diff(left, right);
  474. assert_eq!(
  475. change.edits,
  476. [
  477. PushRoot { root: 3 },
  478. PushRoot { root: 4 },
  479. InsertBefore { n: 2, root: 0 }
  480. ]
  481. );
  482. }
  483. /// Should result in moves onl
  484. #[test]
  485. fn keyed_diffing_out_of_order_adds_3() {
  486. let dom = new_dom();
  487. let left = rsx!({
  488. [/**/ 4, 5, 6, 7, 8 /**/].iter().map(|f| {
  489. rsx! { div { key: "{f}" }}
  490. })
  491. });
  492. let right = rsx!({
  493. [/**/ 4, 8, 7, 5, 6 /**/].iter().map(|f| {
  494. rsx! { div { key: "{f}" }}
  495. })
  496. });
  497. let (_, change) = dom.lazy_diff(left, right);
  498. assert_eq!(
  499. change.edits,
  500. [
  501. PushRoot { root: 4 },
  502. PushRoot { root: 3 },
  503. InsertBefore { n: 2, root: 1 }
  504. ]
  505. );
  506. }
  507. /// Should result in moves onl
  508. #[test]
  509. fn keyed_diffing_out_of_order_adds_4() {
  510. let dom = new_dom();
  511. let left = rsx!({
  512. [/**/ 4, 5, 6, 7, 8 /**/].iter().map(|f| {
  513. rsx! { div { key: "{f}" }}
  514. })
  515. });
  516. let right = rsx!({
  517. [/**/ 4, 5, 8, 7, 6 /**/].iter().map(|f| {
  518. rsx! { div { key: "{f}" }}
  519. })
  520. });
  521. let (_, change) = dom.lazy_diff(left, right);
  522. assert_eq!(
  523. change.edits,
  524. [
  525. PushRoot { root: 4 },
  526. PushRoot { root: 3 },
  527. InsertBefore { n: 2, root: 2 }
  528. ]
  529. );
  530. }
  531. /// Should result in moves onl
  532. #[test]
  533. fn keyed_diffing_out_of_order_adds_5() {
  534. let dom = new_dom();
  535. let left = rsx!({
  536. [/**/ 4, 5, 6, 7, 8 /**/].iter().map(|f| {
  537. rsx! { div { key: "{f}" }}
  538. })
  539. });
  540. let right = rsx!({
  541. [/**/ 4, 5, 6, 8, 7 /**/].iter().map(|f| {
  542. rsx! { div { key: "{f}" }}
  543. })
  544. });
  545. let (_, change) = dom.lazy_diff(left, right);
  546. assert_eq!(
  547. change.edits,
  548. [PushRoot { root: 4 }, InsertBefore { n: 1, root: 3 }]
  549. );
  550. }
  551. #[test]
  552. fn keyed_diffing_additions() {
  553. let dom = new_dom();
  554. let left = rsx!({
  555. [/**/ 4, 5, 6, 7, 8 /**/].iter().map(|f| {
  556. rsx! { div { key: "{f}" }}
  557. })
  558. });
  559. let right = rsx!({
  560. [/**/ 4, 5, 6, 7, 8, 9, 10 /**/].iter().map(|f| {
  561. rsx! { div { key: "{f}" }}
  562. })
  563. });
  564. let (_, change) = dom.lazy_diff(left, right);
  565. assert_eq!(
  566. change.edits,
  567. [
  568. CreateElement {
  569. root: 5,
  570. tag: "div"
  571. },
  572. CreateElement {
  573. root: 6,
  574. tag: "div"
  575. },
  576. InsertAfter { n: 2, root: 4 }
  577. ]
  578. );
  579. }
  580. #[test]
  581. fn keyed_diffing_additions_and_moves_on_ends() {
  582. let dom = new_dom();
  583. let left = rsx!({
  584. [/**/ 4, 5, 6, 7 /**/].iter().map(|f| {
  585. rsx! { div { key: "{f}" }}
  586. })
  587. });
  588. let right = rsx!({
  589. [/**/ 7, 4, 5, 6, 11, 12 /**/].iter().map(|f| {
  590. rsx! { div { key: "{f}" }}
  591. })
  592. });
  593. let (_, change) = dom.lazy_diff(left, right);
  594. log::debug!("{:?}", change);
  595. assert_eq!(
  596. change.edits,
  597. [
  598. // create 11, 12
  599. CreateElement {
  600. tag: "div",
  601. root: 4
  602. },
  603. CreateElement {
  604. tag: "div",
  605. root: 5
  606. },
  607. InsertAfter { root: 2, n: 2 },
  608. // move 7 to the front
  609. PushRoot { root: 3 },
  610. InsertBefore { root: 0, n: 1 }
  611. ]
  612. );
  613. }
  614. #[test]
  615. fn keyed_diffing_additions_and_moves_in_middle() {
  616. let dom = new_dom();
  617. let left = rsx!({
  618. [/**/ 4, 5, 6, 7 /**/].iter().map(|f| {
  619. rsx! { div { key: "{f}" }}
  620. })
  621. });
  622. let right = rsx!({
  623. [/**/ 7, 4, 13, 17, 5, 11, 12, 6 /**/].iter().map(|f| {
  624. rsx! { div { key: "{f}" }}
  625. })
  626. });
  627. // LIS: 4, 5, 6
  628. let (_, change) = dom.lazy_diff(left, right);
  629. log::debug!("{:#?}", change);
  630. assert_eq!(
  631. change.edits,
  632. [
  633. // create 13, 17
  634. CreateElement {
  635. tag: "div",
  636. root: 4
  637. },
  638. CreateElement {
  639. tag: "div",
  640. root: 5
  641. },
  642. InsertBefore { root: 1, n: 2 },
  643. // create 11, 12
  644. CreateElement {
  645. tag: "div",
  646. root: 6
  647. },
  648. CreateElement {
  649. tag: "div",
  650. root: 7
  651. },
  652. InsertBefore { root: 2, n: 2 },
  653. // move 7
  654. PushRoot { root: 3 },
  655. InsertBefore { root: 0, n: 1 }
  656. ]
  657. );
  658. }
  659. #[test]
  660. fn controlled_keyed_diffing_out_of_order() {
  661. let dom = new_dom();
  662. let left = rsx!({
  663. [4, 5, 6, 7].iter().map(|f| {
  664. rsx! { div { key: "{f}" }}
  665. })
  666. });
  667. let right = rsx!({
  668. [0, 5, 9, 6, 4].iter().map(|f| {
  669. rsx! { div { key: "{f}" }}
  670. })
  671. });
  672. // LIS: 5, 6
  673. let (_, changes) = dom.lazy_diff(left, right);
  674. log::debug!("{:#?}", &changes);
  675. assert_eq!(
  676. changes.edits,
  677. [
  678. // move 4 to after 6
  679. PushRoot { root: 0 },
  680. InsertAfter { n: 1, root: 2 },
  681. // remove 7
  682. // create 9 and insert before 6
  683. CreateElement {
  684. root: 4,
  685. tag: "div"
  686. },
  687. InsertBefore { n: 1, root: 2 },
  688. // create 0 and insert before 5
  689. CreateElement {
  690. root: 5,
  691. tag: "div"
  692. },
  693. InsertBefore { n: 1, root: 1 },
  694. ]
  695. );
  696. }
  697. #[test]
  698. fn controlled_keyed_diffing_out_of_order_max_test() {
  699. let dom = new_dom();
  700. let left = rsx!({
  701. [0, 1, 2, 3, 4].iter().map(|f| {
  702. rsx! { div { key: "{f}" }}
  703. })
  704. });
  705. let right = rsx!({
  706. [3, 0, 1, 10, 2].iter().map(|f| {
  707. rsx! { div { key: "{f}" }}
  708. })
  709. });
  710. let (_, changes) = dom.lazy_diff(left, right);
  711. log::debug!("{:#?}", &changes);
  712. assert_eq!(
  713. changes.edits,
  714. [
  715. CreateElement {
  716. root: 5,
  717. tag: "div"
  718. },
  719. InsertBefore { n: 1, root: 2 },
  720. PushRoot { root: 3 },
  721. InsertBefore { n: 1, root: 0 },
  722. ]
  723. );
  724. }
  725. #[test]
  726. fn suspense() {
  727. let dom = new_dom();
  728. let edits = dom.create(LazyNodes::new(|f| {
  729. use std::cell::{Cell, RefCell};
  730. VNode::Suspended(f.bump().alloc(VSuspended {
  731. task_id: 0,
  732. callback: RefCell::new(None),
  733. dom_id: Cell::new(None),
  734. }))
  735. }));
  736. assert_eq!(
  737. edits.edits,
  738. [CreatePlaceholder { root: 0 }, AppendChildren { many: 1 }]
  739. );
  740. }