apiAsyncComponent.spec.ts 25 KB

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