hydration.spec.ts 148 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263
  1. import {
  2. VaporTeleport,
  3. child,
  4. createComponent,
  5. createPlainElement,
  6. createVaporSSRApp,
  7. defineVaporAsyncComponent,
  8. defineVaporComponent,
  9. delegateEvents,
  10. renderEffect,
  11. setStyle,
  12. template,
  13. useVaporCssVars,
  14. } from '../src'
  15. import { defineAsyncComponent, nextTick, reactive, ref } from '@vue/runtime-dom'
  16. import { isString } from '@vue/shared'
  17. import type { VaporComponentInstance } from '../src/component'
  18. import type { TeleportFragment } from '../src/components/Teleport'
  19. import { VueServerRenderer, compile, runtimeDom, runtimeVapor } from './_utils'
  20. const formatHtml = (raw: string) => {
  21. return raw
  22. .replace(/<!--\[/g, '\n<!--[')
  23. .replace(/]-->/g, ']-->\n')
  24. .replace(/\n{2,}/g, '\n')
  25. }
  26. async function testWithVaporApp(
  27. code: string,
  28. components?: Record<string, string | { code: string; vapor: boolean }>,
  29. data?: any,
  30. ) {
  31. return testHydration(code, components, data, {
  32. isVaporApp: true,
  33. interop: true,
  34. })
  35. }
  36. async function testWithVDOMApp(
  37. code: string,
  38. components?: Record<string, string | { code: string; vapor: boolean }>,
  39. data?: any,
  40. ) {
  41. return testHydration(code, components, data, {
  42. isVaporApp: false,
  43. interop: true,
  44. })
  45. }
  46. function compileVaporComponent(
  47. code: string,
  48. data: runtimeDom.Ref<any> = ref({}),
  49. components?: Record<string, any>,
  50. ssr = false,
  51. ) {
  52. if (!code.includes(`<script`)) {
  53. code = `<template>${code}</template>`
  54. }
  55. return compile(code, data, components, {
  56. vapor: true,
  57. ssr,
  58. })
  59. }
  60. async function mountWithHydration(
  61. html: string,
  62. code: string,
  63. data: runtimeDom.Ref<any> = ref({}),
  64. components?: Record<string, any>,
  65. ) {
  66. const container = document.createElement('div')
  67. container.innerHTML = html
  68. document.body.appendChild(container)
  69. const clientComp = compileVaporComponent(code, data, components)
  70. const app = createVaporSSRApp(clientComp)
  71. app.mount(container)
  72. return {
  73. block: (app._instance! as VaporComponentInstance).block,
  74. container,
  75. }
  76. }
  77. async function testHydration(
  78. code: string,
  79. components: Record<string, string | { code: string; vapor: boolean }> = {},
  80. data: any = ref('foo'),
  81. { isVaporApp = true, interop = false } = {},
  82. ) {
  83. const ssrComponents: any = {}
  84. const clientComponents: any = {}
  85. for (const key in components) {
  86. const comp = components[key]
  87. const code = isString(comp) ? comp : comp.code
  88. const isVaporComp = isString(comp) || !!comp.vapor
  89. clientComponents[key] = compile(code, data, clientComponents, {
  90. vapor: isVaporComp,
  91. ssr: false,
  92. })
  93. ssrComponents[key] = compile(code, data, ssrComponents, {
  94. vapor: isVaporComp,
  95. ssr: true,
  96. })
  97. }
  98. const serverComp = compile(code, data, ssrComponents, {
  99. vapor: isVaporApp,
  100. ssr: true,
  101. })
  102. const html = await VueServerRenderer.renderToString(
  103. runtimeDom.createSSRApp(serverComp),
  104. )
  105. const container = document.createElement('div')
  106. document.body.appendChild(container)
  107. container.innerHTML = html
  108. const clientComp = compile(code, data, clientComponents, {
  109. vapor: isVaporApp,
  110. ssr: false,
  111. })
  112. let app
  113. if (isVaporApp) {
  114. app = createVaporSSRApp(clientComp)
  115. } else {
  116. app = runtimeDom.createSSRApp(clientComp)
  117. }
  118. if (interop) {
  119. app.use(runtimeVapor.vaporInteropPlugin)
  120. }
  121. app.mount(container)
  122. return { data, container }
  123. }
  124. const triggerEvent = (type: string, el: Element) => {
  125. const event = new Event(type, { bubbles: true })
  126. el.dispatchEvent(event)
  127. }
  128. delegateEvents('click')
  129. beforeEach(() => {
  130. document.body.innerHTML = ''
  131. })
  132. describe('Vapor Mode hydration', () => {
  133. describe('text', () => {
  134. test('root text', async () => {
  135. const { data, container } = await testHydration(`
  136. <template>{{ data }}</template>
  137. `)
  138. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`"foo"`)
  139. data.value = 'bar'
  140. await nextTick()
  141. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`"bar"`)
  142. })
  143. test('consecutive text nodes', async () => {
  144. const { data, container } = await testHydration(`
  145. <template>{{ data }}{{ data }}</template>
  146. `)
  147. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`"foofoo"`)
  148. data.value = 'bar'
  149. await nextTick()
  150. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`"barbar"`)
  151. })
  152. test('consecutive text nodes with insertion anchor', async () => {
  153. const { data, container } = await testHydration(`
  154. <template><span/>{{ data }}{{ data }}<span/></template>
  155. `)
  156. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  157. `
  158. "
  159. <!--[--><span></span>foofoo<span></span><!--]-->
  160. "
  161. `,
  162. )
  163. data.value = 'bar'
  164. await nextTick()
  165. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  166. `
  167. "
  168. <!--[--><span></span>barbar<span></span><!--]-->
  169. "
  170. `,
  171. )
  172. })
  173. test('mixed text nodes', async () => {
  174. const { data, container } = await testHydration(`
  175. <template>{{ data }}A{{ data }}B{{ data }}</template>
  176. `)
  177. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  178. `"fooAfooBfoo"`,
  179. )
  180. data.value = 'bar'
  181. await nextTick()
  182. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  183. `"barAbarBbar"`,
  184. )
  185. })
  186. test('mixed text nodes with insertion anchor', async () => {
  187. const { data, container } = await testHydration(`
  188. <template><span/>{{ data }}A{{ data }}B{{ data }}<span/></template>
  189. `)
  190. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  191. `
  192. "
  193. <!--[--><span></span>fooAfooBfoo<span></span><!--]-->
  194. "
  195. `,
  196. )
  197. data.value = 'bar'
  198. await nextTick()
  199. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  200. `
  201. "
  202. <!--[--><span></span>barAbarBbar<span></span><!--]-->
  203. "
  204. `,
  205. )
  206. })
  207. test('empty text node', async () => {
  208. const data = reactive({ txt: '' })
  209. const { container } = await testHydration(
  210. `<template><div>{{ data.txt }}</div></template>`,
  211. undefined,
  212. data,
  213. )
  214. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  215. `"<div></div>"`,
  216. )
  217. data.txt = 'foo'
  218. await nextTick()
  219. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  220. `"<div>foo</div>"`,
  221. )
  222. })
  223. test('empty text node in slot', async () => {
  224. const data = reactive({ txt: '' })
  225. const { container } = await testHydration(
  226. `<template><components.Child>{{data.txt}}</components.Child></template>`,
  227. {
  228. Child: `<template><slot/></template>`,
  229. },
  230. data,
  231. )
  232. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  233. `
  234. "
  235. <!--[--><!--]-->
  236. "
  237. `,
  238. )
  239. data.txt = 'foo'
  240. await nextTick()
  241. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  242. `
  243. "
  244. <!--[-->foo<!--]-->
  245. "
  246. `,
  247. )
  248. })
  249. })
  250. describe('element', () => {
  251. test('root comment', async () => {
  252. const { container } = await testHydration(`
  253. <template><!----></template>
  254. `)
  255. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`"<!---->"`)
  256. expect(`mismatch in <div>`).not.toHaveBeenWarned()
  257. })
  258. test('root with mixed element and text', async () => {
  259. const { container, data } = await testHydration(`
  260. <template> A<span>{{ data }}</span>{{ data }}</template>
  261. `)
  262. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  263. `
  264. "
  265. <!--[--> A<span>foo</span>foo<!--]-->
  266. "
  267. `,
  268. )
  269. data.value = 'bar'
  270. await nextTick()
  271. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  272. `
  273. "
  274. <!--[--> A<span>bar</span>bar<!--]-->
  275. "
  276. `,
  277. )
  278. })
  279. test('empty element', async () => {
  280. const { container } = await testHydration(`
  281. <template><div/></template>
  282. `)
  283. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  284. `"<div></div>"`,
  285. )
  286. expect(`mismatch in <div>`).not.toHaveBeenWarned()
  287. })
  288. test('element with binding and text children', async () => {
  289. const { container, data } = await testHydration(`
  290. <template><div :class="data">{{ data }}</div></template>
  291. `)
  292. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  293. `"<div class="foo">foo</div>"`,
  294. )
  295. data.value = 'bar'
  296. await nextTick()
  297. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  298. `"<div class="bar">bar</div>"`,
  299. )
  300. })
  301. test('element with elements children', async () => {
  302. const { container } = await testHydration(`
  303. <template>
  304. <div>
  305. <span>{{ data }}</span>
  306. <span :class="data" @click="data = 'bar'"/>
  307. </div>
  308. </template>
  309. `)
  310. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  311. `"<div><span>foo</span><span class="foo"></span></div>"`,
  312. )
  313. // event handler
  314. triggerEvent('click', container.querySelector('.foo')!)
  315. await nextTick()
  316. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  317. `"<div><span>bar</span><span class="bar"></span></div>"`,
  318. )
  319. })
  320. test('element with ref', async () => {
  321. const { data, container } = await testHydration(
  322. `<template>
  323. <div ref="data">hi</div>
  324. </template>
  325. `,
  326. {},
  327. ref(null),
  328. )
  329. expect(data.value).toBe(container.firstChild)
  330. })
  331. })
  332. describe('component', () => {
  333. test('basic component', async () => {
  334. const { container, data } = await testHydration(
  335. `
  336. <template><div><span></span><components.Child/></div></template>
  337. `,
  338. { Child: `<template>{{ data }}</template>` },
  339. )
  340. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  341. `"<div><span></span>foo</div>"`,
  342. )
  343. data.value = 'bar'
  344. await nextTick()
  345. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  346. `"<div><span></span>bar</div>"`,
  347. )
  348. })
  349. test('fragment component', async () => {
  350. const { container, data } = await testHydration(
  351. `
  352. <template><div><span></span><components.Child/></div></template>
  353. `,
  354. { Child: `<template><div>{{ data }}</div>-{{ data }}-</template>` },
  355. )
  356. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  357. `
  358. "<div><span></span>
  359. <!--[--><div>foo</div>-foo-<!--]-->
  360. </div>"
  361. `,
  362. )
  363. data.value = 'bar'
  364. await nextTick()
  365. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  366. `
  367. "<div><span></span>
  368. <!--[--><div>bar</div>-bar-<!--]-->
  369. </div>"
  370. `,
  371. )
  372. })
  373. test('fragment component with prepend', async () => {
  374. const { container, data } = await testHydration(
  375. `
  376. <template><div><components.Child/><span></span></div></template>
  377. `,
  378. { Child: `<template><div>{{ data }}</div>-{{ data }}-</template>` },
  379. )
  380. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  381. `
  382. "<div>
  383. <!--[--><div>foo</div>-foo-<!--]-->
  384. <span></span></div>"
  385. `,
  386. )
  387. data.value = 'bar'
  388. await nextTick()
  389. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  390. `
  391. "<div>
  392. <!--[--><div>bar</div>-bar-<!--]-->
  393. <span></span></div>"
  394. `,
  395. )
  396. })
  397. test('nested fragment components', async () => {
  398. const { container, data } = await testHydration(
  399. `
  400. <template><div><components.Parent/><span></span></div></template>
  401. `,
  402. {
  403. Parent: `<template><div/><components.Child/><div/></template>`,
  404. Child: `<template><div>{{ data }}</div>-{{ data }}-</template>`,
  405. },
  406. )
  407. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  408. `
  409. "<div>
  410. <!--[--><div></div>
  411. <!--[--><div>foo</div>-foo-<!--]-->
  412. <div></div><!--]-->
  413. <span></span></div>"
  414. `,
  415. )
  416. data.value = 'bar'
  417. await nextTick()
  418. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  419. `
  420. "<div>
  421. <!--[--><div></div>
  422. <!--[--><div>bar</div>-bar-<!--]-->
  423. <div></div><!--]-->
  424. <span></span></div>"
  425. `,
  426. )
  427. })
  428. test('component with insertion anchor', async () => {
  429. const { container, data } = await testHydration(
  430. `<template>
  431. <div>
  432. <span/>
  433. <components.Child/>
  434. <span/>
  435. </div>
  436. </template>
  437. `,
  438. {
  439. Child: `<template>{{ data }}</template>`,
  440. },
  441. )
  442. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  443. `"<div><span></span>foo<span></span></div>"`,
  444. )
  445. data.value = 'bar'
  446. await nextTick()
  447. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  448. `"<div><span></span>bar<span></span></div>"`,
  449. )
  450. })
  451. test('nested components with insertion anchor', async () => {
  452. const { container, data } = await testHydration(
  453. `
  454. <template><components.Parent/></template>
  455. `,
  456. {
  457. Parent: `<template><div><span/><components.Child/><span/></div></template>`,
  458. Child: `<template><div>{{ data }}</div></template>`,
  459. },
  460. )
  461. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  462. `"<div><span></span><div>foo</div><span></span></div>"`,
  463. )
  464. data.value = 'bar'
  465. await nextTick()
  466. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  467. `"<div><span></span><div>bar</div><span></span></div>"`,
  468. )
  469. })
  470. test('nested components with multi level anchor insertion', async () => {
  471. const { container, data } = await testHydration(
  472. `
  473. <template><div><span></span><components.Parent/><span></span></div></template>
  474. `,
  475. {
  476. Parent: `<template><div><span/><components.Child/><span/></div></template>`,
  477. Child: `<template><div>{{ data }}</div></template>`,
  478. },
  479. )
  480. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  481. `"<div><span></span><div><span></span><div>foo</div><span></span></div><span></span></div>"`,
  482. )
  483. data.value = 'bar'
  484. await nextTick()
  485. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  486. `"<div><span></span><div><span></span><div>bar</div><span></span></div><span></span></div>"`,
  487. )
  488. })
  489. test('consecutive components with insertion parent', async () => {
  490. const data = reactive({ foo: 'foo', bar: 'bar' })
  491. const { container } = await testHydration(
  492. `<template>
  493. <div>
  494. <components.Child1/>
  495. <components.Child2/>
  496. </div>
  497. </template>
  498. `,
  499. {
  500. Child1: `<template><span>{{ data.foo }}</span></template>`,
  501. Child2: `<template><span>{{ data.bar }}</span></template>`,
  502. },
  503. data,
  504. )
  505. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  506. `"<div><span>foo</span><span>bar</span></div>"`,
  507. )
  508. data.foo = 'foo1'
  509. data.bar = 'bar1'
  510. await nextTick()
  511. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  512. `"<div><span>foo1</span><span>bar1</span></div>"`,
  513. )
  514. })
  515. test('nested consecutive components with insertion anchor', async () => {
  516. const { container, data } = await testHydration(
  517. `
  518. <template><components.Parent/></template>
  519. `,
  520. {
  521. Parent: `<template><div><span/><components.Child/><components.Child/><span/></div></template>`,
  522. Child: `<template><div>{{ data }}</div></template>`,
  523. },
  524. )
  525. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  526. `"<div><span></span><div>foo</div><div>foo</div><span></span></div>"`,
  527. )
  528. data.value = 'bar'
  529. await nextTick()
  530. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  531. `"<div><span></span><div>bar</div><div>bar</div><span></span></div>"`,
  532. )
  533. })
  534. test('nested consecutive components with multi level anchor insertion', async () => {
  535. const { container, data } = await testHydration(
  536. `
  537. <template><div><span></span><components.Parent/><span></span></div></template>
  538. `,
  539. {
  540. Parent: `<template><div><span/><components.Child/><components.Child/><span/></div></template>`,
  541. Child: `<template><div>{{ data }}</div></template>`,
  542. },
  543. )
  544. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  545. `"<div><span></span><div><span></span><div>foo</div><div>foo</div><span></span></div><span></span></div>"`,
  546. )
  547. data.value = 'bar'
  548. await nextTick()
  549. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  550. `"<div><span></span><div><span></span><div>bar</div><div>bar</div><span></span></div><span></span></div>"`,
  551. )
  552. })
  553. test('mixed component and element with insertion anchor', async () => {
  554. const { container, data } = await testHydration(
  555. `<template>
  556. <div>
  557. <span/>
  558. <components.Child/>
  559. <span/>
  560. <components.Child/>
  561. <span/>
  562. </div>
  563. </template>
  564. `,
  565. {
  566. Child: `<template>{{ data }}</template>`,
  567. },
  568. )
  569. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  570. `"<div><span></span>foo<span></span>foo<span></span></div>"`,
  571. )
  572. data.value = 'bar'
  573. await nextTick()
  574. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  575. `"<div><span></span>bar<span></span>bar<span></span></div>"`,
  576. )
  577. })
  578. test('fragment component with insertion anchor', async () => {
  579. const { container, data } = await testHydration(
  580. `<template>
  581. <div>
  582. <span/>
  583. <components.Child/>
  584. <span/>
  585. </div>
  586. </template>
  587. `,
  588. {
  589. Child: `<template><div>{{ data }}</div>-{{ data }}</template>`,
  590. },
  591. )
  592. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  593. `
  594. "<div><span></span>
  595. <!--[--><div>foo</div>-foo<!--]-->
  596. <span></span></div>"
  597. `,
  598. )
  599. data.value = 'bar'
  600. await nextTick()
  601. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  602. `
  603. "<div><span></span>
  604. <!--[--><div>bar</div>-bar<!--]-->
  605. <span></span></div>"
  606. `,
  607. )
  608. })
  609. test('nested fragment component with insertion anchor', async () => {
  610. const { container, data } = await testHydration(
  611. `
  612. <template><components.Parent/></template>
  613. `,
  614. {
  615. Parent: `<template><div><span/><components.Child/><span/></div></template>`,
  616. Child: `<template><div>{{ data }}</div>-{{ data }}-</template>`,
  617. },
  618. )
  619. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  620. `
  621. "<div><span></span>
  622. <!--[--><div>foo</div>-foo-<!--]-->
  623. <span></span></div>"
  624. `,
  625. )
  626. data.value = 'bar'
  627. await nextTick()
  628. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  629. `
  630. "<div><span></span>
  631. <!--[--><div>bar</div>-bar-<!--]-->
  632. <span></span></div>"
  633. `,
  634. )
  635. })
  636. test('nested fragment component with multi level anchor insertion', async () => {
  637. const { container, data } = await testHydration(
  638. `
  639. <template><div><span/><components.Parent/><span/></div></template>
  640. `,
  641. {
  642. Parent: `<template><div><span/><components.Child/><span/></div></template>`,
  643. Child: `<template><div>{{ data }}</div>-{{ data }}-</template>`,
  644. },
  645. )
  646. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  647. `
  648. "<div><span></span><div><span></span>
  649. <!--[--><div>foo</div>-foo-<!--]-->
  650. <span></span></div><span></span></div>"
  651. `,
  652. )
  653. data.value = 'bar'
  654. await nextTick()
  655. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  656. `
  657. "<div><span></span><div><span></span>
  658. <!--[--><div>bar</div>-bar-<!--]-->
  659. <span></span></div><span></span></div>"
  660. `,
  661. )
  662. })
  663. test('consecutive fragment components with insertion anchor', async () => {
  664. const { container, data } = await testHydration(
  665. `<template>
  666. <div>
  667. <span/>
  668. <components.Child/>
  669. <components.Child/>
  670. <span/>
  671. </div>
  672. </template>
  673. `,
  674. {
  675. Child: `<template><div>{{ data }}</div>-{{ data }}</template>`,
  676. },
  677. )
  678. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  679. `
  680. "<div><span></span>
  681. <!--[--><div>foo</div>-foo<!--]-->
  682. <!--[--><div>foo</div>-foo<!--]-->
  683. <span></span></div>"
  684. `,
  685. )
  686. data.value = 'bar'
  687. await nextTick()
  688. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  689. `
  690. "<div><span></span>
  691. <!--[--><div>bar</div>-bar<!--]-->
  692. <!--[--><div>bar</div>-bar<!--]-->
  693. <span></span></div>"
  694. `,
  695. )
  696. })
  697. test('nested consecutive fragment components with insertion anchor', async () => {
  698. const { container, data } = await testHydration(
  699. `
  700. <template><components.Parent/></template>
  701. `,
  702. {
  703. Parent: `<template><div><span/><components.Child/><components.Child/><span/></div></template>`,
  704. Child: `<template><div>{{ data }}</div>-{{ data }}-</template>`,
  705. },
  706. )
  707. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  708. `
  709. "<div><span></span>
  710. <!--[--><div>foo</div>-foo-<!--]-->
  711. <!--[--><div>foo</div>-foo-<!--]-->
  712. <span></span></div>"
  713. `,
  714. )
  715. data.value = 'bar'
  716. await nextTick()
  717. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  718. `
  719. "<div><span></span>
  720. <!--[--><div>bar</div>-bar-<!--]-->
  721. <!--[--><div>bar</div>-bar-<!--]-->
  722. <span></span></div>"
  723. `,
  724. )
  725. })
  726. test('nested consecutive fragment components with multi level anchor insertion', async () => {
  727. const { container, data } = await testHydration(
  728. `
  729. <template><div><span></span><components.Parent/><span></span></div></template>
  730. `,
  731. {
  732. Parent: `<template><div><span/><components.Child/><components.Child/><span/></div></template>`,
  733. Child: `<template><div>{{ data }}</div>-{{ data }}-</template>`,
  734. },
  735. )
  736. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  737. `
  738. "<div><span></span><div><span></span>
  739. <!--[--><div>foo</div>-foo-<!--]-->
  740. <!--[--><div>foo</div>-foo-<!--]-->
  741. <span></span></div><span></span></div>"
  742. `,
  743. )
  744. data.value = 'bar'
  745. await nextTick()
  746. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  747. `
  748. "<div><span></span><div><span></span>
  749. <!--[--><div>bar</div>-bar-<!--]-->
  750. <!--[--><div>bar</div>-bar-<!--]-->
  751. <span></span></div><span></span></div>"
  752. `,
  753. )
  754. })
  755. test('nested consecutive fragment components with root level anchor insertion', async () => {
  756. const { container, data } = await testHydration(
  757. `
  758. <template><div><span></span><components.Parent/><span></span></div></template>
  759. `,
  760. {
  761. Parent: `<template><components.Child/><components.Child/></template>`,
  762. Child: `<template><div>{{ data }}</div>-{{ data }}-</template>`,
  763. },
  764. )
  765. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  766. `
  767. "<div><span></span>
  768. <!--[-->
  769. <!--[--><div>foo</div>-foo-<!--]-->
  770. <!--[--><div>foo</div>-foo-<!--]-->
  771. <!--]-->
  772. <span></span></div>"
  773. `,
  774. )
  775. data.value = 'bar'
  776. await nextTick()
  777. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  778. `
  779. "<div><span></span>
  780. <!--[-->
  781. <!--[--><div>bar</div>-bar-<!--]-->
  782. <!--[--><div>bar</div>-bar-<!--]-->
  783. <!--]-->
  784. <span></span></div>"
  785. `,
  786. )
  787. })
  788. test('mixed fragment component and element with insertion anchor', async () => {
  789. const { container, data } = await testHydration(
  790. `<template>
  791. <div>
  792. <span/>
  793. <components.Child/>
  794. <span/>
  795. <components.Child/>
  796. <span/>
  797. </div>
  798. </template>
  799. `,
  800. {
  801. Child: `<template><div>{{ data }}</div>-{{ data }}</template>`,
  802. },
  803. )
  804. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  805. `
  806. "<div><span></span>
  807. <!--[--><div>foo</div>-foo<!--]-->
  808. <span></span>
  809. <!--[--><div>foo</div>-foo<!--]-->
  810. <span></span></div>"
  811. `,
  812. )
  813. data.value = 'bar'
  814. await nextTick()
  815. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  816. `
  817. "<div><span></span>
  818. <!--[--><div>bar</div>-bar<!--]-->
  819. <span></span>
  820. <!--[--><div>bar</div>-bar<!--]-->
  821. <span></span></div>"
  822. `,
  823. )
  824. })
  825. test('mixed fragment component and text with insertion anchor', async () => {
  826. const { container, data } = await testHydration(
  827. `<template>
  828. <div>
  829. <span/>
  830. <components.Child/>
  831. {{ data }}
  832. <components.Child/>
  833. <span/>
  834. </div>
  835. </template>
  836. `,
  837. {
  838. Child: `<template><div>{{ data }}</div>-{{ data }}</template>`,
  839. },
  840. )
  841. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  842. `
  843. "<div><span></span>
  844. <!--[--><div>foo</div>-foo<!--]-->
  845. foo
  846. <!--[--><div>foo</div>-foo<!--]-->
  847. <span></span></div>"
  848. `,
  849. )
  850. data.value = 'bar'
  851. await nextTick()
  852. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  853. `
  854. "<div><span></span>
  855. <!--[--><div>bar</div>-bar<!--]-->
  856. bar
  857. <!--[--><div>bar</div>-bar<!--]-->
  858. <span></span></div>"
  859. `,
  860. )
  861. })
  862. })
  863. describe('dynamic component', () => {
  864. test('basic dynamic component', async () => {
  865. const { container, data } = await testHydration(
  866. `<template>
  867. <component :is="components[data]"/>
  868. </template>`,
  869. {
  870. foo: `<template><div>foo</div></template>`,
  871. bar: `<template><div>bar</div></template>`,
  872. },
  873. ref('foo'),
  874. )
  875. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  876. `"<div>foo</div><!--dynamic-component-->"`,
  877. )
  878. data.value = 'bar'
  879. await nextTick()
  880. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  881. `"<div>bar</div><!--dynamic-component-->"`,
  882. )
  883. })
  884. test('dynamic component with insertion anchor', async () => {
  885. const { container, data } = await testHydration(
  886. `<template>
  887. <div>
  888. <span/>
  889. <component :is="components[data]"/>
  890. <span/>
  891. </div>
  892. </template>`,
  893. {
  894. foo: `<template><div>foo</div></template>`,
  895. bar: `<template><div>bar</div></template>`,
  896. },
  897. ref('foo'),
  898. )
  899. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  900. `"<div><span></span><div>foo</div><!--dynamic-component--><span></span></div>"`,
  901. )
  902. data.value = 'bar'
  903. await nextTick()
  904. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  905. `"<div><span></span><div>bar</div><!--dynamic-component--><span></span></div>"`,
  906. )
  907. })
  908. test('consecutive dynamic components with insertion anchor', async () => {
  909. const { container, data } = await testHydration(
  910. `<template>
  911. <div>
  912. <span/>
  913. <component :is="components[data]"/>
  914. <component :is="components[data]"/>
  915. <span/>
  916. </div>
  917. </template>`,
  918. {
  919. foo: `<template><div>foo</div></template>`,
  920. bar: `<template><div>bar</div></template>`,
  921. },
  922. ref('foo'),
  923. )
  924. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  925. `"<div><span></span><div>foo</div><!--dynamic-component--><div>foo</div><!--dynamic-component--><span></span></div>"`,
  926. )
  927. data.value = 'bar'
  928. await nextTick()
  929. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  930. `"<div><span></span><div>bar</div><!--dynamic-component--><div>bar</div><!--dynamic-component--><span></span></div>"`,
  931. )
  932. })
  933. test('dynamic component fallback', async () => {
  934. const { container, data } = await testHydration(
  935. `<template>
  936. <component :is="'button'">
  937. <span>{{ data }}</span>
  938. </component>
  939. </template>`,
  940. {},
  941. ref('foo'),
  942. )
  943. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  944. `"<button><span>foo</span></button><!--dynamic-component-->"`,
  945. )
  946. data.value = 'bar'
  947. await nextTick()
  948. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  949. `"<button><span>bar</span></button><!--dynamic-component-->"`,
  950. )
  951. })
  952. test('in ssr slot vnode fallback', async () => {
  953. const { container, data } = await testHydration(
  954. `<template>
  955. <components.Child>
  956. <span>{{ data }}</span>
  957. </components.Child>
  958. </template>`,
  959. {
  960. Child: `
  961. <template>
  962. <component :is="'div'">
  963. <slot />
  964. </component>
  965. </template>`,
  966. },
  967. ref('foo'),
  968. )
  969. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  970. `
  971. "<div>
  972. <!--[--><span>foo</span><!--]-->
  973. </div><!--dynamic-component-->"
  974. `,
  975. )
  976. data.value = 'bar'
  977. await nextTick()
  978. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  979. `
  980. "<div>
  981. <!--[--><span>bar</span><!--]-->
  982. </div><!--dynamic-component-->"
  983. `,
  984. )
  985. })
  986. test('dynamic component fallback with dynamic slots', async () => {
  987. const data = ref({
  988. name: 'default',
  989. msg: 'foo',
  990. })
  991. const { container } = await testHydration(
  992. `<template>
  993. <component :is="'div'">
  994. <template v-slot:[data.name]>
  995. <span>{{ data.msg }}</span>
  996. </template>
  997. </component>
  998. </template>`,
  999. {},
  1000. data,
  1001. )
  1002. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1003. `"<div><span>foo</span><!----></div><!--dynamic-component-->"`,
  1004. )
  1005. data.value.msg = 'bar'
  1006. await nextTick()
  1007. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1008. `"<div><span>bar</span><!----></div><!--dynamic-component-->"`,
  1009. )
  1010. })
  1011. })
  1012. describe('if', () => {
  1013. test('basic toggle - true -> false', async () => {
  1014. const data = ref(true)
  1015. const { container } = await testHydration(
  1016. `<template>
  1017. <div v-if="data">foo</div>
  1018. </template>`,
  1019. undefined,
  1020. data,
  1021. )
  1022. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1023. `"<div>foo</div><!--if-->"`,
  1024. )
  1025. data.value = false
  1026. await nextTick()
  1027. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1028. `"<!--if-->"`,
  1029. )
  1030. })
  1031. test('basic toggle - false -> true', async () => {
  1032. const data = ref(false)
  1033. const { container } = await testHydration(
  1034. `<template>
  1035. <div v-if="data">foo</div>
  1036. </template>`,
  1037. undefined,
  1038. data,
  1039. )
  1040. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1041. `"<!--if-->"`,
  1042. )
  1043. data.value = true
  1044. await nextTick()
  1045. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1046. `"<div>foo</div><!--if-->"`,
  1047. )
  1048. })
  1049. test('v-if on insertion parent', async () => {
  1050. const data = ref(true)
  1051. const { container } = await testHydration(
  1052. `<template>
  1053. <div v-if="data">
  1054. <components.Child/>
  1055. </div>
  1056. </template>`,
  1057. { Child: `<template>foo</template>` },
  1058. data,
  1059. )
  1060. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1061. `"<div>foo</div><!--if-->"`,
  1062. )
  1063. data.value = false
  1064. await nextTick()
  1065. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1066. `"<!--if-->"`,
  1067. )
  1068. data.value = true
  1069. await nextTick()
  1070. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1071. `"<div>foo</div><!--if-->"`,
  1072. )
  1073. })
  1074. test('v-if/else-if/else chain - switch branches', async () => {
  1075. const data = ref('a')
  1076. const { container } = await testHydration(
  1077. `<template>
  1078. <div v-if="data === 'a'">foo</div>
  1079. <div v-else-if="data === 'b'">bar</div>
  1080. <div v-else>baz</div>
  1081. </template>`,
  1082. undefined,
  1083. data,
  1084. )
  1085. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1086. `"<div>foo</div><!--if-->"`,
  1087. )
  1088. data.value = 'b'
  1089. await nextTick()
  1090. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1091. `"<div>bar</div><!--if--><!--if-->"`,
  1092. )
  1093. data.value = 'c'
  1094. await nextTick()
  1095. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1096. `"<div>baz</div><!--if--><!--if-->"`,
  1097. )
  1098. data.value = 'a'
  1099. await nextTick()
  1100. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1101. `"<div>foo</div><!--if-->"`,
  1102. )
  1103. })
  1104. test('nested if', async () => {
  1105. const data = reactive({ outer: true, inner: true })
  1106. const { container } = await testHydration(
  1107. `<template>
  1108. <div v-if="data.outer">
  1109. <span>outer</span>
  1110. <div v-if="data.inner">inner</div>
  1111. </div>
  1112. </template>`,
  1113. undefined,
  1114. data,
  1115. )
  1116. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1117. `"<div><span>outer</span><div>inner</div><!--if--></div><!--if-->"`,
  1118. )
  1119. data.inner = false
  1120. await nextTick()
  1121. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1122. `"<div><span>outer</span><!--if--></div><!--if-->"`,
  1123. )
  1124. data.outer = false
  1125. await nextTick()
  1126. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1127. `"<!--if-->"`,
  1128. )
  1129. })
  1130. test('on component', async () => {
  1131. const data = ref(true)
  1132. const { container } = await testHydration(
  1133. `<template>
  1134. <components.Child v-if="data"/>
  1135. </template>`,
  1136. { Child: `<template>foo</template>` },
  1137. data,
  1138. )
  1139. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1140. `"foo<!--if-->"`,
  1141. )
  1142. data.value = false
  1143. await nextTick()
  1144. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1145. `"<!--if-->"`,
  1146. )
  1147. })
  1148. test('consecutive if node', async () => {
  1149. const data = ref(true)
  1150. const { container } = await testHydration(
  1151. `<template>
  1152. <components.Child v-if="data"/>
  1153. </template>`,
  1154. { Child: `<template><div v-if="data">foo</div></template>` },
  1155. data,
  1156. )
  1157. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1158. `"<div>foo</div><!--if--><!--if-->"`,
  1159. )
  1160. data.value = false
  1161. await nextTick()
  1162. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1163. `"<!--if-->"`,
  1164. )
  1165. data.value = true
  1166. await nextTick()
  1167. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1168. `"<div>foo</div><!--if--><!--if-->"`,
  1169. )
  1170. })
  1171. test('mixed prepend and insertion anchor', async () => {
  1172. const data = reactive({
  1173. show: true,
  1174. foo: 'foo',
  1175. bar: 'bar',
  1176. qux: 'qux',
  1177. })
  1178. const { container } = await testHydration(
  1179. `<template>
  1180. <components.Child/>
  1181. </template>`,
  1182. {
  1183. Child: `<template>
  1184. <span v-if="data.show">
  1185. <span v-if="data.show">{{data.foo}}</span>
  1186. <span v-if="data.show">{{data.bar}}</span>
  1187. <span>baz</span>
  1188. <span v-if="data.show">{{data.qux}}</span>
  1189. <span>quux</span>
  1190. </span>
  1191. </template>`,
  1192. },
  1193. data,
  1194. )
  1195. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1196. `"<span><span>foo</span><!--if--><span>bar</span><!--if--><span>baz</span><span>qux</span><!--if--><span>quux</span></span><!--if-->"`,
  1197. )
  1198. data.qux = 'qux1'
  1199. await nextTick()
  1200. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1201. `"<span><span>foo</span><!--if--><span>bar</span><!--if--><span>baz</span><span>qux1</span><!--if--><span>quux</span></span><!--if-->"`,
  1202. )
  1203. data.foo = 'foo1'
  1204. await nextTick()
  1205. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1206. `"<span><span>foo1</span><!--if--><span>bar</span><!--if--><span>baz</span><span>qux1</span><!--if--><span>quux</span></span><!--if-->"`,
  1207. )
  1208. })
  1209. test('v-if/else-if/else chain on component - switch branches', async () => {
  1210. const data = ref('a')
  1211. const { container } = await testHydration(
  1212. `<template>
  1213. <components.Child1 v-if="data === 'a'"/>
  1214. <components.Child2 v-else-if="data === 'b'"/>
  1215. <components.Child3 v-else/>
  1216. </template>`,
  1217. {
  1218. Child1: `<template><span>{{data}} child1</span></template>`,
  1219. Child2: `<template><span>{{data}} child2</span></template>`,
  1220. Child3: `<template><span>{{data}} child3</span></template>`,
  1221. },
  1222. data,
  1223. )
  1224. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1225. `"<span>a child1</span><!--if-->"`,
  1226. )
  1227. data.value = 'b'
  1228. await nextTick()
  1229. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1230. `"<span>b child2</span><!--if--><!--if-->"`,
  1231. )
  1232. data.value = 'c'
  1233. await nextTick()
  1234. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1235. `"<span>c child3</span><!--if--><!--if-->"`,
  1236. )
  1237. data.value = 'a'
  1238. await nextTick()
  1239. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1240. `"<span>a child1</span><!--if-->"`,
  1241. )
  1242. })
  1243. test('on component with insertion anchor', async () => {
  1244. const data = ref(true)
  1245. const { container } = await testHydration(
  1246. `<template>
  1247. <div>
  1248. <span/>
  1249. <components.Child v-if="data"/>
  1250. <span/>
  1251. </div>
  1252. </template>`,
  1253. { Child: `<template>foo</template>` },
  1254. data,
  1255. )
  1256. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1257. `"<div><span></span>foo<!--if--><span></span></div>"`,
  1258. )
  1259. data.value = false
  1260. await nextTick()
  1261. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1262. `"<div><span></span><!--if--><span></span></div>"`,
  1263. )
  1264. })
  1265. test('consecutive component with insertion parent', async () => {
  1266. const data = reactive({
  1267. show: true,
  1268. foo: 'foo',
  1269. bar: 'bar',
  1270. })
  1271. const { container } = await testHydration(
  1272. `<template>
  1273. <div v-if="data.show">
  1274. <components.Child/>
  1275. <components.Child2/>
  1276. </div>
  1277. </template>`,
  1278. {
  1279. Child: `<template><span>{{data.foo}}</span></template>`,
  1280. Child2: `<template><span>{{data.bar}}</span></template>`,
  1281. },
  1282. data,
  1283. )
  1284. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1285. `"<div><span>foo</span><span>bar</span></div><!--if-->"`,
  1286. )
  1287. data.show = false
  1288. await nextTick()
  1289. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1290. `"<!--if-->"`,
  1291. )
  1292. data.show = true
  1293. await nextTick()
  1294. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1295. `"<div><span>foo</span><span>bar</span></div><!--if-->"`,
  1296. )
  1297. data.foo = 'foo1'
  1298. data.bar = 'bar1'
  1299. await nextTick()
  1300. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1301. `"<div><span>foo1</span><span>bar1</span></div><!--if-->"`,
  1302. )
  1303. })
  1304. test('on fragment component', async () => {
  1305. const data = ref(true)
  1306. const { container } = await testHydration(
  1307. `<template>
  1308. <div>
  1309. <components.Child v-if="data"/>
  1310. </div>
  1311. </template>`,
  1312. {
  1313. Child: `<template><div>{{ data }}</div>-{{ data }}-</template>`,
  1314. },
  1315. data,
  1316. )
  1317. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1318. `
  1319. "<div>
  1320. <!--[--><div>true</div>-true-<!--]-->
  1321. <!--if--></div>"
  1322. `,
  1323. )
  1324. data.value = false
  1325. await nextTick()
  1326. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1327. `
  1328. "<div>
  1329. <!--[--><!--]-->
  1330. <!--if--></div>"
  1331. `,
  1332. )
  1333. data.value = true
  1334. await nextTick()
  1335. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1336. `
  1337. "<div>
  1338. <!--[--><!--]-->
  1339. <div>true</div>-true-<!--if--></div>"
  1340. `,
  1341. )
  1342. })
  1343. test('on fragment component with insertion anchor', async () => {
  1344. const data = ref(true)
  1345. const { container } = await testHydration(
  1346. `<template>
  1347. <div>
  1348. <span/>
  1349. <components.Child v-if="data"/>
  1350. <span/>
  1351. </div>
  1352. </template>`,
  1353. {
  1354. Child: `<template><div>{{ data }}</div>-{{ data }}-</template>`,
  1355. },
  1356. data,
  1357. )
  1358. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1359. `
  1360. "<div><span></span>
  1361. <!--[--><div>true</div>-true-<!--]-->
  1362. <!--if--><span></span></div>"
  1363. `,
  1364. )
  1365. data.value = false
  1366. await nextTick()
  1367. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1368. `
  1369. "<div><span></span>
  1370. <!--[--><!--]-->
  1371. <!--if--><span></span></div>"
  1372. `,
  1373. )
  1374. data.value = true
  1375. await nextTick()
  1376. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`
  1377. "<div><span></span>
  1378. <!--[--><!--]-->
  1379. <div>true</div>-true-<!--if--><span></span></div>"
  1380. `)
  1381. })
  1382. test('consecutive v-if on fragment component with insertion anchor', async () => {
  1383. const data = ref(true)
  1384. const { container } = await testHydration(
  1385. `<template>
  1386. <div>
  1387. <span/>
  1388. <components.Child v-if="data"/>
  1389. <components.Child v-if="data"/>
  1390. <span/>
  1391. </div>
  1392. </template>`,
  1393. {
  1394. Child: `<template><div>{{ data }}</div>-{{ data }}-</template>`,
  1395. },
  1396. data,
  1397. )
  1398. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1399. `
  1400. "<div><span></span>
  1401. <!--[--><div>true</div>-true-<!--]-->
  1402. <!--if-->
  1403. <!--[--><div>true</div>-true-<!--]-->
  1404. <!--if--><span></span></div>"
  1405. `,
  1406. )
  1407. data.value = false
  1408. await nextTick()
  1409. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1410. `
  1411. "<div><span></span>
  1412. <!--[--><!--]-->
  1413. <!--if-->
  1414. <!--[--><!--]-->
  1415. <!--if--><span></span></div>"
  1416. `,
  1417. )
  1418. data.value = true
  1419. await nextTick()
  1420. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`
  1421. "<div><span></span>
  1422. <!--[--><!--]-->
  1423. <div>true</div>-true-<!--if-->
  1424. <!--[--><!--]-->
  1425. <div>true</div>-true-<!--if--><span></span></div>"
  1426. `)
  1427. })
  1428. test('on dynamic component with insertion anchor', async () => {
  1429. const data = ref(true)
  1430. const { container } = await testHydration(
  1431. `<template>
  1432. <div>
  1433. <span/>
  1434. <component :is="components.Child" v-if="data"/>
  1435. <span/>
  1436. </div>
  1437. </template>`,
  1438. { Child: `<template>foo</template>` },
  1439. data,
  1440. )
  1441. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1442. `"<div><span></span>foo<!--dynamic-component--><!--if--><span></span></div>"`,
  1443. )
  1444. data.value = false
  1445. await nextTick()
  1446. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1447. `"<div><span></span><!--if--><span></span></div>"`,
  1448. )
  1449. data.value = true
  1450. await nextTick()
  1451. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1452. `"<div><span></span>foo<!--dynamic-component--><!--if--><span></span></div>"`,
  1453. )
  1454. })
  1455. })
  1456. describe('for', () => {
  1457. test('basic v-for', async () => {
  1458. const { container, data } = await testHydration(
  1459. `<template>
  1460. <span v-for="item in data" :key="item">{{ item }}</span>
  1461. </template>`,
  1462. undefined,
  1463. ref(['a', 'b', 'c']),
  1464. )
  1465. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1466. `
  1467. "
  1468. <!--[--><span>a</span><span>b</span><span>c</span><!--]-->
  1469. "
  1470. `,
  1471. )
  1472. data.value.push('d')
  1473. await nextTick()
  1474. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1475. `
  1476. "
  1477. <!--[--><span>a</span><span>b</span><span>c</span><span>d</span><!--]-->
  1478. "
  1479. `,
  1480. )
  1481. })
  1482. test('empty v-for', async () => {
  1483. const { container, data } = await testHydration(
  1484. `<template>
  1485. <span v-for="item in data" :key="item">{{ item }}</span>
  1486. </template>`,
  1487. undefined,
  1488. ref([]),
  1489. )
  1490. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1491. `
  1492. "
  1493. <!--[--><!--]-->
  1494. "
  1495. `,
  1496. )
  1497. data.value.push('a')
  1498. await nextTick()
  1499. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1500. `
  1501. "
  1502. <!--[--><span>a</span><!--]-->
  1503. "
  1504. `,
  1505. )
  1506. })
  1507. test('v-for with insertion parent + sibling component', async () => {
  1508. const { container, data } = await testHydration(
  1509. `<template>
  1510. <div>
  1511. <span v-for="item in data" :key="item">{{ item }}</span>
  1512. </div>
  1513. <components.Child/>
  1514. </template>`,
  1515. {
  1516. Child: `<template><div>{{data.length}}</div></template>`,
  1517. },
  1518. ref(['a', 'b', 'c']),
  1519. )
  1520. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1521. `
  1522. "
  1523. <!--[--><div>
  1524. <!--[--><span>a</span><span>b</span><span>c</span><!--]-->
  1525. </div><div>3</div><!--]-->
  1526. "
  1527. `,
  1528. )
  1529. data.value.push('d')
  1530. await nextTick()
  1531. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1532. `
  1533. "
  1534. <!--[--><div>
  1535. <!--[--><span>a</span><span>b</span><span>c</span><span>d</span><!--]-->
  1536. </div><div>4</div><!--]-->
  1537. "
  1538. `,
  1539. )
  1540. })
  1541. test('v-for with insertion anchor', async () => {
  1542. const { container, data } = await testHydration(
  1543. `<template>
  1544. <div>
  1545. <span/>
  1546. <span v-for="item in data" :key="item">{{ item }}</span>
  1547. <span/>
  1548. </div>
  1549. </template>`,
  1550. undefined,
  1551. ref(['a', 'b', 'c']),
  1552. )
  1553. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1554. `
  1555. "<div><span></span>
  1556. <!--[--><span>a</span><span>b</span><span>c</span><!--]-->
  1557. <span></span></div>"
  1558. `,
  1559. )
  1560. data.value.push('d')
  1561. await nextTick()
  1562. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1563. `
  1564. "<div><span></span>
  1565. <!--[--><span>a</span><span>b</span><span>c</span><span>d</span><!--]-->
  1566. <span></span></div>"
  1567. `,
  1568. )
  1569. data.value.splice(0, 1)
  1570. await nextTick()
  1571. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1572. `
  1573. "<div><span></span>
  1574. <!--[--><span>b</span><span>c</span><span>d</span><!--]-->
  1575. <span></span></div>"
  1576. `,
  1577. )
  1578. })
  1579. test('consecutive v-for with insertion anchor', async () => {
  1580. const { container, data } = await testHydration(
  1581. `<template>
  1582. <div>
  1583. <span/>
  1584. <span v-for="item in data" :key="item">{{ item }}</span>
  1585. <span v-for="item in data" :key="item">{{ item }}</span>
  1586. <span/>
  1587. </div>
  1588. </template>`,
  1589. undefined,
  1590. ref(['a', 'b', 'c']),
  1591. )
  1592. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1593. `
  1594. "<div><span></span>
  1595. <!--[--><span>a</span><span>b</span><span>c</span><!--]-->
  1596. <!--[--><span>a</span><span>b</span><span>c</span><!--]-->
  1597. <span></span></div>"
  1598. `,
  1599. )
  1600. data.value.push('d')
  1601. await nextTick()
  1602. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1603. `
  1604. "<div><span></span>
  1605. <!--[--><span>a</span><span>b</span><span>c</span><span>d</span><!--]-->
  1606. <!--[--><span>a</span><span>b</span><span>c</span><span>d</span><!--]-->
  1607. <span></span></div>"
  1608. `,
  1609. )
  1610. data.value.splice(0, 2)
  1611. await nextTick()
  1612. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1613. `
  1614. "<div><span></span>
  1615. <!--[--><span>c</span><span>d</span><!--]-->
  1616. <!--[--><span>c</span><span>d</span><!--]-->
  1617. <span></span></div>"
  1618. `,
  1619. )
  1620. })
  1621. test('v-for on component', async () => {
  1622. const { container, data } = await testHydration(
  1623. `<template>
  1624. <div>
  1625. <components.Child v-for="item in data" :key="item"/>
  1626. </div>
  1627. </template>`,
  1628. {
  1629. Child: `<template><div>comp</div></template>`,
  1630. },
  1631. ref(['a', 'b', 'c']),
  1632. )
  1633. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1634. `
  1635. "<div>
  1636. <!--[--><div>comp</div><div>comp</div><div>comp</div><!--]-->
  1637. </div>"
  1638. `,
  1639. )
  1640. data.value.push('d')
  1641. await nextTick()
  1642. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1643. `
  1644. "<div>
  1645. <!--[--><div>comp</div><div>comp</div><div>comp</div><div>comp</div><!--]-->
  1646. </div>"
  1647. `,
  1648. )
  1649. })
  1650. test('v-for on component with slots', async () => {
  1651. const { container, data } = await testHydration(
  1652. `<template>
  1653. <div>
  1654. <components.Child v-for="item in data" :key="item">
  1655. <span>{{ item }}</span>
  1656. </components.Child>
  1657. </div>
  1658. </template>`,
  1659. {
  1660. Child: `<template><slot/></template>`,
  1661. },
  1662. ref(['a', 'b', 'c']),
  1663. )
  1664. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1665. `
  1666. "<div>
  1667. <!--[-->
  1668. <!--[--><span>a</span><!--]-->
  1669. <!--[--><span>b</span><!--]-->
  1670. <!--[--><span>c</span><!--]-->
  1671. <!--]-->
  1672. </div>"
  1673. `,
  1674. )
  1675. data.value.push('d')
  1676. await nextTick()
  1677. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1678. `
  1679. "<div>
  1680. <!--[-->
  1681. <!--[--><span>a</span><!--]-->
  1682. <!--[--><span>b</span><!--]-->
  1683. <!--[--><span>c</span><!--]-->
  1684. <span>d</span><!--slot--><!--]-->
  1685. </div>"
  1686. `,
  1687. )
  1688. })
  1689. test('on fragment component', async () => {
  1690. const { container, data } = await testHydration(
  1691. `<template>
  1692. <div>
  1693. <components.Child v-for="item in data" :key="item"/>
  1694. </div>
  1695. </template>`,
  1696. {
  1697. Child: `<template><div>foo</div>-bar-</template>`,
  1698. },
  1699. ref(['a', 'b', 'c']),
  1700. )
  1701. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1702. `
  1703. "<div>
  1704. <!--[-->
  1705. <!--[--><div>foo</div>-bar-<!--]-->
  1706. <!--[--><div>foo</div>-bar-<!--]-->
  1707. <!--[--><div>foo</div>-bar-<!--]-->
  1708. <!--]-->
  1709. </div>"
  1710. `,
  1711. )
  1712. data.value.push('d')
  1713. await nextTick()
  1714. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1715. `
  1716. "<div>
  1717. <!--[-->
  1718. <!--[--><div>foo</div>-bar-<!--]-->
  1719. <!--[--><div>foo</div>-bar-<!--]-->
  1720. <!--[--><div>foo</div>-bar-<!--]-->
  1721. <div>foo</div>-bar-<!--]-->
  1722. </div>"
  1723. `,
  1724. )
  1725. })
  1726. test('on component with non-hydration node', async () => {
  1727. const data = ref({ show: true, msg: 'foo' })
  1728. const { container } = await testHydration(
  1729. `<template>
  1730. <div>
  1731. <components.Child v-for="item in 2" :key="item"/>
  1732. </div>
  1733. </template>`,
  1734. {
  1735. Child: `<template>
  1736. <div>
  1737. <div>
  1738. <div v-if="data.show">{{ data.msg }}</div>
  1739. </div>
  1740. <span>non-hydration node</span>
  1741. </div>
  1742. </template>`,
  1743. },
  1744. data,
  1745. )
  1746. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1747. `
  1748. "<div>
  1749. <!--[--><div><div><div>foo</div><!--if--></div><span>non-hydration node</span></div><div><div><div>foo</div><!--if--></div><span>non-hydration node</span></div><!--]-->
  1750. </div>"
  1751. `,
  1752. )
  1753. data.value.msg = 'bar'
  1754. await nextTick()
  1755. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1756. `
  1757. "<div>
  1758. <!--[--><div><div><div>bar</div><!--if--></div><span>non-hydration node</span></div><div><div><div>bar</div><!--if--></div><span>non-hydration node</span></div><!--]-->
  1759. </div>"
  1760. `,
  1761. )
  1762. data.value.show = false
  1763. await nextTick()
  1764. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`
  1765. "<div>
  1766. <!--[--><div><div><!--if--></div><span>non-hydration node</span></div><div><div><!--if--></div><span>non-hydration node</span></div><!--]-->
  1767. </div>"
  1768. `)
  1769. data.value.show = true
  1770. await nextTick()
  1771. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`
  1772. "<div>
  1773. <!--[--><div><div><div>bar</div><!--if--></div><span>non-hydration node</span></div><div><div><div>bar</div><!--if--></div><span>non-hydration node</span></div><!--]-->
  1774. </div>"
  1775. `)
  1776. })
  1777. test('with non-hydration node', async () => {
  1778. const data = ref({ show: true, msg: 'foo' })
  1779. const { container } = await testHydration(
  1780. `<template>
  1781. <div>
  1782. <div v-for="item in 2">
  1783. <div>
  1784. <div v-if="data.show">{{ data.msg }}</div>
  1785. </div>
  1786. <span>non-hydration node</span>
  1787. </div>
  1788. </div>
  1789. </template>`,
  1790. {},
  1791. data,
  1792. )
  1793. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1794. `
  1795. "<div>
  1796. <!--[--><div><div><div>foo</div><!--if--></div><span>non-hydration node</span></div><div><div><div>foo</div><!--if--></div><span>non-hydration node</span></div><!--]-->
  1797. </div>"
  1798. `,
  1799. )
  1800. data.value.msg = 'bar'
  1801. await nextTick()
  1802. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1803. `
  1804. "<div>
  1805. <!--[--><div><div><div>bar</div><!--if--></div><span>non-hydration node</span></div><div><div><div>bar</div><!--if--></div><span>non-hydration node</span></div><!--]-->
  1806. </div>"
  1807. `,
  1808. )
  1809. data.value.show = false
  1810. await nextTick()
  1811. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`
  1812. "<div>
  1813. <!--[--><div><div><!--if--></div><span>non-hydration node</span></div><div><div><!--if--></div><span>non-hydration node</span></div><!--]-->
  1814. </div>"
  1815. `)
  1816. data.value.show = true
  1817. await nextTick()
  1818. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`
  1819. "<div>
  1820. <!--[--><div><div><div>bar</div><!--if--></div><span>non-hydration node</span></div><div><div><div>bar</div><!--if--></div><span>non-hydration node</span></div><!--]-->
  1821. </div>"
  1822. `)
  1823. })
  1824. })
  1825. describe('slots', () => {
  1826. test('basic slot', async () => {
  1827. const { data, container } = await testHydration(
  1828. `<template>
  1829. <components.Child>
  1830. <span>{{data}}</span>
  1831. </components.Child>
  1832. </template>`,
  1833. {
  1834. Child: `<template><slot/></template>`,
  1835. },
  1836. )
  1837. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1838. `
  1839. "
  1840. <!--[--><span>foo</span><!--]-->
  1841. "
  1842. `,
  1843. )
  1844. data.value = 'bar'
  1845. await nextTick()
  1846. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1847. `
  1848. "
  1849. <!--[--><span>bar</span><!--]-->
  1850. "
  1851. `,
  1852. )
  1853. })
  1854. test('named slot', async () => {
  1855. const { data, container } = await testHydration(
  1856. `<template>
  1857. <components.Child>
  1858. <template #foo>
  1859. <span>{{data}}</span>
  1860. </template>
  1861. </components.Child>
  1862. </template>`,
  1863. {
  1864. Child: `<template><slot/><slot name="foo"/></template>`,
  1865. },
  1866. )
  1867. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1868. `
  1869. "
  1870. <!--[-->
  1871. <!--[--><!--]-->
  1872. <!--[--><span>foo</span><!--]-->
  1873. <!--]-->
  1874. "
  1875. `,
  1876. )
  1877. data.value = 'bar'
  1878. await nextTick()
  1879. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1880. `
  1881. "
  1882. <!--[-->
  1883. <!--[--><!--]-->
  1884. <!--[--><span>bar</span><!--]-->
  1885. <!--]-->
  1886. "
  1887. `,
  1888. )
  1889. })
  1890. test('named slot with v-if', async () => {
  1891. const { data, container } = await testHydration(
  1892. `<template>
  1893. <components.Child>
  1894. <template #foo v-if="data">
  1895. <span>{{data}}</span>
  1896. </template>
  1897. </components.Child>
  1898. </template>`,
  1899. {
  1900. Child: `<template><slot name="foo"/></template>`,
  1901. },
  1902. )
  1903. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1904. `
  1905. "
  1906. <!--[--><span>foo</span><!--]-->
  1907. "
  1908. `,
  1909. )
  1910. data.value = false
  1911. await nextTick()
  1912. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1913. `
  1914. "
  1915. <!--[--><!--]-->
  1916. "
  1917. `,
  1918. )
  1919. data.value = true
  1920. await nextTick()
  1921. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`
  1922. "
  1923. <!--[--><span>true</span><!--]-->
  1924. "
  1925. `)
  1926. })
  1927. test('named slot with v-if and v-for', async () => {
  1928. const data = reactive({
  1929. show: true,
  1930. items: ['a', 'b', 'c'],
  1931. })
  1932. const { container } = await testHydration(
  1933. `<template>
  1934. <components.Child>
  1935. <template #foo v-if="data.show">
  1936. <span v-for="item in data.items" :key="item">{{item}}</span>
  1937. </template>
  1938. </components.Child>
  1939. </template>`,
  1940. {
  1941. Child: `<template><slot name="foo"/></template>`,
  1942. },
  1943. data,
  1944. )
  1945. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1946. `
  1947. "
  1948. <!--[-->
  1949. <!--[--><span>a</span><span>b</span><span>c</span><!--]-->
  1950. <!--]-->
  1951. "
  1952. `,
  1953. )
  1954. data.show = false
  1955. await nextTick()
  1956. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1957. `
  1958. "
  1959. <!--[-->
  1960. <!--[--><!--]-->
  1961. "
  1962. `,
  1963. )
  1964. data.show = true
  1965. await nextTick()
  1966. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1967. `
  1968. "
  1969. <!--[-->
  1970. <!--[--><span>a</span><span>b</span><span>c</span><!--for--><!--]-->
  1971. "
  1972. `,
  1973. )
  1974. })
  1975. test('with insertion anchor', async () => {
  1976. const { data, container } = await testHydration(
  1977. `<template>
  1978. <components.Child>
  1979. <span/>
  1980. <span>{{data}}</span>
  1981. <span/>
  1982. </components.Child>
  1983. </template>`,
  1984. {
  1985. Child: `<template><slot/></template>`,
  1986. },
  1987. )
  1988. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1989. `
  1990. "
  1991. <!--[--><span></span><span>foo</span><span></span><!--]-->
  1992. "
  1993. `,
  1994. )
  1995. data.value = 'bar'
  1996. await nextTick()
  1997. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  1998. `
  1999. "
  2000. <!--[--><span></span><span>bar</span><span></span><!--]-->
  2001. "
  2002. `,
  2003. )
  2004. })
  2005. test('with multi level anchor insertion', async () => {
  2006. const { data, container } = await testHydration(
  2007. `<template>
  2008. <components.Child>
  2009. <span/>
  2010. <span>{{data}}</span>
  2011. <span/>
  2012. </components.Child>
  2013. </template>`,
  2014. {
  2015. Child: `
  2016. <template>
  2017. <div/>
  2018. <div/>
  2019. <slot/>
  2020. <div/>
  2021. </div>
  2022. </template>`,
  2023. },
  2024. )
  2025. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2026. `
  2027. "
  2028. <!--[--><div></div><div></div>
  2029. <!--[--><span></span><span>foo</span><span></span><!--]-->
  2030. <div></div><!--]-->
  2031. "
  2032. `,
  2033. )
  2034. data.value = 'bar'
  2035. await nextTick()
  2036. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2037. `
  2038. "
  2039. <!--[--><div></div><div></div>
  2040. <!--[--><span></span><span>bar</span><span></span><!--]-->
  2041. <div></div><!--]-->
  2042. "
  2043. `,
  2044. )
  2045. })
  2046. test('mixed slot and text node', async () => {
  2047. const data = reactive({
  2048. text: 'foo',
  2049. msg: 'hi',
  2050. })
  2051. const { container } = await testHydration(
  2052. `<template>
  2053. <components.Child>
  2054. <span>{{data.text}}</span>
  2055. </components.Child>
  2056. </template>`,
  2057. {
  2058. Child: `<template><div><slot/>{{data.msg}}</div></template>`,
  2059. },
  2060. data,
  2061. )
  2062. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2063. `
  2064. "<div>
  2065. <!--[--><span>foo</span><!--]-->
  2066. hi</div>"
  2067. `,
  2068. )
  2069. data.msg = 'bar'
  2070. await nextTick()
  2071. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2072. `
  2073. "<div>
  2074. <!--[--><span>foo</span><!--]-->
  2075. bar</div>"
  2076. `,
  2077. )
  2078. })
  2079. test('mixed root slot and text node', async () => {
  2080. const data = reactive({
  2081. text: 'foo',
  2082. msg: 'hi',
  2083. })
  2084. const { container } = await testHydration(
  2085. `<template>
  2086. <components.Child>
  2087. <span>{{data.text}}</span>
  2088. </components.Child>
  2089. </template>`,
  2090. {
  2091. Child: `<template>{{data.text}}<slot/>{{data.msg}}</template>`,
  2092. },
  2093. data,
  2094. )
  2095. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2096. `
  2097. "
  2098. <!--[-->foo
  2099. <!--[--><span>foo</span><!--]-->
  2100. hi<!--]-->
  2101. "
  2102. `,
  2103. )
  2104. data.msg = 'bar'
  2105. await nextTick()
  2106. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2107. `
  2108. "
  2109. <!--[-->foo
  2110. <!--[--><span>foo</span><!--]-->
  2111. bar<!--]-->
  2112. "
  2113. `,
  2114. )
  2115. })
  2116. test('mixed consecutive slot and element', async () => {
  2117. const data = reactive({
  2118. text: 'foo',
  2119. msg: 'hi',
  2120. })
  2121. const { container } = await testHydration(
  2122. `<template>
  2123. <components.Child>
  2124. <template #foo><span>{{data.text}}</span></template>
  2125. <template #bar><span>bar</span></template>
  2126. </components.Child>
  2127. </template>`,
  2128. {
  2129. Child: `<template><div><slot name="foo"/><slot name="bar"/><div>{{data.msg}}</div></div></template>`,
  2130. },
  2131. data,
  2132. )
  2133. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2134. `
  2135. "<div>
  2136. <!--[--><span>foo</span><!--]-->
  2137. <!--[--><span>bar</span><!--]-->
  2138. <div>hi</div></div>"
  2139. `,
  2140. )
  2141. data.msg = 'bar'
  2142. await nextTick()
  2143. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2144. `
  2145. "<div>
  2146. <!--[--><span>foo</span><!--]-->
  2147. <!--[--><span>bar</span><!--]-->
  2148. <div>bar</div></div>"
  2149. `,
  2150. )
  2151. })
  2152. test('mixed slot and element', async () => {
  2153. const data = reactive({
  2154. text: 'foo',
  2155. msg: 'hi',
  2156. })
  2157. const { container } = await testHydration(
  2158. `<template>
  2159. <components.Child>
  2160. <span>{{data.text}}</span>
  2161. </components.Child>
  2162. </template>`,
  2163. {
  2164. Child: `<template><div><slot/><div>{{data.msg}}</div></div></template>`,
  2165. },
  2166. data,
  2167. )
  2168. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2169. `
  2170. "<div>
  2171. <!--[--><span>foo</span><!--]-->
  2172. <div>hi</div></div>"
  2173. `,
  2174. )
  2175. data.msg = 'bar'
  2176. await nextTick()
  2177. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2178. `
  2179. "<div>
  2180. <!--[--><span>foo</span><!--]-->
  2181. <div>bar</div></div>"
  2182. `,
  2183. )
  2184. })
  2185. test('mixed slot and component', async () => {
  2186. const data = reactive({
  2187. msg1: 'foo',
  2188. msg2: 'bar',
  2189. })
  2190. const { container } = await testHydration(
  2191. `<template>
  2192. <components.Child>
  2193. <span>{{data.msg1}}</span>
  2194. </components.Child>
  2195. </template>`,
  2196. {
  2197. Child: `
  2198. <template>
  2199. <div>
  2200. <components.Child2/>
  2201. <slot/>
  2202. <components.Child2/>
  2203. </div>
  2204. </template>`,
  2205. Child2: `
  2206. <template>
  2207. <div>{{data.msg2}}</div>
  2208. </template>`,
  2209. },
  2210. data,
  2211. )
  2212. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2213. `
  2214. "<div><div>bar</div>
  2215. <!--[--><span>foo</span><!--]-->
  2216. <div>bar</div></div>"
  2217. `,
  2218. )
  2219. data.msg2 = 'hello'
  2220. await nextTick()
  2221. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2222. `
  2223. "<div><div>hello</div>
  2224. <!--[--><span>foo</span><!--]-->
  2225. <div>hello</div></div>"
  2226. `,
  2227. )
  2228. })
  2229. test('mixed slot and fragment component', async () => {
  2230. const data = reactive({
  2231. msg1: 'foo',
  2232. msg2: 'bar',
  2233. })
  2234. const { container } = await testHydration(
  2235. `<template>
  2236. <components.Child>
  2237. <span>{{data.msg1}}</span>
  2238. </components.Child>
  2239. </template>`,
  2240. {
  2241. Child: `
  2242. <template>
  2243. <div>
  2244. <components.Child2/>
  2245. <slot/>
  2246. <components.Child2/>
  2247. </div>
  2248. </template>`,
  2249. Child2: `
  2250. <template>
  2251. <div>{{data.msg1}}</div> {{data.msg2}}
  2252. </template>`,
  2253. },
  2254. data,
  2255. )
  2256. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2257. `
  2258. "<div>
  2259. <!--[--><div>foo</div> bar<!--]-->
  2260. <!--[--><span>foo</span><!--]-->
  2261. <!--[--><div>foo</div> bar<!--]-->
  2262. </div>"
  2263. `,
  2264. )
  2265. data.msg1 = 'hello'
  2266. data.msg2 = 'vapor'
  2267. await nextTick()
  2268. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2269. `
  2270. "<div>
  2271. <!--[--><div>hello</div> vapor<!--]-->
  2272. <!--[--><span>hello</span><!--]-->
  2273. <!--[--><div>hello</div> vapor<!--]-->
  2274. </div>"
  2275. `,
  2276. )
  2277. })
  2278. test('mixed slot and v-if', async () => {
  2279. const data = reactive({
  2280. show: true,
  2281. msg: 'foo',
  2282. })
  2283. const { container } = await testHydration(
  2284. `<template>
  2285. <components.Child>
  2286. <span>{{data.msg}}</span>
  2287. </components.Child>
  2288. </template>`,
  2289. {
  2290. Child: `
  2291. <template>
  2292. <div v-if="data.show">{{data.msg}}</div>
  2293. <slot/>
  2294. <div v-if="data.show">{{data.msg}}</div>
  2295. </template>`,
  2296. },
  2297. data,
  2298. )
  2299. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2300. `
  2301. "
  2302. <!--[--><div>foo</div><!--if-->
  2303. <!--[--><span>foo</span><!--]-->
  2304. <div>foo</div><!--if--><!--]-->
  2305. "
  2306. `,
  2307. )
  2308. data.show = false
  2309. await nextTick()
  2310. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2311. `
  2312. "
  2313. <!--[--><!--if-->
  2314. <!--[--><span>foo</span><!--]-->
  2315. <!--if--><!--]-->
  2316. "
  2317. `,
  2318. )
  2319. data.show = true
  2320. await nextTick()
  2321. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`
  2322. "
  2323. <!--[--><div>foo</div><!--if-->
  2324. <!--[--><span>foo</span><!--]-->
  2325. <div>foo</div><!--if--><!--]-->
  2326. "
  2327. `)
  2328. })
  2329. test('mixed slot and v-for', async () => {
  2330. const data = reactive({
  2331. items: ['a', 'b', 'c'],
  2332. msg: 'foo',
  2333. })
  2334. const { container } = await testHydration(
  2335. `<template>
  2336. <components.Child>
  2337. <span>{{data.msg}}</span>
  2338. </components.Child>
  2339. </template>`,
  2340. {
  2341. Child: `
  2342. <template>
  2343. <div v-for="item in data.items" :key="item">{{item}}</div>
  2344. <slot/>
  2345. <div v-for="item in data.items" :key="item">{{item}}</div>
  2346. </template>`,
  2347. },
  2348. data,
  2349. )
  2350. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2351. `
  2352. "
  2353. <!--[-->
  2354. <!--[--><div>a</div><div>b</div><div>c</div><!--]-->
  2355. <!--[--><span>foo</span><!--]-->
  2356. <!--[--><div>a</div><div>b</div><div>c</div><!--]-->
  2357. <!--]-->
  2358. "
  2359. `,
  2360. )
  2361. data.items.push('d')
  2362. await nextTick()
  2363. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2364. `
  2365. "
  2366. <!--[-->
  2367. <!--[--><div>a</div><div>b</div><div>c</div><div>d</div><!--]-->
  2368. <!--[--><span>foo</span><!--]-->
  2369. <!--[--><div>a</div><div>b</div><div>c</div><div>d</div><!--]-->
  2370. <!--]-->
  2371. "
  2372. `,
  2373. )
  2374. })
  2375. test('consecutive slots', async () => {
  2376. const data = reactive({
  2377. msg1: 'foo',
  2378. msg2: 'bar',
  2379. })
  2380. const { container } = await testHydration(
  2381. `<template>
  2382. <components.Child>
  2383. <span>{{data.msg1}}</span>
  2384. <template #bar>
  2385. <span>{{data.msg2}}</span>
  2386. </template>
  2387. </components.Child>
  2388. </template>`,
  2389. {
  2390. Child: `<template><slot/><slot name="bar"/></template>`,
  2391. },
  2392. data,
  2393. )
  2394. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2395. `
  2396. "
  2397. <!--[-->
  2398. <!--[--><span>foo</span><!--]-->
  2399. <!--[--><span>bar</span><!--]-->
  2400. <!--]-->
  2401. "
  2402. `,
  2403. )
  2404. data.msg1 = 'hello'
  2405. data.msg2 = 'vapor'
  2406. await nextTick()
  2407. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2408. `
  2409. "
  2410. <!--[-->
  2411. <!--[--><span>hello</span><!--]-->
  2412. <!--[--><span>vapor</span><!--]-->
  2413. <!--]-->
  2414. "
  2415. `,
  2416. )
  2417. })
  2418. test('consecutive slots with insertion anchor', async () => {
  2419. const data = reactive({
  2420. msg1: 'foo',
  2421. msg2: 'bar',
  2422. })
  2423. const { container } = await testHydration(
  2424. `<template>
  2425. <components.Child>
  2426. <span>{{data.msg1}}</span>
  2427. <template #bar>
  2428. <span>{{data.msg2}}</span>
  2429. </template>
  2430. </components.Child>
  2431. </template>`,
  2432. {
  2433. Child: `<template>
  2434. <div>
  2435. <span/>
  2436. <slot/>
  2437. <slot name="bar"/>
  2438. <span/>
  2439. </div>
  2440. </template>`,
  2441. },
  2442. data,
  2443. )
  2444. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2445. `
  2446. "<div><span></span>
  2447. <!--[--><span>foo</span><!--]-->
  2448. <!--[--><span>bar</span><!--]-->
  2449. <span></span></div>"
  2450. `,
  2451. )
  2452. data.msg1 = 'hello'
  2453. data.msg2 = 'vapor'
  2454. await nextTick()
  2455. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2456. `
  2457. "<div><span></span>
  2458. <!--[--><span>hello</span><!--]-->
  2459. <!--[--><span>vapor</span><!--]-->
  2460. <span></span></div>"
  2461. `,
  2462. )
  2463. })
  2464. test('consecutive slots prepend', async () => {
  2465. const data = reactive({
  2466. msg1: 'foo',
  2467. msg2: 'bar',
  2468. msg3: 'baz',
  2469. })
  2470. const { container } = await testHydration(
  2471. `<template>
  2472. <components.Child>
  2473. <template #foo>
  2474. <span>{{data.msg1}}</span>
  2475. </template>
  2476. <template #bar>
  2477. <span>{{data.msg2}}</span>
  2478. </template>
  2479. </components.Child>
  2480. </template>`,
  2481. {
  2482. Child: `<template>
  2483. <div>
  2484. <slot name="foo"/>
  2485. <slot name="bar"/>
  2486. <div>{{data.msg3}}</div>
  2487. </div>
  2488. </template>`,
  2489. },
  2490. data,
  2491. )
  2492. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2493. `
  2494. "<div>
  2495. <!--[--><span>foo</span><!--]-->
  2496. <!--[--><span>bar</span><!--]-->
  2497. <div>baz</div></div>"
  2498. `,
  2499. )
  2500. data.msg1 = 'hello'
  2501. data.msg2 = 'vapor'
  2502. await nextTick()
  2503. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2504. `
  2505. "<div>
  2506. <!--[--><span>hello</span><!--]-->
  2507. <!--[--><span>vapor</span><!--]-->
  2508. <div>baz</div></div>"
  2509. `,
  2510. )
  2511. })
  2512. test('slot fallback', async () => {
  2513. const data = reactive({
  2514. foo: 'foo',
  2515. })
  2516. const { container } = await testHydration(
  2517. `<template>
  2518. <components.Child>
  2519. </components.Child>
  2520. </template>`,
  2521. {
  2522. Child: `<template><slot><span>{{data.foo}}</span></slot></template>`,
  2523. },
  2524. data,
  2525. )
  2526. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2527. `
  2528. "
  2529. <!--[--><span>foo</span><!--]-->
  2530. "
  2531. `,
  2532. )
  2533. data.foo = 'bar'
  2534. await nextTick()
  2535. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2536. `
  2537. "
  2538. <!--[--><span>bar</span><!--]-->
  2539. "
  2540. `,
  2541. )
  2542. })
  2543. test('forwarded slot', async () => {
  2544. const data = reactive({
  2545. foo: 'foo',
  2546. bar: 'bar',
  2547. })
  2548. const { container } = await testHydration(
  2549. `<template>
  2550. <div>
  2551. <components.Parent>
  2552. <span>{{data.foo}}</span>
  2553. </components.Parent>
  2554. <div>{{data.bar}}</div>
  2555. </div>
  2556. </template>`,
  2557. {
  2558. Parent: `<template><div><components.Child><slot/></components.Child></div></template>`,
  2559. Child: `<template><div><slot/></div></template>`,
  2560. },
  2561. data,
  2562. )
  2563. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2564. `
  2565. "<div><div><div>
  2566. <!--[-->
  2567. <!--[--><span>foo</span><!--]-->
  2568. <!--]-->
  2569. </div></div><div>bar</div></div>"
  2570. `,
  2571. )
  2572. data.foo = 'foo1'
  2573. data.bar = 'bar1'
  2574. await nextTick()
  2575. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2576. `
  2577. "<div><div><div>
  2578. <!--[-->
  2579. <!--[--><span>foo1</span><!--]-->
  2580. <!--]-->
  2581. </div></div><div>bar1</div></div>"
  2582. `,
  2583. )
  2584. })
  2585. test('forwarded slot with fallback', async () => {
  2586. const data = reactive({
  2587. foo: 'foo',
  2588. })
  2589. const { container } = await testHydration(
  2590. `<template>
  2591. <components.Parent/>
  2592. </template>`,
  2593. {
  2594. Parent: `<template><components.Child><slot/></components.Child></template>`,
  2595. Child: `<template><div><slot>{{data.foo}}</slot></div></template>`,
  2596. },
  2597. data,
  2598. )
  2599. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2600. `
  2601. "<div>
  2602. <!--[-->foo<!--]-->
  2603. </div>"
  2604. `,
  2605. )
  2606. data.foo = 'foo1'
  2607. await nextTick()
  2608. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2609. `
  2610. "<div>
  2611. <!--[-->foo1<!--]-->
  2612. </div>"
  2613. `,
  2614. )
  2615. })
  2616. test('forwarded slot with empty content', async () => {
  2617. const data = reactive({
  2618. foo: 'foo',
  2619. })
  2620. const { container } = await testHydration(
  2621. `<template>
  2622. <components.Foo/>
  2623. </template>`,
  2624. {
  2625. Foo: `<template>
  2626. <components.Bar>
  2627. <template #foo>
  2628. <slot name="foo" />
  2629. </template>
  2630. </components.Bar>
  2631. </template>`,
  2632. Bar: `<template>
  2633. <components.Baz>
  2634. <template #foo>
  2635. <slot name="foo" />
  2636. </template>
  2637. </components.Baz>
  2638. </template>`,
  2639. Baz: `<template>
  2640. <components.Qux>
  2641. <template #foo>
  2642. <slot name="foo" />
  2643. </template>
  2644. </components.Qux>
  2645. </template>`,
  2646. Qux: `<template>
  2647. <div>
  2648. <slot name="foo" />
  2649. <div>{{data.foo}}</div>
  2650. </div>
  2651. </template>`,
  2652. },
  2653. data,
  2654. )
  2655. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2656. `
  2657. "<div>
  2658. <!--[--><!--]-->
  2659. <div>foo</div></div>"
  2660. `,
  2661. )
  2662. data.foo = 'bar'
  2663. await nextTick()
  2664. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2665. `
  2666. "<div>
  2667. <!--[--><!--]-->
  2668. <div>bar</div></div>"
  2669. `,
  2670. )
  2671. })
  2672. })
  2673. describe('transition', async () => {
  2674. test('transition appear', async () => {
  2675. const { container } = await testHydration(
  2676. `<template>
  2677. <transition appear>
  2678. <div>foo</div>
  2679. </transition>
  2680. </template>`,
  2681. )
  2682. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2683. `"<div style="" class="v-enter-from v-enter-active">foo</div>"`,
  2684. )
  2685. expect(`mismatch`).not.toHaveBeenWarned()
  2686. })
  2687. test('transition appear work with pre-existing class', async () => {
  2688. const { container } = await testHydration(
  2689. `<template>
  2690. <transition appear>
  2691. <div class="foo">foo</div>
  2692. </transition>
  2693. </template>`,
  2694. )
  2695. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2696. `"<div class="foo v-enter-from v-enter-active" style="">foo</div>"`,
  2697. )
  2698. expect(`mismatch`).not.toHaveBeenWarned()
  2699. })
  2700. test('transition appear work with empty content', async () => {
  2701. const data = ref(true)
  2702. const { container } = await testHydration(
  2703. `<template>
  2704. <transition appear>
  2705. <slot v-if="data"></slot>
  2706. <span v-else>foo</span>
  2707. </transition>
  2708. </template>`,
  2709. undefined,
  2710. data,
  2711. )
  2712. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2713. `"<!--slot--><!--if-->"`,
  2714. )
  2715. expect(`mismatch`).not.toHaveBeenWarned()
  2716. data.value = false
  2717. await nextTick()
  2718. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2719. `"<span class="v-enter-from v-enter-active">foo</span><!--if-->"`,
  2720. )
  2721. })
  2722. test('transition appear with v-if', async () => {
  2723. const data = ref(false)
  2724. const { container } = await testHydration(
  2725. `<template>
  2726. <transition appear>
  2727. <div v-if="data">foo</div>
  2728. </transition>
  2729. </template>`,
  2730. undefined,
  2731. data,
  2732. )
  2733. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2734. `"<!--if-->"`,
  2735. )
  2736. expect(`mismatch`).not.toHaveBeenWarned()
  2737. })
  2738. test('transition appear with v-show', async () => {
  2739. const data = ref(false)
  2740. const { container } = await testHydration(
  2741. `<template>
  2742. <transition appear>
  2743. <div v-show="data">foo</div>
  2744. </transition>
  2745. </template>`,
  2746. undefined,
  2747. data,
  2748. )
  2749. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2750. `"<div style="display:none;" class="v-enter-from v-enter-active v-leave-from v-leave-active">foo</div>"`,
  2751. )
  2752. expect(`mismatch`).not.toHaveBeenWarned()
  2753. })
  2754. test('transition appear w/ event listener', async () => {
  2755. const { container } = await testHydration(
  2756. `<script setup>
  2757. import { ref } from 'vue'
  2758. const count = ref(0)
  2759. </script>
  2760. <template>
  2761. <transition appear>
  2762. <button @click="count++">{{ count }}</button>
  2763. </transition>
  2764. </template>`,
  2765. )
  2766. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2767. `"<button style="" class="v-enter-from v-enter-active">0</button>"`,
  2768. )
  2769. triggerEvent('click', container.querySelector('button')!)
  2770. await nextTick()
  2771. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  2772. `"<button style="" class="v-enter-from v-enter-active">1</button>"`,
  2773. )
  2774. })
  2775. })
  2776. describe('teleport', () => {
  2777. test('basic', async () => {
  2778. const data = ref({
  2779. msg: ref('foo'),
  2780. disabled: ref(false),
  2781. fn: vi.fn(),
  2782. })
  2783. const teleportContainer = document.createElement('div')
  2784. teleportContainer.id = 'teleport'
  2785. teleportContainer.innerHTML =
  2786. `<!--teleport start anchor-->` +
  2787. `<span>foo</span>` +
  2788. `<span class="foo"></span>` +
  2789. `<!--teleport anchor-->`
  2790. document.body.appendChild(teleportContainer)
  2791. const { block, container } = await mountWithHydration(
  2792. '<!--teleport start--><!--teleport end-->',
  2793. `<teleport to="#teleport" :disabled="data.disabled">
  2794. <span>{{data.msg}}</span>
  2795. <span :class="data.msg" @click="data.fn"></span>
  2796. </teleport>`,
  2797. data,
  2798. )
  2799. const teleport = block as TeleportFragment
  2800. expect(teleport.anchor).toBe(container.lastChild)
  2801. expect(teleport.target).toBe(teleportContainer)
  2802. expect(teleport.targetStart).toBe(teleportContainer.childNodes[0])
  2803. expect((teleport.nodes as Node[])[0]).toBe(
  2804. teleportContainer.childNodes[1],
  2805. )
  2806. expect((teleport.nodes as Node[])[1]).toBe(
  2807. teleportContainer.childNodes[2],
  2808. )
  2809. expect(teleport.targetAnchor).toBe(teleportContainer.childNodes[3])
  2810. expect(container.innerHTML).toMatchInlineSnapshot(
  2811. `"<!--teleport start--><!--teleport end-->"`,
  2812. )
  2813. // event handler
  2814. triggerEvent('click', teleportContainer.querySelector('.foo')!)
  2815. expect(data.value.fn).toHaveBeenCalled()
  2816. data.value.msg = 'bar'
  2817. await nextTick()
  2818. expect(formatHtml(teleportContainer.innerHTML)).toBe(
  2819. `<!--teleport start anchor-->` +
  2820. `<span>bar</span>` +
  2821. `<span class="bar"></span>` +
  2822. `<!--teleport anchor-->`,
  2823. )
  2824. data.value.disabled = true
  2825. await nextTick()
  2826. expect(container.innerHTML).toBe(
  2827. `<!--teleport start-->` +
  2828. `<span>bar</span>` +
  2829. `<span class="bar"></span>` +
  2830. `<!--teleport end-->`,
  2831. )
  2832. expect(formatHtml(teleportContainer.innerHTML)).toMatchInlineSnapshot(
  2833. `"<!--teleport start anchor--><!--teleport anchor-->"`,
  2834. )
  2835. data.value.msg = 'baz'
  2836. await nextTick()
  2837. expect(container.innerHTML).toBe(
  2838. `<!--teleport start-->` +
  2839. `<span>baz</span>` +
  2840. `<span class="baz"></span>` +
  2841. `<!--teleport end-->`,
  2842. )
  2843. data.value.disabled = false
  2844. await nextTick()
  2845. expect(container.innerHTML).toMatchInlineSnapshot(
  2846. `"<!--teleport start--><!--teleport end-->"`,
  2847. )
  2848. expect(formatHtml(teleportContainer.innerHTML)).toBe(
  2849. `<!--teleport start anchor-->` +
  2850. `<span>baz</span>` +
  2851. `<span class="baz"></span>` +
  2852. `<!--teleport anchor-->`,
  2853. )
  2854. })
  2855. test('multiple + integration', async () => {
  2856. const data = ref({
  2857. msg: ref('foo'),
  2858. fn1: vi.fn(),
  2859. fn2: vi.fn(),
  2860. })
  2861. const code = `
  2862. <teleport to="#teleport2">
  2863. <span>{{data.msg}}</span>
  2864. <span :class="data.msg" @click="data.fn1"></span>
  2865. </teleport>
  2866. <teleport to="#teleport2">
  2867. <span>{{data.msg}}2</span>
  2868. <span :class="data.msg + 2" @click="data.fn2"></span>
  2869. </teleport>`
  2870. const SSRComp = compileVaporComponent(code, data, undefined, true)
  2871. const teleportContainer = document.createElement('div')
  2872. teleportContainer.id = 'teleport2'
  2873. const ctx = {} as any
  2874. const mainHtml = await VueServerRenderer.renderToString(
  2875. runtimeDom.createSSRApp(SSRComp),
  2876. ctx,
  2877. )
  2878. expect(mainHtml).toBe(
  2879. `<!--[-->` +
  2880. `<!--teleport start--><!--teleport end-->` +
  2881. `<!--teleport start--><!--teleport end-->` +
  2882. `<!--]-->`,
  2883. )
  2884. const teleportHtml = ctx.teleports!['#teleport2']
  2885. expect(teleportHtml).toBe(
  2886. `<!--teleport start anchor-->` +
  2887. `<span>foo</span><span class="foo"></span>` +
  2888. `<!--teleport anchor-->` +
  2889. `<!--teleport start anchor-->` +
  2890. `<span>foo2</span><span class="foo2"></span>` +
  2891. `<!--teleport anchor-->`,
  2892. )
  2893. teleportContainer.innerHTML = teleportHtml
  2894. document.body.appendChild(teleportContainer)
  2895. const { block, container } = await mountWithHydration(
  2896. mainHtml,
  2897. code,
  2898. data,
  2899. )
  2900. const teleports = block as any as TeleportFragment[]
  2901. const teleport1 = teleports[0]
  2902. const teleport2 = teleports[1]
  2903. expect(teleport1.anchor).toBe(container.childNodes[2])
  2904. expect(teleport2.anchor).toBe(container.childNodes[4])
  2905. expect(teleport1.target).toBe(teleportContainer)
  2906. expect(teleport1.targetStart).toBe(teleportContainer.childNodes[0])
  2907. expect((teleport1.nodes as Node[])[0]).toBe(
  2908. teleportContainer.childNodes[1],
  2909. )
  2910. expect(teleport1.targetAnchor).toBe(teleportContainer.childNodes[3])
  2911. expect(teleport2.target).toBe(teleportContainer)
  2912. expect(teleport2.targetStart).toBe(teleportContainer.childNodes[4])
  2913. expect((teleport2.nodes as Node[])[0]).toBe(
  2914. teleportContainer.childNodes[5],
  2915. )
  2916. expect(teleport2.targetAnchor).toBe(teleportContainer.childNodes[7])
  2917. expect(container.innerHTML).toBe(
  2918. `<!--[-->` +
  2919. `<!--teleport start--><!--teleport end-->` +
  2920. `<!--teleport start--><!--teleport end-->` +
  2921. `<!--]-->`,
  2922. )
  2923. // event handler
  2924. triggerEvent('click', teleportContainer.querySelector('.foo')!)
  2925. expect(data.value.fn1).toHaveBeenCalled()
  2926. triggerEvent('click', teleportContainer.querySelector('.foo2')!)
  2927. expect(data.value.fn2).toHaveBeenCalled()
  2928. data.value.msg = 'bar'
  2929. await nextTick()
  2930. expect(teleportContainer.innerHTML).toBe(
  2931. `<!--teleport start anchor-->` +
  2932. `<span>bar</span>` +
  2933. `<span class="bar"></span>` +
  2934. `<!--teleport anchor-->` +
  2935. `<!--teleport start anchor-->` +
  2936. `<span>bar2</span>` +
  2937. `<span class="bar2"></span>` +
  2938. `<!--teleport anchor-->`,
  2939. )
  2940. })
  2941. test('disabled', async () => {
  2942. const data = ref({
  2943. msg: ref('foo'),
  2944. fn1: vi.fn(),
  2945. fn2: vi.fn(),
  2946. })
  2947. const code = `
  2948. <div>foo</div>
  2949. <teleport to="#teleport3" disabled="true">
  2950. <span>{{data.msg}}</span>
  2951. <span :class="data.msg" @click="data.fn1"></span>
  2952. </teleport>
  2953. <div :class="data.msg + 2" @click="data.fn2">bar</div>
  2954. `
  2955. const SSRComp = compileVaporComponent(code, data, undefined, true)
  2956. const teleportContainer = document.createElement('div')
  2957. teleportContainer.id = 'teleport3'
  2958. const ctx = {} as any
  2959. const mainHtml = await VueServerRenderer.renderToString(
  2960. runtimeDom.createSSRApp(SSRComp),
  2961. ctx,
  2962. )
  2963. expect(mainHtml).toBe(
  2964. `<!--[-->` +
  2965. `<div>foo</div>` +
  2966. `<!--teleport start-->` +
  2967. `<span>foo</span>` +
  2968. `<span class="foo"></span>` +
  2969. `<!--teleport end-->` +
  2970. `<div class="foo2">bar</div>` +
  2971. `<!--]-->`,
  2972. )
  2973. const teleportHtml = ctx.teleports!['#teleport3']
  2974. expect(teleportHtml).toMatchInlineSnapshot(
  2975. `"<!--teleport start anchor--><!--teleport anchor-->"`,
  2976. )
  2977. teleportContainer.innerHTML = teleportHtml
  2978. document.body.appendChild(teleportContainer)
  2979. const { block, container } = await mountWithHydration(
  2980. mainHtml,
  2981. code,
  2982. data,
  2983. )
  2984. const blocks = block as any[]
  2985. expect(blocks[0]).toBe(container.childNodes[1])
  2986. const teleport = blocks[1] as TeleportFragment
  2987. expect((teleport.nodes as Node[])[0]).toBe(container.childNodes[3])
  2988. expect((teleport.nodes as Node[])[1]).toBe(container.childNodes[4])
  2989. expect(teleport.anchor).toBe(container.childNodes[5])
  2990. expect(teleport.target).toBe(teleportContainer)
  2991. expect(teleport.targetStart).toBe(teleportContainer.childNodes[0])
  2992. expect(teleport.targetAnchor).toBe(teleportContainer.childNodes[1])
  2993. expect(blocks[2]).toBe(container.childNodes[6])
  2994. expect(container.innerHTML).toBe(
  2995. `<!--[-->` +
  2996. `<div>foo</div>` +
  2997. `<!--teleport start-->` +
  2998. `<span>foo</span>` +
  2999. `<span class="foo"></span>` +
  3000. `<!--teleport end-->` +
  3001. `<div class="foo2">bar</div>` +
  3002. `<!--]-->`,
  3003. )
  3004. // event handler
  3005. triggerEvent('click', container.querySelector('.foo')!)
  3006. expect(data.value.fn1).toHaveBeenCalled()
  3007. triggerEvent('click', container.querySelector('.foo2')!)
  3008. expect(data.value.fn2).toHaveBeenCalled()
  3009. data.value.msg = 'bar'
  3010. await nextTick()
  3011. expect(container.innerHTML).toBe(
  3012. `<!--[-->` +
  3013. `<div>foo</div>` +
  3014. `<!--teleport start-->` +
  3015. `<span>bar</span>` +
  3016. `<span class="bar"></span>` +
  3017. `<!--teleport end-->` +
  3018. `<div class="bar2">bar</div>` +
  3019. `<!--]-->`,
  3020. )
  3021. })
  3022. test('disabled + as component root', async () => {
  3023. const { container } = await mountWithHydration(
  3024. `<!--[-->` +
  3025. `<div>Parent fragment</div>` +
  3026. `<!--teleport start--><div>Teleport content</div><!--teleport end-->` +
  3027. `<!--]-->`,
  3028. `
  3029. <div>Parent fragment</div>
  3030. <teleport to="body" disabled>
  3031. <div>Teleport content</div>
  3032. </teleport>
  3033. `,
  3034. )
  3035. expect(container.innerHTML).toBe(
  3036. `<!--[-->` +
  3037. `<div>Parent fragment</div>` +
  3038. `<!--teleport start-->` +
  3039. `<div>Teleport content</div>` +
  3040. `<!--teleport end-->` +
  3041. `<!--]-->`,
  3042. )
  3043. expect(`mismatch`).not.toHaveBeenWarned()
  3044. })
  3045. test('as component root', async () => {
  3046. const teleportContainer = document.createElement('div')
  3047. teleportContainer.id = 'teleport4'
  3048. teleportContainer.innerHTML = `<!--teleport start anchor-->hello<!--teleport anchor-->`
  3049. document.body.appendChild(teleportContainer)
  3050. const { block, container } = await mountWithHydration(
  3051. '<!--teleport start--><!--teleport end-->',
  3052. `<components.Wrapper></components.Wrapper>`,
  3053. undefined,
  3054. {
  3055. Wrapper: compileVaporComponent(
  3056. `<teleport to="#teleport4">hello</teleport>`,
  3057. ),
  3058. },
  3059. )
  3060. const teleport = (block as VaporComponentInstance)
  3061. .block as TeleportFragment
  3062. expect(teleport.anchor).toBe(container.childNodes[1])
  3063. expect(teleport.target).toBe(teleportContainer)
  3064. expect(teleport.targetStart).toBe(teleportContainer.childNodes[0])
  3065. expect(teleport.nodes).toBe(teleportContainer.childNodes[1])
  3066. expect(teleport.targetAnchor).toBe(teleportContainer.childNodes[2])
  3067. })
  3068. test('nested', async () => {
  3069. const teleportContainer = document.createElement('div')
  3070. teleportContainer.id = 'teleport5'
  3071. teleportContainer.innerHTML =
  3072. `<!--teleport start anchor-->` +
  3073. `<!--teleport start--><!--teleport end-->` +
  3074. `<!--teleport anchor-->` +
  3075. `<!--teleport start anchor-->` +
  3076. `<div>child</div>` +
  3077. `<!--teleport anchor-->`
  3078. document.body.appendChild(teleportContainer)
  3079. const { block, container } = await mountWithHydration(
  3080. '<!--teleport start--><!--teleport end-->',
  3081. `<teleport to="#teleport5">
  3082. <teleport to="#teleport5"><div>child</div></teleport>
  3083. </teleport>`,
  3084. )
  3085. const teleport = block as TeleportFragment
  3086. expect(teleport.anchor).toBe(container.childNodes[1])
  3087. expect(teleport.targetStart).toBe(teleportContainer.childNodes[0])
  3088. expect(teleport.targetAnchor).toBe(teleportContainer.childNodes[3])
  3089. const childTeleport = teleport.nodes as TeleportFragment
  3090. expect(childTeleport.anchor).toBe(teleportContainer.childNodes[2])
  3091. expect(childTeleport.targetStart).toBe(teleportContainer.childNodes[4])
  3092. expect(childTeleport.targetAnchor).toBe(teleportContainer.childNodes[6])
  3093. expect(childTeleport.nodes).toBe(teleportContainer.childNodes[5])
  3094. })
  3095. test('unmount (full integration)', async () => {
  3096. const targetId = 'teleport6'
  3097. const data = ref({
  3098. toggle: ref(true),
  3099. })
  3100. const template1 = `<Teleport to="#${targetId}"><span>Teleported Comp1</span></Teleport>`
  3101. const Comp1 = compileVaporComponent(template1)
  3102. const SSRComp1 = compileVaporComponent(
  3103. template1,
  3104. undefined,
  3105. undefined,
  3106. true,
  3107. )
  3108. const template2 = `<div>Comp2</div>`
  3109. const Comp2 = compileVaporComponent(template2)
  3110. const SSRComp2 = compileVaporComponent(
  3111. template2,
  3112. undefined,
  3113. undefined,
  3114. true,
  3115. )
  3116. const appCode = `
  3117. <div>
  3118. <components.Comp1 v-if="data.toggle"/>
  3119. <components.Comp2 v-else/>
  3120. </div>
  3121. `
  3122. const SSRApp = compileVaporComponent(
  3123. appCode,
  3124. data,
  3125. {
  3126. Comp1: SSRComp1,
  3127. Comp2: SSRComp2,
  3128. },
  3129. true,
  3130. )
  3131. const teleportContainer = document.createElement('div')
  3132. teleportContainer.id = targetId
  3133. document.body.appendChild(teleportContainer)
  3134. const ctx = {} as any
  3135. const mainHtml = await VueServerRenderer.renderToString(
  3136. runtimeDom.createSSRApp(SSRApp),
  3137. ctx,
  3138. )
  3139. expect(mainHtml).toBe(
  3140. '<div><!--teleport start--><!--teleport end--></div>',
  3141. )
  3142. teleportContainer.innerHTML = ctx.teleports![`#${targetId}`]
  3143. const { container } = await mountWithHydration(mainHtml, appCode, data, {
  3144. Comp1,
  3145. Comp2,
  3146. })
  3147. expect(container.innerHTML).toBe(
  3148. '<div><!--teleport start--><!--teleport end--><!--if--></div>',
  3149. )
  3150. expect(teleportContainer.innerHTML).toBe(
  3151. `<!--teleport start anchor-->` +
  3152. `<span>Teleported Comp1</span>` +
  3153. `<!--teleport anchor-->`,
  3154. )
  3155. expect(`mismatch`).not.toHaveBeenWarned()
  3156. data.value.toggle = false
  3157. await nextTick()
  3158. expect(container.innerHTML).toBe('<div><div>Comp2</div><!--if--></div>')
  3159. expect(teleportContainer.innerHTML).toBe('')
  3160. })
  3161. test('unmount (mismatch + full integration)', async () => {
  3162. const targetId = 'teleport7'
  3163. const data = ref({
  3164. toggle: ref(true),
  3165. })
  3166. const template1 = `<Teleport to="#${targetId}"><span>Teleported Comp1</span></Teleport>`
  3167. const Comp1 = compileVaporComponent(template1)
  3168. const SSRComp1 = compileVaporComponent(
  3169. template1,
  3170. undefined,
  3171. undefined,
  3172. true,
  3173. )
  3174. const template2 = `<div>Comp2</div>`
  3175. const Comp2 = compileVaporComponent(template2)
  3176. const SSRComp2 = compileVaporComponent(
  3177. template2,
  3178. undefined,
  3179. undefined,
  3180. true,
  3181. )
  3182. const appCode = `
  3183. <div>
  3184. <components.Comp1 v-if="data.toggle"/>
  3185. <components.Comp2 v-else/>
  3186. </div>
  3187. `
  3188. const SSRApp = compileVaporComponent(
  3189. appCode,
  3190. data,
  3191. {
  3192. Comp1: SSRComp1,
  3193. Comp2: SSRComp2,
  3194. },
  3195. true,
  3196. )
  3197. const teleportContainer = document.createElement('div')
  3198. teleportContainer.id = targetId
  3199. document.body.appendChild(teleportContainer)
  3200. const mainHtml = await VueServerRenderer.renderToString(
  3201. runtimeDom.createSSRApp(SSRApp),
  3202. )
  3203. expect(mainHtml).toBe(
  3204. '<div><!--teleport start--><!--teleport end--></div>',
  3205. )
  3206. expect(teleportContainer.innerHTML).toBe('')
  3207. const { container } = await mountWithHydration(mainHtml, appCode, data, {
  3208. Comp1,
  3209. Comp2,
  3210. })
  3211. expect(container.innerHTML).toBe(
  3212. '<div><!--teleport start--><!--teleport end--><!--if--></div>',
  3213. )
  3214. expect(teleportContainer.innerHTML).toBe(`<span>Teleported Comp1</span>`)
  3215. expect(`Hydration children mismatch`).toHaveBeenWarned()
  3216. data.value.toggle = false
  3217. await nextTick()
  3218. expect(container.innerHTML).toBe('<div><div>Comp2</div><!--if--></div>')
  3219. expect(teleportContainer.innerHTML).toBe('')
  3220. })
  3221. test('target change (mismatch + full integration)', async () => {
  3222. const targetId1 = 'teleport8-1'
  3223. const targetId2 = 'teleport8-2'
  3224. const data = ref({
  3225. target: ref(targetId1),
  3226. msg: ref('foo'),
  3227. })
  3228. const template = `<Teleport :to="'#' + data.target"><span>{{data.msg}}</span></Teleport>`
  3229. const Comp = compileVaporComponent(template, data)
  3230. const SSRComp = compileVaporComponent(template, data, undefined, true)
  3231. const teleportContainer1 = document.createElement('div')
  3232. teleportContainer1.id = targetId1
  3233. const teleportContainer2 = document.createElement('div')
  3234. teleportContainer2.id = targetId2
  3235. document.body.appendChild(teleportContainer1)
  3236. document.body.appendChild(teleportContainer2)
  3237. // server render
  3238. const mainHtml = await VueServerRenderer.renderToString(
  3239. runtimeDom.createSSRApp(SSRComp),
  3240. )
  3241. expect(mainHtml).toBe(`<!--teleport start--><!--teleport end-->`)
  3242. expect(teleportContainer1.innerHTML).toBe('')
  3243. expect(teleportContainer2.innerHTML).toBe('')
  3244. // hydrate
  3245. const { container } = await mountWithHydration(mainHtml, template, data, {
  3246. Comp,
  3247. })
  3248. expect(container.innerHTML).toBe(
  3249. `<!--teleport start--><!--teleport end-->`,
  3250. )
  3251. expect(teleportContainer1.innerHTML).toBe(`<span>foo</span>`)
  3252. expect(teleportContainer2.innerHTML).toBe('')
  3253. expect(`Hydration children mismatch`).toHaveBeenWarned()
  3254. data.value.target = targetId2
  3255. data.value.msg = 'bar'
  3256. await nextTick()
  3257. expect(container.innerHTML).toBe(
  3258. `<!--teleport start--><!--teleport end-->`,
  3259. )
  3260. expect(teleportContainer1.innerHTML).toBe('')
  3261. expect(teleportContainer2.innerHTML).toBe(`<span>bar</span>`)
  3262. })
  3263. test('with disabled teleport + undefined target', async () => {
  3264. const data = ref({
  3265. msg: ref('foo'),
  3266. })
  3267. const { container } = await mountWithHydration(
  3268. '<!--teleport start--><span>foo</span><!--teleport end-->',
  3269. `<teleport :to="undefined" :disabled="true">
  3270. <span>{{data.msg}}</span>
  3271. </teleport>`,
  3272. data,
  3273. )
  3274. expect(container.innerHTML).toBe(
  3275. `<!--teleport start--><span>foo</span><!--teleport end-->`,
  3276. )
  3277. data.value.msg = 'bar'
  3278. await nextTick()
  3279. expect(container.innerHTML).toBe(
  3280. `<!--teleport start--><span>bar</span><!--teleport end-->`,
  3281. )
  3282. })
  3283. test('should apply css vars after hydration', async () => {
  3284. const state = reactive({ color: 'red' })
  3285. const teleportContainer = document.createElement('div')
  3286. teleportContainer.id = 'teleport-css-vars'
  3287. teleportContainer.innerHTML =
  3288. `<!--teleport start anchor-->` +
  3289. `<span>content</span>` +
  3290. `<!--teleport anchor-->`
  3291. document.body.appendChild(teleportContainer)
  3292. const App = defineVaporComponent({
  3293. setup() {
  3294. useVaporCssVars(() => state)
  3295. return createComponent(
  3296. VaporTeleport,
  3297. { to: () => '#teleport-css-vars' },
  3298. { default: () => template('<span>content</span>', true)() },
  3299. )
  3300. },
  3301. })
  3302. const container = document.createElement('div')
  3303. container.innerHTML = '<!--teleport start--><!--teleport end-->'
  3304. document.body.appendChild(container)
  3305. const app = createVaporSSRApp(App)
  3306. app.mount(container)
  3307. await nextTick()
  3308. // css vars should be applied after hydration
  3309. const span = teleportContainer.querySelector('span') as HTMLElement
  3310. expect(span).toBeTruthy()
  3311. expect(span.style.getPropertyValue('--color')).toBe('red')
  3312. expect(span.hasAttribute('data-v-owner')).toBe(true)
  3313. // css vars should update reactively
  3314. state.color = 'green'
  3315. await nextTick()
  3316. expect(span.style.getPropertyValue('--color')).toBe('green')
  3317. })
  3318. })
  3319. describe('async component', async () => {
  3320. test('async component', async () => {
  3321. const data = ref({
  3322. spy: vi.fn(),
  3323. })
  3324. const compCode = `<button @click="data.spy">hello!</button>`
  3325. const SSRComp = compileVaporComponent(compCode, data, undefined, true)
  3326. let serverResolve: any
  3327. // use defineAsyncComponent in SSR
  3328. let AsyncComp = defineAsyncComponent(
  3329. () =>
  3330. new Promise(r => {
  3331. serverResolve = r
  3332. }),
  3333. )
  3334. const appCode = `hello<components.AsyncComp/>world`
  3335. const SSRApp = compileVaporComponent(appCode, data, { AsyncComp }, true)
  3336. // server render
  3337. const htmlPromise = VueServerRenderer.renderToString(
  3338. runtimeDom.createSSRApp(SSRApp),
  3339. )
  3340. serverResolve(SSRComp)
  3341. const html = await htmlPromise
  3342. expect(html).toMatchInlineSnapshot(
  3343. `"<!--[-->hello<button>hello!</button>world<!--]-->"`,
  3344. )
  3345. // hydration
  3346. let clientResolve: any
  3347. AsyncComp = defineVaporAsyncComponent(
  3348. () =>
  3349. new Promise(r => {
  3350. clientResolve = r
  3351. }),
  3352. ) as any
  3353. const Comp = compileVaporComponent(compCode, data)
  3354. const App = compileVaporComponent(appCode, data, { AsyncComp })
  3355. const container = document.createElement('div')
  3356. container.innerHTML = html
  3357. document.body.appendChild(container)
  3358. createVaporSSRApp(App).mount(container)
  3359. // hydration not complete yet
  3360. triggerEvent('click', container.querySelector('button')!)
  3361. expect(data.value.spy).not.toHaveBeenCalled()
  3362. // resolve
  3363. clientResolve(Comp)
  3364. await new Promise(r => setTimeout(r))
  3365. // should be hydrated now
  3366. triggerEvent('click', container.querySelector('button')!)
  3367. expect(data.value.spy).toHaveBeenCalled()
  3368. })
  3369. // No longer needed, parent component updates in vapor mode no longer
  3370. // cause child components to re-render
  3371. // test.todo('update async wrapper before resolve', async () => {})
  3372. test('update async component after parent mount before async component resolve', async () => {
  3373. const data = ref({
  3374. toggle: true,
  3375. })
  3376. const compCode = `
  3377. <script vapor>
  3378. defineProps(['toggle'])
  3379. </script>
  3380. <template>
  3381. <h1>{{ toggle ? 'Async component' : 'Updated async component' }}</h1>
  3382. </template>
  3383. `
  3384. const SSRComp = compileVaporComponent(
  3385. compCode,
  3386. undefined,
  3387. undefined,
  3388. true,
  3389. )
  3390. let serverResolve: any
  3391. // use defineAsyncComponent in SSR
  3392. let AsyncComp = defineAsyncComponent(
  3393. () =>
  3394. new Promise(r => {
  3395. serverResolve = r
  3396. }),
  3397. )
  3398. const appCode = `<components.AsyncComp :toggle="data.toggle"/>`
  3399. const SSRApp = compileVaporComponent(appCode, data, { AsyncComp }, true)
  3400. // server render
  3401. const htmlPromise = VueServerRenderer.renderToString(
  3402. runtimeDom.createSSRApp(SSRApp),
  3403. )
  3404. serverResolve(SSRComp)
  3405. const html = await htmlPromise
  3406. expect(html).toMatchInlineSnapshot(`"<h1>Async component</h1>"`)
  3407. // hydration
  3408. let clientResolve: any
  3409. AsyncComp = defineVaporAsyncComponent(
  3410. () =>
  3411. new Promise(r => {
  3412. clientResolve = r
  3413. }),
  3414. ) as any
  3415. const Comp = compileVaporComponent(compCode)
  3416. const App = compileVaporComponent(appCode, data, { AsyncComp })
  3417. const container = document.createElement('div')
  3418. container.innerHTML = html
  3419. document.body.appendChild(container)
  3420. createVaporSSRApp(App).mount(container)
  3421. // update before resolve
  3422. data.value.toggle = false
  3423. await nextTick()
  3424. // resolve
  3425. clientResolve(Comp)
  3426. await new Promise(r => setTimeout(r))
  3427. // prevent lazy hydration since the component has been patched
  3428. expect('Skipping lazy hydration for component').toHaveBeenWarned()
  3429. expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  3430. expect(container.innerHTML).toMatchInlineSnapshot(
  3431. `"<h1>Updated async component</h1><!--async component-->"`,
  3432. )
  3433. })
  3434. test('update async component (fragment root) after parent mount before async component resolve', async () => {
  3435. const data = ref({
  3436. toggle: true,
  3437. })
  3438. const compCode = `
  3439. <script vapor>
  3440. defineProps(['toggle'])
  3441. </script>
  3442. <template>
  3443. <h1>{{ toggle ? 'Async component' : 'Updated async component' }}</h1>
  3444. <h2>fragment root</h2>
  3445. </template>
  3446. `
  3447. const SSRComp = compileVaporComponent(
  3448. compCode,
  3449. undefined,
  3450. undefined,
  3451. true,
  3452. )
  3453. let serverResolve: any
  3454. // use defineAsyncComponent in SSR
  3455. let AsyncComp = defineAsyncComponent(
  3456. () =>
  3457. new Promise(r => {
  3458. serverResolve = r
  3459. }),
  3460. )
  3461. const appCode = `<components.AsyncComp :toggle="data.toggle"/>`
  3462. const SSRApp = compileVaporComponent(appCode, data, { AsyncComp }, true)
  3463. // server render
  3464. const htmlPromise = VueServerRenderer.renderToString(
  3465. runtimeDom.createSSRApp(SSRApp),
  3466. )
  3467. serverResolve(SSRComp)
  3468. const html = await htmlPromise
  3469. expect(html).toMatchInlineSnapshot(
  3470. `"<!--[--><h1>Async component</h1><h2>fragment root</h2><!--]-->"`,
  3471. )
  3472. // hydration
  3473. let clientResolve: any
  3474. AsyncComp = defineVaporAsyncComponent(
  3475. () =>
  3476. new Promise(r => {
  3477. clientResolve = r
  3478. }),
  3479. ) as any
  3480. const Comp = compileVaporComponent(compCode)
  3481. const App = compileVaporComponent(appCode, data, { AsyncComp })
  3482. const container = document.createElement('div')
  3483. container.innerHTML = html
  3484. document.body.appendChild(container)
  3485. createVaporSSRApp(App).mount(container)
  3486. // update before resolve
  3487. data.value.toggle = false
  3488. await nextTick()
  3489. // resolve
  3490. clientResolve(Comp)
  3491. await new Promise(r => setTimeout(r))
  3492. // prevent lazy hydration since the component has been patched
  3493. expect('Skipping lazy hydration for component').toHaveBeenWarned()
  3494. expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  3495. expect(container.innerHTML).toMatchInlineSnapshot(
  3496. `"<!--[--><h1>Updated async component</h1><h2>fragment root</h2><!--async component--><!--]-->"`,
  3497. )
  3498. })
  3499. // required vapor Suspense
  3500. test.todo('hydrate safely when property used by async setup changed before render', async () => {})
  3501. // required vapor Suspense
  3502. test.todo('hydrate safely when property used by deep nested async setup changed before render', async () => {})
  3503. test('unmount async wrapper before load', async () => {
  3504. const data = ref({
  3505. toggle: true,
  3506. })
  3507. const compCode = `<div>async</div>`
  3508. const appCode = `
  3509. <div>
  3510. <components.AsyncComp v-if="data.toggle"/>
  3511. <div v-else>hi</div>
  3512. </div>
  3513. `
  3514. // hydration
  3515. let clientResolve: any
  3516. const AsyncComp = defineVaporAsyncComponent(
  3517. () =>
  3518. new Promise(r => {
  3519. clientResolve = r
  3520. }),
  3521. )
  3522. const Comp = compileVaporComponent(compCode)
  3523. const App = compileVaporComponent(appCode, data, {
  3524. AsyncComp,
  3525. })
  3526. const container = document.createElement('div')
  3527. container.innerHTML = '<div><div>async</div></div>'
  3528. createVaporSSRApp(App).mount(container)
  3529. // unmount before resolve
  3530. data.value.toggle = false
  3531. await nextTick()
  3532. expect(container.innerHTML).toBe(`<div><div>hi</div><!--if--></div>`)
  3533. // resolve
  3534. clientResolve(Comp)
  3535. await new Promise(r => setTimeout(r))
  3536. // should remain unmounted
  3537. expect(container.innerHTML).toBe(`<div><div>hi</div><!--if--></div>`)
  3538. })
  3539. test('unmount async wrapper before load (fragment)', async () => {
  3540. const data = ref({
  3541. toggle: true,
  3542. })
  3543. const compCode = `<div>async</div><div>fragment</div>`
  3544. const appCode = `
  3545. <div>
  3546. <components.AsyncComp v-if="data.toggle"/>
  3547. <div v-else>hi</div>
  3548. </div>
  3549. `
  3550. // hydration
  3551. let clientResolve: any
  3552. const AsyncComp = defineVaporAsyncComponent(
  3553. () =>
  3554. new Promise(r => {
  3555. clientResolve = r
  3556. }),
  3557. )
  3558. const Comp = compileVaporComponent(compCode)
  3559. const App = compileVaporComponent(appCode, data, {
  3560. AsyncComp,
  3561. })
  3562. const container = document.createElement('div')
  3563. container.innerHTML =
  3564. '<div><!--[--><div>async</div><div>fragment</div><!--]--></div>'
  3565. createVaporSSRApp(App).mount(container)
  3566. // unmount before resolve
  3567. data.value.toggle = false
  3568. await nextTick()
  3569. expect(container.innerHTML).toBe(`<div><div>hi</div><!--if--></div>`)
  3570. // resolve
  3571. clientResolve(Comp)
  3572. await new Promise(r => setTimeout(r))
  3573. // should remain unmounted
  3574. expect(container.innerHTML).toBe(`<div><div>hi</div><!--if--></div>`)
  3575. })
  3576. test('nested async wrapper', async () => {
  3577. const toggleCode = `
  3578. <script vapor>
  3579. import { onMounted, ref, nextTick } from 'vue'
  3580. const show = ref(false)
  3581. onMounted(() => {
  3582. nextTick(() => {
  3583. show.value = true
  3584. })
  3585. })
  3586. </script>
  3587. <template>
  3588. <div v-show="show">
  3589. <slot />
  3590. </div>
  3591. </template>
  3592. `
  3593. const SSRToggle = compileVaporComponent(
  3594. toggleCode,
  3595. undefined,
  3596. undefined,
  3597. true,
  3598. )
  3599. const wrapperCode = `<slot/>`
  3600. const SSRWrapper = compileVaporComponent(
  3601. wrapperCode,
  3602. undefined,
  3603. undefined,
  3604. true,
  3605. )
  3606. const data = ref({
  3607. count: 0,
  3608. fn: vi.fn(),
  3609. })
  3610. const childCode = `
  3611. <script vapor>
  3612. import { onMounted } from 'vue'
  3613. const data = _data; const components = _components;
  3614. onMounted(() => {
  3615. data.value.fn()
  3616. data.value.count++
  3617. })
  3618. </script>
  3619. <template>
  3620. <div>{{data.count}}</div>
  3621. </template>
  3622. `
  3623. const SSRChild = compileVaporComponent(childCode, data, undefined, true)
  3624. const appCode = `
  3625. <components.Toggle>
  3626. <components.Wrapper>
  3627. <components.Wrapper>
  3628. <components.Child/>
  3629. </components.Wrapper>
  3630. </components.Wrapper>
  3631. </components.Toggle>
  3632. `
  3633. const SSRApp = compileVaporComponent(
  3634. appCode,
  3635. undefined,
  3636. {
  3637. Toggle: SSRToggle,
  3638. Wrapper: SSRWrapper,
  3639. Child: SSRChild,
  3640. },
  3641. true,
  3642. )
  3643. const root = document.createElement('div')
  3644. // server render
  3645. root.innerHTML = await VueServerRenderer.renderToString(
  3646. runtimeDom.createSSRApp(SSRApp),
  3647. )
  3648. expect(root.innerHTML).toMatchInlineSnapshot(
  3649. `"<div style="display:none;"><!--[--><!--[--><!--[--><div>0</div><!--]--><!--]--><!--]--></div>"`,
  3650. )
  3651. const Toggle = compileVaporComponent(toggleCode)
  3652. const Wrapper = compileVaporComponent(wrapperCode)
  3653. const Child = compileVaporComponent(childCode, data)
  3654. const App = compileVaporComponent(appCode, undefined, {
  3655. Toggle,
  3656. Wrapper,
  3657. Child,
  3658. })
  3659. // hydration
  3660. createVaporSSRApp(App).mount(root)
  3661. await nextTick()
  3662. await nextTick()
  3663. expect(root.innerHTML).toMatchInlineSnapshot(
  3664. `"<div style=""><!--[--><!--[--><!--[--><div>1</div><!--]--><!--]--><!--]--></div>"`,
  3665. )
  3666. expect(data.value.fn).toBeCalledTimes(1)
  3667. })
  3668. })
  3669. describe.todo('Suspense')
  3670. describe('force hydrate prop', async () => {
  3671. test('force hydrate prop with `.prop` modifier', async () => {
  3672. const { container } = await mountWithHydration(
  3673. '<input type="checkbox">',
  3674. `<input type="checkbox" .indeterminate="true"/>`,
  3675. )
  3676. expect((container.firstChild! as any).indeterminate).toBe(true)
  3677. })
  3678. test('force hydrate input v-model with non-string value bindings', async () => {
  3679. const { container } = await mountWithHydration(
  3680. '<input type="checkbox" value="true">',
  3681. `<input type="checkbox" :true-value="true"/>`,
  3682. )
  3683. expect((container.firstChild as any)._trueValue).toBe(true)
  3684. })
  3685. test('force hydrate checkbox with indeterminate', async () => {
  3686. const { container } = await mountWithHydration(
  3687. '<input type="checkbox" indeterminate/>',
  3688. `<input type="checkbox" :indeterminate="true"/>`,
  3689. )
  3690. expect((container.firstChild! as any).indeterminate).toBe(true)
  3691. })
  3692. test('force hydrate select option with non-string value bindings', async () => {
  3693. const { container } = await mountWithHydration(
  3694. '<select><option value="true">ok</option></select>',
  3695. `<select><option :value="true">ok</option></select>`,
  3696. )
  3697. expect((container.firstChild!.firstChild as any)._value).toBe(true)
  3698. })
  3699. test('force hydrate v-bind with .prop modifiers', async () => {
  3700. const { container } = await mountWithHydration(
  3701. '<div .foo="true"/>',
  3702. `<div v-bind="data"/>`,
  3703. ref({ '.foo': true }),
  3704. )
  3705. expect((container.firstChild! as any).foo).toBe(true)
  3706. })
  3707. test('force hydrate custom element with dynamic props', () => {
  3708. class MyElement extends HTMLElement {
  3709. foo = ''
  3710. constructor() {
  3711. super()
  3712. }
  3713. }
  3714. customElements.define('my-element-7203', MyElement)
  3715. const msg = ref('bar')
  3716. const container = document.createElement('div')
  3717. container.innerHTML = '<my-element-7203></my-element-7203>'
  3718. const app = createVaporSSRApp({
  3719. setup() {
  3720. return createPlainElement('my-element-7203', { foo: () => msg.value })
  3721. },
  3722. })
  3723. app.mount(container)
  3724. expect((container.firstChild as any).foo).toBe(msg.value)
  3725. })
  3726. })
  3727. })
  3728. describe('mismatch handling', () => {
  3729. test('text node', async () => {
  3730. const foo = ref('bar')
  3731. const { container } = await mountWithHydration(`foo`, `{{data}}`, foo)
  3732. expect(container.textContent).toBe('bar')
  3733. expect(`Hydration text mismatch`).toHaveBeenWarned()
  3734. })
  3735. test('element text content', async () => {
  3736. const data = ref({ textContent: 'bar' })
  3737. const { container } = await mountWithHydration(
  3738. `<div>foo</div>`,
  3739. `<div v-bind="data"></div>`,
  3740. data,
  3741. )
  3742. expect(container.innerHTML).toBe('<div>bar</div>')
  3743. expect(`Hydration text content mismatch`).toHaveBeenWarned()
  3744. })
  3745. // test('not enough children', () => {
  3746. // const { container } = mountWithHydration(`<div></div>`, () =>
  3747. // h('div', [h('span', 'foo'), h('span', 'bar')]),
  3748. // )
  3749. // expect(container.innerHTML).toBe(
  3750. // '<div><span>foo</span><span>bar</span></div>',
  3751. // )
  3752. // expect(`Hydration children mismatch`).toHaveBeenWarned()
  3753. // })
  3754. // test('too many children', () => {
  3755. // const { container } = mountWithHydration(
  3756. // `<div><span>foo</span><span>bar</span></div>`,
  3757. // () => h('div', [h('span', 'foo')]),
  3758. // )
  3759. // expect(container.innerHTML).toBe('<div><span>foo</span></div>')
  3760. // expect(`Hydration children mismatch`).toHaveBeenWarned()
  3761. // })
  3762. test('complete mismatch', async () => {
  3763. const data = ref('span')
  3764. const { container } = await mountWithHydration(
  3765. `<div>foo</div>`,
  3766. `<component :is="data">foo</component>`,
  3767. data,
  3768. )
  3769. expect(container.innerHTML).toBe('<span>foo</span><!--dynamic-component-->')
  3770. expect(`Hydration node mismatch`).toHaveBeenWarned()
  3771. })
  3772. // test('fragment mismatch removal', () => {
  3773. // const { container } = mountWithHydration(
  3774. // `<div><!--[--><div>foo</div><div>bar</div><!--]--></div>`,
  3775. // () => h('div', [h('span', 'replaced')]),
  3776. // )
  3777. // expect(container.innerHTML).toBe('<div><span>replaced</span></div>')
  3778. // expect(`Hydration node mismatch`).toHaveBeenWarned()
  3779. // })
  3780. // test('fragment not enough children', () => {
  3781. // const { container } = mountWithHydration(
  3782. // `<div><!--[--><div>foo</div><!--]--><div>baz</div></div>`,
  3783. // () => h('div', [[h('div', 'foo'), h('div', 'bar')], h('div', 'baz')]),
  3784. // )
  3785. // expect(container.innerHTML).toBe(
  3786. // '<div><!--[--><div>foo</div><div>bar</div><!--]--><div>baz</div></div>',
  3787. // )
  3788. // expect(`Hydration node mismatch`).toHaveBeenWarned()
  3789. // })
  3790. // test('fragment too many children', () => {
  3791. // const { container } = mountWithHydration(
  3792. // `<div><!--[--><div>foo</div><div>bar</div><!--]--><div>baz</div></div>`,
  3793. // () => h('div', [[h('div', 'foo')], h('div', 'baz')]),
  3794. // )
  3795. // expect(container.innerHTML).toBe(
  3796. // '<div><!--[--><div>foo</div><!--]--><div>baz</div></div>',
  3797. // )
  3798. // // fragment ends early and attempts to hydrate the extra <div>bar</div>
  3799. // // as 2nd fragment child.
  3800. // expect(`Hydration text content mismatch`).toHaveBeenWarned()
  3801. // // excessive children removal
  3802. // expect(`Hydration children mismatch`).toHaveBeenWarned()
  3803. // })
  3804. // test('Teleport target has empty children', () => {
  3805. // const teleportContainer = document.createElement('div')
  3806. // teleportContainer.id = 'teleport'
  3807. // document.body.appendChild(teleportContainer)
  3808. // mountWithHydration('<!--teleport start--><!--teleport end-->', () =>
  3809. // h(Teleport, { to: '#teleport' }, [h('span', 'value')]),
  3810. // )
  3811. // expect(teleportContainer.innerHTML).toBe(`<span>value</span>`)
  3812. // expect(`Hydration children mismatch`).toHaveBeenWarned()
  3813. // })
  3814. // test('comment mismatch (element)', () => {
  3815. // const { container } = mountWithHydration(`<div><span></span></div>`, () =>
  3816. // h('div', [createCommentVNode('hi')]),
  3817. // )
  3818. // expect(container.innerHTML).toBe('<div><!--hi--></div>')
  3819. // expect(`Hydration node mismatch`).toHaveBeenWarned()
  3820. // })
  3821. // test('comment mismatch (text)', () => {
  3822. // const { container } = mountWithHydration(`<div>foobar</div>`, () =>
  3823. // h('div', [createCommentVNode('hi')]),
  3824. // )
  3825. // expect(container.innerHTML).toBe('<div><!--hi--></div>')
  3826. // expect(`Hydration node mismatch`).toHaveBeenWarned()
  3827. // })
  3828. test('class mismatch', async () => {
  3829. await mountWithHydration(
  3830. `<div class="foo bar"></div>`,
  3831. `<div :class="data"></div>`,
  3832. ref(['foo', 'bar']),
  3833. )
  3834. await mountWithHydration(
  3835. `<div class="foo bar"></div>`,
  3836. `<div :class="data"></div>`,
  3837. ref({ foo: true, bar: true }),
  3838. )
  3839. await mountWithHydration(
  3840. `<div class="foo bar"></div>`,
  3841. `<div :class="data"></div>`,
  3842. ref('foo bar'),
  3843. )
  3844. // svg classes
  3845. await mountWithHydration(
  3846. `<svg class="foo bar"></svg>`,
  3847. `<svg :class="data"></svg>`,
  3848. ref('foo bar'),
  3849. )
  3850. // class with different order
  3851. await mountWithHydration(
  3852. `<div class="foo bar"></div>`,
  3853. `<div :class="data"></div>`,
  3854. ref('bar foo'),
  3855. )
  3856. expect(`Hydration class mismatch`).not.toHaveBeenWarned()
  3857. // single root mismatch
  3858. const { container: root } = await mountWithHydration(
  3859. `<div class="foo bar"></div>`,
  3860. `<div :class="data"></div>`,
  3861. ref('baz'),
  3862. )
  3863. expect(root.innerHTML).toBe('<div class="foo bar baz"></div>')
  3864. expect(`Hydration class mismatch`).toHaveBeenWarned()
  3865. // multiple root mismatch
  3866. const { container } = await mountWithHydration(
  3867. `<div class="foo bar"></div><span/>`,
  3868. `<div :class="data"></div><span/>`,
  3869. ref('foo'),
  3870. )
  3871. expect(container.innerHTML).toBe('<div class="foo"></div><span></span>')
  3872. expect(`Hydration class mismatch`).toHaveBeenWarned()
  3873. })
  3874. test('style mismatch', async () => {
  3875. await mountWithHydration(
  3876. `<div style="color:red;"></div>`,
  3877. `<div :style="data"></div>`,
  3878. ref({ color: 'red' }),
  3879. )
  3880. await mountWithHydration(
  3881. `<div style="color:red;"></div>`,
  3882. `<div :style="data"></div>`,
  3883. ref('color:red;'),
  3884. )
  3885. // style with different order
  3886. await mountWithHydration(
  3887. `<div style="color:red; font-size: 12px;"></div>`,
  3888. `<div :style="data"></div>`,
  3889. ref(`font-size: 12px; color:red;`),
  3890. )
  3891. expect(`Hydration style mismatch`).not.toHaveBeenWarned()
  3892. // single root mismatch
  3893. const { container: root } = await mountWithHydration(
  3894. `<div style="color:red;"></div>`,
  3895. `<div :style="data"></div>`,
  3896. ref({ color: 'green' }),
  3897. )
  3898. expect(root.innerHTML).toBe('<div style="color: green;"></div>')
  3899. expect(`Hydration style mismatch`).toHaveBeenWarned()
  3900. // multiple root mismatch
  3901. const { container } = await mountWithHydration(
  3902. `<div style="color:red;"></div><span/>`,
  3903. `<div :style="data"></div><span/>`,
  3904. ref({ color: 'green' }),
  3905. )
  3906. expect(container.innerHTML).toBe(
  3907. '<div style="color: green;"></div><span></span>',
  3908. )
  3909. expect(`Hydration style mismatch`).toHaveBeenWarned()
  3910. })
  3911. test('style mismatch when no style attribute is present', async () => {
  3912. await mountWithHydration(
  3913. `<div></div>`,
  3914. `<div :style="data"></div>`,
  3915. ref({ color: 'red' }),
  3916. )
  3917. expect(`Hydration style mismatch`).toHaveBeenWarnedTimes(1)
  3918. })
  3919. test('style mismatch w/ v-show', async () => {
  3920. await mountWithHydration(
  3921. `<div style="color:red;display:none"></div>`,
  3922. `<div v-show="data" style="color: red;"></div>`,
  3923. ref(false),
  3924. )
  3925. expect(`Hydration style mismatch`).not.toHaveBeenWarned()
  3926. // mismatch with single root
  3927. const { container: root } = await mountWithHydration(
  3928. `<div style="color:red;"></div>`,
  3929. `<div v-show="data" style="color: red;"></div>`,
  3930. ref(false),
  3931. )
  3932. expect(root.innerHTML).toBe(
  3933. '<div style="color: red; display: none;"></div>',
  3934. )
  3935. expect(`Hydration style mismatch`).toHaveBeenWarned()
  3936. // mismatch with multiple root
  3937. const { container } = await mountWithHydration(
  3938. `<div style="color:red;"></div><span/>`,
  3939. `<div v-show="data.show" :style="data.style"></div><span/>`,
  3940. ref({ show: false, style: 'color: red' }),
  3941. )
  3942. expect(container.innerHTML).toBe(
  3943. '<div style="color: red; display: none;"></div><span></span>',
  3944. )
  3945. expect(`Hydration style mismatch`).toHaveBeenWarned()
  3946. })
  3947. test('attr mismatch', async () => {
  3948. await mountWithHydration(
  3949. `<div id="foo"></div>`,
  3950. `<div :id="data"></div>`,
  3951. ref('foo'),
  3952. )
  3953. await mountWithHydration(
  3954. `<div spellcheck></div>`,
  3955. `<div :spellcheck="data"></div>`,
  3956. ref(''),
  3957. )
  3958. await mountWithHydration(
  3959. `<div></div>`,
  3960. `<div :id="data"></div>`,
  3961. ref(undefined),
  3962. )
  3963. // boolean
  3964. await mountWithHydration(
  3965. `<select multiple></div>`,
  3966. `<select :multiple="data"></select>`,
  3967. ref(true),
  3968. )
  3969. await mountWithHydration(
  3970. `<select multiple></div>`,
  3971. `<select :multiple="data"></select>`,
  3972. ref('multiple'),
  3973. )
  3974. expect(`Hydration attribute mismatch`).not.toHaveBeenWarned()
  3975. await mountWithHydration(
  3976. `<div></div>`,
  3977. `<div :id="data"></div>`,
  3978. ref('foo'),
  3979. )
  3980. expect(`Hydration attribute mismatch`).toHaveBeenWarnedTimes(1)
  3981. await mountWithHydration(
  3982. `<div id="bar"></div>`,
  3983. `<div :id="data"></div>`,
  3984. ref('foo'),
  3985. )
  3986. expect(`Hydration attribute mismatch`).toHaveBeenWarnedTimes(2)
  3987. })
  3988. test('attr special case: textarea value', async () => {
  3989. await mountWithHydration(
  3990. `<textarea>foo</textarea>`,
  3991. `<textarea :value="data"></textarea>`,
  3992. ref('foo'),
  3993. )
  3994. await mountWithHydration(
  3995. `<textarea></textarea>`,
  3996. `<textarea :value="data"></textarea>`,
  3997. ref(''),
  3998. )
  3999. expect(`Hydration attribute mismatch`).not.toHaveBeenWarned()
  4000. await mountWithHydration(
  4001. `<textarea>foo</textarea>`,
  4002. `<textarea :value="data"></textarea>`,
  4003. ref('bar'),
  4004. )
  4005. expect(`Hydration attribute mismatch`).toHaveBeenWarned()
  4006. })
  4007. test('<textarea> with newlines at the beginning', async () => {
  4008. await mountWithHydration(
  4009. `<textarea>\nhello</textarea>`,
  4010. `<textarea :value="data"></textarea>`,
  4011. ref('\nhello'),
  4012. )
  4013. await mountWithHydration(
  4014. `<textarea>\nhello</textarea>`,
  4015. `<textarea v-text="data"></textarea>`,
  4016. ref('\nhello'),
  4017. )
  4018. await mountWithHydration(
  4019. `<textarea>\nhello</textarea>`,
  4020. `<textarea v-bind="data"></textarea>`,
  4021. ref({ textContent: '\nhello' }),
  4022. )
  4023. expect(`Hydration text content mismatch`).not.toHaveBeenWarned()
  4024. })
  4025. test('<pre> with newlines at the beginning', async () => {
  4026. await mountWithHydration(`<pre>\n</pre>`, `<pre>{{data}}</pre>`, ref('\n'))
  4027. await mountWithHydration(
  4028. `<pre>\n</pre>`,
  4029. `<pre v-text="data"></pre>`,
  4030. ref('\n'),
  4031. )
  4032. await mountWithHydration(
  4033. `<pre>\n</pre>`,
  4034. `<pre v-bind="data"></pre>`,
  4035. ref({ textContent: '\n' }),
  4036. )
  4037. expect(`Hydration text content mismatch`).not.toHaveBeenWarned()
  4038. })
  4039. test('boolean attr handling', async () => {
  4040. await mountWithHydration(
  4041. `<input />`,
  4042. `<input :readonly="data" />`,
  4043. ref(false),
  4044. )
  4045. await mountWithHydration(
  4046. `<input readonly />`,
  4047. `<input :readonly="data" />`,
  4048. ref(true),
  4049. )
  4050. await mountWithHydration(
  4051. `<input readonly="readonly" />`,
  4052. `<input :readonly="data" />`,
  4053. ref(true),
  4054. )
  4055. expect(`Hydration attribute mismatch`).not.toHaveBeenWarned()
  4056. })
  4057. test('client value is null or undefined', async () => {
  4058. await mountWithHydration(
  4059. `<div></div>`,
  4060. `<div :draggable="data"></div>`,
  4061. ref(undefined),
  4062. )
  4063. expect(`Hydration attribute mismatch`).not.toHaveBeenWarned()
  4064. await mountWithHydration(`<input />`, `<input :type="data" />`, ref(null))
  4065. expect(`Hydration attribute mismatch`).not.toHaveBeenWarned()
  4066. })
  4067. test('should not warn against object values', async () => {
  4068. await mountWithHydration(`<input />`, `<input :from="data" />`, ref({}))
  4069. expect(`Hydration attribute mismatch`).not.toHaveBeenWarned()
  4070. })
  4071. test('should not warn on falsy bindings of non-property keys', async () => {
  4072. await mountWithHydration(
  4073. `<button></button>`,
  4074. `<button :href="data"></button>`,
  4075. ref(undefined),
  4076. )
  4077. expect(`Hydration attribute mismatch`).not.toHaveBeenWarned()
  4078. })
  4079. test('should not warn on non-renderable option values', async () => {
  4080. await mountWithHydration(
  4081. `<select><option>hello</option></select>`,
  4082. `<select><option :value="data">hello</option></select>`,
  4083. ref(['foo']),
  4084. )
  4085. expect(`Hydration attribute mismatch`).not.toHaveBeenWarned()
  4086. })
  4087. test('should not warn css v-bind', async () => {
  4088. const container = document.createElement('div')
  4089. container.innerHTML = `<div style="--foo:red;color:var(--foo);" />`
  4090. const app = createVaporSSRApp({
  4091. setup() {
  4092. useVaporCssVars(() => ({ foo: 'red' }))
  4093. const n0 = template('<div></div>', true)() as any
  4094. renderEffect(() => setStyle(n0, { color: 'var(--foo)' }))
  4095. return n0
  4096. },
  4097. })
  4098. app.mount(container)
  4099. expect(`Hydration style mismatch`).not.toHaveBeenWarned()
  4100. })
  4101. test('css vars should only be added to expected on component root dom', () => {
  4102. const container = document.createElement('div')
  4103. container.innerHTML = `<div style="--foo:red;"><div style="color:var(--foo);" /></div>`
  4104. const app = createVaporSSRApp({
  4105. setup() {
  4106. useVaporCssVars(() => ({ foo: 'red' }))
  4107. const n0 = template('<div><div></div></div>', true)() as any
  4108. const n1 = child(n0) as any
  4109. renderEffect(() => setStyle(n1, { color: 'var(--foo)' }))
  4110. return n0
  4111. },
  4112. })
  4113. app.mount(container)
  4114. expect(`Hydration style mismatch`).not.toHaveBeenWarned()
  4115. })
  4116. test('css vars support fallthrough', () => {
  4117. const container = document.createElement('div')
  4118. container.innerHTML = `<div style="padding: 4px;--foo:red;"></div>`
  4119. const app = createVaporSSRApp({
  4120. setup() {
  4121. useVaporCssVars(() => ({ foo: 'red' }))
  4122. return createComponent(Child)
  4123. },
  4124. })
  4125. const Child = defineVaporComponent({
  4126. setup() {
  4127. const n0 = template('<div></div>', true)() as any
  4128. renderEffect(() => setStyle(n0, { padding: '4px' }))
  4129. return n0
  4130. },
  4131. })
  4132. app.mount(container)
  4133. expect(`Hydration style mismatch`).not.toHaveBeenWarned()
  4134. })
  4135. // vapor directive does not have a created hook
  4136. test('should not warn for directives that mutate DOM in created', () => {
  4137. // const container = document.createElement('div')
  4138. // container.innerHTML = `<div class="test red"></div>`
  4139. // const vColor: ObjectDirective = {
  4140. // created(el, binding) {
  4141. // el.classList.add(binding.value)
  4142. // },
  4143. // }
  4144. // const app = createSSRApp({
  4145. // setup() {
  4146. // return () =>
  4147. // withDirectives(h('div', { class: 'test' }), [[vColor, 'red']])
  4148. // },
  4149. // })
  4150. // app.mount(container)
  4151. // expect(`Hydration style mismatch`).not.toHaveBeenWarned()
  4152. })
  4153. test('escape css var name', () => {
  4154. const container = document.createElement('div')
  4155. container.innerHTML = `<div style="padding: 4px;--foo\\.bar:red;"></div>`
  4156. const app = createVaporSSRApp({
  4157. setup() {
  4158. useVaporCssVars(() => ({ 'foo.bar': 'red' }))
  4159. return createComponent(Child)
  4160. },
  4161. })
  4162. const Child = defineVaporComponent({
  4163. setup() {
  4164. const n0 = template('<div></div>', true)() as any
  4165. renderEffect(() => setStyle(n0, { padding: '4px' }))
  4166. return n0
  4167. },
  4168. })
  4169. app.mount(container)
  4170. expect(`Hydration style mismatch`).not.toHaveBeenWarned()
  4171. })
  4172. })
  4173. describe('data-allow-mismatch', () => {
  4174. test('element text content', async () => {
  4175. const data = ref({ textContent: 'bar' })
  4176. const { container } = await mountWithHydration(
  4177. `<div data-allow-mismatch="text">foo</div>`,
  4178. `<div v-bind="data"></div>`,
  4179. data,
  4180. )
  4181. expect(container.innerHTML).toBe(
  4182. '<div data-allow-mismatch="text">bar</div>',
  4183. )
  4184. expect(`Hydration text content mismatch`).not.toHaveBeenWarned()
  4185. })
  4186. // test('not enough children', () => {
  4187. // const { container } = mountWithHydration(
  4188. // `<div data-allow-mismatch="children"></div>`,
  4189. // () => h('div', [h('span', 'foo'), h('span', 'bar')]),
  4190. // )
  4191. // expect(container.innerHTML).toBe(
  4192. // '<div data-allow-mismatch="children"><span>foo</span><span>bar</span></div>',
  4193. // )
  4194. // expect(`Hydration children mismatch`).not.toHaveBeenWarned()
  4195. // })
  4196. // test('too many children', () => {
  4197. // const { container } = mountWithHydration(
  4198. // `<div data-allow-mismatch="children"><span>foo</span><span>bar</span></div>`,
  4199. // () => h('div', [h('span', 'foo')]),
  4200. // )
  4201. // expect(container.innerHTML).toBe(
  4202. // '<div data-allow-mismatch="children"><span>foo</span></div>',
  4203. // )
  4204. // expect(`Hydration children mismatch`).not.toHaveBeenWarned()
  4205. // })
  4206. test('complete mismatch', async () => {
  4207. const { container } = await mountWithHydration(
  4208. `<div data-allow-mismatch="children"><div>foo</div></div>`,
  4209. `<div><component :is="data">foo</component></div>`,
  4210. ref('span'),
  4211. )
  4212. expect(container.innerHTML).toBe(
  4213. '<div data-allow-mismatch="children"><span>foo</span><!--dynamic-component--></div>',
  4214. )
  4215. expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  4216. })
  4217. // test('fragment mismatch removal', () => {
  4218. // const { container } = mountWithHydration(
  4219. // `<div data-allow-mismatch="children"><!--[--><div>foo</div><div>bar</div><!--]--></div>`,
  4220. // () => h('div', [h('span', 'replaced')]),
  4221. // )
  4222. // expect(container.innerHTML).toBe(
  4223. // '<div data-allow-mismatch="children"><span>replaced</span></div>',
  4224. // )
  4225. // expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  4226. // })
  4227. // test('fragment not enough children', () => {
  4228. // const { container } = mountWithHydration(
  4229. // `<div data-allow-mismatch="children"><!--[--><div>foo</div><!--]--><div>baz</div></div>`,
  4230. // () => h('div', [[h('div', 'foo'), h('div', 'bar')], h('div', 'baz')]),
  4231. // )
  4232. // expect(container.innerHTML).toBe(
  4233. // '<div data-allow-mismatch="children"><!--[--><div>foo</div><div>bar</div><!--]--><div>baz</div></div>',
  4234. // )
  4235. // expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  4236. // })
  4237. // test('fragment too many children', () => {
  4238. // const { container } = mountWithHydration(
  4239. // `<div data-allow-mismatch="children"><!--[--><div>foo</div><div>bar</div><!--]--><div>baz</div></div>`,
  4240. // () => h('div', [[h('div', 'foo')], h('div', 'baz')]),
  4241. // )
  4242. // expect(container.innerHTML).toBe(
  4243. // '<div data-allow-mismatch="children"><!--[--><div>foo</div><!--]--><div>baz</div></div>',
  4244. // )
  4245. // // fragment ends early and attempts to hydrate the extra <div>bar</div>
  4246. // // as 2nd fragment child.
  4247. // expect(`Hydration text content mismatch`).not.toHaveBeenWarned()
  4248. // // excessive children removal
  4249. // expect(`Hydration children mismatch`).not.toHaveBeenWarned()
  4250. // })
  4251. // test('comment mismatch (element)', () => {
  4252. // const { container } = mountWithHydration(
  4253. // `<div data-allow-mismatch="children"><span></span></div>`,
  4254. // () => h('div', [createCommentVNode('hi')]),
  4255. // )
  4256. // expect(container.innerHTML).toBe(
  4257. // '<div data-allow-mismatch="children"><!--hi--></div>',
  4258. // )
  4259. // expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  4260. // })
  4261. // test('comment mismatch (text)', () => {
  4262. // const { container } = mountWithHydration(
  4263. // `<div data-allow-mismatch="children">foobar</div>`,
  4264. // () => h('div', [createCommentVNode('hi')]),
  4265. // )
  4266. // expect(container.innerHTML).toBe(
  4267. // '<div data-allow-mismatch="children"><!--hi--></div>',
  4268. // )
  4269. // expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  4270. // })
  4271. test('class mismatch', async () => {
  4272. await mountWithHydration(
  4273. `<div class="foo bar" data-allow-mismatch="class"></div>`,
  4274. `<div :class="data"></div>`,
  4275. ref('foo'),
  4276. )
  4277. expect(`Hydration class mismatch`).not.toHaveBeenWarned()
  4278. })
  4279. test('style mismatch', async () => {
  4280. await mountWithHydration(
  4281. `<div style="color:red;" data-allow-mismatch="style"></div>`,
  4282. `<div :style="data"></div>`,
  4283. ref({ color: 'green' }),
  4284. )
  4285. expect(`Hydration style mismatch`).not.toHaveBeenWarned()
  4286. })
  4287. test('attr mismatch', async () => {
  4288. await mountWithHydration(
  4289. `<div data-allow-mismatch="attribute"></div>`,
  4290. `<div :id="data"></div>`,
  4291. ref('foo'),
  4292. )
  4293. await mountWithHydration(
  4294. `<div id="bar" data-allow-mismatch="attribute"></div>`,
  4295. `<div :id="data"></div>`,
  4296. ref('foo'),
  4297. )
  4298. expect(`Hydration attribute mismatch`).not.toHaveBeenWarned()
  4299. })
  4300. })
  4301. describe('VDOM interop', () => {
  4302. test('basic render vapor component', async () => {
  4303. const data = ref(true)
  4304. const { container } = await testWithVDOMApp(
  4305. `<script setup>const data = _data; const components = _components;</script>
  4306. <template>
  4307. <components.VaporChild/>
  4308. </template>`,
  4309. {
  4310. VaporChild: {
  4311. code: `<template>{{ data }}</template>`,
  4312. vapor: true,
  4313. },
  4314. },
  4315. data,
  4316. )
  4317. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`"true"`)
  4318. data.value = false
  4319. await nextTick()
  4320. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`"false"`)
  4321. })
  4322. test('nested components (VDOM -> Vapor -> VDOM)', async () => {
  4323. const data = ref(true)
  4324. const { container } = await testWithVDOMApp(
  4325. `<script setup>const data = _data; const components = _components;</script>
  4326. <template>
  4327. <components.VaporChild/>
  4328. </template>`,
  4329. {
  4330. VaporChild: {
  4331. code: `<template><components.VdomChild/></template>`,
  4332. vapor: true,
  4333. },
  4334. VdomChild: {
  4335. code: `<script setup>const data = _data;</script>
  4336. <template>{{ data }}</template>`,
  4337. vapor: false,
  4338. },
  4339. },
  4340. data,
  4341. )
  4342. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`"true"`)
  4343. data.value = false
  4344. await nextTick()
  4345. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(`"false"`)
  4346. })
  4347. test('nested components (VDOM -> Vapor(multi-root) -> VDOM)', async () => {
  4348. const data = ref('foo')
  4349. const { container } = await testWithVDOMApp(
  4350. `<script setup>const data = _data; const components = _components;</script>
  4351. <template>
  4352. <components.VaporChild/>
  4353. </template>`,
  4354. {
  4355. // Vapor component with multiple root nodes, VDOM child as first element
  4356. // This ensures hydration starts at <!--[--> and tests skipFragmentAnchor
  4357. VaporChild: {
  4358. code: `<template><components.VdomChild/><div>second</div></template>`,
  4359. vapor: true,
  4360. },
  4361. VdomChild: {
  4362. code: `<script setup>const data = _data;</script>
  4363. <template><span>{{ data }}</span></template>`,
  4364. vapor: false,
  4365. },
  4366. },
  4367. data,
  4368. )
  4369. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4370. `
  4371. "
  4372. <!--[--><span>foo</span><div>second</div><!--]-->
  4373. "
  4374. `,
  4375. )
  4376. expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  4377. data.value = 'bar'
  4378. await nextTick()
  4379. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4380. `
  4381. "
  4382. <!--[--><span>bar</span><div>second</div><!--]-->
  4383. "
  4384. `,
  4385. )
  4386. })
  4387. test('nested components (VDOM -> Vapor -> VDOM (with slot fallback))', async () => {
  4388. const data = ref(true)
  4389. const { container } = await testWithVDOMApp(
  4390. `<script setup>const data = _data; const components = _components;</script>
  4391. <template>
  4392. <components.VaporChild/>
  4393. </template>`,
  4394. {
  4395. VaporChild: {
  4396. code: `<template><components.VdomChild/></template>`,
  4397. vapor: true,
  4398. },
  4399. VdomChild: {
  4400. code: `<script setup>const data = _data;</script>
  4401. <template><slot><span>{{data}}</span></slot></template>`,
  4402. vapor: false,
  4403. },
  4404. },
  4405. data,
  4406. )
  4407. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4408. `
  4409. "
  4410. <!--[--><span>true</span><!--]-->
  4411. "
  4412. `,
  4413. )
  4414. data.value = false
  4415. await nextTick()
  4416. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4417. `
  4418. "
  4419. <!--[--><span>false</span><!--]-->
  4420. "
  4421. `,
  4422. )
  4423. })
  4424. test('nested components (VDOM -> Vapor(with slot content) -> VDOM)', async () => {
  4425. const data = ref(true)
  4426. const { container } = await testWithVDOMApp(
  4427. `<script setup>const data = _data; const components = _components;</script>
  4428. <template>
  4429. <components.VaporChild/>
  4430. </template>`,
  4431. {
  4432. VaporChild: {
  4433. code: `<template>
  4434. <components.VdomChild>
  4435. <template #default>
  4436. <span>{{data}} vapor fallback</span>
  4437. </template>
  4438. </components.VdomChild>
  4439. </template>`,
  4440. vapor: true,
  4441. },
  4442. VdomChild: {
  4443. code: `<script setup>const data = _data;</script>
  4444. <template><slot><span>vdom fallback</span></slot></template>`,
  4445. vapor: false,
  4446. },
  4447. },
  4448. data,
  4449. )
  4450. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4451. `
  4452. "
  4453. <!--[--><span>true vapor fallback</span><!--]-->
  4454. "
  4455. `,
  4456. )
  4457. data.value = false
  4458. await nextTick()
  4459. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4460. `
  4461. "
  4462. <!--[--><span>false vapor fallback</span><!--]-->
  4463. "
  4464. `,
  4465. )
  4466. })
  4467. test('nested components (VDOM -> Vapor(with slot content) -> Vapor)', async () => {
  4468. const data = ref(true)
  4469. const { container } = await testWithVDOMApp(
  4470. `<script setup>const data = _data; const components = _components;</script>
  4471. <template>
  4472. <components.VaporChild/>
  4473. </template>`,
  4474. {
  4475. VaporChild: {
  4476. code: `<template>
  4477. <components.VaporChild2>
  4478. <template #default>
  4479. <span>{{data}} vapor fallback</span>
  4480. </template>
  4481. </components.VaporChild2>
  4482. </template>`,
  4483. vapor: true,
  4484. },
  4485. VaporChild2: {
  4486. code: `<template><slot><span>vapor fallback2</span></slot></template>`,
  4487. vapor: true,
  4488. },
  4489. },
  4490. data,
  4491. )
  4492. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4493. `
  4494. "
  4495. <!--[--><span>true vapor fallback</span><!--]-->
  4496. "
  4497. `,
  4498. )
  4499. data.value = false
  4500. await nextTick()
  4501. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4502. `
  4503. "
  4504. <!--[--><span>false vapor fallback</span><!--]-->
  4505. "
  4506. `,
  4507. )
  4508. })
  4509. test('vapor slot render vdom component', async () => {
  4510. const data = ref(true)
  4511. const { container } = await testWithVDOMApp(
  4512. `<script setup>const data = _data; const components = _components;</script>
  4513. <template>
  4514. <components.VaporChild>
  4515. <components.VdomChild/>
  4516. </components.VaporChild>
  4517. </template>`,
  4518. {
  4519. VaporChild: {
  4520. code: `<template><div><slot/></div></template>`,
  4521. vapor: true,
  4522. },
  4523. VdomChild: {
  4524. code: `<script setup>const data = _data;</script>
  4525. <template>{{ data }}</template>`,
  4526. vapor: false,
  4527. },
  4528. },
  4529. data,
  4530. )
  4531. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4532. `
  4533. "<div>
  4534. <!--[-->true<!--]-->
  4535. </div>"
  4536. `,
  4537. )
  4538. data.value = false
  4539. await nextTick()
  4540. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4541. `
  4542. "<div>
  4543. <!--[-->false<!--]-->
  4544. </div>"
  4545. `,
  4546. )
  4547. })
  4548. test('vapor slot render vdom component (multi-root slot content)', async () => {
  4549. const data = ref('foo')
  4550. const { container } = await testWithVaporApp(
  4551. `<script setup>const data = _data; const components = _components;</script>
  4552. <template>
  4553. <components.VaporChild>
  4554. <components.VdomChild/>
  4555. <div>vapor content</div>
  4556. </components.VaporChild>
  4557. </template>`,
  4558. {
  4559. VaporChild: {
  4560. code: `<template><div><slot/></div></template>`,
  4561. vapor: true,
  4562. },
  4563. VdomChild: {
  4564. code: `<script setup>const data = _data;</script>
  4565. <template><span>{{ data }}</span></template>`,
  4566. vapor: false,
  4567. },
  4568. },
  4569. data,
  4570. )
  4571. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4572. `
  4573. "<div>
  4574. <!--[--><span>foo</span><div>vapor content</div><!--]-->
  4575. </div>"
  4576. `,
  4577. )
  4578. expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  4579. data.value = 'bar'
  4580. await nextTick()
  4581. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4582. `
  4583. "<div>
  4584. <!--[--><span>bar</span><div>vapor content</div><!--]-->
  4585. </div>"
  4586. `,
  4587. )
  4588. })
  4589. test('vapor slot render vdom component (render function)', async () => {
  4590. const data = ref(true)
  4591. const { container } = await testWithVaporApp(
  4592. `<script setup>
  4593. import { h } from 'vue'
  4594. const data = _data; const components = _components;
  4595. const VdomChild = {
  4596. setup() {
  4597. return () => h('div', null, [h('div', [String(data.value)])])
  4598. }
  4599. }
  4600. </script>
  4601. <template>
  4602. <components.VaporChild>
  4603. <VdomChild/>
  4604. </components.VaporChild>
  4605. </template>`,
  4606. {
  4607. VaporChild: {
  4608. code: `<template><div><slot/></div></template>`,
  4609. vapor: true,
  4610. },
  4611. },
  4612. data,
  4613. )
  4614. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4615. `
  4616. "<div>
  4617. <!--[--><div><div>true</div></div><!--]-->
  4618. </div>"
  4619. `,
  4620. )
  4621. expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  4622. data.value = false
  4623. await nextTick()
  4624. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4625. `
  4626. "<div>
  4627. <!--[--><div><div>false</div></div><!--]-->
  4628. </div>"
  4629. `,
  4630. )
  4631. })
  4632. test('hydrate VNode rendered via createDynamicComponent', async () => {
  4633. const data = ref('foo')
  4634. const { container } = await testWithVaporApp(
  4635. `<script setup>
  4636. import { h } from 'vue'
  4637. const data = _data; const components = _components;
  4638. // Simulating RouterView pattern: VDOM component passes VNode through slot
  4639. const RouterView = {
  4640. setup(_, { slots }) {
  4641. return () => {
  4642. const component = h(components.VaporChild)
  4643. return slots.default({ Component: component })
  4644. }
  4645. }
  4646. }
  4647. </script>
  4648. <template>
  4649. <RouterView v-slot="{ Component }">
  4650. <component :is="Component" />
  4651. </RouterView>
  4652. </template>`,
  4653. {
  4654. VaporChild: {
  4655. code: `<template><div>{{ data }}</div></template>`,
  4656. vapor: true,
  4657. },
  4658. },
  4659. data,
  4660. )
  4661. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4662. `
  4663. "
  4664. <!--[--><div>foo</div><!--dynamic-component--><!--]-->
  4665. "
  4666. `,
  4667. )
  4668. expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  4669. data.value = 'bar'
  4670. await nextTick()
  4671. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4672. `
  4673. "
  4674. <!--[--><div>bar</div><!--dynamic-component--><!--]-->
  4675. "
  4676. `,
  4677. )
  4678. })
  4679. test('hydrate VDOM slot content', async () => {
  4680. const data = ref('foo')
  4681. const { container } = await testWithVaporApp(
  4682. `<script setup>
  4683. const data = _data; const components = _components;
  4684. </script>
  4685. <template>
  4686. <components.VdomWrapper>
  4687. <div>{{ data }}</div>
  4688. </components.VdomWrapper>
  4689. </template>`,
  4690. {
  4691. VdomWrapper: {
  4692. code: `<script setup>const data = _data;</script>
  4693. <template><slot /></template>`,
  4694. vapor: false,
  4695. },
  4696. },
  4697. data,
  4698. )
  4699. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4700. `
  4701. "
  4702. <!--[--><div>foo</div><!--]-->
  4703. "
  4704. `,
  4705. )
  4706. expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  4707. data.value = 'bar'
  4708. await nextTick()
  4709. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4710. `
  4711. "
  4712. <!--[--><div>bar</div><!--]-->
  4713. "
  4714. `,
  4715. )
  4716. })
  4717. test('hydrate VDOM slot fallback', async () => {
  4718. const data = ref('foo')
  4719. const { container } = await testWithVaporApp(
  4720. `<script setup>
  4721. const data = _data; const components = _components;
  4722. </script>
  4723. <template>
  4724. <components.VdomWrapper />
  4725. </template>`,
  4726. {
  4727. VdomWrapper: {
  4728. code: `<script setup>const data = _data;</script>
  4729. <template><slot><div>{{ data }}</div></slot></template>`,
  4730. vapor: false,
  4731. },
  4732. },
  4733. data,
  4734. )
  4735. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4736. `
  4737. "
  4738. <!--[--><div>foo</div><!--]-->
  4739. "
  4740. `,
  4741. )
  4742. expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  4743. data.value = 'bar'
  4744. await nextTick()
  4745. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4746. `
  4747. "
  4748. <!--[--><div>bar</div><!--]-->
  4749. "
  4750. `,
  4751. )
  4752. })
  4753. test('hydrate VDOM component returning Fragment', async () => {
  4754. const data = ref('foo')
  4755. const { container } = await testWithVaporApp(
  4756. `<script setup>
  4757. const data = _data; const components = _components;
  4758. </script>
  4759. <template>
  4760. <components.VdomFragmentComp />
  4761. </template>`,
  4762. {
  4763. // VDOM component that returns a Fragment (multiple root nodes)
  4764. VdomFragmentComp: {
  4765. code: `<script setup>const data = _data;</script>
  4766. <template><div>first {{ data }}</div><div>second {{ data }}</div></template>`,
  4767. vapor: false,
  4768. },
  4769. },
  4770. data,
  4771. )
  4772. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4773. `
  4774. "
  4775. <!--[--><div>first foo</div><div>second foo</div><!--]-->
  4776. "
  4777. `,
  4778. )
  4779. expect(`Hydration node mismatch`).not.toHaveBeenWarned()
  4780. data.value = 'bar'
  4781. await nextTick()
  4782. expect(formatHtml(container.innerHTML)).toMatchInlineSnapshot(
  4783. `
  4784. "
  4785. <!--[--><div>first bar</div><div>second bar</div><!--]-->
  4786. "
  4787. `,
  4788. )
  4789. })
  4790. })