apiAsyncComponent.spec.ts 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847
  1. import { vi } from 'vitest'
  2. import {
  3. defineAsyncComponent,
  4. h,
  5. Component,
  6. ref,
  7. nextTick,
  8. Suspense,
  9. KeepAlive
  10. } from '../src'
  11. import { createApp, nodeOps, serializeInner } from '@vue/runtime-test'
  12. import { onActivated } from '../src/components/KeepAlive'
  13. const timeout = (n: number = 0) => new Promise(r => setTimeout(r, n))
  14. describe('api: defineAsyncComponent', () => {
  15. test('simple usage', async () => {
  16. let resolve: (comp: Component) => void
  17. const Foo = defineAsyncComponent(
  18. () =>
  19. new Promise(r => {
  20. resolve = r as any
  21. })
  22. )
  23. const toggle = ref(true)
  24. const root = nodeOps.createElement('div')
  25. createApp({
  26. render: () => (toggle.value ? h(Foo) : null)
  27. }).mount(root)
  28. expect(serializeInner(root)).toBe('<!---->')
  29. resolve!(() => 'resolved')
  30. // first time resolve, wait for macro task since there are multiple
  31. // microtasks / .then() calls
  32. await timeout()
  33. expect(serializeInner(root)).toBe('resolved')
  34. toggle.value = false
  35. await nextTick()
  36. expect(serializeInner(root)).toBe('<!---->')
  37. // already resolved component should update on nextTick
  38. toggle.value = true
  39. await nextTick()
  40. expect(serializeInner(root)).toBe('resolved')
  41. })
  42. test('with loading component', async () => {
  43. let resolve: (comp: Component) => void
  44. const Foo = defineAsyncComponent({
  45. loader: () =>
  46. new Promise(r => {
  47. resolve = r as any
  48. }),
  49. loadingComponent: () => 'loading',
  50. delay: 1 // defaults to 200
  51. })
  52. const toggle = ref(true)
  53. const root = nodeOps.createElement('div')
  54. createApp({
  55. render: () => (toggle.value ? h(Foo) : null)
  56. }).mount(root)
  57. // due to the delay, initial mount should be empty
  58. expect(serializeInner(root)).toBe('<!---->')
  59. // loading show up after delay
  60. await timeout(1)
  61. expect(serializeInner(root)).toBe('loading')
  62. resolve!(() => 'resolved')
  63. await timeout()
  64. expect(serializeInner(root)).toBe('resolved')
  65. toggle.value = false
  66. await nextTick()
  67. expect(serializeInner(root)).toBe('<!---->')
  68. // already resolved component should update on nextTick without loading
  69. // state
  70. toggle.value = true
  71. await nextTick()
  72. expect(serializeInner(root)).toBe('resolved')
  73. })
  74. test('with loading component + explicit delay (0)', async () => {
  75. let resolve: (comp: Component) => void
  76. const Foo = defineAsyncComponent({
  77. loader: () =>
  78. new Promise(r => {
  79. resolve = r as any
  80. }),
  81. loadingComponent: () => 'loading',
  82. delay: 0
  83. })
  84. const toggle = ref(true)
  85. const root = nodeOps.createElement('div')
  86. createApp({
  87. render: () => (toggle.value ? h(Foo) : null)
  88. }).mount(root)
  89. // with delay: 0, should show loading immediately
  90. expect(serializeInner(root)).toBe('loading')
  91. resolve!(() => 'resolved')
  92. await timeout()
  93. expect(serializeInner(root)).toBe('resolved')
  94. toggle.value = false
  95. await nextTick()
  96. expect(serializeInner(root)).toBe('<!---->')
  97. // already resolved component should update on nextTick without loading
  98. // state
  99. toggle.value = true
  100. await nextTick()
  101. expect(serializeInner(root)).toBe('resolved')
  102. })
  103. test('error without error component', async () => {
  104. let resolve: (comp: Component) => void
  105. let reject: (e: Error) => void
  106. const Foo = defineAsyncComponent(
  107. () =>
  108. new Promise((_resolve, _reject) => {
  109. resolve = _resolve as any
  110. reject = _reject
  111. })
  112. )
  113. const toggle = ref(true)
  114. const root = nodeOps.createElement('div')
  115. const app = createApp({
  116. render: () => (toggle.value ? h(Foo) : null)
  117. })
  118. const handler = (app.config.errorHandler = vi.fn())
  119. app.mount(root)
  120. expect(serializeInner(root)).toBe('<!---->')
  121. const err = new Error('foo')
  122. reject!(err)
  123. await timeout()
  124. expect(handler).toHaveBeenCalled()
  125. expect(handler.mock.calls[0][0]).toBe(err)
  126. expect(serializeInner(root)).toBe('<!---->')
  127. toggle.value = false
  128. await nextTick()
  129. expect(serializeInner(root)).toBe('<!---->')
  130. // errored out on previous load, toggle and mock success this time
  131. toggle.value = true
  132. await nextTick()
  133. expect(serializeInner(root)).toBe('<!---->')
  134. // should render this time
  135. resolve!(() => 'resolved')
  136. await timeout()
  137. expect(serializeInner(root)).toBe('resolved')
  138. })
  139. test('error with error component', async () => {
  140. let resolve: (comp: Component) => void
  141. let reject: (e: Error) => void
  142. const Foo = defineAsyncComponent({
  143. loader: () =>
  144. new Promise((_resolve, _reject) => {
  145. resolve = _resolve as any
  146. reject = _reject
  147. }),
  148. errorComponent: (props: { error: Error }) => props.error.message
  149. })
  150. const toggle = ref(true)
  151. const root = nodeOps.createElement('div')
  152. const app = createApp({
  153. render: () => (toggle.value ? h(Foo) : null)
  154. })
  155. const handler = (app.config.errorHandler = vi.fn())
  156. app.mount(root)
  157. expect(serializeInner(root)).toBe('<!---->')
  158. const err = new Error('errored out')
  159. reject!(err)
  160. await timeout()
  161. expect(handler).toHaveBeenCalled()
  162. expect(serializeInner(root)).toBe('errored out')
  163. toggle.value = false
  164. await nextTick()
  165. expect(serializeInner(root)).toBe('<!---->')
  166. // errored out on previous load, toggle and mock success this time
  167. toggle.value = true
  168. await nextTick()
  169. expect(serializeInner(root)).toBe('<!---->')
  170. // should render this time
  171. resolve!(() => 'resolved')
  172. await timeout()
  173. expect(serializeInner(root)).toBe('resolved')
  174. })
  175. // #2129
  176. test('error with error component, without global handler', async () => {
  177. let resolve: (comp: Component) => void
  178. let reject: (e: Error) => void
  179. const Foo = defineAsyncComponent({
  180. loader: () =>
  181. new Promise((_resolve, _reject) => {
  182. resolve = _resolve as any
  183. reject = _reject
  184. }),
  185. errorComponent: (props: { error: Error }) => props.error.message
  186. })
  187. const toggle = ref(true)
  188. const root = nodeOps.createElement('div')
  189. const app = createApp({
  190. render: () => (toggle.value ? h(Foo) : null)
  191. })
  192. app.mount(root)
  193. expect(serializeInner(root)).toBe('<!---->')
  194. const err = new Error('errored out')
  195. reject!(err)
  196. await timeout()
  197. expect(serializeInner(root)).toBe('errored out')
  198. expect(
  199. 'Unhandled error during execution of async component loader'
  200. ).toHaveBeenWarned()
  201. toggle.value = false
  202. await nextTick()
  203. expect(serializeInner(root)).toBe('<!---->')
  204. // errored out on previous load, toggle and mock success this time
  205. toggle.value = true
  206. await nextTick()
  207. expect(serializeInner(root)).toBe('<!---->')
  208. // should render this time
  209. resolve!(() => 'resolved')
  210. await timeout()
  211. expect(serializeInner(root)).toBe('resolved')
  212. })
  213. test('error with error + loading components', async () => {
  214. let resolve: (comp: Component) => void
  215. let reject: (e: Error) => void
  216. const Foo = defineAsyncComponent({
  217. loader: () =>
  218. new Promise((_resolve, _reject) => {
  219. resolve = _resolve as any
  220. reject = _reject
  221. }),
  222. errorComponent: (props: { error: Error }) => props.error.message,
  223. loadingComponent: () => 'loading',
  224. delay: 1
  225. })
  226. const toggle = ref(true)
  227. const root = nodeOps.createElement('div')
  228. const app = createApp({
  229. render: () => (toggle.value ? h(Foo) : null)
  230. })
  231. const handler = (app.config.errorHandler = vi.fn())
  232. app.mount(root)
  233. // due to the delay, initial mount should be empty
  234. expect(serializeInner(root)).toBe('<!---->')
  235. // loading show up after delay
  236. await timeout(1)
  237. expect(serializeInner(root)).toBe('loading')
  238. const err = new Error('errored out')
  239. reject!(err)
  240. await timeout()
  241. expect(handler).toHaveBeenCalled()
  242. expect(serializeInner(root)).toBe('errored out')
  243. toggle.value = false
  244. await nextTick()
  245. expect(serializeInner(root)).toBe('<!---->')
  246. // errored out on previous load, toggle and mock success this time
  247. toggle.value = true
  248. await nextTick()
  249. expect(serializeInner(root)).toBe('<!---->')
  250. // loading show up after delay
  251. await timeout(1)
  252. expect(serializeInner(root)).toBe('loading')
  253. // should render this time
  254. resolve!(() => 'resolved')
  255. await timeout()
  256. expect(serializeInner(root)).toBe('resolved')
  257. })
  258. test('timeout without error component', async () => {
  259. let resolve: (comp: Component) => void
  260. const Foo = defineAsyncComponent({
  261. loader: () =>
  262. new Promise(_resolve => {
  263. resolve = _resolve as any
  264. }),
  265. timeout: 1
  266. })
  267. const root = nodeOps.createElement('div')
  268. const app = createApp({
  269. render: () => h(Foo)
  270. })
  271. const handler = vi.fn()
  272. app.config.errorHandler = handler
  273. app.mount(root)
  274. expect(serializeInner(root)).toBe('<!---->')
  275. await timeout(1)
  276. expect(handler).toHaveBeenCalled()
  277. expect(handler.mock.calls[0][0].message).toMatch(
  278. `Async component timed out after 1ms.`
  279. )
  280. expect(serializeInner(root)).toBe('<!---->')
  281. // if it resolved after timeout, should still work
  282. resolve!(() => 'resolved')
  283. await timeout()
  284. expect(serializeInner(root)).toBe('resolved')
  285. })
  286. test('timeout with error component', async () => {
  287. let resolve: (comp: Component) => void
  288. const Foo = defineAsyncComponent({
  289. loader: () =>
  290. new Promise(_resolve => {
  291. resolve = _resolve as any
  292. }),
  293. timeout: 1,
  294. errorComponent: () => 'timed out'
  295. })
  296. const root = nodeOps.createElement('div')
  297. const app = createApp({
  298. render: () => h(Foo)
  299. })
  300. const handler = (app.config.errorHandler = vi.fn())
  301. app.mount(root)
  302. expect(serializeInner(root)).toBe('<!---->')
  303. await timeout(1)
  304. expect(handler).toHaveBeenCalled()
  305. expect(serializeInner(root)).toBe('timed out')
  306. // if it resolved after timeout, should still work
  307. resolve!(() => 'resolved')
  308. await timeout()
  309. expect(serializeInner(root)).toBe('resolved')
  310. })
  311. test('timeout with error + loading components', async () => {
  312. let resolve: (comp: Component) => void
  313. const Foo = defineAsyncComponent({
  314. loader: () =>
  315. new Promise(_resolve => {
  316. resolve = _resolve as any
  317. }),
  318. delay: 1,
  319. timeout: 16,
  320. errorComponent: () => 'timed out',
  321. loadingComponent: () => 'loading'
  322. })
  323. const root = nodeOps.createElement('div')
  324. const app = createApp({
  325. render: () => h(Foo)
  326. })
  327. const handler = (app.config.errorHandler = vi.fn())
  328. app.mount(root)
  329. expect(serializeInner(root)).toBe('<!---->')
  330. await timeout(1)
  331. expect(serializeInner(root)).toBe('loading')
  332. await timeout(16)
  333. expect(serializeInner(root)).toBe('timed out')
  334. expect(handler).toHaveBeenCalled()
  335. resolve!(() => 'resolved')
  336. await timeout()
  337. expect(serializeInner(root)).toBe('resolved')
  338. })
  339. test('timeout without error component, but with loading component', async () => {
  340. let resolve: (comp: Component) => void
  341. const Foo = defineAsyncComponent({
  342. loader: () =>
  343. new Promise(_resolve => {
  344. resolve = _resolve as any
  345. }),
  346. delay: 1,
  347. timeout: 16,
  348. loadingComponent: () => 'loading'
  349. })
  350. const root = nodeOps.createElement('div')
  351. const app = createApp({
  352. render: () => h(Foo)
  353. })
  354. const handler = vi.fn()
  355. app.config.errorHandler = handler
  356. app.mount(root)
  357. expect(serializeInner(root)).toBe('<!---->')
  358. await timeout(1)
  359. expect(serializeInner(root)).toBe('loading')
  360. await timeout(16)
  361. expect(handler).toHaveBeenCalled()
  362. expect(handler.mock.calls[0][0].message).toMatch(
  363. `Async component timed out after 16ms.`
  364. )
  365. // should still display loading
  366. expect(serializeInner(root)).toBe('loading')
  367. resolve!(() => 'resolved')
  368. await timeout()
  369. expect(serializeInner(root)).toBe('resolved')
  370. })
  371. test('with suspense', async () => {
  372. let resolve: (comp: Component) => void
  373. const Foo = defineAsyncComponent(
  374. () =>
  375. new Promise(_resolve => {
  376. resolve = _resolve as any
  377. })
  378. )
  379. const root = nodeOps.createElement('div')
  380. const app = createApp({
  381. render: () =>
  382. h(Suspense, null, {
  383. default: () => h('div', [h(Foo), ' & ', h(Foo)]),
  384. fallback: () => 'loading'
  385. })
  386. })
  387. app.mount(root)
  388. expect(serializeInner(root)).toBe('loading')
  389. resolve!(() => 'resolved')
  390. await timeout()
  391. expect(serializeInner(root)).toBe('<div>resolved & resolved</div>')
  392. })
  393. test('suspensible: false', async () => {
  394. let resolve: (comp: Component) => void
  395. const Foo = defineAsyncComponent({
  396. loader: () =>
  397. new Promise(_resolve => {
  398. resolve = _resolve as any
  399. }),
  400. suspensible: false
  401. })
  402. const root = nodeOps.createElement('div')
  403. const app = createApp({
  404. render: () =>
  405. h(Suspense, null, {
  406. default: () => h('div', [h(Foo), ' & ', h(Foo)]),
  407. fallback: () => 'loading'
  408. })
  409. })
  410. app.mount(root)
  411. // should not show suspense fallback
  412. expect(serializeInner(root)).toBe('<div><!----> & <!----></div>')
  413. resolve!(() => 'resolved')
  414. await timeout()
  415. expect(serializeInner(root)).toBe('<div>resolved & resolved</div>')
  416. })
  417. test('suspense with error handling', async () => {
  418. let reject: (e: Error) => void
  419. const Foo = defineAsyncComponent(
  420. () =>
  421. new Promise((_resolve, _reject) => {
  422. reject = _reject
  423. })
  424. )
  425. const root = nodeOps.createElement('div')
  426. const app = createApp({
  427. render: () =>
  428. h(Suspense, null, {
  429. default: () => h('div', [h(Foo), ' & ', h(Foo)]),
  430. fallback: () => 'loading'
  431. })
  432. })
  433. const handler = (app.config.errorHandler = vi.fn())
  434. app.mount(root)
  435. expect(serializeInner(root)).toBe('loading')
  436. reject!(new Error('no'))
  437. await timeout()
  438. expect(handler).toHaveBeenCalled()
  439. expect(serializeInner(root)).toBe('<div><!----> & <!----></div>')
  440. })
  441. test('retry (success)', async () => {
  442. let loaderCallCount = 0
  443. let resolve: (comp: Component) => void
  444. let reject: (e: Error) => void
  445. const Foo = defineAsyncComponent({
  446. loader: () => {
  447. loaderCallCount++
  448. return new Promise((_resolve, _reject) => {
  449. resolve = _resolve as any
  450. reject = _reject
  451. })
  452. },
  453. onError(error, retry, fail) {
  454. if (error.message.match(/foo/)) {
  455. retry()
  456. } else {
  457. fail()
  458. }
  459. }
  460. })
  461. const root = nodeOps.createElement('div')
  462. const app = createApp({
  463. render: () => h(Foo)
  464. })
  465. const handler = (app.config.errorHandler = vi.fn())
  466. app.mount(root)
  467. expect(serializeInner(root)).toBe('<!---->')
  468. expect(loaderCallCount).toBe(1)
  469. const err = new Error('foo')
  470. reject!(err)
  471. await timeout()
  472. expect(handler).not.toHaveBeenCalled()
  473. expect(loaderCallCount).toBe(2)
  474. expect(serializeInner(root)).toBe('<!---->')
  475. // should render this time
  476. resolve!(() => 'resolved')
  477. await timeout()
  478. expect(handler).not.toHaveBeenCalled()
  479. expect(serializeInner(root)).toBe('resolved')
  480. })
  481. test('retry (skipped)', async () => {
  482. let loaderCallCount = 0
  483. let reject: (e: Error) => void
  484. const Foo = defineAsyncComponent({
  485. loader: () => {
  486. loaderCallCount++
  487. return new Promise((_resolve, _reject) => {
  488. reject = _reject
  489. })
  490. },
  491. onError(error, retry, fail) {
  492. if (error.message.match(/bar/)) {
  493. retry()
  494. } else {
  495. fail()
  496. }
  497. }
  498. })
  499. const root = nodeOps.createElement('div')
  500. const app = createApp({
  501. render: () => h(Foo)
  502. })
  503. const handler = (app.config.errorHandler = vi.fn())
  504. app.mount(root)
  505. expect(serializeInner(root)).toBe('<!---->')
  506. expect(loaderCallCount).toBe(1)
  507. const err = new Error('foo')
  508. reject!(err)
  509. await timeout()
  510. // should fail because retryWhen returns false
  511. expect(handler).toHaveBeenCalled()
  512. expect(handler.mock.calls[0][0]).toBe(err)
  513. expect(loaderCallCount).toBe(1)
  514. expect(serializeInner(root)).toBe('<!---->')
  515. })
  516. test('retry (fail w/ max retry attempts)', async () => {
  517. let loaderCallCount = 0
  518. let reject: (e: Error) => void
  519. const Foo = defineAsyncComponent({
  520. loader: () => {
  521. loaderCallCount++
  522. return new Promise((_resolve, _reject) => {
  523. reject = _reject
  524. })
  525. },
  526. onError(error, retry, fail, attempts) {
  527. if (error.message.match(/foo/) && attempts <= 1) {
  528. retry()
  529. } else {
  530. fail()
  531. }
  532. }
  533. })
  534. const root = nodeOps.createElement('div')
  535. const app = createApp({
  536. render: () => h(Foo)
  537. })
  538. const handler = (app.config.errorHandler = vi.fn())
  539. app.mount(root)
  540. expect(serializeInner(root)).toBe('<!---->')
  541. expect(loaderCallCount).toBe(1)
  542. // first retry
  543. const err = new Error('foo')
  544. reject!(err)
  545. await timeout()
  546. expect(handler).not.toHaveBeenCalled()
  547. expect(loaderCallCount).toBe(2)
  548. expect(serializeInner(root)).toBe('<!---->')
  549. // 2nd retry, should fail due to reaching maxRetries
  550. reject!(err)
  551. await timeout()
  552. expect(handler).toHaveBeenCalled()
  553. expect(handler.mock.calls[0][0]).toBe(err)
  554. expect(loaderCallCount).toBe(2)
  555. expect(serializeInner(root)).toBe('<!---->')
  556. })
  557. test('template ref forwarding', async () => {
  558. let resolve: (comp: Component) => void
  559. const Foo = defineAsyncComponent(
  560. () =>
  561. new Promise(r => {
  562. resolve = r as any
  563. })
  564. )
  565. const fooRef = ref<any>(null)
  566. const toggle = ref(true)
  567. const root = nodeOps.createElement('div')
  568. createApp({
  569. render: () => (toggle.value ? h(Foo, { ref: fooRef }) : null)
  570. }).mount(root)
  571. expect(serializeInner(root)).toBe('<!---->')
  572. expect(fooRef.value).toBe(null)
  573. resolve!({
  574. data() {
  575. return {
  576. id: 'foo'
  577. }
  578. },
  579. render: () => 'resolved'
  580. })
  581. // first time resolve, wait for macro task since there are multiple
  582. // microtasks / .then() calls
  583. await timeout()
  584. expect(serializeInner(root)).toBe('resolved')
  585. expect(fooRef.value.id).toBe('foo')
  586. toggle.value = false
  587. await nextTick()
  588. expect(serializeInner(root)).toBe('<!---->')
  589. expect(fooRef.value).toBe(null)
  590. // already resolved component should update on nextTick
  591. toggle.value = true
  592. await nextTick()
  593. expect(serializeInner(root)).toBe('resolved')
  594. expect(fooRef.value.id).toBe('foo')
  595. })
  596. // #3188
  597. test('the forwarded template ref should always exist when doing multi patching', async () => {
  598. let resolve: (comp: Component) => void
  599. const Foo = defineAsyncComponent(
  600. () =>
  601. new Promise(r => {
  602. resolve = r as any
  603. })
  604. )
  605. const fooRef = ref<any>(null)
  606. const toggle = ref(true)
  607. const updater = ref(0)
  608. const root = nodeOps.createElement('div')
  609. createApp({
  610. render: () =>
  611. toggle.value ? [h(Foo, { ref: fooRef }), updater.value] : null
  612. }).mount(root)
  613. expect(serializeInner(root)).toBe('<!---->0')
  614. expect(fooRef.value).toBe(null)
  615. resolve!({
  616. data() {
  617. return {
  618. id: 'foo'
  619. }
  620. },
  621. render: () => 'resolved'
  622. })
  623. await timeout()
  624. expect(serializeInner(root)).toBe('resolved0')
  625. expect(fooRef.value.id).toBe('foo')
  626. updater.value++
  627. await nextTick()
  628. expect(serializeInner(root)).toBe('resolved1')
  629. expect(fooRef.value.id).toBe('foo')
  630. toggle.value = false
  631. await nextTick()
  632. expect(serializeInner(root)).toBe('<!---->')
  633. expect(fooRef.value).toBe(null)
  634. })
  635. test('vnode hooks on async wrapper', async () => {
  636. let resolve: (comp: Component) => void
  637. const Foo = defineAsyncComponent(
  638. () =>
  639. new Promise(r => {
  640. resolve = r as any
  641. })
  642. )
  643. const updater = ref(0)
  644. const vnodeHooks = {
  645. onVnodeBeforeMount: vi.fn(),
  646. onVnodeMounted: vi.fn(),
  647. onVnodeBeforeUpdate: vi.fn(),
  648. onVnodeUpdated: vi.fn(),
  649. onVnodeBeforeUnmount: vi.fn(),
  650. onVnodeUnmounted: vi.fn()
  651. }
  652. const toggle = ref(true)
  653. const root = nodeOps.createElement('div')
  654. createApp({
  655. render: () => (toggle.value ? [h(Foo, vnodeHooks), updater.value] : null)
  656. }).mount(root)
  657. expect(serializeInner(root)).toBe('<!---->0')
  658. resolve!({
  659. data() {
  660. return {
  661. id: 'foo'
  662. }
  663. },
  664. render: () => 'resolved'
  665. })
  666. await timeout()
  667. expect(serializeInner(root)).toBe('resolved0')
  668. expect(vnodeHooks.onVnodeBeforeMount).toHaveBeenCalledTimes(1)
  669. expect(vnodeHooks.onVnodeMounted).toHaveBeenCalledTimes(1)
  670. updater.value++
  671. await nextTick()
  672. expect(serializeInner(root)).toBe('resolved1')
  673. expect(vnodeHooks.onVnodeBeforeUpdate).toHaveBeenCalledTimes(1)
  674. expect(vnodeHooks.onVnodeUpdated).toHaveBeenCalledTimes(1)
  675. toggle.value = false
  676. await nextTick()
  677. expect(serializeInner(root)).toBe('<!---->')
  678. expect(vnodeHooks.onVnodeBeforeUnmount).toHaveBeenCalledTimes(1)
  679. expect(vnodeHooks.onVnodeUnmounted).toHaveBeenCalledTimes(1)
  680. })
  681. test('with KeepAlive', async () => {
  682. const spy = vi.fn()
  683. let resolve: (comp: Component) => void
  684. const Foo = defineAsyncComponent(
  685. () =>
  686. new Promise(r => {
  687. resolve = r as any
  688. })
  689. )
  690. const Bar = defineAsyncComponent(() => Promise.resolve(() => 'Bar'))
  691. const toggle = ref(true)
  692. const root = nodeOps.createElement('div')
  693. const app = createApp({
  694. render: () => h(KeepAlive, [toggle.value ? h(Foo) : h(Bar)])
  695. })
  696. app.mount(root)
  697. await nextTick()
  698. resolve!({
  699. setup() {
  700. onActivated(() => {
  701. spy()
  702. })
  703. return () => 'Foo'
  704. }
  705. })
  706. await timeout()
  707. expect(serializeInner(root)).toBe('Foo')
  708. expect(spy).toBeCalledTimes(1)
  709. toggle.value = false
  710. await timeout()
  711. expect(serializeInner(root)).toBe('Bar')
  712. })
  713. })