From 129324bf7585a05899f957ef4189bd70bf20d34b Mon Sep 17 00:00:00 2001 From: Sohdai Yokokawa Date: Sat, 31 Jan 2026 20:03:41 -0800 Subject: [PATCH] Added hyperlink for HackMerced X --- src/X/404.html | 1 + src/X/asset-manifest.json | 46 ++++++++ src/X/discord.html | 18 +++ src/X/favicon.ico | Bin 0 -> 95504 bytes src/X/index.html | 1 + src/X/logo192.png | Bin 0 -> 5347 bytes src/X/logo512.png | Bin 0 -> 9664 bytes src/X/manifest.json | 25 ++++ src/X/robots.txt | 3 + src/X/static/css/main.143e02ec.css | 2 + src/X/static/css/main.143e02ec.css.map | 1 + src/X/static/js/488.4d6d0dc5.chunk.js | 2 + src/X/static/js/488.4d6d0dc5.chunk.js.map | 1 + src/X/static/js/main.273f6e3b.js | 3 + src/X/static/js/main.273f6e3b.js.LICENSE.txt | 99 ++++++++++++++++ src/X/static/js/main.273f6e3b.js.map | 1 + .../media/Aaron.d9742a0139f85aae0882.webp | Bin 0 -> 25312 bytes .../media/Alex.b27a90623c5a49c4997e.webp | Bin 0 -> 40742 bytes .../media/Alisson.b97c246bbf0f90c582bf.webp | Bin 0 -> 44886 bytes .../media/Andre.fa5649344a2a22ab64f0.webp | Bin 0 -> 28444 bytes .../media/Brandon.ebe25f19deed183bcd5e.webp | Bin 0 -> 41252 bytes .../media/Drishya.d320d733ac3130f4fffb.webp | Bin 0 -> 24586 bytes .../media/Elaine.2621346587810ae7fb18.webp | Bin 0 -> 39552 bytes .../HackMercedTower.4fdbb51d75e6819b17fd.png | Bin 0 -> 228952 bytes .../media/Isha.2ef5dc7c3022261b6390.webp | Bin 0 -> 31072 bytes .../media/Ishan.44b15b9d28ced5d5fd2f.webp | Bin 0 -> 23814 bytes .../media/Jeevan.91f5b36c62fba78fd0ff.webp | Bin 0 -> 28696 bytes .../media/Jet.4bd5a66f55130571edee.webp | Bin 0 -> 31288 bytes .../media/Marian.03625411afe42f1aef2c.webp | Bin 0 -> 39536 bytes .../media/Megan.b1be85544a7f0603b17b.webp | Bin 0 -> 357640 bytes ...ainPageBackground.0f23da999859c41b6f1a.svg | 107 ++++++++++++++++++ .../SponsorsCollage.dcb671ad749e970cf0fd.png | Bin 0 -> 1517071 bytes .../media/Team2.a9ce980bfd4d78cc3b47.webp | Bin 0 -> 115026 bytes ...tor 1.dfe7bd48083770efdefca5ab71127068.svg | 3 + .../ag.63b60df20946c9323fc476570af66de5.svg | 17 +++ .../media/akhil.2da83b52a10f3f4d234d.JPG | Bin 0 -> 1848993 bytes ...right.797afb4bb4bfd0b5293bba1cdb904dd4.svg | 1 + ...kground-buildings.5a49afed817462164433.png | Bin 0 -> 967893 bytes .../bus.ba0656c48b52645d022bd7d137d5fecf.svg | 3 + ...ord 1.e290b67d8b19e02a28652e2b2c90c81f.svg | 10 ++ ...ook 1.a49b4357e5c204d5cdf81d0d0ed7974e.svg | 3 + .../hackmerced-tower.06913f705c04ff3fab04.png | Bin 0 -> 224324 bytes ...ealth.52082d88db33c341b551985fdac44892.svg | 17 +++ ...ram 1.611fcf3b3dac69c4b8163932828624b5.svg | 11 ++ .../media/lauren.5e02914aa3521f325759.jpg | Bin 0 -> 1241297 bytes .../media/shyam.ab4a0d3e5458068b8f2e.JPG | Bin 0 -> 1460714 bytes ...cmmap.f648cfe35bbe2d337db1b512443ac0a4.svg | 9 ++ src/index.njk | 4 +- 48 files changed, 386 insertions(+), 2 deletions(-) create mode 100644 src/X/404.html create mode 100644 src/X/asset-manifest.json create mode 100644 src/X/discord.html create mode 100644 src/X/favicon.ico create mode 100644 src/X/index.html create mode 100644 src/X/logo192.png create mode 100644 src/X/logo512.png create mode 100644 src/X/manifest.json create mode 100644 src/X/robots.txt create mode 100644 src/X/static/css/main.143e02ec.css create mode 100644 src/X/static/css/main.143e02ec.css.map create mode 100644 src/X/static/js/488.4d6d0dc5.chunk.js create mode 100644 src/X/static/js/488.4d6d0dc5.chunk.js.map create mode 100644 src/X/static/js/main.273f6e3b.js create mode 100644 src/X/static/js/main.273f6e3b.js.LICENSE.txt create mode 100644 src/X/static/js/main.273f6e3b.js.map create mode 100644 src/X/static/media/Aaron.d9742a0139f85aae0882.webp create mode 100644 src/X/static/media/Alex.b27a90623c5a49c4997e.webp create mode 100644 src/X/static/media/Alisson.b97c246bbf0f90c582bf.webp create mode 100644 src/X/static/media/Andre.fa5649344a2a22ab64f0.webp create mode 100644 src/X/static/media/Brandon.ebe25f19deed183bcd5e.webp create mode 100644 src/X/static/media/Drishya.d320d733ac3130f4fffb.webp create mode 100644 src/X/static/media/Elaine.2621346587810ae7fb18.webp create mode 100644 src/X/static/media/HackMercedTower.4fdbb51d75e6819b17fd.png create mode 100644 src/X/static/media/Isha.2ef5dc7c3022261b6390.webp create mode 100644 src/X/static/media/Ishan.44b15b9d28ced5d5fd2f.webp create mode 100644 src/X/static/media/Jeevan.91f5b36c62fba78fd0ff.webp create mode 100644 src/X/static/media/Jet.4bd5a66f55130571edee.webp create mode 100644 src/X/static/media/Marian.03625411afe42f1aef2c.webp create mode 100644 src/X/static/media/Megan.b1be85544a7f0603b17b.webp create mode 100644 src/X/static/media/NewMainPageBackground.0f23da999859c41b6f1a.svg create mode 100644 src/X/static/media/SponsorsCollage.dcb671ad749e970cf0fd.png create mode 100644 src/X/static/media/Team2.a9ce980bfd4d78cc3b47.webp create mode 100644 src/X/static/media/Vector 1.dfe7bd48083770efdefca5ab71127068.svg create mode 100644 src/X/static/media/ag.63b60df20946c9323fc476570af66de5.svg create mode 100644 src/X/static/media/akhil.2da83b52a10f3f4d234d.JPG create mode 100644 src/X/static/media/arrow_right.797afb4bb4bfd0b5293bba1cdb904dd4.svg create mode 100644 src/X/static/media/background-buildings.5a49afed817462164433.png create mode 100644 src/X/static/media/bus.ba0656c48b52645d022bd7d137d5fecf.svg create mode 100644 src/X/static/media/discord 1.e290b67d8b19e02a28652e2b2c90c81f.svg create mode 100644 src/X/static/media/facebook 1.a49b4357e5c204d5cdf81d0d0ed7974e.svg create mode 100644 src/X/static/media/hackmerced-tower.06913f705c04ff3fab04.png create mode 100644 src/X/static/media/health.52082d88db33c341b551985fdac44892.svg create mode 100644 src/X/static/media/instagram 1.611fcf3b3dac69c4b8163932828624b5.svg create mode 100644 src/X/static/media/lauren.5e02914aa3521f325759.jpg create mode 100644 src/X/static/media/shyam.ab4a0d3e5458068b8f2e.JPG create mode 100644 src/X/static/media/ucmmap.f648cfe35bbe2d337db1b512443ac0a4.svg diff --git a/src/X/404.html b/src/X/404.html new file mode 100644 index 0000000..96e21ce --- /dev/null +++ b/src/X/404.html @@ -0,0 +1 @@ +HackMerced X
\ No newline at end of file diff --git a/src/X/asset-manifest.json b/src/X/asset-manifest.json new file mode 100644 index 0000000..05ee06c --- /dev/null +++ b/src/X/asset-manifest.json @@ -0,0 +1,46 @@ +{ + "files": { + "main.css": "/X/static/css/main.143e02ec.css", + "main.js": "/X/static/js/main.273f6e3b.js", + "static/js/488.4d6d0dc5.chunk.js": "/X/static/js/488.4d6d0dc5.chunk.js", + "static/media/NewMainPageBackground.svg": "/X/static/media/NewMainPageBackground.0f23da999859c41b6f1a.svg", + "static/media/akhil.JPG": "/X/static/media/akhil.2da83b52a10f3f4d234d.JPG", + "static/media/SponsorsCollage.png": "/X/static/media/SponsorsCollage.dcb671ad749e970cf0fd.png", + "static/media/shyam.JPG": "/X/static/media/shyam.ab4a0d3e5458068b8f2e.JPG", + "static/media/lauren.jpg": "/X/static/media/lauren.5e02914aa3521f325759.jpg", + "static/media/background-buildings.png": "/X/static/media/background-buildings.5a49afed817462164433.png", + "static/media/ucmmap.svg": "/X/static/media/ucmmap.f648cfe35bbe2d337db1b512443ac0a4.svg", + "static/media/Megan.webp": "/X/static/media/Megan.b1be85544a7f0603b17b.webp", + "static/media/HackMercedTower.png": "/X/static/media/HackMercedTower.4fdbb51d75e6819b17fd.png", + "static/media/hackmerced-tower.png": "/X/static/media/hackmerced-tower.06913f705c04ff3fab04.png", + "static/media/Team2.webp": "/X/static/media/Team2.a9ce980bfd4d78cc3b47.webp", + "static/media/Alisson.webp": "/X/static/media/Alisson.b97c246bbf0f90c582bf.webp", + "static/media/Brandon.webp": "/X/static/media/Brandon.ebe25f19deed183bcd5e.webp", + "static/media/Alex.webp": "/X/static/media/Alex.b27a90623c5a49c4997e.webp", + "static/media/Elaine.webp": "/X/static/media/Elaine.2621346587810ae7fb18.webp", + "static/media/Marian.webp": "/X/static/media/Marian.03625411afe42f1aef2c.webp", + "static/media/Jet.webp": "/X/static/media/Jet.4bd5a66f55130571edee.webp", + "static/media/Isha.webp": "/X/static/media/Isha.2ef5dc7c3022261b6390.webp", + "static/media/Jeevan.webp": "/X/static/media/Jeevan.91f5b36c62fba78fd0ff.webp", + "static/media/Andre.webp": "/X/static/media/Andre.fa5649344a2a22ab64f0.webp", + "static/media/Aaron.webp": "/X/static/media/Aaron.d9742a0139f85aae0882.webp", + "static/media/Drishya.webp": "/X/static/media/Drishya.d320d733ac3130f4fffb.webp", + "static/media/Ishan.webp": "/X/static/media/Ishan.44b15b9d28ced5d5fd2f.webp", + "static/media/ag.svg": "/X/static/media/ag.63b60df20946c9323fc476570af66de5.svg", + "static/media/instagram 1.svg": "/X/static/media/instagram 1.611fcf3b3dac69c4b8163932828624b5.svg", + "static/media/discord 1.svg": "/X/static/media/discord 1.e290b67d8b19e02a28652e2b2c90c81f.svg", + "static/media/bus.svg": "/X/static/media/bus.ba0656c48b52645d022bd7d137d5fecf.svg", + "static/media/health.svg": "/X/static/media/health.52082d88db33c341b551985fdac44892.svg", + "static/media/facebook 1.svg": "/X/static/media/facebook 1.a49b4357e5c204d5cdf81d0d0ed7974e.svg", + "index.html": "/X/index.html", + "static/media/Vector 1.svg": "/X/static/media/Vector 1.dfe7bd48083770efdefca5ab71127068.svg", + "static/media/arrow_right.svg": "/X/static/media/arrow_right.797afb4bb4bfd0b5293bba1cdb904dd4.svg", + "main.143e02ec.css.map": "/X/static/css/main.143e02ec.css.map", + "main.273f6e3b.js.map": "/X/static/js/main.273f6e3b.js.map", + "488.4d6d0dc5.chunk.js.map": "/X/static/js/488.4d6d0dc5.chunk.js.map" + }, + "entrypoints": [ + "static/css/main.143e02ec.css", + "static/js/main.273f6e3b.js" + ] +} \ No newline at end of file diff --git a/src/X/discord.html b/src/X/discord.html new file mode 100644 index 0000000..841fea2 --- /dev/null +++ b/src/X/discord.html @@ -0,0 +1,18 @@ + + + + + + + Redirecting + + + +

Redirecting you to the Discord Invite. If you are not directed automatically, Click Here.

+ + + + + \ No newline at end of file diff --git a/src/X/favicon.ico b/src/X/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..c936cd55d3d60cd2634bf725a079b3f36fc0fb0c GIT binary patch literal 95504 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 P00000NkvXXu0mjf?7(D| literal 0 HcmV?d00001 diff --git a/src/X/index.html b/src/X/index.html new file mode 100644 index 0000000..96e21ce --- /dev/null +++ b/src/X/index.html @@ -0,0 +1 @@ +HackMerced X

\ No newline at end of file diff --git a/src/X/logo192.png b/src/X/logo192.png new file mode 100644 index 0000000000000000000000000000000000000000..fc44b0a3796c0e0a64c3d858ca038bd4570465d9 GIT binary patch literal 5347 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 b{|{}U^kRn2wo}j7|0ATu<;8xA7zX}7|B6mN literal 0 HcmV?d00001 diff --git a/src/X/manifest.json b/src/X/manifest.json new file mode 100644 index 0000000..080d6c7 --- /dev/null +++ b/src/X/manifest.json @@ -0,0 +1,25 @@ +{ + "short_name": "React App", + "name": "Create React App Sample", + "icons": [ + { + "src": "favicon.ico", + "sizes": "64x64 32x32 24x24 16x16", + "type": "image/x-icon" + }, + { + "src": "logo192.png", + "type": "image/png", + "sizes": "192x192" + }, + { + "src": "logo512.png", + "type": "image/png", + "sizes": "512x512" + } + ], + "start_url": ".", + "display": "standalone", + "theme_color": "#000000", + "background_color": "#ffffff" +} diff --git a/src/X/robots.txt b/src/X/robots.txt new file mode 100644 index 0000000..e9e57dc --- /dev/null +++ b/src/X/robots.txt @@ -0,0 +1,3 @@ +# https://www.robotstxt.org/robotstxt.html +User-agent: * +Disallow: diff --git a/src/X/static/css/main.143e02ec.css b/src/X/static/css/main.143e02ec.css new file mode 100644 index 0000000..283981a --- /dev/null +++ b/src/X/static/css/main.143e02ec.css @@ -0,0 +1,2 @@ +@import url(https://fonts.googleapis.com/css2?family=K2D:ital,wght@0,400;1,700&display=swap);@import url(https://fonts.googleapis.com/css2?family=Rajdhani:wght@300;400;500;600;700&display=swap);body{-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;font-family:Rajdhani,-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen,Ubuntu,Cantarell,Fira Sans,Droid Sans,Helvetica Neue,sans-serif;margin:0}code{font-family:Rajdhani,source-code-pro,Menlo,Monaco,Consolas,Courier New,monospace}:root{--primary-purple:#639;--primary-orange:#c0593a;--lighter-orange:#cf8139;--secondary-red:#993392;--secondary-purple:#639;--main-page-background:#422162;--primary-purple-transparent:#ff9fffb9;--link-color:#c0fcff;--link-color-2:#c0fcff;--button-hover:#ffa052;--button-hover-gradient-2:#ff7252;--main-white:#fff;--orange-font-color:#ffa052;--main-shadows-1:#313131;--main-shadows-2:#251212;--secondary-shadows:#8f1a1a;--accordian-border:#e0b8ff;--hacktually-infobox:#ffffff80;--hacktually-info:#fff3}.html{overflow-y:hidden}.App{font-family:Rajdhani;font-family:var(--font-family);min-height:100vh}.App-link{color:#c0fcff;color:var(--link-color)}.About-HM,.Tracks,a{color:#fff;color:var(--main-white)}.About-HM,.Tracks{align-items:center;-webkit-backdrop-filter:blur(6px);backdrop-filter:blur(6px);background:radial-gradient(50% 50% at 50% 50%,#cf8139,#c0593a,#639);background:radial-gradient(50% 50% at 50% 50%,var(--lighter-orange),var(--primary-orange),var(--primary-purple));background-blend-mode:overlay;height:-webkit-fit-content;height:fit-content;max-width:100vw;mix-blend-mode:normal;padding-bottom:10vh;text-align:center}.About-HM-Title,.Hacktually-title,.Tracks-title{font-family:Rajdhani;font-family:var(--font-family);font-weight:700;margin-bottom:2em;margin-left:11%;margin-top:0;padding-top:3.5em;position:absolute;text-align:left}.About-HM-Contents,.Tracks-Flex{display:flex;justify-content:center;padding-top:1rem}.About-HM-Desc,.Team-photo,.Track-Card{border-image:linear-gradient(180deg,#639,#c0593a,#cf8139) 30 /1rem;border-image:linear-gradient(180deg,var(--primary-purple),var(--primary-orange),var(--lighter-orange)) 30 /1rem;border-radius:0!important;border-style:groove;box-shadow:inset 0 0 60px -12px #313131;box-shadow:inset var(--main-shadows-1) 0 0 60px -12px;margin:20vh 1em 5vh;padding:5vh 2em 2em;position:relative;width:20vw}.About-HM-Desc,.Team-photo,.Track-Card,.Tracks-Flex .css-1j74uew-MuiPaper-root-MuiCard-root{background:linear-gradient(170.14deg,#251212 1.13%,#8f1a1a 104.83%);background:linear-gradient(170.14deg,var(--main-shadows-2) 1.13%,var(--secondary-shadows) 104.83%)}.Tracks-Flex .Track-Card{background:linear-gradient(170.14deg,#251212 1.13%,#8f1a1a 104.83%)!important;background:linear-gradient(170.14deg,var(--main-shadows-2) 1.13%,var(--secondary-shadows) 104.83%)!important}.Track-Name{color:#ffa052;color:var(--orange-font-color)}.About-HM-Text,.Track-Name,.Track-Text{font-family:Rajdhani;font-family:var(--font-family)}.About-HM-Text,.Track-Text{color:#fff;color:var(--main-white);font-size:.9em;font-style:normal;font-weight:400;margin:1vw;padding-top:2vh;text-align:center}.Track-Logo{filter:drop-shadow(0 0 10px #FFA052);filter:drop-shadow(0 0 10px var(--orange-font-color));margin-top:5vh;min-height:10vh;width:10vw}.Sponsors{align-items:center;-webkit-backdrop-filter:blur(6px);backdrop-filter:blur(6px);background:radial-gradient(60% 45% at 50% 60%,#639,#c0593a,#cf8139);background:radial-gradient(60% 45% at 50% 60%,var(--primary-purple),var(--primary-orange),var(--lighter-orange));background-blend-mode:overlay;color:#fff;color:var(--main-white);height:0;height:-webkit-fit-content;height:fit-content;max-width:100vw;mix-blend-mode:normal;text-align:center}.Sponsors-Logo{left:121px;padding-bottom:5.5em;padding-top:4em;width:70vw}.Sponsors-Description{font-size:1.2em;font-style:normal;font-weight:500;left:10vw;margin-right:10vw;position:absolute}.Sponsors-Description,.Sponsors-title{font-family:Rajdhani;font-family:var(--font-family);text-align:left}.Sponsors-title{font-weight:400;margin-top:0;padding-left:10vw;padding-top:3em}.FAQ,.Team{align-items:center;background:linear-gradient(180deg,#639,#c0593a,#cf8139);background:linear-gradient(180deg,var(--primary-purple),var(--primary-orange),var(--lighter-orange));display:flex;justify-content:center;padding:5em 0}.FAQ-container{width:80vw}.FAQ-grid{margin:0}.FAQ-container .title{color:#fff;color:var(--main-white);font-family:Rajdhani;font-family:var(--font-family);font-weight:400;margin-bottom:3em;margin-top:0;width:-webkit-fit-content;width:fit-content}.FAQ-accordion{border:2px solid #e0b8ff;border:2px solid var(--accordian-border)}.FAQ-container .FAQ-accordion.MuiAccordion-root{background:linear-gradient(122.2deg,#639 -.42%,#993392 126.58%);background:linear-gradient(122.2deg,var(--secondary-purple) -.42%,var(--secondary-red) 126.58%);border-radius:1.875em}.FAQ-summary{flex-direction:row-reverse}.FAQ-summary .MuiAccordionSummary-expandIconWrapper.Mui-expanded{transform:rotate(90deg)}.accordion-link{color:#fff;color:var(--main-white)}.accordion-title{font-size:1em;font-weight:700}.accordion-description,.accordion-title{color:#fff;color:var(--main-white);font-family:Rajdhani;font-family:var(--font-family);margin:0;text-align:left}.accordion-description{font-size:.9em;font-style:normal}@keyframes App-logo-spin{0%{transform:rotate(0deg)}to{transform:rotate(1turn)}}.NavBar{align-content:center;background:linear-gradient(122.2deg,#639 -.42%,#993392 126.58%);background:linear-gradient(122.2deg,var(--secondary-purple) -.42%,var(--secondary-red) 126.58%);display:flex;justify-content:flex-end;position:-webkit-sticky;position:sticky;width:100%;z-index:2}.nav-items{align-items:center;display:flex;margin-top:.8rem;white-space:nowrap}.nav-items .button,.nav-items .registerButton{font-family:Rajdhani;font-family:var(--font-family);font-size:1em;font-weight:700;width:-webkit-fit-content;width:fit-content}.nav-items>li{list-style-type:none;margin-right:2.5rem}.button{background:none;border:none}.button,.registerButton{color:#fff;color:var(--main-white);font-size:1em;transition-duration:.4s}.registerButton{background-color:#639;background-color:var(--primary-purple);border:.18rem solid #639;border:.18rem solid var(--primary-purple);border-radius:3rem;padding:.8rem .8rem 1rem;text-align:center;width:8rem}.button:hover,.registerButton:hover{color:#ffa052;color:var(--button-hover);cursor:pointer}.registerButton:hover{background-color:#fff;background-color:var(--main-white);border:.18rem solid #ffa052;border:.18rem solid var(--button-hover);transform:translate(.1rem,-.1rem)}.MainPage{background-blend-mode:overlay;background-color:#422162;background-color:var(--main-page-background);display:flex;font-size:calc(1em + 2vmin);font-weight:700;height:100vh;justify-content:center;mix-blend-mode:normal;overflow-x:hidden;width:100%}.Main-hackmerced-tower{left:3rem;position:absolute;top:20%;z-index:-1}.Main-text-background{background-image:url(/X/static/media/NewMainPageBackground.0f23da999859c41b6f1a.svg);background-repeat:round;background-size:100% 90%;filter:drop-shadow(0 0 40px #FFA052);filter:drop-shadow(0 0 40px var(--orange-font-color));height:90vh;transform:scale(90%);width:100vw}.Main-text{align-items:center;display:flex;flex-direction:column;height:inherit;justify-content:center;z-index:2}.Main-title{animation:glide-in .2s ease-out 1;color:#fff;color:var(--main-white);transition-duration:.4s;z-index:2}@keyframes glide-in{0%{filter:blur(100px) opacity(10%);transform:translateY(-30rem)}to{filter:blur(0) opacity(100%);transform:translateY(0)}}.Main-title:hover{filter:drop-shadow(0 0 10px #FFA052) saturate(150%);filter:drop-shadow(0 0 10px var(--orange-font-color)) saturate(150%);transform:scale(103%);transition-duration:1s}.Main-subtitle{animation:glide-in .3s ease-out 1;color:#fff;color:var(--main-white);display:flex;margin:1rem;text-align:center;z-index:2}.Main-sub2{font-size:24px;font-style:small;line-height:31px}.Main-footer,.Main-sub2{align-items:center;color:#fff;color:var(--main-white);display:flex;font-weight:500;z-index:2}.Main-footer{animation:glide-in .5s ease-out 1}.Main-footer,.Main-footerlink{font-size:18px;font-style:normal;line-height:24px}.Main-footerlink{align-items:center;animation:glide-in .6s ease-out 1;color:#c0fcff;color:var(--link-color-2);display:flex;font-weight:500;padding-left:5px;-webkit-text-decoration:wavy;text-decoration:wavy;z-index:2}.emailEmoji{animation:glide-in .7s ease-out 1;filter:grayscale(100%);padding-left:.4rem}.Main-footerlink:hover{color:#993392;color:var(--secondary-red);transition-duration:.4s}.Main-footerlink:hover .emailEmoji{filter:grayscale(0);transform:rotateY(1turn) scale(150%);transition-duration:.4s}.registerButton2{animation:glide-in .4s ease-out 1;background-color:#639;background-color:var(--primary-purple);border-radius:100px;box-shadow:17px 22px #c0593a;box-shadow:17px 22px var(--primary-orange);font-family:Rajdhani;font-family:var(--font-family);font-size:2rem;font-weight:700;height:10vh;transition-duration:.4s;width:50vw;z-index:10}.registerButton2,.registerButton2:hover{border:none;color:#fff;color:var(--main-white)}.registerButton2:hover{background:linear-gradient(122.2deg,#639 -.42%,#ff7252);background:linear-gradient(122.2deg,var(--secondary-purple) -.42%,var(--button-hover-gradient-2) 100%);cursor:pointer;filter:saturate(200%);transform:translate(.1rem,.1rem)}.registerButton2Stars{position:relative}#star{color:#fff;color:var(--main-white);filter:blur(.1rem) opacity(0);font-size:.5em;position:absolute;right:50%;top:50%}@keyframes move-stars-x{0%{filter:blur(.1rem) opacity(0);transform:translateX(-20vw)}10%{filter:blur(.1rem) opacity(100%)}99.9%{filter:blur(.1rem) opacity(100%)}to{filter:blur(.1rem) opacity(0);transform:translateX(20vw)}}@keyframes move-stars-y{0%{transform:translateY(0)}12.5%{transform:translateY(-10rem)}25%{transform:translateY(0)}37.5%{transform:translateY(-10rem)}50%{transform:translateY(0)}62.5%{transform:translateY(-10rem)}75%{transform:translateY(0)}87.5%{transform:translateY(-10rem)}to{transform:translateY(0)}}.registerButton2:hover~.starWrapper #star{display:block}.registerButton2:hover~.starWrapper{animation:move-stars-y 1.8s ease-in-out 2}.registerButton2:hover~.starWrapper .star1{animation:move-stars-x 1s ease-in-out 1 both}.registerButton2:hover~.starWrapper .star2{animation:move-stars-x 1.2s ease-in-out 1s 1 both}.registerButton2:hover~.starWrapper .star3{animation:move-stars-x 1.6s ease-in-out 2s 1 both}.starBursts{display:flex;font-size:0;position:absolute;right:0;top:0}.registerButton2:hover~.starBursts .starBurst1{animation:starburst 1s linear .93s backwards}.registerButton2:hover~.starBursts .starBurst2{animation:starburst 1s linear 2.2s backwards}.registerButton2:hover~.starBursts .starBurst3{animation:starburst 1s linear 3.6s backwards}@keyframes starburst{0%{display:none;font-size:4rem;transform:rotate(0deg)}0.1%{display:flex}to{display:none;transform:rotate(1turn)}}.Footer-Comp{background:linear-gradient(180deg,#cf8139,#c0593a,#639);background:linear-gradient(180deg,var(--lighter-orange),var(--primary-orange),var(--primary-purple));color:#fff;color:var(--main-white);display:flex;font-family:Rajdhani;font-family:var(--font-family);font-size:.9em;font-style:normal;font-weight:400;max-width:100vw;min-height:33vh;padding-bottom:2rem;position:relative}.Footer-container{display:flex;padding-top:10vh}.Footer-socialmedia{padding-left:10%;text-align:left}.Footer-Vector{padding-left:40vw}.Footer-Logo{padding-right:1em}.Footer-description{padding-left:1.5em;text-align:left}.Team-container{min-height:70vh;position:relative;top:4rem;width:80vw}.title{text-align:left}.Team-container .title{color:#fff;color:var(--main-white);font-family:Rajdhani;font-family:var(--font-family);font-weight:400;height:10vh;left:1.5rem;margin-top:0;position:absolute;top:-3.5rem;width:-webkit-fit-content;width:fit-content}.MuiGrid-root.MuiGrid-item{padding-left:none}.Team-individual{padding:1em}.Team-Flex{align-items:flex-start;display:flex;flex-direction:row;justify-content:center}.Team-Img{border-radius:50%;filter:drop-shadow(0 0 10px #FFA052) saturate(150%);filter:drop-shadow(0 0 10px var(--orange-font-color)) saturate(150%);max-height:20vh;max-width:20vw;overflow:hidden}.Team-Name{color:#fff;color:var(--main-white);font-family:Rajdhani;font-family:var(--font-family)}.Contact-Us-container{height:80vh}.About-HM-Title{padding-top:4em}.archive-Container{display:inherit}.About-HM-Contents{display:flex}.About-HM-Desc{align-items:center;display:flex;justify-content:center}.About-HM-Contents .css-1j74uew-MuiPaper-root-MuiCard-root,.css-q8lpi9,MuiPaper-root MuiPaper-elevation MuiPaper-rounded MuiPaper-elevation1 MuiCard-root Track-Card css-q8lpi9{background:linear-gradient(170.14deg,#251212 1.13%,#8f1a1a 104.83%);background:linear-gradient(170.14deg,var(--main-shadows-2) 1.13%,var(--secondary-shadows) 104.83%)}.About-HM-Desc.css-q8lpi9{background:linear-gradient(170.14deg,#251212 1.13%,#8f1a1a 104.83%)!important;background:linear-gradient(170.14deg,var(--main-shadows-2) 1.13%,var(--secondary-shadows) 104.83%)!important}.About-HM-Text{font-size:large;padding:0;text-align:center}.Team-photo{height:auto;width:50vw}#mlh-trust-badge{left:50px}.Hacktually{align-items:center;-webkit-backdrop-filter:blur(6px);backdrop-filter:blur(6px);background:radial-gradient(50% 50% at 50% 50%,#cf8139,#c0593a,#639);background:radial-gradient(50% 50% at 50% 50%,var(--lighter-orange),var(--primary-orange),var(--primary-purple));background-blend-mode:overlay;color:#fff;color:var(--main-white);height:100vh;margin-top:0;max-width:100vw;mix-blend-mode:normal;padding-bottom:10vh;text-align:center}.Hacktually-Grid{grid-column-gap:10vw;grid-row-gap:8vw;align-items:center;column-gap:10vw;display:grid;font-size:1.2em;grid-template-columns:repeat(2,minmax(20vw,35vw));grid-template-rows:3;justify-items:center;margin:5vw 0 0 10vw;row-gap:8vw}.Hacktually-title{position:inherit}.Hacktually-infobox{background:#ffffff80;border-radius:10% 20% 30% 40%/40% 30% 20% 10%;margin:0 5em 0 15%;text-align:justify}.Hacktually-info{background:#fff3;background-blend-mode:soft-light;padding:4em;transition:border-radius .3s ease}.Hacktually-info:nth-of-type(odd){border-radius:50% 70% 35%}.Hacktually-info:nth-of-type(2n){border-radius:90% 50% 70%}.Hacktually-info:nth-of-type(2n):hover{border-radius:40% 25% 85%}.Hacktually-info:nth-of-type(odd):hover{border-radius:85% 40% 25%}.menu-buttons ul li{list-style-type:none}.menu-buttons{display:none}@media screen and (min-width:300px) and (max-width:880px){.MainPage{display:block;overflow-y:hidden}.Main-video{left:0;overflow:hidden;padding-left:0;position:-webkit-sticky;position:sticky;z-index:1}.Main-video:-webkit-scrollbar{display:none}.Main-hackmerced-tower{opacity:20%}.Main-text-background{font-size:smaller;padding:inherit}.Main-text{padding:2rem}.title-flair{display:block}button.registerButton2{margin-bottom:1.2rem;margin-top:0}.Main-footer{display:block;margin-top:0;padding-right:1rem}.Main-footerlink{margin:-1rem;padding-left:1rem}.NavBar{justify-content:center}.nav-items{margin:0;padding:.5em}.nav-items>li{border:hidden;font-size:.8rem;margin-right:.8vw}.registerButton{border-radius:3rem;padding:.5rem;text-align:center;width:4rem}.registerButton2{box-shadow:10px 10px #c0593a;box-shadow:10px 10px var(--primary-orange);font-size:x-large;height:10vh;margin:2rem;width:15rem}.About-HM-Contents,.Team-Flex{flex-direction:column;height:-webkit-fit-content;height:fit-content}.About-HM-Desc{width:auto}.Team-photo{margin-bottom:auto;margin-top:auto;width:auto}.Team-container .title{height:0;left:0;padding-bottom:3em;padding-left:0}.Team-Img{max-width:none;overflow:visible}.Team-Name{padding-top:5em}.Team-individual{display:flex;gap:5vw;margin-bottom:2em}.Team-container{height:-webkit-max-content;height:max-content}.MuiGrid-root.Team-Flex{column-count:1;display:block;gap:0}.Tracks-Flex{display:inline-block}.Track-Card{width:auto}.Tracks{height:auto;margin-top:0;padding-top:0;position:relative;z-index:3}.Tracks-title .title{padding-left:1.5em;position:static;text-align:left}.Tracks-title{margin-left:7%}.FAQ-accordion{flex-direction:column;position:relative}.MuiGrid2-root.FAQ-grid{column-count:1;display:block;gap:0}.FAQ-accordion{width:80vw}.FAQ{position:relative;text-align:left;z-index:3}.Footer-Vector{padding-left:3em}.Footer-socialmedia{padding-left:1.5em}.Footer-container{padding-top:3vh}.Footer-description{padding-left:2em}.Sponsors-Logo{margin-top:15vh;width:80vw}.Tracks{padding-bottom:0}.Track-Card{max-width:70vw}.Track-Logo{height:auto;width:auto}.Contact-Us-container{height:auto}#mlh-trust-badge{left:0}#mlh-trust-badge img{width:-webkit-max-content;width:max-content}.Hacktually{height:-webkit-fit-content!important;height:fit-content!important}.Hacktually-info{border-radius:4px!important;padding:30px}.Hacktually-Grid{column-gap:5vw;font-size:1em;grid-template-columns:repeat(2,40vw);margin:5vw 0 0 8vw;row-gap:8vw}.burger-buttons{display:inherit}#burger-home #burger-about #burger-contact #burger-sponsor{display:none}.menu-buttons,nav{display:flex}nav{height:3em;justify-content:flex-end;padding:1em 10px 1em 1em;width:100%}.burger-menu{cursor:pointer;display:flex;flex-direction:column;height:100%;justify-content:space-between;margin-right:-185px;width:3.5em;z-index:9999}.burger-bar{background-color:#fff;background-color:var(--main-white);float:right;height:.5em;position:relative;width:3.5em}.menu{background:linear-gradient(180deg,#693499,#8b3494);background-color:#000;height:-webkit-fit-content;height:fit-content;padding-bottom:30px;padding-top:50px;position:absolute;top:0;width:100%;z-index:9998!important}.burger-button{background-color:initial;border-radius:0;border-style:none;color:#fff;font-family:Rajdhani,"sans-serif";font-size:2em;font-weight:1000;justify-content:center;margin:0;padding:1rem clamp(1rem,7rem,7rem);text-align:center;text-indent:100px}.burger-bar.clicked:first-child{background-color:#ff9fffb9;background-color:var(--primary-purple-transparent);transform:rotate(45deg) translate(.75em,1.25em);transition-duration:.4s}.burger-bar.clicked:nth-child(2){transform:scale(.1);transition-duration:.4s}.burger-bar.clicked:nth-child(3){background-color:#ffa052;background-color:var(--button-hover);transform:rotate(135deg) translate(-.5em,1em);transition-duration:.4s}.burger-bar.unclicked{transform:rotate(0) translate(0);transition-duration:.4s}.hidden{display:none}.visible{display:inherit}#about-us-button,#contact-us-button,#hackmerced-live-button,#mentor-button,#sponsor-us-button,#sponsors-button{display:none}#burger-home #burger-about #burger-contact #burger-sponsor{display:inherit}.mlh-banner{margin-left:50px;position:absolute}.registerButton{margin-right:100px}}@media (max-width:414px){.menu-buttons{display:flex;font-size:1.3em;transform:translateX(-10%)}.burger-button{padding-left:1rem;padding-right:1rem}}@media (max-width:375px){.menu-buttons{display:flex;font-size:1.2em;transform:translateX(-15%)}.burger-button{padding-left:1rem;padding-right:1rem}}@media (max-width:320px){.menu-buttons{display:flex;font-size:.9em;transform:translateX(-20%)}.burger-button{padding-left:1rem;padding-right:1rem}}:root{--font-family:"Rajdhani"}.contain-first-page{display:flex;height:90vh;width:100%}.first-page-box{display:flex;height:100%;overflow:hidden;position:relative;width:50%}.contain-gradients{height:100%;position:absolute;width:100%}.gradient1{background:linear-gradient(#639,#ff5c01);height:100%;opacity:100%;position:inherit;width:100%}.gradient2{background:linear-gradient(-45deg,#d77b26,#c0593a,#639)}.gradient2,.gradient3{height:100%;opacity:85%;position:inherit;width:100%}.gradient3{background:linear-gradient(45deg,#d77b26,#c0593a,#639)}.hackmerced-tower{animation:pop-up .8s ease-in-out forwards;bottom:-600px;height:500px;left:100px;position:absolute;width:500px;z-index:2}.background-buildings{right:-50px}.background-buildings,.background-buildings2{object-fit:cover;position:absolute;z-index:1}.text-contents{color:#fff;display:flex;flex-direction:column;font-family:Rajdhani;font-family:var(--font-family);position:absolute;right:200px;top:50px;z-index:1}.first-box-text{right:60px}.social-link{align-items:center;display:flex;position:relative;right:-500px;transition:.3s}.social-link:hover{color:#f3934a;text-shadow:0 0 20px red}.socials-logo{color:#fff!important;font-size:30px;margin-right:10px}.first-page-links{text-decoration:none}.first-page-link-0{animation:slide-left .8s ease-in-out forwards}.first-page-link-1{animation:slide-left 1s ease-in-out forwards}.first-page-link-2{animation:slide-left 1.2s ease-in-out forwards}.contain-second-page{align-items:center;background:radial-gradient(#d77b26,#c0593a,#639);display:flex;flex-direction:column;overflow:hidden;padding:25px 0 50px}.contain-second-page h1{color:#fff;font-family:Rajdhani;font-family:var(--font-family);font-size:32px;text-shadow:-1px -1px 0 #c37705,1px -1px 0 #c37705,-1px 1px 0 #c37705,1px 1px 0 #c37705}.contain-second-page p{font-size:24px}.second-page-text-line{color:#fff;font-family:Rajdhani;font-family:var(--font-family);height:256px;margin:10px 0;position:relative;width:100%}.second-page-text-box1{animation:hover 2.6s ease-in-out infinite;background:#639;border-radius:25px;height:110%;padding:0 20px;position:absolute;right:60px;text-align:center;width:420px}.second-page-text-box2{animation:hover 2.8s ease-in-out infinite;left:60px;width:505px}.second-page-text-box2,.second-page-text-box3{align-items:center;background:#639;border-radius:25px;display:flex;flex-direction:column;height:100%;padding:0 20px;position:absolute;text-align:center}.second-page-text-box3{animation:hover 3s ease-in-out infinite;right:60px;width:510px}.contain-third-page{align-items:center;background:linear-gradient(-80deg,#d77b26 5%,#c0593a,#639);display:flex;flex-direction:column;overflow:hidden;padding:25px 0 0}.contain-third-page h1{color:#fff;font-family:Rajdhani;font-family:var(--font-family);font-size:32px;text-shadow:-1px -1px 0 #c37705,1px -1px 0 #c37705,-1px 1px 0 #c37705,1px 1px 0 #c37705}.contain-third-page span{color:#fff}.tier-container{align-items:stretch;display:flex}.tier-h3{font-size:180%;font-weight:700;margin:2% 0 0;padding-left:7%}.sponsor-tiers{background-color:#1c0235;border-radius:2rem;color:#fff;font-family:Rajdhani;font-family:var(--font-family);margin-bottom:3%;opacity:80%;width:80%}.sponsor-tiers img{height:auto;padding:5%;width:15%}.tier-description{font-size:130%;padding-top:2%;width:75%}.tier-items{width:100%}.tier-items-row{display:flex;margin-bottom:10px}.tier-items-row div{box-shadow:0 0 5px #000;margin-right:30px}.text-background{align-self:center;background:#1c0235;border-radius:23px;display:flex;flex-direction:column;font-weight:400;height:auto;justify-content:center;padding:0 20px;text-align:center;width:auto}.third-page-footer{background-color:#1c0235;font-size:130%;opacity:80%;padding:5% 0;text-align:center;width:100%}@keyframes pop-up{to{bottom:0}}@keyframes slide-left{to{right:0}}@keyframes hover{0%{bottom:-20px}25%{bottom:-10px}50%{bottom:-20px}75%{bottom:-10px}to{bottom:-20px}}@media (min-width:982px){.first-box-text{display:none}}@media (max-width:981px){.first-page-box{width:100%}.first-page-box:last-child{display:none}}@media (max-width:500px){.background-buildings{height:100%;right:0;width:100%}.hackmerced-tower{left:0}.first-box-text{left:30px;right:0}}@media (max-width:705px){.contain-second-page h1{font-size:clamp(23px,1rem + 2vw,32px)}.contain-second-page h2,.contain-second-page p{font-size:clamp(12px,1rem + 1.5vw,24px)}.second-page-text-line{align-items:center;display:flex;height:auto;justify-content:center}.second-page-text-box1,.second-page-text-box2,.second-page-text-box3{height:auto;left:0;margin:0 10px;position:relative;right:0;width:clamp(100px,100vw,420px)}}@media (max-width:809px){.tier-items-row{flex-wrap:wrap;margin:0}.text-background{margin:5px 0}.sponsor-tiers{height:auto;margin:30px 0}.contain-third-page span{text-align:center}} +/*# sourceMappingURL=main.143e02ec.css.map*/ \ No newline at end of file diff --git a/src/X/static/css/main.143e02ec.css.map b/src/X/static/css/main.143e02ec.css.map new file mode 100644 index 0000000..a5d33a3 --- /dev/null +++ b/src/X/static/css/main.143e02ec.css.map @@ -0,0 +1 @@ +{"version":3,"file":"static/css/main.143e02ec.css","mappings":"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","sources":["index.css","App.css","Components/Sponsorship-Deck/SponsorshipDeck.css"],"sourcesContent":["@import url('https://fonts.googleapis.com/css2?family=Rajdhani:wght@300;400;500;600;700&display=swap');\r\n\r\n\r\nbody {\r\n margin: 0;\r\n font-family: 'Rajdhani', -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',\r\n 'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue',\r\n sans-serif;\r\n -webkit-font-smoothing: antialiased;\r\n -moz-osx-font-smoothing: grayscale;\r\n}\r\n\r\ncode {\r\n font-family: 'Rajdhani', source-code-pro, Menlo, Monaco, Consolas, 'Courier New',\r\n monospace;\r\n}\r\n","\r\n@import url('https://fonts.googleapis.com/css2?family=K2D:ital,wght@0,400;1,700&display=swap');\r\n\r\n@import url('https://fonts.googleapis.com/css2?family=Rajdhani:wght@300;400;500;600;700&display=swap');\r\n\r\n\r\n:root{\r\n\r\n \r\n /*HackMerced IX Colors*/\r\n /* --primary-purple: #342153;\r\n --primary-orange: #C0593A;\r\n --lighter-orange: #CF8139;\r\n\r\n --secondary-red: #B01F69; /*linear gradient on navbar\r\n --secondary-purple: #301942;\r\n\r\n --main-page-background: #000000;\r\n\r\n --link-color: #61dafb;\r\n --link-color-2: #dbbaf0; /*i think i prefer this one\r\n --button-hover: #AD0624; /*jet- i dont really like this color lol i thikn this was a tmepoarary color that never got changed\r\n --button-hover-gradient-2: #7a4332; /*who made the register button \r\n\r\n --main-font-color: white; /*not changign this\r\n --orange-font-color: #FF5E54; \r\n\r\n --main-shadows-1: rgb(49, 49, 49);\r\n --main-shadows-2: rgba(37, 18, 18);\r\n --secondary-shadows: rgba(143, 26, 26);\r\n\r\n --accordian-border: #E0B8FF;\r\n\r\n --hacktually-infobox: rgba(255, 255, 255, 0.5); /*have not changed these\r\n --hacktually-info: rgba(255, 255, 255, 0.2); */\r\n \r\n\r\n /*HackMerced X Colors*/\r\n --primary-purple: #663399;\r\n --primary-orange: #C0593A; /*C0593A, FF7252*/\r\n --lighter-orange: #CF8139; /*CF8139, FFA052*/\r\n\r\n --secondary-red: #993392; /*linear gradient on navbar*/\r\n --secondary-purple: #663399;\r\n\r\n --main-page-background: #422162;\r\n\r\n --primary-purple-transparent: #ff9fffb9;\r\n\r\n --link-color: #C0FCFF;\r\n --link-color-2: #C0FCFF; /*i think i prefer this one*/\r\n --button-hover: #FFA052; /*jet- i dont really like this color lol i thikn this was a tmepoarary color that never got changed*/\r\n --button-hover-gradient-2: #FF7252; /*who made the register button */\r\n\r\n --main-white: white; /*not changign this*/\r\n --orange-font-color: #FFA052; \r\n\r\n --main-shadows-1: rgb(49, 49, 49);\r\n --main-shadows-2: rgba(37, 18, 18);\r\n --secondary-shadows: rgba(143, 26, 26);\r\n\r\n --accordian-border: #E0B8FF;\r\n\r\n --hacktually-infobox: rgba(255, 255, 255, 0.5); /*have not changed these*/\r\n --hacktually-info: rgba(255, 255, 255, 0.2);\r\n\r\n --font-family: 'Rajdhani';\r\n\r\n}\r\n\r\n.html{\r\n overflow-y: hidden;\r\n}\r\n.App {\r\n min-height: 100vh;\r\n font-family: var(--font-family);\r\n}\r\n\r\n.App-link {\r\n /* color: #61dafb; */\r\n color: var(--link-color);\r\n}\r\n\r\na {\r\n /* color: white; */\r\n color: var(--main-white);\r\n}\r\n\r\n.Tracks, .About-HM{\r\n /* background: radial-gradient(50% 50% at 50% 50%, #CF8139, #C0593A, #342153); */\r\n background: radial-gradient(50% 50% at 50% 50%, var(--lighter-orange), var(--primary-orange), var(--primary-purple));\r\n background-blend-mode: overlay;\r\n mix-blend-mode: normal;\r\n backdrop-filter: blur(6px);\r\n /* color:white; */\r\n color: var(--main-white);\r\n height: fit-content;\r\n max-width: 100vw;\r\n align-items: center;\r\n text-align: center;\r\n padding-bottom: 10vh ;\r\n \r\n}\r\n.Tracks-title, .About-HM-Title, .Hacktually-title {\r\n margin-top: 0;\r\n font-family: var(--font-family);\r\n font-weight:bold;\r\n margin-bottom: 2em;\r\n margin-left: 11%;\r\n padding-top: 3.5em;\r\n text-align: left;\r\n position: absolute;\r\n \r\n}\r\n.Tracks-Flex, .About-HM-Contents {\r\n justify-content: center;\r\n display: flex;\r\n padding-top: 1rem;\r\n}\r\n.Track-Card, .About-HM-Desc, .Team-photo{\r\n /* box-shadow: inset rgb(49, 49, 49) 0px 0px 60px -12px; */\r\n box-shadow: inset var(--main-shadows-1) 0px 0px 60px -12px;\r\n /* background: linear-gradient(170.14deg, rgba(37, 18, 18) 1.13%, rgba(143, 26, 26) 104.83%); */\r\n background: linear-gradient(170.14deg, var(--main-shadows-2) 1.13%, var(--secondary-shadows) 104.83%);\r\n\r\n width: 20vw;\r\n padding:2em;\r\n padding-top: 5vh;\r\n margin:1em;\r\n margin-bottom: 5vh;\r\n margin-top: 20vh;\r\n /* border-image: linear-gradient(180deg,#342153, #C0593A, #CF8139) 30 / 1rem; */\r\n border-image: linear-gradient(180deg,var(--primary-purple), var(--primary-orange), var(--lighter-orange)) 30 / 1rem;\r\n border-style: groove;\r\n border-radius: 0 !important;\r\n position: relative;\r\n}\r\n.Tracks-Flex .css-1j74uew-MuiPaper-root-MuiCard-root {\r\n background: linear-gradient(170.14deg, var(--main-shadows-2) 1.13%, var(--secondary-shadows) 104.83%);\r\n}\r\n.Tracks-Flex .Track-Card {\r\n background: linear-gradient(170.14deg, var(--main-shadows-2) 1.13%, var(--secondary-shadows) 104.83%) !important;\r\n}\r\n\r\n\r\n.Track-Name{\r\n /* color:#FF5E54; */\r\n color: var(--orange-font-color);\r\n font-family: var(--font-family);\r\n}\r\n.Track-Text, .About-HM-Text{\r\n padding-top: 2vh;\r\n margin:1vw;\r\n font-family: var(--font-family);\r\n font-style: normal;\r\n font-weight: 400;\r\n font-size: 0.9em;\r\n text-align: center;\r\n /* color:white; */\r\n color: var(--main-white);\r\n}\r\n.Track-Logo{\r\n /* filter: drop-shadow(0px 0px 10px #FF5E54); */\r\n filter: drop-shadow(0px 0px 10px var(--orange-font-color));\r\n min-height: 10vh;\r\n width: 10vw;\r\n margin-top:5vh;\r\n \r\n}\r\n.Sponsors{\r\n height:0;\r\n /* background: radial-gradient(60% 45% at 50% 60%,#342153, #C0593A, #CF8139); */\r\n background: radial-gradient(60% 45% at 50% 60%,var(--primary-purple), var(--primary-orange), var(--lighter-orange)); \r\n background-blend-mode: overlay;\r\n mix-blend-mode: normal;\r\n backdrop-filter: blur(6px);\r\n /* color:white; */\r\n color: var(--main-white);\r\n height:fit-content;\r\n max-width: 100vw;\r\n align-items: center;\r\n text-align: center;\r\n}\r\n.Sponsors-Logo{\r\n width: 70vw;\r\n left: 121px;\r\n padding-top: 4em;\r\n padding-bottom: 5.5em;\r\n}\r\n.Sponsors-Description{\r\n left: 10vw;\r\n position: absolute;\r\n font-family: var(--font-family);\r\n font-style: normal;\r\n font-weight: 500;\r\n font-size: 1.2em;\r\n margin-right: 10vw;\r\n text-align: left;\r\n}\r\n.Sponsors-title{\r\n margin-top: 0;\r\n font-family: var(--font-family);\r\n font-weight: normal;\r\n padding-top: 3em;\r\n text-align: left;\r\n padding-left: 10vw;\r\n}\r\n\r\n.FAQ, .Team{\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n padding: 5em 0;\r\n /* background: linear-gradient(180deg,#342153, #C0593A, #CF8139); */\r\n background: linear-gradient(180deg,var(--primary-purple), var(--primary-orange), var(--lighter-orange));\r\n}\r\n\r\n.FAQ-container{\r\n width: 80vw;\r\n}\r\n\r\n\r\n.FAQ-grid{\r\n margin: 0;\r\n}\r\n\r\n.FAQ-container .title{\r\n font-family: var(--font-family);\r\n font-weight: normal;\r\n width: fit-content;\r\n margin-top: 0;\r\n margin-bottom: 3em;\r\n /* color: white; */\r\n color: var(--main-white);\r\n}\r\n\r\n.FAQ-accordion{\r\n /* border: 2px solid #E0B8FF; */\r\n border: 2px solid var(--accordian-border);\r\n}\r\n\r\n.FAQ-container .FAQ-accordion.MuiAccordion-root {\r\n border-radius: 1.875em;\r\n /* background: linear-gradient(122.2deg, #301942 -0.42%, #B01F69 126.58%); */\r\n background: linear-gradient(122.2deg, var(--secondary-purple) -0.42%, var(--secondary-red) 126.58%);\r\n}\r\n\r\n.FAQ-summary{\r\n flex-direction: row-reverse;\r\n}\r\n\r\n.FAQ-summary .MuiAccordionSummary-expandIconWrapper.Mui-expanded {\r\n transform: rotate(90deg);\r\n}\r\n.accordion-link{\r\n /* color: white; */\r\n color: var(--main-white);\r\n}\r\n\r\n.accordion-title{\r\n font-family: var(--font-family);\r\n font-weight: bold;\r\n font-size: 1em;\r\n /* color: white; */\r\n color: var(--main-white);\r\n text-align: left;\r\n margin: 0;\r\n}\r\n\r\n.accordion-description{\r\n font-family: var(--font-family);\r\n font-style: normal;\r\n font-size: 0.9em;\r\n /* color: white; */\r\n color: var(--main-white);\r\n text-align: left;\r\n margin: 0;\r\n} \r\n\r\n@keyframes App-logo-spin {\r\n from {\r\n transform: rotate(0deg);\r\n }\r\n to {\r\n transform: rotate(360deg);\r\n }\r\n}\r\n\r\n.NavBar{\r\n position: sticky;\r\n width: 100%;\r\n display: flex;\r\n justify-content: flex-end;\r\n align-content: center;\r\n /* background: linear-gradient(122.2deg, #301942 -0.42%, #B01F69 126.58%); */\r\n background: linear-gradient(122.2deg, var(--secondary-purple) -0.42%, var(--secondary-red) 126.58%);\r\n z-index: 2;\r\n}\r\n.nav-items{\r\n display: flex;\r\n align-items: center;\r\n margin-top: 0.8rem;\r\n white-space: nowrap;\r\n}\r\n\r\n.nav-items .button, .nav-items .registerButton{\r\n font-family: var(--font-family);\r\n font-weight: bold;\r\n font-size: 1em;\r\n width:fit-content;\r\n}\r\n\r\n.nav-items > li{\r\n margin-right: 2.5rem;\r\n list-style-type: none;\r\n}\r\n.button{\r\n background: none;\r\n border: none;\r\n font-size: 1em;\r\n transition-duration: 0.4s;\r\n /* color: white; */\r\n color: var(--main-white);\r\n}\r\n.registerButton{\r\n /* background-color: #342153; */\r\n background-color: var(--primary-purple);\r\n /* border: 0.18rem solid #342153; */\r\n border: 0.18rem solid var(--primary-purple);\r\n border-radius: 3rem;\r\n padding:0.8rem;\r\n padding-bottom: 1rem;\r\n width:8rem;\r\n text-align: center;\r\n font-size: 1em;\r\n transition-duration: 0.4s;\r\n /* color: white; */\r\n color: var(--main-white);\r\n}\r\n.button:hover{\r\n /* color:#AD0624; */\r\n color: var(--button-hover);\r\n cursor: pointer;\r\n}\r\n.registerButton:hover{\r\n /* background-color: #FFFFFF; */\r\n background-color: var(--main-white);\r\n /* color:#AD0624; */\r\n color: var(--button-hover);\r\n /* border: 0.18rem solid #AD0624; */\r\n border: 0.18rem solid var(--button-hover);\r\n cursor: pointer;\r\n transform: translate(0.1rem, -0.1rem);\r\n}\r\n\r\n.MainPage{\r\n /* background-color: #000000; */\r\n background-color: var(--main-page-background);\r\n background-blend-mode: overlay;\r\n mix-blend-mode: normal;\r\n display: flex;\r\n width: 100%;\r\n height: 100vh;\r\n font-size: calc(1em + 2vmin);\r\n font-weight: bold;\r\n overflow-x:hidden;\r\n justify-content: center;\r\n}\r\n\r\n.Main-hackmerced-tower{\r\n /* border: red solid 3px; */\r\n position: absolute;\r\n top: 20%;\r\n left: 3rem;\r\n z-index: -1;\r\n}\r\n\r\n.Main-text-background{\r\n width: 100vw;\r\n height: 90vh;\r\n background-repeat: round;\r\n background-size: 100% 90%;\r\n /* filter: drop-shadow(0px 0px 40px #FF5E54); */\r\n filter: drop-shadow(0px 0px 40px var(--orange-font-color));\r\n /* background-image: url('./Assets/background.png') ; */\r\n background-image: url('./Assets/NewMainPageBackground.svg') ;\r\n transform: scale(90%);\r\n}\r\n.Main-text{\r\n flex-direction: column;\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n height: inherit;\r\n z-index: 2;\r\n}\r\n.Main-title{\r\n /* color:#FFFFFF; */\r\n color: var(--main-white);\r\n z-index: 2;\r\n transition-duration: 0.4s;\r\n animation: glide-in 0.2s ease-out 1;\r\n}\r\n@keyframes glide-in {\r\n 0%{\r\n filter: blur(100px) opacity(10%);\r\n transform:translateY(-30rem)\r\n }\r\n 100%{\r\n filter: blur(0px) opacity(100%);\r\n transform:translateY(0rem)\r\n }\r\n}\r\n.Main-title:hover{\r\n transition-duration: 1s;\r\n transform: scale(103%);\r\n /* filter: drop-shadow(0px 0px 10px #FF5E54) saturate(150%); */\r\n filter: drop-shadow(0px 0px 10px var(--orange-font-color)) saturate(150%);\r\n}\r\n.Main-subtitle{\r\n /* color:#FFFFFF; */\r\n color: var(--main-white);\r\n display: flex;\r\n text-align: center;\r\n animation: glide-in 0.3s ease-out 1;\r\n margin: 1rem;\r\n z-index: 2;\r\n /* border: red 3px solid; */\r\n /* background: var(--primary-purple-transparent);\r\n padding: 1rem; */\r\n}\r\n.Main-sub2{\r\n font-style: small;\r\n font-weight: 500;\r\n font-size: 24px;\r\n line-height: 31px;\r\n display: flex;\r\n align-items: center;\r\n /* color: #FFFFFF; */\r\n color: var(--main-white);\r\n z-index: 2;\r\n /* border: blue 3px solid; */\r\n /* background: var(--main-page-background); */\r\n}\r\n.Main-footer{\r\n font-style: normal;\r\n font-weight: 500;\r\n font-size: 18px;\r\n line-height: 24px;\r\n display: flex;\r\n align-items: center;\r\n /* color: #FFFFFF; */\r\n color: var(--main-white);\r\n animation: glide-in 0.5s ease-out 1;\r\n z-index: 2;\r\n /* border: green 3px solid; */\r\n /* background: var(--primary-purple-transparent);\r\n padding: 1rem; */\r\n}\r\n.Main-footerlink{\r\n font-style: normal;\r\n font-weight: 500;\r\n font-size: 18px;\r\n line-height: 24px;\r\n display: flex;\r\n align-items: center;\r\n /* color: #dbbaf0; */\r\n color: var(--link-color-2);\r\n padding-left: 5px;\r\n text-decoration: wavy;\r\n animation: glide-in 0.6s ease-out 1;\r\n z-index: 2;\r\n}\r\n.emailEmoji{\r\n padding-left: 0.4rem;\r\n filter: grayscale(100%);\r\n animation: glide-in 0.7s ease-out 1;\r\n}\r\n.Main-footerlink:hover{\r\n /* color: #B01F69; */\r\n color: var(--secondary-red);\r\n\r\n transition-duration: 0.4s;\r\n}\r\n.Main-footerlink:hover .emailEmoji{\r\n transition-duration: 0.4s;\r\n transform: rotateY(360deg) scale(150%);\r\n filter: grayscale(0%);\r\n}\r\n\r\n.registerButton2{\r\n font-family: var(--font-family);\r\n font-weight: bold;\r\n font-size: 2rem;\r\n /* background-color: #342153; */\r\n background-color: var(--primary-purple);\r\n /* box-shadow: 17px 22px #C0593A; */\r\n box-shadow: 17px 22px var(--primary-orange);\r\n border-radius: 100px;\r\n /* color: white; */\r\n color: var(--main-white);\r\n width: 50vw;\r\n height:10vh;\r\n transition-duration: .4s;\r\n border: none;\r\n animation: glide-in 0.4s ease-out 1;\r\n z-index: 10;\r\n}\r\n.registerButton2:hover{\r\n /* background: linear-gradient(122.2deg, #301942 -0.42%, #7a4332 100%); */\r\n background: linear-gradient(122.2deg, var(--secondary-purple) -0.42%, var(--button-hover-gradient-2) 100%);\r\n /* border: 0.18rem solid white; */\r\n border: 0.18rem solid var(--main-white);\r\n /* color: #FFFFFF; */\r\n color: var(--main-white);\r\n cursor: pointer;\r\n transform: translate(0.1rem, 0.1rem);\r\n border: none;\r\n filter: saturate(200%);\r\n}\r\n\r\n.registerButton2Stars{\r\n position: relative;\r\n}\r\n#star{\r\n /* display: none; */\r\n /* color: white; */\r\n color: var(--main-white);\r\n filter:blur(0.1rem) opacity(0%);\r\n position: absolute;\r\n font-size: .5em;\r\n top: 50%;\r\n right: 50%;\r\n}\r\n@keyframes move-stars-x {\r\n 0% {\r\n transform: translateX(-20vw);\r\n filter: blur(0.1rem) opacity(0%);\r\n }\r\n 10%{\r\n filter: blur(0.1rem) opacity(100%);\r\n }\r\n 99.9%{\r\n filter: blur(0.1rem) opacity(100%);\r\n }\r\n 100% {\r\n transform: translateX(20vw);\r\n filter: blur(0.1rem) opacity(0%);\r\n }\r\n}\r\n@keyframes move-stars-y {\r\n 0% {\r\n transform: translateY(0rem);\r\n }\r\n 12.5% {\r\n transform: translateY(-10rem);\r\n }\r\n 25% {\r\n transform: translateY(0rem);\r\n }\r\n 37.5% {\r\n transform: translateY(-10rem);\r\n }\r\n 50% {\r\n transform: translateY(0rem);\r\n }\r\n 62.5% {\r\n transform: translateY(-10rem);\r\n }\r\n 75% {\r\n transform: translateY(0rem);\r\n }\r\n 87.5% {\r\n transform: translateY(-10rem);\r\n }\r\n 100% {\r\n transform: translateY(0rem);\r\n }\r\n}\r\n.registerButton2:hover ~ .starWrapper #star{\r\n display: block;\r\n}\r\n.registerButton2:hover ~ .starWrapper{\r\n animation: move-stars-y 1.8s ease-in-out 2;\r\n\r\n}\r\n.registerButton2:hover ~ .starWrapper .star1{\r\n animation: move-stars-x 1s ease-in-out 1 both;\r\n\r\n}\r\n.registerButton2:hover ~ .starWrapper .star2{\r\n animation: move-stars-x 1.2s ease-in-out 1s 1 both;\r\n\r\n}\r\n.registerButton2:hover ~ .starWrapper .star3{\r\n animation: move-stars-x 1.6s ease-in-out 2s 1 both;\r\n\r\n}\r\n.starBursts{\r\n position: absolute;\r\n top: 0%;\r\n right: 0%;\r\n display: flex;\r\n font-size: 0px;\r\n}\r\n.registerButton2:hover ~ .starBursts .starBurst1{\r\n animation: starburst 1s linear 0.93s backwards;\r\n}\r\n.registerButton2:hover ~ .starBursts .starBurst2{\r\n animation: starburst 1s linear 2.2s backwards;\r\n}\r\n.registerButton2:hover ~ .starBursts .starBurst3{\r\n animation: starburst 1s linear 3.6s backwards;\r\n}\r\n@keyframes starburst {\r\n 0%{\r\n font-size: 4rem;\r\n display: none;\r\n transform: rotate(0deg);\r\n }\r\n 0.1%{\r\n display: flex;\r\n }\r\n 100%{\r\n display: none;\r\n transform: rotate(360deg);\r\n }\r\n}\r\n.Footer-Comp{\r\n position: relative;\r\n display: flex;\r\n padding-bottom: 2rem;\r\n /* color: white; */\r\n color: var(--main-white);\r\n font-family: var(--font-family);\r\n font-style: normal;\r\n font-weight: 400;\r\n font-size: 0.9em;\r\n max-width: 100vw;\r\n min-height: 33vh;\r\n /* background: linear-gradient(180deg, #CC7F3A, #C0593A, #342153); */\r\n background: linear-gradient(180deg, var(--lighter-orange), var(--primary-orange), var(--primary-purple));\r\n}\r\n.Footer-container{\r\n padding-top: 10vh;\r\n display: flex;\r\n}\r\n.Footer-socialmedia{\r\n text-align: left;\r\n padding-left: 10%;\r\n \r\n}\r\n.Footer-Vector{\r\n padding-left: 40vw;\r\n}\r\n.Footer-Logo{\r\n padding-right: 1em;\r\n}\r\n.Footer-description{\r\n padding-left: 1.5em;\r\n text-align: left;\r\n}\r\n/*TEAM SECTION*/\r\n.Team-container{\r\n min-height: 70vh;\r\n width: 80vw;\r\n top: 4rem;\r\n position: relative;\r\n}\r\n.title{\r\n text-align: left;\r\n}\r\n.Team-container .title{\r\n height: 10vh;\r\n font-family: var(--font-family);\r\n font-weight: normal;\r\n width: fit-content;\r\n margin-top: 0rem;\r\n position: absolute;\r\n top: -3.5rem;\r\n left: 1.5rem;\r\n /* color: white; */\r\n color: var(--main-white);\r\n}\r\n.MuiGrid-root.MuiGrid-item{\r\n padding-left: none;\r\n}\r\n.Team-individual{\r\n padding: 1em;\r\n \r\n}\r\n.Team-Flex{\r\n display: flex;\r\n align-items: flex-start;\r\n flex-direction:row;\r\n justify-content: center;\r\n}\r\n\r\n.Team-Img{\r\n border-radius: 50%;\r\n max-height: 20vh;\r\n max-width: 20vw;\r\n overflow: hidden;\r\n /* filter: drop-shadow(0px 0px 10px #FF5E54) saturate(150%); */\r\n filter: drop-shadow(0px 0px 10px var(--orange-font-color)) saturate(150%);\r\n}\r\n.Team-Name{\r\n /* color: #FFFFFF; */\r\n color: var(--main-white);\r\n font-family: var(--font-family);\r\n}\r\n/*CONTACT US*/\r\n.Contact-Us-container{\r\n height: 80vh;\r\n}\r\n/*ABOUT US*/\r\n.About-HM-Title{\r\n padding-top: 4em;\r\n}\r\n.archive-Container{\r\n display: inherit;\r\n}\r\n.About-HM-Contents{\r\n display:flex;\r\n}\r\n.About-HM-Desc {\r\n display: flex;\r\n justify-content: center;\r\n align-items: center;\r\n}\r\n.css-q8lpi9 {\r\n background: linear-gradient(170.14deg, var(--main-shadows-2) 1.13%, var(--secondary-shadows) 104.83%);\r\n}\r\nMuiPaper-root MuiPaper-elevation MuiPaper-rounded MuiPaper-elevation1 MuiCard-root Track-Card css-q8lpi9{\r\n background: linear-gradient(170.14deg, var(--main-shadows-2) 1.13%, var(--secondary-shadows) 104.83%);\r\n}\r\n.About-HM-Contents .css-1j74uew-MuiPaper-root-MuiCard-root{\r\n /* background: linear-gradient(170.14deg, rgba(37, 18, 18) 1.13%, rgba(143, 26, 26) 104.83%); */\r\n background: linear-gradient(170.14deg, var(--main-shadows-2) 1.13%, var(--secondary-shadows) 104.83%);\r\n}\r\n.About-HM-Desc.css-q8lpi9 {\r\n background: linear-gradient(170.14deg, var(--main-shadows-2) 1.13%, var(--secondary-shadows) 104.83%) !important;\r\n}\r\n\r\n\r\n.About-HM-Text{\r\n padding: 0;\r\n font-size: large;\r\n text-align: center;\r\n}\r\n.Team-photo{\r\n width:50vw;\r\n height: auto;\r\n}\r\n#mlh-trust-badge{\r\n left: 50px;\r\n}\r\n\r\n.Hacktually {\r\n /* border:1px solid white; */\r\n /* background: radial-gradient(50% 50% at 50% 50%, #CF8139, #C0593A, #342153); */\r\n background: radial-gradient(50% 50% at 50% 50%, var(--lighter-orange), var(--primary-orange), var(--primary-purple));\r\n background-blend-mode: overlay;\r\n mix-blend-mode: normal;\r\n backdrop-filter: blur(6px);\r\n /* color:white; */\r\n color: var(--main-white);\r\n height:100vh;\r\n max-width: 100vw;\r\n align-items: center;\r\n text-align: center;\r\n padding-bottom: 10vh ;\r\n margin-top:0;\r\n}\r\n\r\n.Hacktually-Grid {\r\n display:grid;\r\n grid-template-columns:repeat(2, minmax(20vw, 35vw));\r\n grid-template-rows:3;\r\n column-gap: 10vw;\r\n row-gap: 8vw;\r\n margin:5vw 0 0 10vw;\r\n justify-items:center;\r\n align-items:center;\r\n font-size:1.2em;\r\n}\r\n\r\n.Hacktually-title {\r\n position:inherit;\r\n}\r\n\r\n.Hacktually-infobox {\r\n text-align:justify;\r\n margin:0 5em 0 15%;\r\n /* border:2px solid rgba(255, 255, 255, 0.5); */\r\n background:rgba(255, 255, 255, 0.5);\r\n border-radius: 10% 20% 30% 40% / 40% 30% 20% 10%;\r\n}\r\n\r\n.Hacktually-info {\r\n padding:8vw;\r\n background:rgba(255, 255, 255, 0.2);\r\n padding:4em;\r\n background-blend-mode: soft-light;\r\n transition:border-radius 0.3s ease;\r\n}\r\n.Hacktually-info:nth-of-type(odd) { border-radius: 50% 70% 35%; }\r\n.Hacktually-info:nth-of-type(even) { border-radius:90% 50% 70%; }\r\n.Hacktually-info:nth-of-type(even):hover { border-radius:40% 25% 85%; }\r\n.Hacktually-info:nth-of-type(odd):hover { border-radius:85% 40% 25%; }\r\n\r\n.menu-buttons ul li{\r\n list-style-type: none;\r\n}\r\n\r\n.menu-buttons {\r\n display: none;\r\n}\r\n\r\n/*MOBILE VIEWPORTS*/\r\n@media screen and (min-width: 300px) and (max-width: 880px) {\r\n .MainPage{\r\n overflow-y: hidden;\r\n display: block;\r\n }\r\n .Main-video{\r\n padding-left: 0;\r\n position:sticky;\r\n left:0;\r\n z-index: 1;\r\n overflow: hidden;\r\n }\r\n .Main-video:-webkit-scrollbar {\r\n display: none;\r\n \r\n }\r\n .Main-hackmerced-tower{\r\n opacity: 20%;\r\n }\r\n .Main-text-background{\r\n padding: inherit;\r\n font-size: smaller;\r\n }\r\n .Main-text{\r\n padding: 2rem;\r\n }\r\n\r\n .title-flair{\r\n display: block;\r\n }\r\n button.registerButton2{\r\n margin-top: 0rem;\r\n margin-bottom: 1.2rem;\r\n }\r\n .Main-footer{\r\n display: block;\r\n padding-right: 1rem;\r\n margin-top: 0rem;\r\n }\r\n .Main-footerlink{\r\n margin: -1rem;\r\n padding-left: 1rem;\r\n }\r\n .NavBar{\r\n justify-content: center;\r\n }\r\n .nav-items{\r\n margin: 0;\r\n padding: 0.5em;\r\n\r\n }\r\n .nav-items > li{\r\n margin-right: 0.8vw;\r\n font-size: 0.8rem;\r\n border: hidden;\r\n }\r\n .registerButton{\r\n border-radius: 3rem;\r\n padding: 0.5rem;\r\n text-align: center;\r\n width: 4rem;\r\n }\r\n .registerButton2{\r\n margin: 2rem;\r\n height: 10vh;\r\n width: 15rem;\r\n font-size: x-large;\r\n box-shadow: 10px 10px var(--primary-orange);\r\n }\r\n .Team-Flex, .About-HM-Contents{\r\n flex-direction: column;\r\n height: fit-content;\r\n }\r\n .About-HM-Desc{\r\n width: auto;\r\n }\r\n .Team-photo{\r\n width: auto;\r\n margin-top: auto;\r\n margin-bottom: auto;\r\n }\r\n .Team-container .title{\r\n height: 0;\r\n padding-bottom: 3em;\r\n padding-left: 0;\r\n left: 0;\r\n }\r\n .Team-Img{\r\n overflow: visible;\r\n max-width: none;\r\n }\r\n .Team-Name{\r\n padding-top: 5em;\r\n }\r\n .Team-individual{\r\n display: flex;\r\n margin-bottom: 2em;\r\n gap:5vw;\r\n }\r\n .Team-container{\r\n height: max-content;\r\n }\r\n .MuiGrid-root.Team-Flex{\r\n display: block;\r\n columns: 1;\r\n gap: 0rem;\r\n \r\n }\r\n .Tracks-Flex{\r\n display: inline-block;\r\n }\r\n .Track-Card{\r\n width: auto;\r\n }\r\n .Tracks{\r\n height: auto;\r\n margin-top: 0;\r\n padding-top: 0;\r\n z-index: 3;\r\n position: relative;\r\n }\r\n .Tracks-title .title{\r\n text-align: left;\r\n padding-left:1.5em;\r\n position:initial;\r\n }\r\n .Tracks-title {\r\n margin-left: 7%;\r\n }\r\n .FAQ-accordion{\r\n position: relative;\r\n flex-direction: column;\r\n }\r\n .MuiGrid2-root.FAQ-grid{\r\n display: block;\r\n columns: 1;\r\n gap: 0rem;\r\n \r\n }\r\n .FAQ-accordion{\r\n width: 80vw;\r\n }\r\n .FAQ{\r\n position: relative;\r\n z-index: 3;\r\n text-align: left;\r\n }\r\n .Footer-Vector{\r\n padding-left: 3em;\r\n }\r\n .Footer-socialmedia{\r\n padding-left: 1.5em;\r\n }\r\n .Footer-container{\r\n padding-top: 3vh;\r\n }\r\n .Footer-description{\r\n padding-left: 2em;\r\n }\r\n .Sponsors-Logo{ \r\n margin-top: 15vh;\r\n width: 80vw;\r\n }\r\n .Tracks{\r\n padding-bottom: 0;\r\n }\r\n .Track-Card{\r\n max-width: 70vw;\r\n }\r\n .Track-Logo{\r\n height: auto;\r\n width: auto;\r\n }\r\n .Contact-Us-container{\r\n height: auto;\r\n }\r\n #mlh-trust-badge{\r\n left: 0px;\r\n }\r\n #mlh-trust-badge img{\r\n width:max-content;\r\n }\r\n .Hacktually {\r\n height:fit-content !important;\r\n }\r\n .Hacktually-info {\r\n padding:30px;\r\n border-radius:4px !important;\r\n }\r\n .Hacktually-Grid {\r\n column-gap: 5vw;\r\n grid-template-columns: repeat(2, 40vw);\r\n row-gap: 8vw;\r\n margin:5vw 0 0 8vw;\r\n font-size:1em;\r\n }\r\n\r\n /*Burger menu chaos from Brandon*/\r\n .burger-buttons {\r\n display: inherit;\r\n }\r\n #burger-home #burger-about #burger-contact #burger-sponsor {\r\n display: none;\r\n }\r\n\r\n .menu-buttons {\r\n display: flex;\r\n }\r\n nav {\r\n width: 100%;\r\n height: 3em;\r\n display: flex;\r\n padding: 1em;\r\n justify-content: flex-end;\r\n padding-right: 10px;\r\n }\r\n .burger-menu {\r\n height: 100%;\r\n width: 3.5em;\r\n display: flex;\r\n flex-direction: column;\r\n justify-content: space-between;\r\n cursor: pointer;\r\n z-index: 9999;\r\n margin-right: -185px;\r\n }\r\n .burger-bar {\r\n width: 3.5em;\r\n height: 0.5em;\r\n background-color: var(--main-white);\r\n float: right;\r\n position: relative;\r\n }\r\n .menu {\r\n background: linear-gradient(180deg, #693499, #8B3494);\r\n width: 100%;\r\n height:fit-content;\r\n background-color: black;\r\n top: 0;\r\n z-index: 9998 !important;\r\n position: absolute;\r\n padding-top: 50px;\r\n padding-bottom: 30px;\r\n }\r\n .burger-button {\r\n border-style: none;\r\n background-color: transparent;\r\n font-size: 2em;\r\n font-weight: 1000;\r\n text-align: center;\r\n font-family: 'Rajdhani', 'sans-serif';\r\n padding: 1rem 7rem;\r\n padding: 1rem clamp(1rem, 7rem, 7rem);\r\n margin: 0rem;\r\n justify-content: center;\r\n margin-bottom: 0rem;\r\n color: white;\r\n border-radius: 0px;\r\n text-indent: 100px;\r\n}\r\n .burger-bar.clicked:nth-child(1){\r\n transform: rotate(45deg) translate(0.75em, 1.25em);\r\n transition-duration: 0.4s;\r\n background-color: var(--primary-purple-transparent);\r\n }\r\n .burger-bar.clicked:nth-child(2){\r\n transform: scale(0.1);\r\n transition-duration: 0.4s;\r\n }\r\n .burger-bar.clicked:nth-child(3){\r\n transform: rotate(135deg) translate(-0.5em, 1em);\r\n transition-duration: 0.4s;\r\n background-color: var(--button-hover);\r\n }\r\n .burger-bar.unclicked {\r\n transform: rotate(0) translate(0);\r\n transition-duration: 0.4s;\r\n \r\n }\r\n .hidden {\r\n display: none;\r\n }\r\n .visible {\r\n display: inherit;\r\n }\r\n #sponsors-button {\r\n display: none;\r\n }\r\n #mentor-button {\r\n display: none;\r\n }\r\n #hackmerced-live-button {\r\n display: none;\r\n }\r\n #about-us-button {\r\n display: none;\r\n }\r\n #contact-us-button {\r\n display: none;\r\n }\r\n #sponsor-us-button {\r\n display: none;\r\n }\r\n #burger-home #burger-about #burger-contact #burger-sponsor {\r\n display: inherit;\r\n }\r\n .mlh-banner {\r\n position: absolute;\r\n margin-left: 50px;\r\n }\r\n .registerButton{\r\n margin-right: 100px;\r\n }\r\n}\r\n\r\n\r\n/* Extra extra small devices (very small phones, 480px and down) */\r\n@media (max-width: 414px) {\r\n .menu-buttons{\r\n display: flex;\r\n transform: translateX(-10%);\r\n font-size: 1.3em;\r\n }\r\n .burger-button {\r\n padding-left: 1rem;\r\n padding-right: 1rem;\r\n }\r\n}\r\n@media (max-width: 375px) {\r\n .menu-buttons{\r\n display: flex;\r\n transform: translateX(-15%);\r\n font-size: 1.2em;\r\n }\r\n .burger-button {\r\n padding-left: 1rem;\r\n padding-right: 1rem;\r\n }\r\n}\r\n@media (max-width: 320px) {\r\n.menu-buttons{\r\n display: flex;\r\n transform: translateX(-20%);\r\n font-size: 0.9em;\r\n }\r\n .burger-button {\r\n padding-left: 1rem;\r\n padding-right: 1rem;\r\n }\r\n}","@import url('https://fonts.googleapis.com/css2?family=Rajdhani:wght@300;400;500;600;700&display=swap');\r\n\r\n:root{\r\n --font-family: 'Rajdhani';\r\n}\r\n\r\n/* First Page CSS */\r\n.contain-first-page {\r\n width: 100%;\r\n height: 90vh;\r\n display: flex;\r\n}\r\n\r\n.first-page-box {\r\n height: 100%;\r\n width: 50%;\r\n overflow: hidden;\r\n display: flex;\r\n position: relative;\r\n /* outline: red solid 3px; */\r\n}\r\n\r\n.contain-gradients {\r\n position: absolute;\r\n width: 100%;\r\n height: 100%;\r\n}\r\n\r\n.gradient1 {\r\n height: 100%;\r\n width: 100%;\r\n background: linear-gradient(#663399, #FF5C01);\r\n opacity: 100%;\r\n position: inherit;\r\n\r\n}\r\n.gradient2 {\r\n height: 100%;\r\n width: 100%;\r\n background: linear-gradient(-45deg, #D77B26, #C0593A, #663399);\r\n opacity: 85%;\r\n position: inherit;\r\n}\r\n\r\n.gradient3 {\r\n height: 100%;\r\n width: 100%;\r\n background: linear-gradient(45deg, #D77B26, #C0593A, #663399);\r\n opacity: 85%;\r\n position: inherit;\r\n}\r\n\r\n.hackmerced-tower {\r\n width: 500px;\r\n height: 500px;\r\n position: absolute;\r\n bottom: -600px;\r\n left: 100px;\r\n animation: pop-up .8s ease-in-out forwards;\r\n /*animation: bop 4s ease-in-out infinite;*/\r\n z-index: 2;\r\n}\r\n\r\n.background-buildings {\r\n object-fit: cover;\r\n z-index: 1;\r\n position: absolute;\r\n right: -50px;\r\n}\r\n\r\n.background-buildings2 {\r\n position: absolute;\r\n object-fit: cover;\r\n z-index: 1;\r\n}\r\n\r\n.text-contents {\r\n display: flex;\r\n flex-direction: column;\r\n z-index: 1;\r\n color: #FFFF;\r\n font-family: var(--font-family);\r\n position: absolute;\r\n right: 200px;\r\n top: 50px;\r\n}\r\n\r\n.first-box-text {\r\n right: 60px;\r\n}\r\n\r\n.social-link {\r\n display: flex;\r\n position: relative;\r\n align-items: center;\r\n right: -500px;\r\n transition: 0.3s;\r\n}\r\n\r\n.social-link:hover {\r\n color: #f3934a;\r\n text-shadow: 0 0 20px #ff0000;\r\n}\r\n\r\n.socials-logo {\r\n font-size: 30px;\r\n margin-right: 10px;\r\n color: #FFFF !important;\r\n}\r\n\r\n\r\n.first-page-links {\r\n text-decoration: none;\r\n /* outline: red solid 3px; */\r\n}\r\n\r\n.first-page-link-0 {\r\n animation: slide-left 0.8s ease-in-out forwards;\r\n /* border: blue solid 5px; */\r\n}\r\n\r\n.first-page-link-1 {\r\n animation: slide-left 1s ease-in-out forwards;\r\n /* border: red solid 3px; */\r\n}\r\n\r\n.first-page-link-2 {\r\n animation: slide-left 1.2s ease-in-out forwards;\r\n}\r\n\r\n\r\n\r\n\r\n\r\n\r\n/* Second Page CSS */\r\n.contain-second-page {\r\n display: flex;\r\n flex-direction: column;\r\n background: radial-gradient(#D77B26, #C0593A, #663399);\r\n align-items: center;\r\n padding: 25px 0 50px 0;\r\n overflow: hidden;\r\n}\r\n\r\n.contain-second-page h1 {\r\n font-size: 32px;\r\n color: white;\r\n font-family: var(--font-family);\r\n text-shadow: -1px -1px 0 #C37705, 1px -1px 0 #C37705, -1px 1px 0 #C37705, 1px 1px 0 #C37705;\r\n}\r\n\r\n.contain-second-page p {\r\n font-size: 24px;\r\n}\r\n\r\n.second-page-text-line {\r\n width: 100%;\r\n height: 256px;\r\n position: relative;\r\n margin: 10px 0 10px 0;\r\n color: #FFFF;\r\n font-family: var(--font-family);\r\n}\r\n\r\n.second-page-text-box1 {\r\n background: #663399;\r\n height: 110%;\r\n width: 420px;\r\n position: absolute;\r\n right: 60px;\r\n border-radius: 25px;\r\n animation: hover 2.6s ease-in-out infinite;\r\n text-align: center;\r\n padding: 0 20px 0 20px;\r\n /* border: red solid 3px; */\r\n}\r\n\r\n.second-page-text-box2 {\r\n background: #663399;\r\n height: 100%;\r\n width: 505px;\r\n position: absolute;\r\n left: 60px;\r\n border-radius: 25px;\r\n animation: hover 2.8s ease-in-out infinite;\r\n text-align: center;\r\n padding: 0 20px 0 20px;\r\n display: flex;\r\n flex-direction: column;\r\n align-items: center;\r\n}\r\n\r\n.second-page-text-box3 {\r\n background: #663399;\r\n height: 100%;\r\n width: 510px;\r\n position: absolute;\r\n right: 60px;\r\n border-radius: 25px;\r\n animation: hover 3s ease-in-out infinite;\r\n text-align: center;\r\n padding: 0 20px 0 20px;\r\n display: flex;\r\n flex-direction: column;\r\n align-items: center;\r\n}\r\n\r\n\r\n\r\n\r\n\r\n\r\n/* Third Page CSS */\r\n.contain-third-page {\r\n display: flex;\r\n flex-direction: column;\r\n background: linear-gradient(-80deg, #D77B26 5%, #C0593A, #663399);\r\n align-items: center;\r\n padding: 25px 0 0 0;\r\n overflow: hidden;\r\n}\r\n\r\n.contain-third-page h1 {\r\n font-size: 32px;\r\n color: white;\r\n font-family: var(--font-family);\r\n text-shadow: -1px -1px 0 #C37705, 1px -1px 0 #C37705, -1px 1px 0 #C37705, 1px 1px 0 #C37705;\r\n}\r\n\r\n.contain-third-page span {\r\n color: white;\r\n}\r\n\r\n.tier-container {\r\n display: flex;\r\n align-items: stretch;\r\n}\r\n\r\n.tier-h3 {\r\n padding-left: 7%;\r\n font-weight: bold;\r\n font-size: 180%;\r\n margin: 2% 0 0 0;\r\n}\r\n\r\n.sponsor-tiers {\r\n width: 80%;\r\n /* height: 230px; */\r\n color: #FFFF;\r\n font-family: var(--font-family);\r\n /* display: flex; */\r\n /* align-items: center; */\r\n background-color: #1C0235;\r\n opacity: 80%;\r\n border-radius: 2rem;\r\n margin-bottom: 3%;\r\n}\r\n\r\n.sponsor-tiers img {\r\n width: 15%;\r\n padding: 5%;\r\n height: auto;\r\n}\r\n\r\n.tier-description {\r\n width: 75%;\r\n font-size: 130%;\r\n padding-top: 2%;\r\n}\r\n\r\n/* .tier-description h3 {\r\n font-size: 28px;\r\n margin: 0;\r\n padding: 0;\r\n} */\r\n\r\n.tier-items {\r\n /* display: flex;\r\n flex-direction: column;\r\n margin: 30px 0 0 0; */\r\n width: 100%;\r\n}\r\n\r\n.tier-items-row {\r\n display: flex;\r\n margin-bottom: 10px;\r\n}\r\n\r\n.tier-items-row div {\r\n margin-right: 30px;\r\n box-shadow: 0 0 5px black;\r\n}\r\n\r\n.text-background {\r\n background: #1C0235;\r\n border-radius: 23px;\r\n width: auto;\r\n height: auto;\r\n display: flex;\r\n flex-direction: column;\r\n justify-content: center;\r\n align-self: center;\r\n text-align: center;\r\n font-weight: 400;\r\n padding: 0 20px 0 20px;\r\n}\r\n\r\n.third-page-footer {\r\n background-color: #1C0235;\r\n width: 100%;\r\n text-align: center;\r\n padding: 5% 0;\r\n opacity: 80%;\r\n font-size: 130%;\r\n}\r\n\r\n\r\n\r\n/* Animations */\r\n@keyframes pop-up {\r\n 100% {\r\n bottom: 0px;\r\n }\r\n}\r\n\r\n@keyframes slide-left {\r\n 100% {\r\n right: 0px;\r\n }\r\n}\r\n\r\n@keyframes hover {\r\n 0% {\r\n bottom: -20px;\r\n }\r\n 25% {\r\n bottom: -10px;\r\n }\r\n 50% {\r\n bottom: -20px;\r\n }\r\n 75% {\r\n bottom: -10px;\r\n }\r\n 100% {\r\n bottom: -20px;\r\n }\r\n}\r\n\r\n\r\n\r\n\r\n\r\n\r\n/* Responsive Media Queries */\r\n /* Responsive First Page */\r\n @media (min-width: 982px) {\r\n .first-box-text {\r\n display: none;\r\n }\r\n }\r\n\r\n @media (max-width: 981px) {\r\n .first-page-box {\r\n width: 100%;\r\n }\r\n\r\n .first-page-box:last-child {\r\n display: none;\r\n }\r\n }\r\n\r\n @media (max-width: 500px) {\r\n .background-buildings {\r\n width: 100%;\r\n height: 100%;\r\n right: 0;\r\n }\r\n\r\n .hackmerced-tower {\r\n left: 0;\r\n }\r\n\r\n .first-box-text {\r\n right: 0px;\r\n left: 30px;\r\n }\r\n }\r\n\r\n\r\n\r\n /* Responsive Second Page */\r\n @media (max-width: 705px) {\r\n .contain-second-page h1 { \r\n font-size: clamp(23px, 1rem + 2vw, 32px);\r\n }\r\n\r\n .contain-second-page h2 {\r\n font-size: clamp(12px, 1rem + 1.5vw, 24px);\r\n }\r\n\r\n .contain-second-page p {\r\n font-size: clamp(12px, 1rem + 1.5vw, 24px);\r\n }\r\n\r\n .second-page-text-line {\r\n display: flex;\r\n justify-content: center;\r\n align-items: center;\r\n height: auto;\r\n }\r\n\r\n .second-page-text-box1, .second-page-text-box2, .second-page-text-box3 {\r\n width: clamp(100px, 100vw, 420px);\r\n right: 0;\r\n left: 0;\r\n position: relative;\r\n margin: 0 10px 0 10px;\r\n height: auto;\r\n }\r\n }\r\n\r\n\r\n\r\n /* Responsive Third Page */\r\n @media (max-width: 809px) {\r\n .tier-items-row {\r\n flex-wrap: wrap;\r\n margin: 0;\r\n }\r\n\r\n .text-background {\r\n margin: 5px 0 5px 0;\r\n }\r\n\r\n .sponsor-tiers {\r\n height: auto;\r\n margin: 30px 0 30px 0;\r\n }\r\n\r\n .contain-third-page span {\r\n text-align: center;\r\n }\r\n }"],"names":[],"sourceRoot":""} \ No newline at end of file diff --git a/src/X/static/js/488.4d6d0dc5.chunk.js b/src/X/static/js/488.4d6d0dc5.chunk.js new file mode 100644 index 0000000..c71988a --- /dev/null +++ b/src/X/static/js/488.4d6d0dc5.chunk.js @@ -0,0 +1,2 @@ +"use strict";(self.webpackChunkhackmercedx=self.webpackChunkhackmercedx||[]).push([[488],{107:(e,n,t)=>{t.r(n),t.d(n,{CLSThresholds:()=>P,FCPThresholds:()=>w,FIDThresholds:()=>ne,INPThresholds:()=>_,LCPThresholds:()=>G,TTFBThresholds:()=>Q,onCLS:()=>k,onFCP:()=>I,onFID:()=>te,onINP:()=>z,onLCP:()=>K,onTTFB:()=>V});var r,i,o,a,c,u=-1,s=function(e){addEventListener("pageshow",(function(n){n.persisted&&(u=n.timeStamp,e(n))}),!0)},f=function(){var e=self.performance&&performance.getEntriesByType&&performance.getEntriesByType("navigation")[0];if(e&&e.responseStart>0&&e.responseStart=0?r="back-forward-cache":t&&(document.prerendering||d()>0?r="prerender":document.wasDiscarded?r="restore":t.type&&(r=t.type.replace(/_/g,"-"))),{name:e,value:void 0===n?-1:n,rating:"good",delta:0,entries:[],id:"v4-".concat(Date.now(),"-").concat(Math.floor(8999999999999*Math.random())+1e12),navigationType:r}},p=function(e,n,t){try{if(PerformanceObserver.supportedEntryTypes.includes(e)){var r=new PerformanceObserver((function(e){Promise.resolve().then((function(){n(e.getEntries())}))}));return r.observe(Object.assign({type:e,buffered:!0},t||{})),r}}catch(e){}},v=function(e,n,t,r){var i,o;return function(a){n.value>=0&&(a||r)&&((o=n.value-(i||0))||void 0===i)&&(i=n.value,n.delta=o,n.rating=function(e,n){return e>n[1]?"poor":e>n[0]?"needs-improvement":"good"}(n.value,t),e(n))}},m=function(e){requestAnimationFrame((function(){return requestAnimationFrame((function(){return e()}))}))},h=function(e){document.addEventListener("visibilitychange",(function(){"hidden"===document.visibilityState&&e()}))},g=function(e){var n=!1;return function(){n||(e(),n=!0)}},T=-1,y=function(){return"hidden"!==document.visibilityState||document.prerendering?1/0:0},E=function(e){"hidden"===document.visibilityState&&T>-1&&(T="visibilitychange"===e.type?e.timeStamp:0,b())},C=function(){addEventListener("visibilitychange",E,!0),addEventListener("prerenderingchange",E,!0)},b=function(){removeEventListener("visibilitychange",E,!0),removeEventListener("prerenderingchange",E,!0)},L=function(){return T<0&&(T=y(),C(),s((function(){setTimeout((function(){T=y(),C()}),0)}))),{get firstHiddenTime(){return T}}},S=function(e){document.prerendering?addEventListener("prerenderingchange",(function(){return e()}),!0):e()},w=[1800,3e3],I=function(e,n){n=n||{},S((function(){var t,r=L(),i=l("FCP"),o=p("paint",(function(e){e.forEach((function(e){"first-contentful-paint"===e.name&&(o.disconnect(),e.startTimer.value&&(r.value=i,r.entries=o,t())},c=p("layout-shift",a);c&&(t=v(e,r,P,n.reportAllChanges),h((function(){a(c.takeRecords()),t(!0)})),s((function(){i=0,r=l("CLS",0),t=v(e,r,P,n.reportAllChanges),m((function(){return t()}))})),setTimeout(t,0))})))},A=0,F=1/0,M=0,D=function(e){e.forEach((function(e){e.interactionId&&(F=Math.min(F,e.interactionId),M=Math.max(M,e.interactionId),A=M?(M-F)/7+1:0)}))},x=function(){return r?A:performance.interactionCount||0},B=function(){"interactionCount"in performance||r||(r=p("event",D,{type:"event",buffered:!0,durationThreshold:0}))},R=[],H=new Map,N=0,q=[],O=function(e){if(q.forEach((function(n){return n(e)})),e.interactionId||"first-input"===e.entryType){var n=R[R.length-1],t=H.get(e.interactionId);if(t||R.length<10||e.duration>n.latency){if(t)e.duration>t.latency?(t.entries=[e],t.latency=e.duration):e.duration===t.latency&&e.startTime===t.entries[0].startTime&&t.entries.push(e);else{var r={id:e.interactionId,latency:e.duration,entries:[e]};H.set(r.id,r),R.push(r)}R.sort((function(e,n){return n.latency-e.latency})),R.length>10&&R.splice(10).forEach((function(e){return H.delete(e.id)}))}}},j=function(e){var n=self.requestIdleCallback||self.setTimeout,t=-1;return e=g(e),"hidden"===document.visibilityState?e():(t=n(e),h(e)),t},_=[200,500],z=function(e,n){"PerformanceEventTiming"in self&&"interactionId"in PerformanceEventTiming.prototype&&(n=n||{},S((function(){var t;B();var r,i=l("INP"),o=function(e){j((function(){e.forEach(O);var n=function(){var e=Math.min(R.length-1,Math.floor((x()-N)/50));return R[e]}();n&&n.latency!==i.value&&(i.value=n.latency,i.entries=n.entries,r())}))},a=p("event",o,{durationThreshold:null!==(t=n.durationThreshold)&&void 0!==t?t:40});r=v(e,i,_,n.reportAllChanges),a&&(a.observe({type:"first-input",buffered:!0}),h((function(){o(a.takeRecords()),r(!0)})),s((function(){N=x(),R.length=0,H.clear(),i=l("INP"),r=v(e,i,_,n.reportAllChanges)})))})))},G=[2500,4e3],J={},K=function(e,n){n=n||{},S((function(){var t,r=L(),i=l("LCP"),o=function(e){n.reportAllChanges||(e=e.slice(-1)),e.forEach((function(e){e.startTime=0&&o1e12?new Date:performance.now())-e.timeStamp;"pointerdown"==e.type?function(e,n){var t=function(){Y(e,n),i()},r=function(){i()},i=function(){removeEventListener("pointerup",t,W),removeEventListener("pointercancel",r,W)};addEventListener("pointerup",t,W),addEventListener("pointercancel",r,W)}(n,e):Y(n,e)}},ee=function(e){["mousedown","keydown","touchstart","pointerdown"].forEach((function(n){return e(n,$,W)}))},ne=[100,300],te=function(e,n){n=n||{},S((function(){var t,r=L(),a=l("FID"),u=function(e){e.startTime0&&e.responseStart=0?r=\"back-forward-cache\":t&&(document.prerendering||u()>0?r=\"prerender\":document.wasDiscarded?r=\"restore\":t.type&&(r=t.type.replace(/_/g,\"-\")));return{name:e,value:void 0===n?-1:n,rating:\"good\",delta:0,entries:[],id:\"v4-\".concat(Date.now(),\"-\").concat(Math.floor(8999999999999*Math.random())+1e12),navigationType:r}},s=function(e,n,t){try{if(PerformanceObserver.supportedEntryTypes.includes(e)){var r=new PerformanceObserver((function(e){Promise.resolve().then((function(){n(e.getEntries())}))}));return r.observe(Object.assign({type:e,buffered:!0},t||{})),r}}catch(e){}},d=function(e,n,t,r){var i,o;return function(a){n.value>=0&&(a||r)&&((o=n.value-(i||0))||void 0===i)&&(i=n.value,n.delta=o,n.rating=function(e,n){return e>n[1]?\"poor\":e>n[0]?\"needs-improvement\":\"good\"}(n.value,t),e(n))}},l=function(e){requestAnimationFrame((function(){return requestAnimationFrame((function(){return e()}))}))},p=function(e){document.addEventListener(\"visibilitychange\",(function(){\"hidden\"===document.visibilityState&&e()}))},v=function(e){var n=!1;return function(){n||(e(),n=!0)}},m=-1,h=function(){return\"hidden\"!==document.visibilityState||document.prerendering?1/0:0},g=function(e){\"hidden\"===document.visibilityState&&m>-1&&(m=\"visibilitychange\"===e.type?e.timeStamp:0,T())},y=function(){addEventListener(\"visibilitychange\",g,!0),addEventListener(\"prerenderingchange\",g,!0)},T=function(){removeEventListener(\"visibilitychange\",g,!0),removeEventListener(\"prerenderingchange\",g,!0)},E=function(){return m<0&&(m=h(),y(),a((function(){setTimeout((function(){m=h(),y()}),0)}))),{get firstHiddenTime(){return m}}},C=function(e){document.prerendering?addEventListener(\"prerenderingchange\",(function(){return e()}),!0):e()},b=[1800,3e3],S=function(e,n){n=n||{},C((function(){var t,r=E(),i=f(\"FCP\"),o=s(\"paint\",(function(e){e.forEach((function(e){\"first-contentful-paint\"===e.name&&(o.disconnect(),e.startTimer.value&&(r.value=i,r.entries=o,t())},u=s(\"layout-shift\",c);u&&(t=d(e,r,L,n.reportAllChanges),p((function(){c(u.takeRecords()),t(!0)})),a((function(){i=0,r=f(\"CLS\",0),t=d(e,r,L,n.reportAllChanges),l((function(){return t()}))})),setTimeout(t,0))})))},A=0,I=1/0,P=0,M=function(e){e.forEach((function(e){e.interactionId&&(I=Math.min(I,e.interactionId),P=Math.max(P,e.interactionId),A=P?(P-I)/7+1:0)}))},k=function(){return e?A:performance.interactionCount||0},F=function(){\"interactionCount\"in performance||e||(e=s(\"event\",M,{type:\"event\",buffered:!0,durationThreshold:0}))},D=[],x=new Map,R=0,B=function(){var e=Math.min(D.length-1,Math.floor((k()-R)/50));return D[e]},H=[],q=function(e){if(H.forEach((function(n){return n(e)})),e.interactionId||\"first-input\"===e.entryType){var n=D[D.length-1],t=x.get(e.interactionId);if(t||D.length<10||e.duration>n.latency){if(t)e.duration>t.latency?(t.entries=[e],t.latency=e.duration):e.duration===t.latency&&e.startTime===t.entries[0].startTime&&t.entries.push(e);else{var r={id:e.interactionId,latency:e.duration,entries:[e]};x.set(r.id,r),D.push(r)}D.sort((function(e,n){return n.latency-e.latency})),D.length>10&&D.splice(10).forEach((function(e){return x.delete(e.id)}))}}},O=function(e){var n=self.requestIdleCallback||self.setTimeout,t=-1;return e=v(e),\"hidden\"===document.visibilityState?e():(t=n(e),p(e)),t},N=[200,500],j=function(e,n){\"PerformanceEventTiming\"in self&&\"interactionId\"in PerformanceEventTiming.prototype&&(n=n||{},C((function(){var t;F();var r,i=f(\"INP\"),o=function(e){O((function(){e.forEach(q);var n=B();n&&n.latency!==i.value&&(i.value=n.latency,i.entries=n.entries,r())}))},c=s(\"event\",o,{durationThreshold:null!==(t=n.durationThreshold)&&void 0!==t?t:40});r=d(e,i,N,n.reportAllChanges),c&&(c.observe({type:\"first-input\",buffered:!0}),p((function(){o(c.takeRecords()),r(!0)})),a((function(){R=k(),D.length=0,x.clear(),i=f(\"INP\"),r=d(e,i,N,n.reportAllChanges)})))})))},_=[2500,4e3],z={},G=function(e,n){n=n||{},C((function(){var t,r=E(),i=f(\"LCP\"),o=function(e){n.reportAllChanges||(e=e.slice(-1)),e.forEach((function(e){e.startTime=0&&t1e12?new Date:performance.now())-e.timeStamp;\"pointerdown\"==e.type?function(e,n){var t=function(){W(e,n),i()},r=function(){i()},i=function(){removeEventListener(\"pointerup\",t,U),removeEventListener(\"pointercancel\",r,U)};addEventListener(\"pointerup\",t,U),addEventListener(\"pointercancel\",r,U)}(n,e):W(n,e)}},Z=function(e){[\"mousedown\",\"keydown\",\"touchstart\",\"pointerdown\"].forEach((function(n){return e(n,Y,U)}))},$=[100,300],ee=function(e,r){r=r||{},C((function(){var o,c=E(),u=f(\"FID\"),l=function(e){e.startTime{"use strict";var e={219:(e,t,n)=>{var r=n(763),a={childContextTypes:!0,contextType:!0,contextTypes:!0,defaultProps:!0,displayName:!0,getDefaultProps:!0,getDerivedStateFromError:!0,getDerivedStateFromProps:!0,mixins:!0,propTypes:!0,type:!0},o={name:!0,length:!0,prototype:!0,caller:!0,callee:!0,arguments:!0,arity:!0},i={$$typeof:!0,compare:!0,defaultProps:!0,displayName:!0,propTypes:!0,type:!0},l={};function s(e){return r.isMemo(e)?i:l[e.$$typeof]||a}l[r.ForwardRef]={$$typeof:!0,render:!0,defaultProps:!0,displayName:!0,propTypes:!0},l[r.Memo]=i;var c=Object.defineProperty,u=Object.getOwnPropertyNames,d=Object.getOwnPropertySymbols,f=Object.getOwnPropertyDescriptor,p=Object.getPrototypeOf,h=Object.prototype;e.exports=function e(t,n,r){if("string"!==typeof n){if(h){var a=p(n);a&&a!==h&&e(t,a,r)}var i=u(n);d&&(i=i.concat(d(n)));for(var l=s(t),m=s(n),g=0;g{var n="function"===typeof Symbol&&Symbol.for,r=n?Symbol.for("react.element"):60103,a=n?Symbol.for("react.portal"):60106,o=n?Symbol.for("react.fragment"):60107,i=n?Symbol.for("react.strict_mode"):60108,l=n?Symbol.for("react.profiler"):60114,s=n?Symbol.for("react.provider"):60109,c=n?Symbol.for("react.context"):60110,u=n?Symbol.for("react.async_mode"):60111,d=n?Symbol.for("react.concurrent_mode"):60111,f=n?Symbol.for("react.forward_ref"):60112,p=n?Symbol.for("react.suspense"):60113,h=n?Symbol.for("react.suspense_list"):60120,m=n?Symbol.for("react.memo"):60115,g=n?Symbol.for("react.lazy"):60116,v=n?Symbol.for("react.block"):60121,y=n?Symbol.for("react.fundamental"):60117,b=n?Symbol.for("react.responder"):60118,x=n?Symbol.for("react.scope"):60119;function w(e){if("object"===typeof e&&null!==e){var t=e.$$typeof;switch(t){case r:switch(e=e.type){case u:case d:case o:case l:case i:case p:return e;default:switch(e=e&&e.$$typeof){case c:case f:case g:case m:case s:return e;default:return t}}case a:return t}}}function k(e){return w(e)===d}t.AsyncMode=u,t.ConcurrentMode=d,t.ContextConsumer=c,t.ContextProvider=s,t.Element=r,t.ForwardRef=f,t.Fragment=o,t.Lazy=g,t.Memo=m,t.Portal=a,t.Profiler=l,t.StrictMode=i,t.Suspense=p,t.isAsyncMode=function(e){return k(e)||w(e)===u},t.isConcurrentMode=k,t.isContextConsumer=function(e){return w(e)===c},t.isContextProvider=function(e){return w(e)===s},t.isElement=function(e){return"object"===typeof e&&null!==e&&e.$$typeof===r},t.isForwardRef=function(e){return w(e)===f},t.isFragment=function(e){return w(e)===o},t.isLazy=function(e){return w(e)===g},t.isMemo=function(e){return w(e)===m},t.isPortal=function(e){return w(e)===a},t.isProfiler=function(e){return w(e)===l},t.isStrictMode=function(e){return w(e)===i},t.isSuspense=function(e){return w(e)===p},t.isValidElementType=function(e){return"string"===typeof e||"function"===typeof e||e===o||e===d||e===l||e===i||e===p||e===h||"object"===typeof e&&null!==e&&(e.$$typeof===g||e.$$typeof===m||e.$$typeof===s||e.$$typeof===c||e.$$typeof===f||e.$$typeof===y||e.$$typeof===b||e.$$typeof===x||e.$$typeof===v)},t.typeOf=w},763:(e,t,n)=>{e.exports=n(983)},730:(e,t,n)=>{var r=n(43),a=n(853);function o(e){for(var t="https://reactjs.org/docs/error-decoder.html?invariant="+e,n=1;n attribute.\n *\n * @param to - The destination URL\n */\n createHref(to: To): string;\n\n /**\n * Returns a URL for the given `to` value\n *\n * @param to - The destination URL\n */\n createURL(to: To): URL;\n\n /**\n * Encode a location the same way window.history would do (no-op for memory\n * history) so we ensure our PUSH/REPLACE navigations for data routers\n * behave the same as POP\n *\n * @param to Unencoded path\n */\n encodeLocation(to: To): Path;\n\n /**\n * Pushes a new location onto the history stack, increasing its length by one.\n * If there were any entries in the stack after the current one, they are\n * lost.\n *\n * @param to - The new URL\n * @param state - Data to associate with the new location\n */\n push(to: To, state?: any): void;\n\n /**\n * Replaces the current location in the history stack with a new one. The\n * location that was replaced will no longer be available.\n *\n * @param to - The new URL\n * @param state - Data to associate with the new location\n */\n replace(to: To, state?: any): void;\n\n /**\n * Navigates `n` entries backward/forward in the history stack relative to the\n * current index. For example, a \"back\" navigation would use go(-1).\n *\n * @param delta - The delta in the stack index\n */\n go(delta: number): void;\n\n /**\n * Sets up a listener that will be called whenever the current location\n * changes.\n *\n * @param listener - A function that will be called when the location changes\n * @returns unlisten - A function that may be used to stop listening\n */\n listen(listener: Listener): () => void;\n}\n\ntype HistoryState = {\n usr: any;\n key?: string;\n idx: number;\n};\n\nconst PopStateEventType = \"popstate\";\n//#endregion\n\n////////////////////////////////////////////////////////////////////////////////\n//#region Memory History\n////////////////////////////////////////////////////////////////////////////////\n\n/**\n * A user-supplied object that describes a location. Used when providing\n * entries to `createMemoryHistory` via its `initialEntries` option.\n */\nexport type InitialEntry = string | Partial;\n\nexport type MemoryHistoryOptions = {\n initialEntries?: InitialEntry[];\n initialIndex?: number;\n v5Compat?: boolean;\n};\n\n/**\n * A memory history stores locations in memory. This is useful in stateful\n * environments where there is no web browser, such as node tests or React\n * Native.\n */\nexport interface MemoryHistory extends History {\n /**\n * The current index in the history stack.\n */\n readonly index: number;\n}\n\n/**\n * Memory history stores the current location in memory. It is designed for use\n * in stateful non-browser environments like tests and React Native.\n */\nexport function createMemoryHistory(\n options: MemoryHistoryOptions = {}\n): MemoryHistory {\n let { initialEntries = [\"/\"], initialIndex, v5Compat = false } = options;\n let entries: Location[]; // Declare so we can access from createMemoryLocation\n entries = initialEntries.map((entry, index) =>\n createMemoryLocation(\n entry,\n typeof entry === \"string\" ? null : entry.state,\n index === 0 ? \"default\" : undefined\n )\n );\n let index = clampIndex(\n initialIndex == null ? entries.length - 1 : initialIndex\n );\n let action = Action.Pop;\n let listener: Listener | null = null;\n\n function clampIndex(n: number): number {\n return Math.min(Math.max(n, 0), entries.length - 1);\n }\n function getCurrentLocation(): Location {\n return entries[index];\n }\n function createMemoryLocation(\n to: To,\n state: any = null,\n key?: string\n ): Location {\n let location = createLocation(\n entries ? getCurrentLocation().pathname : \"/\",\n to,\n state,\n key\n );\n warning(\n location.pathname.charAt(0) === \"/\",\n `relative pathnames are not supported in memory history: ${JSON.stringify(\n to\n )}`\n );\n return location;\n }\n\n function createHref(to: To) {\n return typeof to === \"string\" ? to : createPath(to);\n }\n\n let history: MemoryHistory = {\n get index() {\n return index;\n },\n get action() {\n return action;\n },\n get location() {\n return getCurrentLocation();\n },\n createHref,\n createURL(to) {\n return new URL(createHref(to), \"http://localhost\");\n },\n encodeLocation(to: To) {\n let path = typeof to === \"string\" ? parsePath(to) : to;\n return {\n pathname: path.pathname || \"\",\n search: path.search || \"\",\n hash: path.hash || \"\",\n };\n },\n push(to, state) {\n action = Action.Push;\n let nextLocation = createMemoryLocation(to, state);\n index += 1;\n entries.splice(index, entries.length, nextLocation);\n if (v5Compat && listener) {\n listener({ action, location: nextLocation, delta: 1 });\n }\n },\n replace(to, state) {\n action = Action.Replace;\n let nextLocation = createMemoryLocation(to, state);\n entries[index] = nextLocation;\n if (v5Compat && listener) {\n listener({ action, location: nextLocation, delta: 0 });\n }\n },\n go(delta) {\n action = Action.Pop;\n let nextIndex = clampIndex(index + delta);\n let nextLocation = entries[nextIndex];\n index = nextIndex;\n if (listener) {\n listener({ action, location: nextLocation, delta });\n }\n },\n listen(fn: Listener) {\n listener = fn;\n return () => {\n listener = null;\n };\n },\n };\n\n return history;\n}\n//#endregion\n\n////////////////////////////////////////////////////////////////////////////////\n//#region Browser History\n////////////////////////////////////////////////////////////////////////////////\n\n/**\n * A browser history stores the current location in regular URLs in a web\n * browser environment. This is the standard for most web apps and provides the\n * cleanest URLs the browser's address bar.\n *\n * @see https://github.com/remix-run/history/tree/main/docs/api-reference.md#browserhistory\n */\nexport interface BrowserHistory extends UrlHistory {}\n\nexport type BrowserHistoryOptions = UrlHistoryOptions;\n\n/**\n * Browser history stores the location in regular URLs. This is the standard for\n * most web apps, but it requires some configuration on the server to ensure you\n * serve the same app at multiple URLs.\n *\n * @see https://github.com/remix-run/history/tree/main/docs/api-reference.md#createbrowserhistory\n */\nexport function createBrowserHistory(\n options: BrowserHistoryOptions = {}\n): BrowserHistory {\n function createBrowserLocation(\n window: Window,\n globalHistory: Window[\"history\"]\n ) {\n let { pathname, search, hash } = window.location;\n return createLocation(\n \"\",\n { pathname, search, hash },\n // state defaults to `null` because `window.history.state` does\n (globalHistory.state && globalHistory.state.usr) || null,\n (globalHistory.state && globalHistory.state.key) || \"default\"\n );\n }\n\n function createBrowserHref(window: Window, to: To) {\n return typeof to === \"string\" ? to : createPath(to);\n }\n\n return getUrlBasedHistory(\n createBrowserLocation,\n createBrowserHref,\n null,\n options\n );\n}\n//#endregion\n\n////////////////////////////////////////////////////////////////////////////////\n//#region Hash History\n////////////////////////////////////////////////////////////////////////////////\n\n/**\n * A hash history stores the current location in the fragment identifier portion\n * of the URL in a web browser environment.\n *\n * This is ideal for apps that do not control the server for some reason\n * (because the fragment identifier is never sent to the server), including some\n * shared hosting environments that do not provide fine-grained controls over\n * which pages are served at which URLs.\n *\n * @see https://github.com/remix-run/history/tree/main/docs/api-reference.md#hashhistory\n */\nexport interface HashHistory extends UrlHistory {}\n\nexport type HashHistoryOptions = UrlHistoryOptions;\n\n/**\n * Hash history stores the location in window.location.hash. This makes it ideal\n * for situations where you don't want to send the location to the server for\n * some reason, either because you do cannot configure it or the URL space is\n * reserved for something else.\n *\n * @see https://github.com/remix-run/history/tree/main/docs/api-reference.md#createhashhistory\n */\nexport function createHashHistory(\n options: HashHistoryOptions = {}\n): HashHistory {\n function createHashLocation(\n window: Window,\n globalHistory: Window[\"history\"]\n ) {\n let {\n pathname = \"/\",\n search = \"\",\n hash = \"\",\n } = parsePath(window.location.hash.substr(1));\n\n // Hash URL should always have a leading / just like window.location.pathname\n // does, so if an app ends up at a route like /#something then we add a\n // leading slash so all of our path-matching behaves the same as if it would\n // in a browser router. This is particularly important when there exists a\n // root splat route () since that matches internally against\n // \"/*\" and we'd expect /#something to 404 in a hash router app.\n if (!pathname.startsWith(\"/\") && !pathname.startsWith(\".\")) {\n pathname = \"/\" + pathname;\n }\n\n return createLocation(\n \"\",\n { pathname, search, hash },\n // state defaults to `null` because `window.history.state` does\n (globalHistory.state && globalHistory.state.usr) || null,\n (globalHistory.state && globalHistory.state.key) || \"default\"\n );\n }\n\n function createHashHref(window: Window, to: To) {\n let base = window.document.querySelector(\"base\");\n let href = \"\";\n\n if (base && base.getAttribute(\"href\")) {\n let url = window.location.href;\n let hashIndex = url.indexOf(\"#\");\n href = hashIndex === -1 ? url : url.slice(0, hashIndex);\n }\n\n return href + \"#\" + (typeof to === \"string\" ? to : createPath(to));\n }\n\n function validateHashLocation(location: Location, to: To) {\n warning(\n location.pathname.charAt(0) === \"/\",\n `relative pathnames are not supported in hash history.push(${JSON.stringify(\n to\n )})`\n );\n }\n\n return getUrlBasedHistory(\n createHashLocation,\n createHashHref,\n validateHashLocation,\n options\n );\n}\n//#endregion\n\n////////////////////////////////////////////////////////////////////////////////\n//#region UTILS\n////////////////////////////////////////////////////////////////////////////////\n\n/**\n * @private\n */\nexport function invariant(value: boolean, message?: string): asserts value;\nexport function invariant(\n value: T | null | undefined,\n message?: string\n): asserts value is T;\nexport function invariant(value: any, message?: string) {\n if (value === false || value === null || typeof value === \"undefined\") {\n throw new Error(message);\n }\n}\n\nexport function warning(cond: any, message: string) {\n if (!cond) {\n // eslint-disable-next-line no-console\n if (typeof console !== \"undefined\") console.warn(message);\n\n try {\n // Welcome to debugging history!\n //\n // This error is thrown as a convenience, so you can more easily\n // find the source for a warning that appears in the console by\n // enabling \"pause on exceptions\" in your JavaScript debugger.\n throw new Error(message);\n // eslint-disable-next-line no-empty\n } catch (e) {}\n }\n}\n\nfunction createKey() {\n return Math.random().toString(36).substr(2, 8);\n}\n\n/**\n * For browser-based histories, we combine the state and key into an object\n */\nfunction getHistoryState(location: Location, index: number): HistoryState {\n return {\n usr: location.state,\n key: location.key,\n idx: index,\n };\n}\n\n/**\n * Creates a Location object with a unique key from the given Path\n */\nexport function createLocation(\n current: string | Location,\n to: To,\n state: any = null,\n key?: string\n): Readonly {\n let location: Readonly = {\n pathname: typeof current === \"string\" ? current : current.pathname,\n search: \"\",\n hash: \"\",\n ...(typeof to === \"string\" ? parsePath(to) : to),\n state,\n // TODO: This could be cleaned up. push/replace should probably just take\n // full Locations now and avoid the need to run through this flow at all\n // But that's a pretty big refactor to the current test suite so going to\n // keep as is for the time being and just let any incoming keys take precedence\n key: (to && (to as Location).key) || key || createKey(),\n };\n return location;\n}\n\n/**\n * Creates a string URL path from the given pathname, search, and hash components.\n */\nexport function createPath({\n pathname = \"/\",\n search = \"\",\n hash = \"\",\n}: Partial) {\n if (search && search !== \"?\")\n pathname += search.charAt(0) === \"?\" ? search : \"?\" + search;\n if (hash && hash !== \"#\")\n pathname += hash.charAt(0) === \"#\" ? hash : \"#\" + hash;\n return pathname;\n}\n\n/**\n * Parses a string URL path into its separate pathname, search, and hash components.\n */\nexport function parsePath(path: string): Partial {\n let parsedPath: Partial = {};\n\n if (path) {\n let hashIndex = path.indexOf(\"#\");\n if (hashIndex >= 0) {\n parsedPath.hash = path.substr(hashIndex);\n path = path.substr(0, hashIndex);\n }\n\n let searchIndex = path.indexOf(\"?\");\n if (searchIndex >= 0) {\n parsedPath.search = path.substr(searchIndex);\n path = path.substr(0, searchIndex);\n }\n\n if (path) {\n parsedPath.pathname = path;\n }\n }\n\n return parsedPath;\n}\n\nexport interface UrlHistory extends History {}\n\nexport type UrlHistoryOptions = {\n window?: Window;\n v5Compat?: boolean;\n};\n\nfunction getUrlBasedHistory(\n getLocation: (window: Window, globalHistory: Window[\"history\"]) => Location,\n createHref: (window: Window, to: To) => string,\n validateLocation: ((location: Location, to: To) => void) | null,\n options: UrlHistoryOptions = {}\n): UrlHistory {\n let { window = document.defaultView!, v5Compat = false } = options;\n let globalHistory = window.history;\n let action = Action.Pop;\n let listener: Listener | null = null;\n\n let index = getIndex()!;\n // Index should only be null when we initialize. If not, it's because the\n // user called history.pushState or history.replaceState directly, in which\n // case we should log a warning as it will result in bugs.\n if (index == null) {\n index = 0;\n globalHistory.replaceState({ ...globalHistory.state, idx: index }, \"\");\n }\n\n function getIndex(): number {\n let state = globalHistory.state || { idx: null };\n return state.idx;\n }\n\n function handlePop() {\n action = Action.Pop;\n let nextIndex = getIndex();\n let delta = nextIndex == null ? null : nextIndex - index;\n index = nextIndex;\n if (listener) {\n listener({ action, location: history.location, delta });\n }\n }\n\n function push(to: To, state?: any) {\n action = Action.Push;\n let location = createLocation(history.location, to, state);\n if (validateLocation) validateLocation(location, to);\n\n index = getIndex() + 1;\n let historyState = getHistoryState(location, index);\n let url = history.createHref(location);\n\n // try...catch because iOS limits us to 100 pushState calls :/\n try {\n globalHistory.pushState(historyState, \"\", url);\n } catch (error) {\n // If the exception is because `state` can't be serialized, let that throw\n // outwards just like a replace call would so the dev knows the cause\n // https://html.spec.whatwg.org/multipage/nav-history-apis.html#shared-history-push/replace-state-steps\n // https://html.spec.whatwg.org/multipage/structured-data.html#structuredserializeinternal\n if (error instanceof DOMException && error.name === \"DataCloneError\") {\n throw error;\n }\n // They are going to lose state here, but there is no real\n // way to warn them about it since the page will refresh...\n window.location.assign(url);\n }\n\n if (v5Compat && listener) {\n listener({ action, location: history.location, delta: 1 });\n }\n }\n\n function replace(to: To, state?: any) {\n action = Action.Replace;\n let location = createLocation(history.location, to, state);\n if (validateLocation) validateLocation(location, to);\n\n index = getIndex();\n let historyState = getHistoryState(location, index);\n let url = history.createHref(location);\n globalHistory.replaceState(historyState, \"\", url);\n\n if (v5Compat && listener) {\n listener({ action, location: history.location, delta: 0 });\n }\n }\n\n function createURL(to: To): URL {\n // window.location.origin is \"null\" (the literal string value) in Firefox\n // under certain conditions, notably when serving from a local HTML file\n // See https://bugzilla.mozilla.org/show_bug.cgi?id=878297\n let base =\n window.location.origin !== \"null\"\n ? window.location.origin\n : window.location.href;\n\n let href = typeof to === \"string\" ? to : createPath(to);\n // Treating this as a full URL will strip any trailing spaces so we need to\n // pre-encode them since they might be part of a matching splat param from\n // an ancestor route\n href = href.replace(/ $/, \"%20\");\n invariant(\n base,\n `No window.location.(origin|href) available to create URL for href: ${href}`\n );\n return new URL(href, base);\n }\n\n let history: History = {\n get action() {\n return action;\n },\n get location() {\n return getLocation(window, globalHistory);\n },\n listen(fn: Listener) {\n if (listener) {\n throw new Error(\"A history only accepts one active listener\");\n }\n window.addEventListener(PopStateEventType, handlePop);\n listener = fn;\n\n return () => {\n window.removeEventListener(PopStateEventType, handlePop);\n listener = null;\n };\n },\n createHref(to) {\n return createHref(window, to);\n },\n createURL,\n encodeLocation(to) {\n // Encode a Location the same way window.location would\n let url = createURL(to);\n return {\n pathname: url.pathname,\n search: url.search,\n hash: url.hash,\n };\n },\n push,\n replace,\n go(n) {\n return globalHistory.go(n);\n },\n };\n\n return history;\n}\n\n//#endregion\n","import type { Location, Path, To } from \"./history\";\nimport { invariant, parsePath, warning } from \"./history\";\n\n/**\n * Map of routeId -> data returned from a loader/action/error\n */\nexport interface RouteData {\n [routeId: string]: any;\n}\n\nexport enum ResultType {\n data = \"data\",\n deferred = \"deferred\",\n redirect = \"redirect\",\n error = \"error\",\n}\n\n/**\n * Successful result from a loader or action\n */\nexport interface SuccessResult {\n type: ResultType.data;\n data: unknown;\n statusCode?: number;\n headers?: Headers;\n}\n\n/**\n * Successful defer() result from a loader or action\n */\nexport interface DeferredResult {\n type: ResultType.deferred;\n deferredData: DeferredData;\n statusCode?: number;\n headers?: Headers;\n}\n\n/**\n * Redirect result from a loader or action\n */\nexport interface RedirectResult {\n type: ResultType.redirect;\n // We keep the raw Response for redirects so we can return it verbatim\n response: Response;\n}\n\n/**\n * Unsuccessful result from a loader or action\n */\nexport interface ErrorResult {\n type: ResultType.error;\n error: unknown;\n statusCode?: number;\n headers?: Headers;\n}\n\n/**\n * Result from a loader or action - potentially successful or unsuccessful\n */\nexport type DataResult =\n | SuccessResult\n | DeferredResult\n | RedirectResult\n | ErrorResult;\n\ntype LowerCaseFormMethod = \"get\" | \"post\" | \"put\" | \"patch\" | \"delete\";\ntype UpperCaseFormMethod = Uppercase;\n\n/**\n * Users can specify either lowercase or uppercase form methods on `
`,\n * useSubmit(), ``, etc.\n */\nexport type HTMLFormMethod = LowerCaseFormMethod | UpperCaseFormMethod;\n\n/**\n * Active navigation/fetcher form methods are exposed in lowercase on the\n * RouterState\n */\nexport type FormMethod = LowerCaseFormMethod;\nexport type MutationFormMethod = Exclude;\n\n/**\n * In v7, active navigation/fetcher form methods are exposed in uppercase on the\n * RouterState. This is to align with the normalization done via fetch().\n */\nexport type V7_FormMethod = UpperCaseFormMethod;\nexport type V7_MutationFormMethod = Exclude;\n\nexport type FormEncType =\n | \"application/x-www-form-urlencoded\"\n | \"multipart/form-data\"\n | \"application/json\"\n | \"text/plain\";\n\n// Thanks https://github.com/sindresorhus/type-fest!\ntype JsonObject = { [Key in string]: JsonValue } & {\n [Key in string]?: JsonValue | undefined;\n};\ntype JsonArray = JsonValue[] | readonly JsonValue[];\ntype JsonPrimitive = string | number | boolean | null;\ntype JsonValue = JsonPrimitive | JsonObject | JsonArray;\n\n/**\n * @private\n * Internal interface to pass around for action submissions, not intended for\n * external consumption\n */\nexport type Submission =\n | {\n formMethod: FormMethod | V7_FormMethod;\n formAction: string;\n formEncType: FormEncType;\n formData: FormData;\n json: undefined;\n text: undefined;\n }\n | {\n formMethod: FormMethod | V7_FormMethod;\n formAction: string;\n formEncType: FormEncType;\n formData: undefined;\n json: JsonValue;\n text: undefined;\n }\n | {\n formMethod: FormMethod | V7_FormMethod;\n formAction: string;\n formEncType: FormEncType;\n formData: undefined;\n json: undefined;\n text: string;\n };\n\n/**\n * @private\n * Arguments passed to route loader/action functions. Same for now but we keep\n * this as a private implementation detail in case they diverge in the future.\n */\ninterface DataFunctionArgs {\n request: Request;\n params: Params;\n context?: Context;\n}\n\n// TODO: (v7) Change the defaults from any to unknown in and remove Remix wrappers:\n// ActionFunction, ActionFunctionArgs, LoaderFunction, LoaderFunctionArgs\n// Also, make them a type alias instead of an interface\n\n/**\n * Arguments passed to loader functions\n */\nexport interface LoaderFunctionArgs\n extends DataFunctionArgs {}\n\n/**\n * Arguments passed to action functions\n */\nexport interface ActionFunctionArgs\n extends DataFunctionArgs {}\n\n/**\n * Loaders and actions can return anything except `undefined` (`null` is a\n * valid return value if there is no data to return). Responses are preferred\n * and will ease any future migration to Remix\n */\ntype DataFunctionValue = Response | NonNullable | null;\n\ntype DataFunctionReturnValue = Promise | DataFunctionValue;\n\n/**\n * Route loader function signature\n */\nexport type LoaderFunction = {\n (\n args: LoaderFunctionArgs,\n handlerCtx?: unknown\n ): DataFunctionReturnValue;\n} & { hydrate?: boolean };\n\n/**\n * Route action function signature\n */\nexport interface ActionFunction {\n (\n args: ActionFunctionArgs,\n handlerCtx?: unknown\n ): DataFunctionReturnValue;\n}\n\n/**\n * Arguments passed to shouldRevalidate function\n */\nexport interface ShouldRevalidateFunctionArgs {\n currentUrl: URL;\n currentParams: AgnosticDataRouteMatch[\"params\"];\n nextUrl: URL;\n nextParams: AgnosticDataRouteMatch[\"params\"];\n formMethod?: Submission[\"formMethod\"];\n formAction?: Submission[\"formAction\"];\n formEncType?: Submission[\"formEncType\"];\n text?: Submission[\"text\"];\n formData?: Submission[\"formData\"];\n json?: Submission[\"json\"];\n actionStatus?: number;\n actionResult?: any;\n defaultShouldRevalidate: boolean;\n}\n\n/**\n * Route shouldRevalidate function signature. This runs after any submission\n * (navigation or fetcher), so we flatten the navigation/fetcher submission\n * onto the arguments. It shouldn't matter whether it came from a navigation\n * or a fetcher, what really matters is the URLs and the formData since loaders\n * have to re-run based on the data models that were potentially mutated.\n */\nexport interface ShouldRevalidateFunction {\n (args: ShouldRevalidateFunctionArgs): boolean;\n}\n\n/**\n * Function provided by the framework-aware layers to set `hasErrorBoundary`\n * from the framework-aware `errorElement` prop\n *\n * @deprecated Use `mapRouteProperties` instead\n */\nexport interface DetectErrorBoundaryFunction {\n (route: AgnosticRouteObject): boolean;\n}\n\nexport interface DataStrategyMatch\n extends AgnosticRouteMatch {\n shouldLoad: boolean;\n resolve: (\n handlerOverride?: (\n handler: (ctx?: unknown) => DataFunctionReturnValue\n ) => DataFunctionReturnValue\n ) => Promise;\n}\n\nexport interface DataStrategyFunctionArgs\n extends DataFunctionArgs {\n matches: DataStrategyMatch[];\n fetcherKey: string | null;\n}\n\n/**\n * Result from a loader or action called via dataStrategy\n */\nexport interface DataStrategyResult {\n type: \"data\" | \"error\";\n result: unknown; // data, Error, Response, DeferredData, DataWithResponseInit\n}\n\nexport interface DataStrategyFunction {\n (args: DataStrategyFunctionArgs): Promise>;\n}\n\nexport type AgnosticPatchRoutesOnNavigationFunctionArgs<\n O extends AgnosticRouteObject = AgnosticRouteObject,\n M extends AgnosticRouteMatch = AgnosticRouteMatch\n> = {\n signal: AbortSignal;\n path: string;\n matches: M[];\n patch: (routeId: string | null, children: O[]) => void;\n};\n\nexport type AgnosticPatchRoutesOnNavigationFunction<\n O extends AgnosticRouteObject = AgnosticRouteObject,\n M extends AgnosticRouteMatch = AgnosticRouteMatch\n> = (\n opts: AgnosticPatchRoutesOnNavigationFunctionArgs\n) => void | Promise;\n\n/**\n * Function provided by the framework-aware layers to set any framework-specific\n * properties from framework-agnostic properties\n */\nexport interface MapRoutePropertiesFunction {\n (route: AgnosticRouteObject): {\n hasErrorBoundary: boolean;\n } & Record;\n}\n\n/**\n * Keys we cannot change from within a lazy() function. We spread all other keys\n * onto the route. Either they're meaningful to the router, or they'll get\n * ignored.\n */\nexport type ImmutableRouteKey =\n | \"lazy\"\n | \"caseSensitive\"\n | \"path\"\n | \"id\"\n | \"index\"\n | \"children\";\n\nexport const immutableRouteKeys = new Set([\n \"lazy\",\n \"caseSensitive\",\n \"path\",\n \"id\",\n \"index\",\n \"children\",\n]);\n\ntype RequireOne = Exclude<\n {\n [K in keyof T]: K extends Key ? Omit & Required> : never;\n }[keyof T],\n undefined\n>;\n\n/**\n * lazy() function to load a route definition, which can add non-matching\n * related properties to a route\n */\nexport interface LazyRouteFunction {\n (): Promise>>;\n}\n\n/**\n * Base RouteObject with common props shared by all types of routes\n */\ntype AgnosticBaseRouteObject = {\n caseSensitive?: boolean;\n path?: string;\n id?: string;\n loader?: LoaderFunction | boolean;\n action?: ActionFunction | boolean;\n hasErrorBoundary?: boolean;\n shouldRevalidate?: ShouldRevalidateFunction;\n handle?: any;\n lazy?: LazyRouteFunction;\n};\n\n/**\n * Index routes must not have children\n */\nexport type AgnosticIndexRouteObject = AgnosticBaseRouteObject & {\n children?: undefined;\n index: true;\n};\n\n/**\n * Non-index routes may have children, but cannot have index\n */\nexport type AgnosticNonIndexRouteObject = AgnosticBaseRouteObject & {\n children?: AgnosticRouteObject[];\n index?: false;\n};\n\n/**\n * A route object represents a logical route, with (optionally) its child\n * routes organized in a tree-like structure.\n */\nexport type AgnosticRouteObject =\n | AgnosticIndexRouteObject\n | AgnosticNonIndexRouteObject;\n\nexport type AgnosticDataIndexRouteObject = AgnosticIndexRouteObject & {\n id: string;\n};\n\nexport type AgnosticDataNonIndexRouteObject = AgnosticNonIndexRouteObject & {\n children?: AgnosticDataRouteObject[];\n id: string;\n};\n\n/**\n * A data route object, which is just a RouteObject with a required unique ID\n */\nexport type AgnosticDataRouteObject =\n | AgnosticDataIndexRouteObject\n | AgnosticDataNonIndexRouteObject;\n\nexport type RouteManifest = Record;\n\n// Recursive helper for finding path parameters in the absence of wildcards\ntype _PathParam =\n // split path into individual path segments\n Path extends `${infer L}/${infer R}`\n ? _PathParam | _PathParam\n : // find params after `:`\n Path extends `:${infer Param}`\n ? Param extends `${infer Optional}?`\n ? Optional\n : Param\n : // otherwise, there aren't any params present\n never;\n\n/**\n * Examples:\n * \"/a/b/*\" -> \"*\"\n * \":a\" -> \"a\"\n * \"/a/:b\" -> \"b\"\n * \"/a/blahblahblah:b\" -> \"b\"\n * \"/:a/:b\" -> \"a\" | \"b\"\n * \"/:a/b/:c/*\" -> \"a\" | \"c\" | \"*\"\n */\nexport type PathParam =\n // check if path is just a wildcard\n Path extends \"*\" | \"/*\"\n ? \"*\"\n : // look for wildcard at the end of the path\n Path extends `${infer Rest}/*`\n ? \"*\" | _PathParam\n : // look for params in the absence of wildcards\n _PathParam;\n\n// Attempt to parse the given string segment. If it fails, then just return the\n// plain string type as a default fallback. Otherwise, return the union of the\n// parsed string literals that were referenced as dynamic segments in the route.\nexport type ParamParseKey =\n // if you could not find path params, fallback to `string`\n [PathParam] extends [never] ? string : PathParam;\n\n/**\n * The parameters that were parsed from the URL path.\n */\nexport type Params = {\n readonly [key in Key]: string | undefined;\n};\n\n/**\n * A RouteMatch contains info about how a route matched a URL.\n */\nexport interface AgnosticRouteMatch<\n ParamKey extends string = string,\n RouteObjectType extends AgnosticRouteObject = AgnosticRouteObject\n> {\n /**\n * The names and values of dynamic parameters in the URL.\n */\n params: Params;\n /**\n * The portion of the URL pathname that was matched.\n */\n pathname: string;\n /**\n * The portion of the URL pathname that was matched before child routes.\n */\n pathnameBase: string;\n /**\n * The route object that was used to match.\n */\n route: RouteObjectType;\n}\n\nexport interface AgnosticDataRouteMatch\n extends AgnosticRouteMatch {}\n\nfunction isIndexRoute(\n route: AgnosticRouteObject\n): route is AgnosticIndexRouteObject {\n return route.index === true;\n}\n\n// Walk the route tree generating unique IDs where necessary, so we are working\n// solely with AgnosticDataRouteObject's within the Router\nexport function convertRoutesToDataRoutes(\n routes: AgnosticRouteObject[],\n mapRouteProperties: MapRoutePropertiesFunction,\n parentPath: string[] = [],\n manifest: RouteManifest = {}\n): AgnosticDataRouteObject[] {\n return routes.map((route, index) => {\n let treePath = [...parentPath, String(index)];\n let id = typeof route.id === \"string\" ? route.id : treePath.join(\"-\");\n invariant(\n route.index !== true || !route.children,\n `Cannot specify children on an index route`\n );\n invariant(\n !manifest[id],\n `Found a route id collision on id \"${id}\". Route ` +\n \"id's must be globally unique within Data Router usages\"\n );\n\n if (isIndexRoute(route)) {\n let indexRoute: AgnosticDataIndexRouteObject = {\n ...route,\n ...mapRouteProperties(route),\n id,\n };\n manifest[id] = indexRoute;\n return indexRoute;\n } else {\n let pathOrLayoutRoute: AgnosticDataNonIndexRouteObject = {\n ...route,\n ...mapRouteProperties(route),\n id,\n children: undefined,\n };\n manifest[id] = pathOrLayoutRoute;\n\n if (route.children) {\n pathOrLayoutRoute.children = convertRoutesToDataRoutes(\n route.children,\n mapRouteProperties,\n treePath,\n manifest\n );\n }\n\n return pathOrLayoutRoute;\n }\n });\n}\n\n/**\n * Matches the given routes to a location and returns the match data.\n *\n * @see https://reactrouter.com/v6/utils/match-routes\n */\nexport function matchRoutes<\n RouteObjectType extends AgnosticRouteObject = AgnosticRouteObject\n>(\n routes: RouteObjectType[],\n locationArg: Partial | string,\n basename = \"/\"\n): AgnosticRouteMatch[] | null {\n return matchRoutesImpl(routes, locationArg, basename, false);\n}\n\nexport function matchRoutesImpl<\n RouteObjectType extends AgnosticRouteObject = AgnosticRouteObject\n>(\n routes: RouteObjectType[],\n locationArg: Partial | string,\n basename: string,\n allowPartial: boolean\n): AgnosticRouteMatch[] | null {\n let location =\n typeof locationArg === \"string\" ? parsePath(locationArg) : locationArg;\n\n let pathname = stripBasename(location.pathname || \"/\", basename);\n\n if (pathname == null) {\n return null;\n }\n\n let branches = flattenRoutes(routes);\n rankRouteBranches(branches);\n\n let matches = null;\n for (let i = 0; matches == null && i < branches.length; ++i) {\n // Incoming pathnames are generally encoded from either window.location\n // or from router.navigate, but we want to match against the unencoded\n // paths in the route definitions. Memory router locations won't be\n // encoded here but there also shouldn't be anything to decode so this\n // should be a safe operation. This avoids needing matchRoutes to be\n // history-aware.\n let decoded = decodePath(pathname);\n matches = matchRouteBranch(\n branches[i],\n decoded,\n allowPartial\n );\n }\n\n return matches;\n}\n\nexport interface UIMatch {\n id: string;\n pathname: string;\n params: AgnosticRouteMatch[\"params\"];\n data: Data;\n handle: Handle;\n}\n\nexport function convertRouteMatchToUiMatch(\n match: AgnosticDataRouteMatch,\n loaderData: RouteData\n): UIMatch {\n let { route, pathname, params } = match;\n return {\n id: route.id,\n pathname,\n params,\n data: loaderData[route.id],\n handle: route.handle,\n };\n}\n\ninterface RouteMeta<\n RouteObjectType extends AgnosticRouteObject = AgnosticRouteObject\n> {\n relativePath: string;\n caseSensitive: boolean;\n childrenIndex: number;\n route: RouteObjectType;\n}\n\ninterface RouteBranch<\n RouteObjectType extends AgnosticRouteObject = AgnosticRouteObject\n> {\n path: string;\n score: number;\n routesMeta: RouteMeta[];\n}\n\nfunction flattenRoutes<\n RouteObjectType extends AgnosticRouteObject = AgnosticRouteObject\n>(\n routes: RouteObjectType[],\n branches: RouteBranch[] = [],\n parentsMeta: RouteMeta[] = [],\n parentPath = \"\"\n): RouteBranch[] {\n let flattenRoute = (\n route: RouteObjectType,\n index: number,\n relativePath?: string\n ) => {\n let meta: RouteMeta = {\n relativePath:\n relativePath === undefined ? route.path || \"\" : relativePath,\n caseSensitive: route.caseSensitive === true,\n childrenIndex: index,\n route,\n };\n\n if (meta.relativePath.startsWith(\"/\")) {\n invariant(\n meta.relativePath.startsWith(parentPath),\n `Absolute route path \"${meta.relativePath}\" nested under path ` +\n `\"${parentPath}\" is not valid. An absolute child route path ` +\n `must start with the combined path of all its parent routes.`\n );\n\n meta.relativePath = meta.relativePath.slice(parentPath.length);\n }\n\n let path = joinPaths([parentPath, meta.relativePath]);\n let routesMeta = parentsMeta.concat(meta);\n\n // Add the children before adding this route to the array, so we traverse the\n // route tree depth-first and child routes appear before their parents in\n // the \"flattened\" version.\n if (route.children && route.children.length > 0) {\n invariant(\n // Our types know better, but runtime JS may not!\n // @ts-expect-error\n route.index !== true,\n `Index routes must not have child routes. Please remove ` +\n `all child routes from route path \"${path}\".`\n );\n flattenRoutes(route.children, branches, routesMeta, path);\n }\n\n // Routes without a path shouldn't ever match by themselves unless they are\n // index routes, so don't add them to the list of possible branches.\n if (route.path == null && !route.index) {\n return;\n }\n\n branches.push({\n path,\n score: computeScore(path, route.index),\n routesMeta,\n });\n };\n routes.forEach((route, index) => {\n // coarse-grain check for optional params\n if (route.path === \"\" || !route.path?.includes(\"?\")) {\n flattenRoute(route, index);\n } else {\n for (let exploded of explodeOptionalSegments(route.path)) {\n flattenRoute(route, index, exploded);\n }\n }\n });\n\n return branches;\n}\n\n/**\n * Computes all combinations of optional path segments for a given path,\n * excluding combinations that are ambiguous and of lower priority.\n *\n * For example, `/one/:two?/three/:four?/:five?` explodes to:\n * - `/one/three`\n * - `/one/:two/three`\n * - `/one/three/:four`\n * - `/one/three/:five`\n * - `/one/:two/three/:four`\n * - `/one/:two/three/:five`\n * - `/one/three/:four/:five`\n * - `/one/:two/three/:four/:five`\n */\nfunction explodeOptionalSegments(path: string): string[] {\n let segments = path.split(\"/\");\n if (segments.length === 0) return [];\n\n let [first, ...rest] = segments;\n\n // Optional path segments are denoted by a trailing `?`\n let isOptional = first.endsWith(\"?\");\n // Compute the corresponding required segment: `foo?` -> `foo`\n let required = first.replace(/\\?$/, \"\");\n\n if (rest.length === 0) {\n // Intepret empty string as omitting an optional segment\n // `[\"one\", \"\", \"three\"]` corresponds to omitting `:two` from `/one/:two?/three` -> `/one/three`\n return isOptional ? [required, \"\"] : [required];\n }\n\n let restExploded = explodeOptionalSegments(rest.join(\"/\"));\n\n let result: string[] = [];\n\n // All child paths with the prefix. Do this for all children before the\n // optional version for all children, so we get consistent ordering where the\n // parent optional aspect is preferred as required. Otherwise, we can get\n // child sections interspersed where deeper optional segments are higher than\n // parent optional segments, where for example, /:two would explode _earlier_\n // then /:one. By always including the parent as required _for all children_\n // first, we avoid this issue\n result.push(\n ...restExploded.map((subpath) =>\n subpath === \"\" ? required : [required, subpath].join(\"/\")\n )\n );\n\n // Then, if this is an optional value, add all child versions without\n if (isOptional) {\n result.push(...restExploded);\n }\n\n // for absolute paths, ensure `/` instead of empty segment\n return result.map((exploded) =>\n path.startsWith(\"/\") && exploded === \"\" ? \"/\" : exploded\n );\n}\n\nfunction rankRouteBranches(branches: RouteBranch[]): void {\n branches.sort((a, b) =>\n a.score !== b.score\n ? b.score - a.score // Higher score first\n : compareIndexes(\n a.routesMeta.map((meta) => meta.childrenIndex),\n b.routesMeta.map((meta) => meta.childrenIndex)\n )\n );\n}\n\nconst paramRe = /^:[\\w-]+$/;\nconst dynamicSegmentValue = 3;\nconst indexRouteValue = 2;\nconst emptySegmentValue = 1;\nconst staticSegmentValue = 10;\nconst splatPenalty = -2;\nconst isSplat = (s: string) => s === \"*\";\n\nfunction computeScore(path: string, index: boolean | undefined): number {\n let segments = path.split(\"/\");\n let initialScore = segments.length;\n if (segments.some(isSplat)) {\n initialScore += splatPenalty;\n }\n\n if (index) {\n initialScore += indexRouteValue;\n }\n\n return segments\n .filter((s) => !isSplat(s))\n .reduce(\n (score, segment) =>\n score +\n (paramRe.test(segment)\n ? dynamicSegmentValue\n : segment === \"\"\n ? emptySegmentValue\n : staticSegmentValue),\n initialScore\n );\n}\n\nfunction compareIndexes(a: number[], b: number[]): number {\n let siblings =\n a.length === b.length && a.slice(0, -1).every((n, i) => n === b[i]);\n\n return siblings\n ? // If two routes are siblings, we should try to match the earlier sibling\n // first. This allows people to have fine-grained control over the matching\n // behavior by simply putting routes with identical paths in the order they\n // want them tried.\n a[a.length - 1] - b[b.length - 1]\n : // Otherwise, it doesn't really make sense to rank non-siblings by index,\n // so they sort equally.\n 0;\n}\n\nfunction matchRouteBranch<\n ParamKey extends string = string,\n RouteObjectType extends AgnosticRouteObject = AgnosticRouteObject\n>(\n branch: RouteBranch,\n pathname: string,\n allowPartial = false\n): AgnosticRouteMatch[] | null {\n let { routesMeta } = branch;\n\n let matchedParams = {};\n let matchedPathname = \"/\";\n let matches: AgnosticRouteMatch[] = [];\n for (let i = 0; i < routesMeta.length; ++i) {\n let meta = routesMeta[i];\n let end = i === routesMeta.length - 1;\n let remainingPathname =\n matchedPathname === \"/\"\n ? pathname\n : pathname.slice(matchedPathname.length) || \"/\";\n let match = matchPath(\n { path: meta.relativePath, caseSensitive: meta.caseSensitive, end },\n remainingPathname\n );\n\n let route = meta.route;\n\n if (\n !match &&\n end &&\n allowPartial &&\n !routesMeta[routesMeta.length - 1].route.index\n ) {\n match = matchPath(\n {\n path: meta.relativePath,\n caseSensitive: meta.caseSensitive,\n end: false,\n },\n remainingPathname\n );\n }\n\n if (!match) {\n return null;\n }\n\n Object.assign(matchedParams, match.params);\n\n matches.push({\n // TODO: Can this as be avoided?\n params: matchedParams as Params,\n pathname: joinPaths([matchedPathname, match.pathname]),\n pathnameBase: normalizePathname(\n joinPaths([matchedPathname, match.pathnameBase])\n ),\n route,\n });\n\n if (match.pathnameBase !== \"/\") {\n matchedPathname = joinPaths([matchedPathname, match.pathnameBase]);\n }\n }\n\n return matches;\n}\n\n/**\n * Returns a path with params interpolated.\n *\n * @see https://reactrouter.com/v6/utils/generate-path\n */\nexport function generatePath(\n originalPath: Path,\n params: {\n [key in PathParam]: string | null;\n } = {} as any\n): string {\n let path: string = originalPath;\n if (path.endsWith(\"*\") && path !== \"*\" && !path.endsWith(\"/*\")) {\n warning(\n false,\n `Route path \"${path}\" will be treated as if it were ` +\n `\"${path.replace(/\\*$/, \"/*\")}\" because the \\`*\\` character must ` +\n `always follow a \\`/\\` in the pattern. To get rid of this warning, ` +\n `please change the route path to \"${path.replace(/\\*$/, \"/*\")}\".`\n );\n path = path.replace(/\\*$/, \"/*\") as Path;\n }\n\n // ensure `/` is added at the beginning if the path is absolute\n const prefix = path.startsWith(\"/\") ? \"/\" : \"\";\n\n const stringify = (p: any) =>\n p == null ? \"\" : typeof p === \"string\" ? p : String(p);\n\n const segments = path\n .split(/\\/+/)\n .map((segment, index, array) => {\n const isLastSegment = index === array.length - 1;\n\n // only apply the splat if it's the last segment\n if (isLastSegment && segment === \"*\") {\n const star = \"*\" as PathParam;\n // Apply the splat\n return stringify(params[star]);\n }\n\n const keyMatch = segment.match(/^:([\\w-]+)(\\??)$/);\n if (keyMatch) {\n const [, key, optional] = keyMatch;\n let param = params[key as PathParam];\n invariant(optional === \"?\" || param != null, `Missing \":${key}\" param`);\n return stringify(param);\n }\n\n // Remove any optional markers from optional static segments\n return segment.replace(/\\?$/g, \"\");\n })\n // Remove empty segments\n .filter((segment) => !!segment);\n\n return prefix + segments.join(\"/\");\n}\n\n/**\n * A PathPattern is used to match on some portion of a URL pathname.\n */\nexport interface PathPattern {\n /**\n * A string to match against a URL pathname. May contain `:id`-style segments\n * to indicate placeholders for dynamic parameters. May also end with `/*` to\n * indicate matching the rest of the URL pathname.\n */\n path: Path;\n /**\n * Should be `true` if the static portions of the `path` should be matched in\n * the same case.\n */\n caseSensitive?: boolean;\n /**\n * Should be `true` if this pattern should match the entire URL pathname.\n */\n end?: boolean;\n}\n\n/**\n * A PathMatch contains info about how a PathPattern matched on a URL pathname.\n */\nexport interface PathMatch {\n /**\n * The names and values of dynamic parameters in the URL.\n */\n params: Params;\n /**\n * The portion of the URL pathname that was matched.\n */\n pathname: string;\n /**\n * The portion of the URL pathname that was matched before child routes.\n */\n pathnameBase: string;\n /**\n * The pattern that was used to match.\n */\n pattern: PathPattern;\n}\n\ntype Mutable = {\n -readonly [P in keyof T]: T[P];\n};\n\n/**\n * Performs pattern matching on a URL pathname and returns information about\n * the match.\n *\n * @see https://reactrouter.com/v6/utils/match-path\n */\nexport function matchPath<\n ParamKey extends ParamParseKey,\n Path extends string\n>(\n pattern: PathPattern | Path,\n pathname: string\n): PathMatch | null {\n if (typeof pattern === \"string\") {\n pattern = { path: pattern, caseSensitive: false, end: true };\n }\n\n let [matcher, compiledParams] = compilePath(\n pattern.path,\n pattern.caseSensitive,\n pattern.end\n );\n\n let match = pathname.match(matcher);\n if (!match) return null;\n\n let matchedPathname = match[0];\n let pathnameBase = matchedPathname.replace(/(.)\\/+$/, \"$1\");\n let captureGroups = match.slice(1);\n let params: Params = compiledParams.reduce>(\n (memo, { paramName, isOptional }, index) => {\n // We need to compute the pathnameBase here using the raw splat value\n // instead of using params[\"*\"] later because it will be decoded then\n if (paramName === \"*\") {\n let splatValue = captureGroups[index] || \"\";\n pathnameBase = matchedPathname\n .slice(0, matchedPathname.length - splatValue.length)\n .replace(/(.)\\/+$/, \"$1\");\n }\n\n const value = captureGroups[index];\n if (isOptional && !value) {\n memo[paramName] = undefined;\n } else {\n memo[paramName] = (value || \"\").replace(/%2F/g, \"/\");\n }\n return memo;\n },\n {}\n );\n\n return {\n params,\n pathname: matchedPathname,\n pathnameBase,\n pattern,\n };\n}\n\ntype CompiledPathParam = { paramName: string; isOptional?: boolean };\n\nfunction compilePath(\n path: string,\n caseSensitive = false,\n end = true\n): [RegExp, CompiledPathParam[]] {\n warning(\n path === \"*\" || !path.endsWith(\"*\") || path.endsWith(\"/*\"),\n `Route path \"${path}\" will be treated as if it were ` +\n `\"${path.replace(/\\*$/, \"/*\")}\" because the \\`*\\` character must ` +\n `always follow a \\`/\\` in the pattern. To get rid of this warning, ` +\n `please change the route path to \"${path.replace(/\\*$/, \"/*\")}\".`\n );\n\n let params: CompiledPathParam[] = [];\n let regexpSource =\n \"^\" +\n path\n .replace(/\\/*\\*?$/, \"\") // Ignore trailing / and /*, we'll handle it below\n .replace(/^\\/*/, \"/\") // Make sure it has a leading /\n .replace(/[\\\\.*+^${}|()[\\]]/g, \"\\\\$&\") // Escape special regex chars\n .replace(\n /\\/:([\\w-]+)(\\?)?/g,\n (_: string, paramName: string, isOptional) => {\n params.push({ paramName, isOptional: isOptional != null });\n return isOptional ? \"/?([^\\\\/]+)?\" : \"/([^\\\\/]+)\";\n }\n );\n\n if (path.endsWith(\"*\")) {\n params.push({ paramName: \"*\" });\n regexpSource +=\n path === \"*\" || path === \"/*\"\n ? \"(.*)$\" // Already matched the initial /, just match the rest\n : \"(?:\\\\/(.+)|\\\\/*)$\"; // Don't include the / in params[\"*\"]\n } else if (end) {\n // When matching to the end, ignore trailing slashes\n regexpSource += \"\\\\/*$\";\n } else if (path !== \"\" && path !== \"/\") {\n // If our path is non-empty and contains anything beyond an initial slash,\n // then we have _some_ form of path in our regex, so we should expect to\n // match only if we find the end of this path segment. Look for an optional\n // non-captured trailing slash (to match a portion of the URL) or the end\n // of the path (if we've matched to the end). We used to do this with a\n // word boundary but that gives false positives on routes like\n // /user-preferences since `-` counts as a word boundary.\n regexpSource += \"(?:(?=\\\\/|$))\";\n } else {\n // Nothing to match for \"\" or \"/\"\n }\n\n let matcher = new RegExp(regexpSource, caseSensitive ? undefined : \"i\");\n\n return [matcher, params];\n}\n\nexport function decodePath(value: string) {\n try {\n return value\n .split(\"/\")\n .map((v) => decodeURIComponent(v).replace(/\\//g, \"%2F\"))\n .join(\"/\");\n } catch (error) {\n warning(\n false,\n `The URL path \"${value}\" could not be decoded because it is is a ` +\n `malformed URL segment. This is probably due to a bad percent ` +\n `encoding (${error}).`\n );\n\n return value;\n }\n}\n\n/**\n * @private\n */\nexport function stripBasename(\n pathname: string,\n basename: string\n): string | null {\n if (basename === \"/\") return pathname;\n\n if (!pathname.toLowerCase().startsWith(basename.toLowerCase())) {\n return null;\n }\n\n // We want to leave trailing slash behavior in the user's control, so if they\n // specify a basename with a trailing slash, we should support it\n let startIndex = basename.endsWith(\"/\")\n ? basename.length - 1\n : basename.length;\n let nextChar = pathname.charAt(startIndex);\n if (nextChar && nextChar !== \"/\") {\n // pathname does not start with basename/\n return null;\n }\n\n return pathname.slice(startIndex) || \"/\";\n}\n\n/**\n * Returns a resolved path object relative to the given pathname.\n *\n * @see https://reactrouter.com/v6/utils/resolve-path\n */\nexport function resolvePath(to: To, fromPathname = \"/\"): Path {\n let {\n pathname: toPathname,\n search = \"\",\n hash = \"\",\n } = typeof to === \"string\" ? parsePath(to) : to;\n\n let pathname = toPathname\n ? toPathname.startsWith(\"/\")\n ? toPathname\n : resolvePathname(toPathname, fromPathname)\n : fromPathname;\n\n return {\n pathname,\n search: normalizeSearch(search),\n hash: normalizeHash(hash),\n };\n}\n\nfunction resolvePathname(relativePath: string, fromPathname: string): string {\n let segments = fromPathname.replace(/\\/+$/, \"\").split(\"/\");\n let relativeSegments = relativePath.split(\"/\");\n\n relativeSegments.forEach((segment) => {\n if (segment === \"..\") {\n // Keep the root \"\" segment so the pathname starts at /\n if (segments.length > 1) segments.pop();\n } else if (segment !== \".\") {\n segments.push(segment);\n }\n });\n\n return segments.length > 1 ? segments.join(\"/\") : \"/\";\n}\n\nfunction getInvalidPathError(\n char: string,\n field: string,\n dest: string,\n path: Partial\n) {\n return (\n `Cannot include a '${char}' character in a manually specified ` +\n `\\`to.${field}\\` field [${JSON.stringify(\n path\n )}]. Please separate it out to the ` +\n `\\`to.${dest}\\` field. Alternatively you may provide the full path as ` +\n `a string in and the router will parse it for you.`\n );\n}\n\n/**\n * @private\n *\n * When processing relative navigation we want to ignore ancestor routes that\n * do not contribute to the path, such that index/pathless layout routes don't\n * interfere.\n *\n * For example, when moving a route element into an index route and/or a\n * pathless layout route, relative link behavior contained within should stay\n * the same. Both of the following examples should link back to the root:\n *\n * \n * \n * \n *\n * \n * \n * }> // <-- Does not contribute\n * // <-- Does not contribute\n * \n * \n */\nexport function getPathContributingMatches<\n T extends AgnosticRouteMatch = AgnosticRouteMatch\n>(matches: T[]) {\n return matches.filter(\n (match, index) =>\n index === 0 || (match.route.path && match.route.path.length > 0)\n );\n}\n\n// Return the array of pathnames for the current route matches - used to\n// generate the routePathnames input for resolveTo()\nexport function getResolveToMatches<\n T extends AgnosticRouteMatch = AgnosticRouteMatch\n>(matches: T[], v7_relativeSplatPath: boolean) {\n let pathMatches = getPathContributingMatches(matches);\n\n // When v7_relativeSplatPath is enabled, use the full pathname for the leaf\n // match so we include splat values for \".\" links. See:\n // https://github.com/remix-run/react-router/issues/11052#issuecomment-1836589329\n if (v7_relativeSplatPath) {\n return pathMatches.map((match, idx) =>\n idx === pathMatches.length - 1 ? match.pathname : match.pathnameBase\n );\n }\n\n return pathMatches.map((match) => match.pathnameBase);\n}\n\n/**\n * @private\n */\nexport function resolveTo(\n toArg: To,\n routePathnames: string[],\n locationPathname: string,\n isPathRelative = false\n): Path {\n let to: Partial;\n if (typeof toArg === \"string\") {\n to = parsePath(toArg);\n } else {\n to = { ...toArg };\n\n invariant(\n !to.pathname || !to.pathname.includes(\"?\"),\n getInvalidPathError(\"?\", \"pathname\", \"search\", to)\n );\n invariant(\n !to.pathname || !to.pathname.includes(\"#\"),\n getInvalidPathError(\"#\", \"pathname\", \"hash\", to)\n );\n invariant(\n !to.search || !to.search.includes(\"#\"),\n getInvalidPathError(\"#\", \"search\", \"hash\", to)\n );\n }\n\n let isEmptyPath = toArg === \"\" || to.pathname === \"\";\n let toPathname = isEmptyPath ? \"/\" : to.pathname;\n\n let from: string;\n\n // Routing is relative to the current pathname if explicitly requested.\n //\n // If a pathname is explicitly provided in `to`, it should be relative to the\n // route context. This is explained in `Note on `` values` in our\n // migration guide from v5 as a means of disambiguation between `to` values\n // that begin with `/` and those that do not. However, this is problematic for\n // `to` values that do not provide a pathname. `to` can simply be a search or\n // hash string, in which case we should assume that the navigation is relative\n // to the current location's pathname and *not* the route pathname.\n if (toPathname == null) {\n from = locationPathname;\n } else {\n let routePathnameIndex = routePathnames.length - 1;\n\n // With relative=\"route\" (the default), each leading .. segment means\n // \"go up one route\" instead of \"go up one URL segment\". This is a key\n // difference from how works and a major reason we call this a\n // \"to\" value instead of a \"href\".\n if (!isPathRelative && toPathname.startsWith(\"..\")) {\n let toSegments = toPathname.split(\"/\");\n\n while (toSegments[0] === \"..\") {\n toSegments.shift();\n routePathnameIndex -= 1;\n }\n\n to.pathname = toSegments.join(\"/\");\n }\n\n from = routePathnameIndex >= 0 ? routePathnames[routePathnameIndex] : \"/\";\n }\n\n let path = resolvePath(to, from);\n\n // Ensure the pathname has a trailing slash if the original \"to\" had one\n let hasExplicitTrailingSlash =\n toPathname && toPathname !== \"/\" && toPathname.endsWith(\"/\");\n // Or if this was a link to the current path which has a trailing slash\n let hasCurrentTrailingSlash =\n (isEmptyPath || toPathname === \".\") && locationPathname.endsWith(\"/\");\n if (\n !path.pathname.endsWith(\"/\") &&\n (hasExplicitTrailingSlash || hasCurrentTrailingSlash)\n ) {\n path.pathname += \"/\";\n }\n\n return path;\n}\n\n/**\n * @private\n */\nexport function getToPathname(to: To): string | undefined {\n // Empty strings should be treated the same as / paths\n return to === \"\" || (to as Path).pathname === \"\"\n ? \"/\"\n : typeof to === \"string\"\n ? parsePath(to).pathname\n : to.pathname;\n}\n\n/**\n * @private\n */\nexport const joinPaths = (paths: string[]): string =>\n paths.join(\"/\").replace(/\\/\\/+/g, \"/\");\n\n/**\n * @private\n */\nexport const normalizePathname = (pathname: string): string =>\n pathname.replace(/\\/+$/, \"\").replace(/^\\/*/, \"/\");\n\n/**\n * @private\n */\nexport const normalizeSearch = (search: string): string =>\n !search || search === \"?\"\n ? \"\"\n : search.startsWith(\"?\")\n ? search\n : \"?\" + search;\n\n/**\n * @private\n */\nexport const normalizeHash = (hash: string): string =>\n !hash || hash === \"#\" ? \"\" : hash.startsWith(\"#\") ? hash : \"#\" + hash;\n\nexport type JsonFunction = (\n data: Data,\n init?: number | ResponseInit\n) => Response;\n\n/**\n * This is a shortcut for creating `application/json` responses. Converts `data`\n * to JSON and sets the `Content-Type` header.\n *\n * @deprecated The `json` method is deprecated in favor of returning raw objects.\n * This method will be removed in v7.\n */\nexport const json: JsonFunction = (data, init = {}) => {\n let responseInit = typeof init === \"number\" ? { status: init } : init;\n\n let headers = new Headers(responseInit.headers);\n if (!headers.has(\"Content-Type\")) {\n headers.set(\"Content-Type\", \"application/json; charset=utf-8\");\n }\n\n return new Response(JSON.stringify(data), {\n ...responseInit,\n headers,\n });\n};\n\nexport class DataWithResponseInit {\n type: string = \"DataWithResponseInit\";\n data: D;\n init: ResponseInit | null;\n\n constructor(data: D, init?: ResponseInit) {\n this.data = data;\n this.init = init || null;\n }\n}\n\n/**\n * Create \"responses\" that contain `status`/`headers` without forcing\n * serialization into an actual `Response` - used by Remix single fetch\n */\nexport function data(data: D, init?: number | ResponseInit) {\n return new DataWithResponseInit(\n data,\n typeof init === \"number\" ? { status: init } : init\n );\n}\n\nexport interface TrackedPromise extends Promise {\n _tracked?: boolean;\n _data?: any;\n _error?: any;\n}\n\nexport class AbortedDeferredError extends Error {}\n\nexport class DeferredData {\n private pendingKeysSet: Set = new Set();\n private controller: AbortController;\n private abortPromise: Promise;\n private unlistenAbortSignal: () => void;\n private subscribers: Set<(aborted: boolean, settledKey?: string) => void> =\n new Set();\n data: Record;\n init?: ResponseInit;\n deferredKeys: string[] = [];\n\n constructor(data: Record, responseInit?: ResponseInit) {\n invariant(\n data && typeof data === \"object\" && !Array.isArray(data),\n \"defer() only accepts plain objects\"\n );\n\n // Set up an AbortController + Promise we can race against to exit early\n // cancellation\n let reject: (e: AbortedDeferredError) => void;\n this.abortPromise = new Promise((_, r) => (reject = r));\n this.controller = new AbortController();\n let onAbort = () =>\n reject(new AbortedDeferredError(\"Deferred data aborted\"));\n this.unlistenAbortSignal = () =>\n this.controller.signal.removeEventListener(\"abort\", onAbort);\n this.controller.signal.addEventListener(\"abort\", onAbort);\n\n this.data = Object.entries(data).reduce(\n (acc, [key, value]) =>\n Object.assign(acc, {\n [key]: this.trackPromise(key, value),\n }),\n {}\n );\n\n if (this.done) {\n // All incoming values were resolved\n this.unlistenAbortSignal();\n }\n\n this.init = responseInit;\n }\n\n private trackPromise(\n key: string,\n value: Promise | unknown\n ): TrackedPromise | unknown {\n if (!(value instanceof Promise)) {\n return value;\n }\n\n this.deferredKeys.push(key);\n this.pendingKeysSet.add(key);\n\n // We store a little wrapper promise that will be extended with\n // _data/_error props upon resolve/reject\n let promise: TrackedPromise = Promise.race([value, this.abortPromise]).then(\n (data) => this.onSettle(promise, key, undefined, data as unknown),\n (error) => this.onSettle(promise, key, error as unknown)\n );\n\n // Register rejection listeners to avoid uncaught promise rejections on\n // errors or aborted deferred values\n promise.catch(() => {});\n\n Object.defineProperty(promise, \"_tracked\", { get: () => true });\n return promise;\n }\n\n private onSettle(\n promise: TrackedPromise,\n key: string,\n error: unknown,\n data?: unknown\n ): unknown {\n if (\n this.controller.signal.aborted &&\n error instanceof AbortedDeferredError\n ) {\n this.unlistenAbortSignal();\n Object.defineProperty(promise, \"_error\", { get: () => error });\n return Promise.reject(error);\n }\n\n this.pendingKeysSet.delete(key);\n\n if (this.done) {\n // Nothing left to abort!\n this.unlistenAbortSignal();\n }\n\n // If the promise was resolved/rejected with undefined, we'll throw an error as you\n // should always resolve with a value or null\n if (error === undefined && data === undefined) {\n let undefinedError = new Error(\n `Deferred data for key \"${key}\" resolved/rejected with \\`undefined\\`, ` +\n `you must resolve/reject with a value or \\`null\\`.`\n );\n Object.defineProperty(promise, \"_error\", { get: () => undefinedError });\n this.emit(false, key);\n return Promise.reject(undefinedError);\n }\n\n if (data === undefined) {\n Object.defineProperty(promise, \"_error\", { get: () => error });\n this.emit(false, key);\n return Promise.reject(error);\n }\n\n Object.defineProperty(promise, \"_data\", { get: () => data });\n this.emit(false, key);\n return data;\n }\n\n private emit(aborted: boolean, settledKey?: string) {\n this.subscribers.forEach((subscriber) => subscriber(aborted, settledKey));\n }\n\n subscribe(fn: (aborted: boolean, settledKey?: string) => void) {\n this.subscribers.add(fn);\n return () => this.subscribers.delete(fn);\n }\n\n cancel() {\n this.controller.abort();\n this.pendingKeysSet.forEach((v, k) => this.pendingKeysSet.delete(k));\n this.emit(true);\n }\n\n async resolveData(signal: AbortSignal) {\n let aborted = false;\n if (!this.done) {\n let onAbort = () => this.cancel();\n signal.addEventListener(\"abort\", onAbort);\n aborted = await new Promise((resolve) => {\n this.subscribe((aborted) => {\n signal.removeEventListener(\"abort\", onAbort);\n if (aborted || this.done) {\n resolve(aborted);\n }\n });\n });\n }\n return aborted;\n }\n\n get done() {\n return this.pendingKeysSet.size === 0;\n }\n\n get unwrappedData() {\n invariant(\n this.data !== null && this.done,\n \"Can only unwrap data on initialized and settled deferreds\"\n );\n\n return Object.entries(this.data).reduce(\n (acc, [key, value]) =>\n Object.assign(acc, {\n [key]: unwrapTrackedPromise(value),\n }),\n {}\n );\n }\n\n get pendingKeys() {\n return Array.from(this.pendingKeysSet);\n }\n}\n\nfunction isTrackedPromise(value: any): value is TrackedPromise {\n return (\n value instanceof Promise && (value as TrackedPromise)._tracked === true\n );\n}\n\nfunction unwrapTrackedPromise(value: any) {\n if (!isTrackedPromise(value)) {\n return value;\n }\n\n if (value._error) {\n throw value._error;\n }\n return value._data;\n}\n\nexport type DeferFunction = (\n data: Record,\n init?: number | ResponseInit\n) => DeferredData;\n\n/**\n * @deprecated The `defer` method is deprecated in favor of returning raw\n * objects. This method will be removed in v7.\n */\nexport const defer: DeferFunction = (data, init = {}) => {\n let responseInit = typeof init === \"number\" ? { status: init } : init;\n\n return new DeferredData(data, responseInit);\n};\n\nexport type RedirectFunction = (\n url: string,\n init?: number | ResponseInit\n) => Response;\n\n/**\n * A redirect response. Sets the status code and the `Location` header.\n * Defaults to \"302 Found\".\n */\nexport const redirect: RedirectFunction = (url, init = 302) => {\n let responseInit = init;\n if (typeof responseInit === \"number\") {\n responseInit = { status: responseInit };\n } else if (typeof responseInit.status === \"undefined\") {\n responseInit.status = 302;\n }\n\n let headers = new Headers(responseInit.headers);\n headers.set(\"Location\", url);\n\n return new Response(null, {\n ...responseInit,\n headers,\n });\n};\n\n/**\n * A redirect response that will force a document reload to the new location.\n * Sets the status code and the `Location` header.\n * Defaults to \"302 Found\".\n */\nexport const redirectDocument: RedirectFunction = (url, init) => {\n let response = redirect(url, init);\n response.headers.set(\"X-Remix-Reload-Document\", \"true\");\n return response;\n};\n\n/**\n * A redirect response that will perform a `history.replaceState` instead of a\n * `history.pushState` for client-side navigation redirects.\n * Sets the status code and the `Location` header.\n * Defaults to \"302 Found\".\n */\nexport const replace: RedirectFunction = (url, init) => {\n let response = redirect(url, init);\n response.headers.set(\"X-Remix-Replace\", \"true\");\n return response;\n};\n\nexport type ErrorResponse = {\n status: number;\n statusText: string;\n data: any;\n};\n\n/**\n * @private\n * Utility class we use to hold auto-unwrapped 4xx/5xx Response bodies\n *\n * We don't export the class for public use since it's an implementation\n * detail, but we export the interface above so folks can build their own\n * abstractions around instances via isRouteErrorResponse()\n */\nexport class ErrorResponseImpl implements ErrorResponse {\n status: number;\n statusText: string;\n data: any;\n private error?: Error;\n private internal: boolean;\n\n constructor(\n status: number,\n statusText: string | undefined,\n data: any,\n internal = false\n ) {\n this.status = status;\n this.statusText = statusText || \"\";\n this.internal = internal;\n if (data instanceof Error) {\n this.data = data.toString();\n this.error = data;\n } else {\n this.data = data;\n }\n }\n}\n\n/**\n * Check if the given error is an ErrorResponse generated from a 4xx/5xx\n * Response thrown from an action/loader\n */\nexport function isRouteErrorResponse(error: any): error is ErrorResponse {\n return (\n error != null &&\n typeof error.status === \"number\" &&\n typeof error.statusText === \"string\" &&\n typeof error.internal === \"boolean\" &&\n \"data\" in error\n );\n}\n","import type { History, Location, Path, To } from \"./history\";\nimport {\n Action as HistoryAction,\n createLocation,\n createPath,\n invariant,\n parsePath,\n warning,\n} from \"./history\";\nimport type {\n AgnosticDataRouteMatch,\n AgnosticDataRouteObject,\n DataStrategyMatch,\n AgnosticRouteObject,\n DataResult,\n DataStrategyFunction,\n DataStrategyFunctionArgs,\n DeferredData,\n DeferredResult,\n DetectErrorBoundaryFunction,\n ErrorResult,\n FormEncType,\n FormMethod,\n HTMLFormMethod,\n DataStrategyResult,\n ImmutableRouteKey,\n MapRoutePropertiesFunction,\n MutationFormMethod,\n RedirectResult,\n RouteData,\n RouteManifest,\n ShouldRevalidateFunctionArgs,\n Submission,\n SuccessResult,\n UIMatch,\n V7_FormMethod,\n V7_MutationFormMethod,\n AgnosticPatchRoutesOnNavigationFunction,\n DataWithResponseInit,\n} from \"./utils\";\nimport {\n ErrorResponseImpl,\n ResultType,\n convertRouteMatchToUiMatch,\n convertRoutesToDataRoutes,\n getPathContributingMatches,\n getResolveToMatches,\n immutableRouteKeys,\n isRouteErrorResponse,\n joinPaths,\n matchRoutes,\n matchRoutesImpl,\n resolveTo,\n stripBasename,\n} from \"./utils\";\n\n////////////////////////////////////////////////////////////////////////////////\n//#region Types and Constants\n////////////////////////////////////////////////////////////////////////////////\n\n/**\n * A Router instance manages all navigation and data loading/mutations\n */\nexport interface Router {\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Return the basename for the router\n */\n get basename(): RouterInit[\"basename\"];\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Return the future config for the router\n */\n get future(): FutureConfig;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Return the current state of the router\n */\n get state(): RouterState;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Return the routes for this router instance\n */\n get routes(): AgnosticDataRouteObject[];\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Return the window associated with the router\n */\n get window(): RouterInit[\"window\"];\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Initialize the router, including adding history listeners and kicking off\n * initial data fetches. Returns a function to cleanup listeners and abort\n * any in-progress loads\n */\n initialize(): Router;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Subscribe to router.state updates\n *\n * @param fn function to call with the new state\n */\n subscribe(fn: RouterSubscriber): () => void;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Enable scroll restoration behavior in the router\n *\n * @param savedScrollPositions Object that will manage positions, in case\n * it's being restored from sessionStorage\n * @param getScrollPosition Function to get the active Y scroll position\n * @param getKey Function to get the key to use for restoration\n */\n enableScrollRestoration(\n savedScrollPositions: Record,\n getScrollPosition: GetScrollPositionFunction,\n getKey?: GetScrollRestorationKeyFunction\n ): () => void;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Navigate forward/backward in the history stack\n * @param to Delta to move in the history stack\n */\n navigate(to: number): Promise;\n\n /**\n * Navigate to the given path\n * @param to Path to navigate to\n * @param opts Navigation options (method, submission, etc.)\n */\n navigate(to: To | null, opts?: RouterNavigateOptions): Promise;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Trigger a fetcher load/submission\n *\n * @param key Fetcher key\n * @param routeId Route that owns the fetcher\n * @param href href to fetch\n * @param opts Fetcher options, (method, submission, etc.)\n */\n fetch(\n key: string,\n routeId: string,\n href: string | null,\n opts?: RouterFetchOptions\n ): void;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Trigger a revalidation of all current route loaders and fetcher loads\n */\n revalidate(): void;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Utility function to create an href for the given location\n * @param location\n */\n createHref(location: Location | URL): string;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Utility function to URL encode a destination path according to the internal\n * history implementation\n * @param to\n */\n encodeLocation(to: To): Path;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Get/create a fetcher for the given key\n * @param key\n */\n getFetcher(key: string): Fetcher;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Delete the fetcher for a given key\n * @param key\n */\n deleteFetcher(key: string): void;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Cleanup listeners and abort any in-progress loads\n */\n dispose(): void;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Get a navigation blocker\n * @param key The identifier for the blocker\n * @param fn The blocker function implementation\n */\n getBlocker(key: string, fn: BlockerFunction): Blocker;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Delete a navigation blocker\n * @param key The identifier for the blocker\n */\n deleteBlocker(key: string): void;\n\n /**\n * @internal\n * PRIVATE DO NOT USE\n *\n * Patch additional children routes into an existing parent route\n * @param routeId The parent route id or a callback function accepting `patch`\n * to perform batch patching\n * @param children The additional children routes\n */\n patchRoutes(routeId: string | null, children: AgnosticRouteObject[]): void;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * HMR needs to pass in-flight route updates to React Router\n * TODO: Replace this with granular route update APIs (addRoute, updateRoute, deleteRoute)\n */\n _internalSetRoutes(routes: AgnosticRouteObject[]): void;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Internal fetch AbortControllers accessed by unit tests\n */\n _internalFetchControllers: Map;\n\n /**\n * @internal\n * PRIVATE - DO NOT USE\n *\n * Internal pending DeferredData instances accessed by unit tests\n */\n _internalActiveDeferreds: Map;\n}\n\n/**\n * State maintained internally by the router. During a navigation, all states\n * reflect the the \"old\" location unless otherwise noted.\n */\nexport interface RouterState {\n /**\n * The action of the most recent navigation\n */\n historyAction: HistoryAction;\n\n /**\n * The current location reflected by the router\n */\n location: Location;\n\n /**\n * The current set of route matches\n */\n matches: AgnosticDataRouteMatch[];\n\n /**\n * Tracks whether we've completed our initial data load\n */\n initialized: boolean;\n\n /**\n * Current scroll position we should start at for a new view\n * - number -> scroll position to restore to\n * - false -> do not restore scroll at all (used during submissions)\n * - null -> don't have a saved position, scroll to hash or top of page\n */\n restoreScrollPosition: number | false | null;\n\n /**\n * Indicate whether this navigation should skip resetting the scroll position\n * if we are unable to restore the scroll position\n */\n preventScrollReset: boolean;\n\n /**\n * Tracks the state of the current navigation\n */\n navigation: Navigation;\n\n /**\n * Tracks any in-progress revalidations\n */\n revalidation: RevalidationState;\n\n /**\n * Data from the loaders for the current matches\n */\n loaderData: RouteData;\n\n /**\n * Data from the action for the current matches\n */\n actionData: RouteData | null;\n\n /**\n * Errors caught from loaders for the current matches\n */\n errors: RouteData | null;\n\n /**\n * Map of current fetchers\n */\n fetchers: Map;\n\n /**\n * Map of current blockers\n */\n blockers: Map;\n}\n\n/**\n * Data that can be passed into hydrate a Router from SSR\n */\nexport type HydrationState = Partial<\n Pick\n>;\n\n/**\n * Future flags to toggle new feature behavior\n */\nexport interface FutureConfig {\n v7_fetcherPersist: boolean;\n v7_normalizeFormMethod: boolean;\n v7_partialHydration: boolean;\n v7_prependBasename: boolean;\n v7_relativeSplatPath: boolean;\n v7_skipActionErrorRevalidation: boolean;\n}\n\n/**\n * Initialization options for createRouter\n */\nexport interface RouterInit {\n routes: AgnosticRouteObject[];\n history: History;\n basename?: string;\n /**\n * @deprecated Use `mapRouteProperties` instead\n */\n detectErrorBoundary?: DetectErrorBoundaryFunction;\n mapRouteProperties?: MapRoutePropertiesFunction;\n future?: Partial;\n hydrationData?: HydrationState;\n window?: Window;\n dataStrategy?: DataStrategyFunction;\n patchRoutesOnNavigation?: AgnosticPatchRoutesOnNavigationFunction;\n}\n\n/**\n * State returned from a server-side query() call\n */\nexport interface StaticHandlerContext {\n basename: Router[\"basename\"];\n location: RouterState[\"location\"];\n matches: RouterState[\"matches\"];\n loaderData: RouterState[\"loaderData\"];\n actionData: RouterState[\"actionData\"];\n errors: RouterState[\"errors\"];\n statusCode: number;\n loaderHeaders: Record;\n actionHeaders: Record;\n activeDeferreds: Record | null;\n _deepestRenderedBoundaryId?: string | null;\n}\n\n/**\n * A StaticHandler instance manages a singular SSR navigation/fetch event\n */\nexport interface StaticHandler {\n dataRoutes: AgnosticDataRouteObject[];\n query(\n request: Request,\n opts?: {\n requestContext?: unknown;\n skipLoaderErrorBubbling?: boolean;\n dataStrategy?: DataStrategyFunction;\n }\n ): Promise;\n queryRoute(\n request: Request,\n opts?: {\n routeId?: string;\n requestContext?: unknown;\n dataStrategy?: DataStrategyFunction;\n }\n ): Promise;\n}\n\ntype ViewTransitionOpts = {\n currentLocation: Location;\n nextLocation: Location;\n};\n\n/**\n * Subscriber function signature for changes to router state\n */\nexport interface RouterSubscriber {\n (\n state: RouterState,\n opts: {\n deletedFetchers: string[];\n viewTransitionOpts?: ViewTransitionOpts;\n flushSync: boolean;\n }\n ): void;\n}\n\n/**\n * Function signature for determining the key to be used in scroll restoration\n * for a given location\n */\nexport interface GetScrollRestorationKeyFunction {\n (location: Location, matches: UIMatch[]): string | null;\n}\n\n/**\n * Function signature for determining the current scroll position\n */\nexport interface GetScrollPositionFunction {\n (): number;\n}\n\nexport type RelativeRoutingType = \"route\" | \"path\";\n\n// Allowed for any navigation or fetch\ntype BaseNavigateOrFetchOptions = {\n preventScrollReset?: boolean;\n relative?: RelativeRoutingType;\n flushSync?: boolean;\n};\n\n// Only allowed for navigations\ntype BaseNavigateOptions = BaseNavigateOrFetchOptions & {\n replace?: boolean;\n state?: any;\n fromRouteId?: string;\n viewTransition?: boolean;\n};\n\n// Only allowed for submission navigations\ntype BaseSubmissionOptions = {\n formMethod?: HTMLFormMethod;\n formEncType?: FormEncType;\n} & (\n | { formData: FormData; body?: undefined }\n | { formData?: undefined; body: any }\n);\n\n/**\n * Options for a navigate() call for a normal (non-submission) navigation\n */\ntype LinkNavigateOptions = BaseNavigateOptions;\n\n/**\n * Options for a navigate() call for a submission navigation\n */\ntype SubmissionNavigateOptions = BaseNavigateOptions & BaseSubmissionOptions;\n\n/**\n * Options to pass to navigate() for a navigation\n */\nexport type RouterNavigateOptions =\n | LinkNavigateOptions\n | SubmissionNavigateOptions;\n\n/**\n * Options for a fetch() load\n */\ntype LoadFetchOptions = BaseNavigateOrFetchOptions;\n\n/**\n * Options for a fetch() submission\n */\ntype SubmitFetchOptions = BaseNavigateOrFetchOptions & BaseSubmissionOptions;\n\n/**\n * Options to pass to fetch()\n */\nexport type RouterFetchOptions = LoadFetchOptions | SubmitFetchOptions;\n\n/**\n * Potential states for state.navigation\n */\nexport type NavigationStates = {\n Idle: {\n state: \"idle\";\n location: undefined;\n formMethod: undefined;\n formAction: undefined;\n formEncType: undefined;\n formData: undefined;\n json: undefined;\n text: undefined;\n };\n Loading: {\n state: \"loading\";\n location: Location;\n formMethod: Submission[\"formMethod\"] | undefined;\n formAction: Submission[\"formAction\"] | undefined;\n formEncType: Submission[\"formEncType\"] | undefined;\n formData: Submission[\"formData\"] | undefined;\n json: Submission[\"json\"] | undefined;\n text: Submission[\"text\"] | undefined;\n };\n Submitting: {\n state: \"submitting\";\n location: Location;\n formMethod: Submission[\"formMethod\"];\n formAction: Submission[\"formAction\"];\n formEncType: Submission[\"formEncType\"];\n formData: Submission[\"formData\"];\n json: Submission[\"json\"];\n text: Submission[\"text\"];\n };\n};\n\nexport type Navigation = NavigationStates[keyof NavigationStates];\n\nexport type RevalidationState = \"idle\" | \"loading\";\n\n/**\n * Potential states for fetchers\n */\ntype FetcherStates = {\n Idle: {\n state: \"idle\";\n formMethod: undefined;\n formAction: undefined;\n formEncType: undefined;\n text: undefined;\n formData: undefined;\n json: undefined;\n data: TData | undefined;\n };\n Loading: {\n state: \"loading\";\n formMethod: Submission[\"formMethod\"] | undefined;\n formAction: Submission[\"formAction\"] | undefined;\n formEncType: Submission[\"formEncType\"] | undefined;\n text: Submission[\"text\"] | undefined;\n formData: Submission[\"formData\"] | undefined;\n json: Submission[\"json\"] | undefined;\n data: TData | undefined;\n };\n Submitting: {\n state: \"submitting\";\n formMethod: Submission[\"formMethod\"];\n formAction: Submission[\"formAction\"];\n formEncType: Submission[\"formEncType\"];\n text: Submission[\"text\"];\n formData: Submission[\"formData\"];\n json: Submission[\"json\"];\n data: TData | undefined;\n };\n};\n\nexport type Fetcher =\n FetcherStates[keyof FetcherStates];\n\ninterface BlockerBlocked {\n state: \"blocked\";\n reset(): void;\n proceed(): void;\n location: Location;\n}\n\ninterface BlockerUnblocked {\n state: \"unblocked\";\n reset: undefined;\n proceed: undefined;\n location: undefined;\n}\n\ninterface BlockerProceeding {\n state: \"proceeding\";\n reset: undefined;\n proceed: undefined;\n location: Location;\n}\n\nexport type Blocker = BlockerUnblocked | BlockerBlocked | BlockerProceeding;\n\nexport type BlockerFunction = (args: {\n currentLocation: Location;\n nextLocation: Location;\n historyAction: HistoryAction;\n}) => boolean;\n\ninterface ShortCircuitable {\n /**\n * startNavigation does not need to complete the navigation because we\n * redirected or got interrupted\n */\n shortCircuited?: boolean;\n}\n\ntype PendingActionResult = [string, SuccessResult | ErrorResult];\n\ninterface HandleActionResult extends ShortCircuitable {\n /**\n * Route matches which may have been updated from fog of war discovery\n */\n matches?: RouterState[\"matches\"];\n /**\n * Tuple for the returned or thrown value from the current action. The routeId\n * is the action route for success and the bubbled boundary route for errors.\n */\n pendingActionResult?: PendingActionResult;\n}\n\ninterface HandleLoadersResult extends ShortCircuitable {\n /**\n * Route matches which may have been updated from fog of war discovery\n */\n matches?: RouterState[\"matches\"];\n /**\n * loaderData returned from the current set of loaders\n */\n loaderData?: RouterState[\"loaderData\"];\n /**\n * errors thrown from the current set of loaders\n */\n errors?: RouterState[\"errors\"];\n}\n\n/**\n * Cached info for active fetcher.load() instances so they can participate\n * in revalidation\n */\ninterface FetchLoadMatch {\n routeId: string;\n path: string;\n}\n\n/**\n * Identified fetcher.load() calls that need to be revalidated\n */\ninterface RevalidatingFetcher extends FetchLoadMatch {\n key: string;\n match: AgnosticDataRouteMatch | null;\n matches: AgnosticDataRouteMatch[] | null;\n controller: AbortController | null;\n}\n\nconst validMutationMethodsArr: MutationFormMethod[] = [\n \"post\",\n \"put\",\n \"patch\",\n \"delete\",\n];\nconst validMutationMethods = new Set(\n validMutationMethodsArr\n);\n\nconst validRequestMethodsArr: FormMethod[] = [\n \"get\",\n ...validMutationMethodsArr,\n];\nconst validRequestMethods = new Set(validRequestMethodsArr);\n\nconst redirectStatusCodes = new Set([301, 302, 303, 307, 308]);\nconst redirectPreserveMethodStatusCodes = new Set([307, 308]);\n\nexport const IDLE_NAVIGATION: NavigationStates[\"Idle\"] = {\n state: \"idle\",\n location: undefined,\n formMethod: undefined,\n formAction: undefined,\n formEncType: undefined,\n formData: undefined,\n json: undefined,\n text: undefined,\n};\n\nexport const IDLE_FETCHER: FetcherStates[\"Idle\"] = {\n state: \"idle\",\n data: undefined,\n formMethod: undefined,\n formAction: undefined,\n formEncType: undefined,\n formData: undefined,\n json: undefined,\n text: undefined,\n};\n\nexport const IDLE_BLOCKER: BlockerUnblocked = {\n state: \"unblocked\",\n proceed: undefined,\n reset: undefined,\n location: undefined,\n};\n\nconst ABSOLUTE_URL_REGEX = /^(?:[a-z][a-z0-9+.-]*:|\\/\\/)/i;\n\nconst defaultMapRouteProperties: MapRoutePropertiesFunction = (route) => ({\n hasErrorBoundary: Boolean(route.hasErrorBoundary),\n});\n\nconst TRANSITIONS_STORAGE_KEY = \"remix-router-transitions\";\n\n//#endregion\n\n////////////////////////////////////////////////////////////////////////////////\n//#region createRouter\n////////////////////////////////////////////////////////////////////////////////\n\n/**\n * Create a router and listen to history POP navigations\n */\nexport function createRouter(init: RouterInit): Router {\n const routerWindow = init.window\n ? init.window\n : typeof window !== \"undefined\"\n ? window\n : undefined;\n const isBrowser =\n typeof routerWindow !== \"undefined\" &&\n typeof routerWindow.document !== \"undefined\" &&\n typeof routerWindow.document.createElement !== \"undefined\";\n const isServer = !isBrowser;\n\n invariant(\n init.routes.length > 0,\n \"You must provide a non-empty routes array to createRouter\"\n );\n\n let mapRouteProperties: MapRoutePropertiesFunction;\n if (init.mapRouteProperties) {\n mapRouteProperties = init.mapRouteProperties;\n } else if (init.detectErrorBoundary) {\n // If they are still using the deprecated version, wrap it with the new API\n let detectErrorBoundary = init.detectErrorBoundary;\n mapRouteProperties = (route) => ({\n hasErrorBoundary: detectErrorBoundary(route),\n });\n } else {\n mapRouteProperties = defaultMapRouteProperties;\n }\n\n // Routes keyed by ID\n let manifest: RouteManifest = {};\n // Routes in tree format for matching\n let dataRoutes = convertRoutesToDataRoutes(\n init.routes,\n mapRouteProperties,\n undefined,\n manifest\n );\n let inFlightDataRoutes: AgnosticDataRouteObject[] | undefined;\n let basename = init.basename || \"/\";\n let dataStrategyImpl = init.dataStrategy || defaultDataStrategy;\n let patchRoutesOnNavigationImpl = init.patchRoutesOnNavigation;\n\n // Config driven behavior flags\n let future: FutureConfig = {\n v7_fetcherPersist: false,\n v7_normalizeFormMethod: false,\n v7_partialHydration: false,\n v7_prependBasename: false,\n v7_relativeSplatPath: false,\n v7_skipActionErrorRevalidation: false,\n ...init.future,\n };\n // Cleanup function for history\n let unlistenHistory: (() => void) | null = null;\n // Externally-provided functions to call on all state changes\n let subscribers = new Set();\n // Externally-provided object to hold scroll restoration locations during routing\n let savedScrollPositions: Record | null = null;\n // Externally-provided function to get scroll restoration keys\n let getScrollRestorationKey: GetScrollRestorationKeyFunction | null = null;\n // Externally-provided function to get current scroll position\n let getScrollPosition: GetScrollPositionFunction | null = null;\n // One-time flag to control the initial hydration scroll restoration. Because\n // we don't get the saved positions from until _after_\n // the initial render, we need to manually trigger a separate updateState to\n // send along the restoreScrollPosition\n // Set to true if we have `hydrationData` since we assume we were SSR'd and that\n // SSR did the initial scroll restoration.\n let initialScrollRestored = init.hydrationData != null;\n\n let initialMatches = matchRoutes(dataRoutes, init.history.location, basename);\n let initialMatchesIsFOW = false;\n let initialErrors: RouteData | null = null;\n\n if (initialMatches == null && !patchRoutesOnNavigationImpl) {\n // If we do not match a user-provided-route, fall back to the root\n // to allow the error boundary to take over\n let error = getInternalRouterError(404, {\n pathname: init.history.location.pathname,\n });\n let { matches, route } = getShortCircuitMatches(dataRoutes);\n initialMatches = matches;\n initialErrors = { [route.id]: error };\n }\n\n // In SPA apps, if the user provided a patchRoutesOnNavigation implementation and\n // our initial match is a splat route, clear them out so we run through lazy\n // discovery on hydration in case there's a more accurate lazy route match.\n // In SSR apps (with `hydrationData`), we expect that the server will send\n // up the proper matched routes so we don't want to run lazy discovery on\n // initial hydration and want to hydrate into the splat route.\n if (initialMatches && !init.hydrationData) {\n let fogOfWar = checkFogOfWar(\n initialMatches,\n dataRoutes,\n init.history.location.pathname\n );\n if (fogOfWar.active) {\n initialMatches = null;\n }\n }\n\n let initialized: boolean;\n if (!initialMatches) {\n initialized = false;\n initialMatches = [];\n\n // If partial hydration and fog of war is enabled, we will be running\n // `patchRoutesOnNavigation` during hydration so include any partial matches as\n // the initial matches so we can properly render `HydrateFallback`'s\n if (future.v7_partialHydration) {\n let fogOfWar = checkFogOfWar(\n null,\n dataRoutes,\n init.history.location.pathname\n );\n if (fogOfWar.active && fogOfWar.matches) {\n initialMatchesIsFOW = true;\n initialMatches = fogOfWar.matches;\n }\n }\n } else if (initialMatches.some((m) => m.route.lazy)) {\n // All initialMatches need to be loaded before we're ready. If we have lazy\n // functions around still then we'll need to run them in initialize()\n initialized = false;\n } else if (!initialMatches.some((m) => m.route.loader)) {\n // If we've got no loaders to run, then we're good to go\n initialized = true;\n } else if (future.v7_partialHydration) {\n // If partial hydration is enabled, we're initialized so long as we were\n // provided with hydrationData for every route with a loader, and no loaders\n // were marked for explicit hydration\n let loaderData = init.hydrationData ? init.hydrationData.loaderData : null;\n let errors = init.hydrationData ? init.hydrationData.errors : null;\n // If errors exist, don't consider routes below the boundary\n if (errors) {\n let idx = initialMatches.findIndex(\n (m) => errors![m.route.id] !== undefined\n );\n initialized = initialMatches\n .slice(0, idx + 1)\n .every((m) => !shouldLoadRouteOnHydration(m.route, loaderData, errors));\n } else {\n initialized = initialMatches.every(\n (m) => !shouldLoadRouteOnHydration(m.route, loaderData, errors)\n );\n }\n } else {\n // Without partial hydration - we're initialized if we were provided any\n // hydrationData - which is expected to be complete\n initialized = init.hydrationData != null;\n }\n\n let router: Router;\n let state: RouterState = {\n historyAction: init.history.action,\n location: init.history.location,\n matches: initialMatches,\n initialized,\n navigation: IDLE_NAVIGATION,\n // Don't restore on initial updateState() if we were SSR'd\n restoreScrollPosition: init.hydrationData != null ? false : null,\n preventScrollReset: false,\n revalidation: \"idle\",\n loaderData: (init.hydrationData && init.hydrationData.loaderData) || {},\n actionData: (init.hydrationData && init.hydrationData.actionData) || null,\n errors: (init.hydrationData && init.hydrationData.errors) || initialErrors,\n fetchers: new Map(),\n blockers: new Map(),\n };\n\n // -- Stateful internal variables to manage navigations --\n // Current navigation in progress (to be committed in completeNavigation)\n let pendingAction: HistoryAction = HistoryAction.Pop;\n\n // Should the current navigation prevent the scroll reset if scroll cannot\n // be restored?\n let pendingPreventScrollReset = false;\n\n // AbortController for the active navigation\n let pendingNavigationController: AbortController | null;\n\n // Should the current navigation enable document.startViewTransition?\n let pendingViewTransitionEnabled = false;\n\n // Store applied view transitions so we can apply them on POP\n let appliedViewTransitions: Map> = new Map<\n string,\n Set\n >();\n\n // Cleanup function for persisting applied transitions to sessionStorage\n let removePageHideEventListener: (() => void) | null = null;\n\n // We use this to avoid touching history in completeNavigation if a\n // revalidation is entirely uninterrupted\n let isUninterruptedRevalidation = false;\n\n // Use this internal flag to force revalidation of all loaders:\n // - submissions (completed or interrupted)\n // - useRevalidator()\n // - X-Remix-Revalidate (from redirect)\n let isRevalidationRequired = false;\n\n // Use this internal array to capture routes that require revalidation due\n // to a cancelled deferred on action submission\n let cancelledDeferredRoutes: string[] = [];\n\n // Use this internal array to capture fetcher loads that were cancelled by an\n // action navigation and require revalidation\n let cancelledFetcherLoads: Set = new Set();\n\n // AbortControllers for any in-flight fetchers\n let fetchControllers = new Map();\n\n // Track loads based on the order in which they started\n let incrementingLoadId = 0;\n\n // Track the outstanding pending navigation data load to be compared against\n // the globally incrementing load when a fetcher load lands after a completed\n // navigation\n let pendingNavigationLoadId = -1;\n\n // Fetchers that triggered data reloads as a result of their actions\n let fetchReloadIds = new Map();\n\n // Fetchers that triggered redirect navigations\n let fetchRedirectIds = new Set();\n\n // Most recent href/match for fetcher.load calls for fetchers\n let fetchLoadMatches = new Map();\n\n // Ref-count mounted fetchers so we know when it's ok to clean them up\n let activeFetchers = new Map();\n\n // Fetchers that have requested a delete when using v7_fetcherPersist,\n // they'll be officially removed after they return to idle\n let deletedFetchers = new Set();\n\n // Store DeferredData instances for active route matches. When a\n // route loader returns defer() we stick one in here. Then, when a nested\n // promise resolves we update loaderData. If a new navigation starts we\n // cancel active deferreds for eliminated routes.\n let activeDeferreds = new Map();\n\n // Store blocker functions in a separate Map outside of router state since\n // we don't need to update UI state if they change\n let blockerFunctions = new Map();\n\n // Map of pending patchRoutesOnNavigation() promises (keyed by path/matches) so\n // that we only kick them off once for a given combo\n let pendingPatchRoutes = new Map<\n string,\n ReturnType\n >();\n\n // Flag to ignore the next history update, so we can revert the URL change on\n // a POP navigation that was blocked by the user without touching router state\n let unblockBlockerHistoryUpdate: (() => void) | undefined = undefined;\n\n // Initialize the router, all side effects should be kicked off from here.\n // Implemented as a Fluent API for ease of:\n // let router = createRouter(init).initialize();\n function initialize() {\n // If history informs us of a POP navigation, start the navigation but do not update\n // state. We'll update our own state once the navigation completes\n unlistenHistory = init.history.listen(\n ({ action: historyAction, location, delta }) => {\n // Ignore this event if it was just us resetting the URL from a\n // blocked POP navigation\n if (unblockBlockerHistoryUpdate) {\n unblockBlockerHistoryUpdate();\n unblockBlockerHistoryUpdate = undefined;\n return;\n }\n\n warning(\n blockerFunctions.size === 0 || delta != null,\n \"You are trying to use a blocker on a POP navigation to a location \" +\n \"that was not created by @remix-run/router. This will fail silently in \" +\n \"production. This can happen if you are navigating outside the router \" +\n \"via `window.history.pushState`/`window.location.hash` instead of using \" +\n \"router navigation APIs. This can also happen if you are using \" +\n \"createHashRouter and the user manually changes the URL.\"\n );\n\n let blockerKey = shouldBlockNavigation({\n currentLocation: state.location,\n nextLocation: location,\n historyAction,\n });\n\n if (blockerKey && delta != null) {\n // Restore the URL to match the current UI, but don't update router state\n let nextHistoryUpdatePromise = new Promise((resolve) => {\n unblockBlockerHistoryUpdate = resolve;\n });\n init.history.go(delta * -1);\n\n // Put the blocker into a blocked state\n updateBlocker(blockerKey, {\n state: \"blocked\",\n location,\n proceed() {\n updateBlocker(blockerKey!, {\n state: \"proceeding\",\n proceed: undefined,\n reset: undefined,\n location,\n });\n // Re-do the same POP navigation we just blocked, after the url\n // restoration is also complete. See:\n // https://github.com/remix-run/react-router/issues/11613\n nextHistoryUpdatePromise.then(() => init.history.go(delta));\n },\n reset() {\n let blockers = new Map(state.blockers);\n blockers.set(blockerKey!, IDLE_BLOCKER);\n updateState({ blockers });\n },\n });\n return;\n }\n\n return startNavigation(historyAction, location);\n }\n );\n\n if (isBrowser) {\n // FIXME: This feels gross. How can we cleanup the lines between\n // scrollRestoration/appliedTransitions persistance?\n restoreAppliedTransitions(routerWindow, appliedViewTransitions);\n let _saveAppliedTransitions = () =>\n persistAppliedTransitions(routerWindow, appliedViewTransitions);\n routerWindow.addEventListener(\"pagehide\", _saveAppliedTransitions);\n removePageHideEventListener = () =>\n routerWindow.removeEventListener(\"pagehide\", _saveAppliedTransitions);\n }\n\n // Kick off initial data load if needed. Use Pop to avoid modifying history\n // Note we don't do any handling of lazy here. For SPA's it'll get handled\n // in the normal navigation flow. For SSR it's expected that lazy modules are\n // resolved prior to router creation since we can't go into a fallbackElement\n // UI for SSR'd apps\n if (!state.initialized) {\n startNavigation(HistoryAction.Pop, state.location, {\n initialHydration: true,\n });\n }\n\n return router;\n }\n\n // Clean up a router and it's side effects\n function dispose() {\n if (unlistenHistory) {\n unlistenHistory();\n }\n if (removePageHideEventListener) {\n removePageHideEventListener();\n }\n subscribers.clear();\n pendingNavigationController && pendingNavigationController.abort();\n state.fetchers.forEach((_, key) => deleteFetcher(key));\n state.blockers.forEach((_, key) => deleteBlocker(key));\n }\n\n // Subscribe to state updates for the router\n function subscribe(fn: RouterSubscriber) {\n subscribers.add(fn);\n return () => subscribers.delete(fn);\n }\n\n // Update our state and notify the calling context of the change\n function updateState(\n newState: Partial,\n opts: {\n flushSync?: boolean;\n viewTransitionOpts?: ViewTransitionOpts;\n } = {}\n ): void {\n state = {\n ...state,\n ...newState,\n };\n\n // Prep fetcher cleanup so we can tell the UI which fetcher data entries\n // can be removed\n let completedFetchers: string[] = [];\n let deletedFetchersKeys: string[] = [];\n\n if (future.v7_fetcherPersist) {\n state.fetchers.forEach((fetcher, key) => {\n if (fetcher.state === \"idle\") {\n if (deletedFetchers.has(key)) {\n // Unmounted from the UI and can be totally removed\n deletedFetchersKeys.push(key);\n } else {\n // Returned to idle but still mounted in the UI, so semi-remains for\n // revalidations and such\n completedFetchers.push(key);\n }\n }\n });\n }\n\n // Remove any lingering deleted fetchers that have already been removed\n // from state.fetchers\n deletedFetchers.forEach((key) => {\n if (!state.fetchers.has(key) && !fetchControllers.has(key)) {\n deletedFetchersKeys.push(key);\n }\n });\n\n // Iterate over a local copy so that if flushSync is used and we end up\n // removing and adding a new subscriber due to the useCallback dependencies,\n // we don't get ourselves into a loop calling the new subscriber immediately\n [...subscribers].forEach((subscriber) =>\n subscriber(state, {\n deletedFetchers: deletedFetchersKeys,\n viewTransitionOpts: opts.viewTransitionOpts,\n flushSync: opts.flushSync === true,\n })\n );\n\n // Remove idle fetchers from state since we only care about in-flight fetchers.\n if (future.v7_fetcherPersist) {\n completedFetchers.forEach((key) => state.fetchers.delete(key));\n deletedFetchersKeys.forEach((key) => deleteFetcher(key));\n } else {\n // We already called deleteFetcher() on these, can remove them from this\n // Set now that we've handed the keys off to the data layer\n deletedFetchersKeys.forEach((key) => deletedFetchers.delete(key));\n }\n }\n\n // Complete a navigation returning the state.navigation back to the IDLE_NAVIGATION\n // and setting state.[historyAction/location/matches] to the new route.\n // - Location is a required param\n // - Navigation will always be set to IDLE_NAVIGATION\n // - Can pass any other state in newState\n function completeNavigation(\n location: Location,\n newState: Partial>,\n { flushSync }: { flushSync?: boolean } = {}\n ): void {\n // Deduce if we're in a loading/actionReload state:\n // - We have committed actionData in the store\n // - The current navigation was a mutation submission\n // - We're past the submitting state and into the loading state\n // - The location being loaded is not the result of a redirect\n let isActionReload =\n state.actionData != null &&\n state.navigation.formMethod != null &&\n isMutationMethod(state.navigation.formMethod) &&\n state.navigation.state === \"loading\" &&\n location.state?._isRedirect !== true;\n\n let actionData: RouteData | null;\n if (newState.actionData) {\n if (Object.keys(newState.actionData).length > 0) {\n actionData = newState.actionData;\n } else {\n // Empty actionData -> clear prior actionData due to an action error\n actionData = null;\n }\n } else if (isActionReload) {\n // Keep the current data if we're wrapping up the action reload\n actionData = state.actionData;\n } else {\n // Clear actionData on any other completed navigations\n actionData = null;\n }\n\n // Always preserve any existing loaderData from re-used routes\n let loaderData = newState.loaderData\n ? mergeLoaderData(\n state.loaderData,\n newState.loaderData,\n newState.matches || [],\n newState.errors\n )\n : state.loaderData;\n\n // On a successful navigation we can assume we got through all blockers\n // so we can start fresh\n let blockers = state.blockers;\n if (blockers.size > 0) {\n blockers = new Map(blockers);\n blockers.forEach((_, k) => blockers.set(k, IDLE_BLOCKER));\n }\n\n // Always respect the user flag. Otherwise don't reset on mutation\n // submission navigations unless they redirect\n let preventScrollReset =\n pendingPreventScrollReset === true ||\n (state.navigation.formMethod != null &&\n isMutationMethod(state.navigation.formMethod) &&\n location.state?._isRedirect !== true);\n\n // Commit any in-flight routes at the end of the HMR revalidation \"navigation\"\n if (inFlightDataRoutes) {\n dataRoutes = inFlightDataRoutes;\n inFlightDataRoutes = undefined;\n }\n\n if (isUninterruptedRevalidation) {\n // If this was an uninterrupted revalidation then do not touch history\n } else if (pendingAction === HistoryAction.Pop) {\n // Do nothing for POP - URL has already been updated\n } else if (pendingAction === HistoryAction.Push) {\n init.history.push(location, location.state);\n } else if (pendingAction === HistoryAction.Replace) {\n init.history.replace(location, location.state);\n }\n\n let viewTransitionOpts: ViewTransitionOpts | undefined;\n\n // On POP, enable transitions if they were enabled on the original navigation\n if (pendingAction === HistoryAction.Pop) {\n // Forward takes precedence so they behave like the original navigation\n let priorPaths = appliedViewTransitions.get(state.location.pathname);\n if (priorPaths && priorPaths.has(location.pathname)) {\n viewTransitionOpts = {\n currentLocation: state.location,\n nextLocation: location,\n };\n } else if (appliedViewTransitions.has(location.pathname)) {\n // If we don't have a previous forward nav, assume we're popping back to\n // the new location and enable if that location previously enabled\n viewTransitionOpts = {\n currentLocation: location,\n nextLocation: state.location,\n };\n }\n } else if (pendingViewTransitionEnabled) {\n // Store the applied transition on PUSH/REPLACE\n let toPaths = appliedViewTransitions.get(state.location.pathname);\n if (toPaths) {\n toPaths.add(location.pathname);\n } else {\n toPaths = new Set([location.pathname]);\n appliedViewTransitions.set(state.location.pathname, toPaths);\n }\n viewTransitionOpts = {\n currentLocation: state.location,\n nextLocation: location,\n };\n }\n\n updateState(\n {\n ...newState, // matches, errors, fetchers go through as-is\n actionData,\n loaderData,\n historyAction: pendingAction,\n location,\n initialized: true,\n navigation: IDLE_NAVIGATION,\n revalidation: \"idle\",\n restoreScrollPosition: getSavedScrollPosition(\n location,\n newState.matches || state.matches\n ),\n preventScrollReset,\n blockers,\n },\n {\n viewTransitionOpts,\n flushSync: flushSync === true,\n }\n );\n\n // Reset stateful navigation vars\n pendingAction = HistoryAction.Pop;\n pendingPreventScrollReset = false;\n pendingViewTransitionEnabled = false;\n isUninterruptedRevalidation = false;\n isRevalidationRequired = false;\n cancelledDeferredRoutes = [];\n }\n\n // Trigger a navigation event, which can either be a numerical POP or a PUSH\n // replace with an optional submission\n async function navigate(\n to: number | To | null,\n opts?: RouterNavigateOptions\n ): Promise {\n if (typeof to === \"number\") {\n init.history.go(to);\n return;\n }\n\n let normalizedPath = normalizeTo(\n state.location,\n state.matches,\n basename,\n future.v7_prependBasename,\n to,\n future.v7_relativeSplatPath,\n opts?.fromRouteId,\n opts?.relative\n );\n let { path, submission, error } = normalizeNavigateOptions(\n future.v7_normalizeFormMethod,\n false,\n normalizedPath,\n opts\n );\n\n let currentLocation = state.location;\n let nextLocation = createLocation(state.location, path, opts && opts.state);\n\n // When using navigate as a PUSH/REPLACE we aren't reading an already-encoded\n // URL from window.location, so we need to encode it here so the behavior\n // remains the same as POP and non-data-router usages. new URL() does all\n // the same encoding we'd get from a history.pushState/window.location read\n // without having to touch history\n nextLocation = {\n ...nextLocation,\n ...init.history.encodeLocation(nextLocation),\n };\n\n let userReplace = opts && opts.replace != null ? opts.replace : undefined;\n\n let historyAction = HistoryAction.Push;\n\n if (userReplace === true) {\n historyAction = HistoryAction.Replace;\n } else if (userReplace === false) {\n // no-op\n } else if (\n submission != null &&\n isMutationMethod(submission.formMethod) &&\n submission.formAction === state.location.pathname + state.location.search\n ) {\n // By default on submissions to the current location we REPLACE so that\n // users don't have to double-click the back button to get to the prior\n // location. If the user redirects to a different location from the\n // action/loader this will be ignored and the redirect will be a PUSH\n historyAction = HistoryAction.Replace;\n }\n\n let preventScrollReset =\n opts && \"preventScrollReset\" in opts\n ? opts.preventScrollReset === true\n : undefined;\n\n let flushSync = (opts && opts.flushSync) === true;\n\n let blockerKey = shouldBlockNavigation({\n currentLocation,\n nextLocation,\n historyAction,\n });\n\n if (blockerKey) {\n // Put the blocker into a blocked state\n updateBlocker(blockerKey, {\n state: \"blocked\",\n location: nextLocation,\n proceed() {\n updateBlocker(blockerKey!, {\n state: \"proceeding\",\n proceed: undefined,\n reset: undefined,\n location: nextLocation,\n });\n // Send the same navigation through\n navigate(to, opts);\n },\n reset() {\n let blockers = new Map(state.blockers);\n blockers.set(blockerKey!, IDLE_BLOCKER);\n updateState({ blockers });\n },\n });\n return;\n }\n\n return await startNavigation(historyAction, nextLocation, {\n submission,\n // Send through the formData serialization error if we have one so we can\n // render at the right error boundary after we match routes\n pendingError: error,\n preventScrollReset,\n replace: opts && opts.replace,\n enableViewTransition: opts && opts.viewTransition,\n flushSync,\n });\n }\n\n // Revalidate all current loaders. If a navigation is in progress or if this\n // is interrupted by a navigation, allow this to \"succeed\" by calling all\n // loaders during the next loader round\n function revalidate() {\n interruptActiveLoads();\n updateState({ revalidation: \"loading\" });\n\n // If we're currently submitting an action, we don't need to start a new\n // navigation, we'll just let the follow up loader execution call all loaders\n if (state.navigation.state === \"submitting\") {\n return;\n }\n\n // If we're currently in an idle state, start a new navigation for the current\n // action/location and mark it as uninterrupted, which will skip the history\n // update in completeNavigation\n if (state.navigation.state === \"idle\") {\n startNavigation(state.historyAction, state.location, {\n startUninterruptedRevalidation: true,\n });\n return;\n }\n\n // Otherwise, if we're currently in a loading state, just start a new\n // navigation to the navigation.location but do not trigger an uninterrupted\n // revalidation so that history correctly updates once the navigation completes\n startNavigation(\n pendingAction || state.historyAction,\n state.navigation.location,\n {\n overrideNavigation: state.navigation,\n // Proxy through any rending view transition\n enableViewTransition: pendingViewTransitionEnabled === true,\n }\n );\n }\n\n // Start a navigation to the given action/location. Can optionally provide a\n // overrideNavigation which will override the normalLoad in the case of a redirect\n // navigation\n async function startNavigation(\n historyAction: HistoryAction,\n location: Location,\n opts?: {\n initialHydration?: boolean;\n submission?: Submission;\n fetcherSubmission?: Submission;\n overrideNavigation?: Navigation;\n pendingError?: ErrorResponseImpl;\n startUninterruptedRevalidation?: boolean;\n preventScrollReset?: boolean;\n replace?: boolean;\n enableViewTransition?: boolean;\n flushSync?: boolean;\n }\n ): Promise {\n // Abort any in-progress navigations and start a new one. Unset any ongoing\n // uninterrupted revalidations unless told otherwise, since we want this\n // new navigation to update history normally\n pendingNavigationController && pendingNavigationController.abort();\n pendingNavigationController = null;\n pendingAction = historyAction;\n isUninterruptedRevalidation =\n (opts && opts.startUninterruptedRevalidation) === true;\n\n // Save the current scroll position every time we start a new navigation,\n // and track whether we should reset scroll on completion\n saveScrollPosition(state.location, state.matches);\n pendingPreventScrollReset = (opts && opts.preventScrollReset) === true;\n\n pendingViewTransitionEnabled = (opts && opts.enableViewTransition) === true;\n\n let routesToUse = inFlightDataRoutes || dataRoutes;\n let loadingNavigation = opts && opts.overrideNavigation;\n let matches =\n opts?.initialHydration &&\n state.matches &&\n state.matches.length > 0 &&\n !initialMatchesIsFOW\n ? // `matchRoutes()` has already been called if we're in here via `router.initialize()`\n state.matches\n : matchRoutes(routesToUse, location, basename);\n let flushSync = (opts && opts.flushSync) === true;\n\n let fogOfWar = checkFogOfWar(matches, routesToUse, location.pathname);\n if (fogOfWar.active && fogOfWar.matches) {\n matches = fogOfWar.matches;\n }\n\n // Short circuit with a 404 on the root error boundary if we match nothing\n if (!matches) {\n let { error, notFoundMatches, route } = handleNavigational404(\n location.pathname\n );\n completeNavigation(\n location,\n {\n matches: notFoundMatches,\n loaderData: {},\n errors: {\n [route.id]: error,\n },\n },\n { flushSync }\n );\n return;\n }\n\n // Short circuit if it's only a hash change and not a revalidation or\n // mutation submission.\n //\n // Ignore on initial page loads because since the initial hydration will always\n // be \"same hash\". For example, on /page#hash and submit a \n // which will default to a navigation to /page\n if (\n state.initialized &&\n !isRevalidationRequired &&\n isHashChangeOnly(state.location, location) &&\n !(opts && opts.submission && isMutationMethod(opts.submission.formMethod))\n ) {\n completeNavigation(location, { matches }, { flushSync });\n return;\n }\n\n // Create a controller/Request for this navigation\n pendingNavigationController = new AbortController();\n let request = createClientSideRequest(\n init.history,\n location,\n pendingNavigationController.signal,\n opts && opts.submission\n );\n let pendingActionResult: PendingActionResult | undefined;\n\n if (opts && opts.pendingError) {\n // If we have a pendingError, it means the user attempted a GET submission\n // with binary FormData so assign here and skip to handleLoaders. That\n // way we handle calling loaders above the boundary etc. It's not really\n // different from an actionError in that sense.\n pendingActionResult = [\n findNearestBoundary(matches).route.id,\n { type: ResultType.error, error: opts.pendingError },\n ];\n } else if (\n opts &&\n opts.submission &&\n isMutationMethod(opts.submission.formMethod)\n ) {\n // Call action if we received an action submission\n let actionResult = await handleAction(\n request,\n location,\n opts.submission,\n matches,\n fogOfWar.active,\n { replace: opts.replace, flushSync }\n );\n\n if (actionResult.shortCircuited) {\n return;\n }\n\n // If we received a 404 from handleAction, it's because we couldn't lazily\n // discover the destination route so we don't want to call loaders\n if (actionResult.pendingActionResult) {\n let [routeId, result] = actionResult.pendingActionResult;\n if (\n isErrorResult(result) &&\n isRouteErrorResponse(result.error) &&\n result.error.status === 404\n ) {\n pendingNavigationController = null;\n\n completeNavigation(location, {\n matches: actionResult.matches,\n loaderData: {},\n errors: {\n [routeId]: result.error,\n },\n });\n return;\n }\n }\n\n matches = actionResult.matches || matches;\n pendingActionResult = actionResult.pendingActionResult;\n loadingNavigation = getLoadingNavigation(location, opts.submission);\n flushSync = false;\n // No need to do fog of war matching again on loader execution\n fogOfWar.active = false;\n\n // Create a GET request for the loaders\n request = createClientSideRequest(\n init.history,\n request.url,\n request.signal\n );\n }\n\n // Call loaders\n let {\n shortCircuited,\n matches: updatedMatches,\n loaderData,\n errors,\n } = await handleLoaders(\n request,\n location,\n matches,\n fogOfWar.active,\n loadingNavigation,\n opts && opts.submission,\n opts && opts.fetcherSubmission,\n opts && opts.replace,\n opts && opts.initialHydration === true,\n flushSync,\n pendingActionResult\n );\n\n if (shortCircuited) {\n return;\n }\n\n // Clean up now that the action/loaders have completed. Don't clean up if\n // we short circuited because pendingNavigationController will have already\n // been assigned to a new controller for the next navigation\n pendingNavigationController = null;\n\n completeNavigation(location, {\n matches: updatedMatches || matches,\n ...getActionDataForCommit(pendingActionResult),\n loaderData,\n errors,\n });\n }\n\n // Call the action matched by the leaf route for this navigation and handle\n // redirects/errors\n async function handleAction(\n request: Request,\n location: Location,\n submission: Submission,\n matches: AgnosticDataRouteMatch[],\n isFogOfWar: boolean,\n opts: { replace?: boolean; flushSync?: boolean } = {}\n ): Promise {\n interruptActiveLoads();\n\n // Put us in a submitting state\n let navigation = getSubmittingNavigation(location, submission);\n updateState({ navigation }, { flushSync: opts.flushSync === true });\n\n if (isFogOfWar) {\n let discoverResult = await discoverRoutes(\n matches,\n location.pathname,\n request.signal\n );\n if (discoverResult.type === \"aborted\") {\n return { shortCircuited: true };\n } else if (discoverResult.type === \"error\") {\n let boundaryId = findNearestBoundary(discoverResult.partialMatches)\n .route.id;\n return {\n matches: discoverResult.partialMatches,\n pendingActionResult: [\n boundaryId,\n {\n type: ResultType.error,\n error: discoverResult.error,\n },\n ],\n };\n } else if (!discoverResult.matches) {\n let { notFoundMatches, error, route } = handleNavigational404(\n location.pathname\n );\n return {\n matches: notFoundMatches,\n pendingActionResult: [\n route.id,\n {\n type: ResultType.error,\n error,\n },\n ],\n };\n } else {\n matches = discoverResult.matches;\n }\n }\n\n // Call our action and get the result\n let result: DataResult;\n let actionMatch = getTargetMatch(matches, location);\n\n if (!actionMatch.route.action && !actionMatch.route.lazy) {\n result = {\n type: ResultType.error,\n error: getInternalRouterError(405, {\n method: request.method,\n pathname: location.pathname,\n routeId: actionMatch.route.id,\n }),\n };\n } else {\n let results = await callDataStrategy(\n \"action\",\n state,\n request,\n [actionMatch],\n matches,\n null\n );\n result = results[actionMatch.route.id];\n\n if (request.signal.aborted) {\n return { shortCircuited: true };\n }\n }\n\n if (isRedirectResult(result)) {\n let replace: boolean;\n if (opts && opts.replace != null) {\n replace = opts.replace;\n } else {\n // If the user didn't explicity indicate replace behavior, replace if\n // we redirected to the exact same location we're currently at to avoid\n // double back-buttons\n let location = normalizeRedirectLocation(\n result.response.headers.get(\"Location\")!,\n new URL(request.url),\n basename\n );\n replace = location === state.location.pathname + state.location.search;\n }\n await startRedirectNavigation(request, result, true, {\n submission,\n replace,\n });\n return { shortCircuited: true };\n }\n\n if (isDeferredResult(result)) {\n throw getInternalRouterError(400, { type: \"defer-action\" });\n }\n\n if (isErrorResult(result)) {\n // Store off the pending error - we use it to determine which loaders\n // to call and will commit it when we complete the navigation\n let boundaryMatch = findNearestBoundary(matches, actionMatch.route.id);\n\n // By default, all submissions to the current location are REPLACE\n // navigations, but if the action threw an error that'll be rendered in\n // an errorElement, we fall back to PUSH so that the user can use the\n // back button to get back to the pre-submission form location to try\n // again\n if ((opts && opts.replace) !== true) {\n pendingAction = HistoryAction.Push;\n }\n\n return {\n matches,\n pendingActionResult: [boundaryMatch.route.id, result],\n };\n }\n\n return {\n matches,\n pendingActionResult: [actionMatch.route.id, result],\n };\n }\n\n // Call all applicable loaders for the given matches, handling redirects,\n // errors, etc.\n async function handleLoaders(\n request: Request,\n location: Location,\n matches: AgnosticDataRouteMatch[],\n isFogOfWar: boolean,\n overrideNavigation?: Navigation,\n submission?: Submission,\n fetcherSubmission?: Submission,\n replace?: boolean,\n initialHydration?: boolean,\n flushSync?: boolean,\n pendingActionResult?: PendingActionResult\n ): Promise {\n // Figure out the right navigation we want to use for data loading\n let loadingNavigation =\n overrideNavigation || getLoadingNavigation(location, submission);\n\n // If this was a redirect from an action we don't have a \"submission\" but\n // we have it on the loading navigation so use that if available\n let activeSubmission =\n submission ||\n fetcherSubmission ||\n getSubmissionFromNavigation(loadingNavigation);\n\n // If this is an uninterrupted revalidation, we remain in our current idle\n // state. If not, we need to switch to our loading state and load data,\n // preserving any new action data or existing action data (in the case of\n // a revalidation interrupting an actionReload)\n // If we have partialHydration enabled, then don't update the state for the\n // initial data load since it's not a \"navigation\"\n let shouldUpdateNavigationState =\n !isUninterruptedRevalidation &&\n (!future.v7_partialHydration || !initialHydration);\n\n // When fog of war is enabled, we enter our `loading` state earlier so we\n // can discover new routes during the `loading` state. We skip this if\n // we've already run actions since we would have done our matching already.\n // If the children() function threw then, we want to proceed with the\n // partial matches it discovered.\n if (isFogOfWar) {\n if (shouldUpdateNavigationState) {\n let actionData = getUpdatedActionData(pendingActionResult);\n updateState(\n {\n navigation: loadingNavigation,\n ...(actionData !== undefined ? { actionData } : {}),\n },\n {\n flushSync,\n }\n );\n }\n\n let discoverResult = await discoverRoutes(\n matches,\n location.pathname,\n request.signal\n );\n\n if (discoverResult.type === \"aborted\") {\n return { shortCircuited: true };\n } else if (discoverResult.type === \"error\") {\n let boundaryId = findNearestBoundary(discoverResult.partialMatches)\n .route.id;\n return {\n matches: discoverResult.partialMatches,\n loaderData: {},\n errors: {\n [boundaryId]: discoverResult.error,\n },\n };\n } else if (!discoverResult.matches) {\n let { error, notFoundMatches, route } = handleNavigational404(\n location.pathname\n );\n return {\n matches: notFoundMatches,\n loaderData: {},\n errors: {\n [route.id]: error,\n },\n };\n } else {\n matches = discoverResult.matches;\n }\n }\n\n let routesToUse = inFlightDataRoutes || dataRoutes;\n let [matchesToLoad, revalidatingFetchers] = getMatchesToLoad(\n init.history,\n state,\n matches,\n activeSubmission,\n location,\n future.v7_partialHydration && initialHydration === true,\n future.v7_skipActionErrorRevalidation,\n isRevalidationRequired,\n cancelledDeferredRoutes,\n cancelledFetcherLoads,\n deletedFetchers,\n fetchLoadMatches,\n fetchRedirectIds,\n routesToUse,\n basename,\n pendingActionResult\n );\n\n // Cancel pending deferreds for no-longer-matched routes or routes we're\n // about to reload. Note that if this is an action reload we would have\n // already cancelled all pending deferreds so this would be a no-op\n cancelActiveDeferreds(\n (routeId) =>\n !(matches && matches.some((m) => m.route.id === routeId)) ||\n (matchesToLoad && matchesToLoad.some((m) => m.route.id === routeId))\n );\n\n pendingNavigationLoadId = ++incrementingLoadId;\n\n // Short circuit if we have no loaders to run\n if (matchesToLoad.length === 0 && revalidatingFetchers.length === 0) {\n let updatedFetchers = markFetchRedirectsDone();\n completeNavigation(\n location,\n {\n matches,\n loaderData: {},\n // Commit pending error if we're short circuiting\n errors:\n pendingActionResult && isErrorResult(pendingActionResult[1])\n ? { [pendingActionResult[0]]: pendingActionResult[1].error }\n : null,\n ...getActionDataForCommit(pendingActionResult),\n ...(updatedFetchers ? { fetchers: new Map(state.fetchers) } : {}),\n },\n { flushSync }\n );\n return { shortCircuited: true };\n }\n\n if (shouldUpdateNavigationState) {\n let updates: Partial = {};\n if (!isFogOfWar) {\n // Only update navigation/actionNData if we didn't already do it above\n updates.navigation = loadingNavigation;\n let actionData = getUpdatedActionData(pendingActionResult);\n if (actionData !== undefined) {\n updates.actionData = actionData;\n }\n }\n if (revalidatingFetchers.length > 0) {\n updates.fetchers = getUpdatedRevalidatingFetchers(revalidatingFetchers);\n }\n updateState(updates, { flushSync });\n }\n\n revalidatingFetchers.forEach((rf) => {\n abortFetcher(rf.key);\n if (rf.controller) {\n // Fetchers use an independent AbortController so that aborting a fetcher\n // (via deleteFetcher) does not abort the triggering navigation that\n // triggered the revalidation\n fetchControllers.set(rf.key, rf.controller);\n }\n });\n\n // Proxy navigation abort through to revalidation fetchers\n let abortPendingFetchRevalidations = () =>\n revalidatingFetchers.forEach((f) => abortFetcher(f.key));\n if (pendingNavigationController) {\n pendingNavigationController.signal.addEventListener(\n \"abort\",\n abortPendingFetchRevalidations\n );\n }\n\n let { loaderResults, fetcherResults } =\n await callLoadersAndMaybeResolveData(\n state,\n matches,\n matchesToLoad,\n revalidatingFetchers,\n request\n );\n\n if (request.signal.aborted) {\n return { shortCircuited: true };\n }\n\n // Clean up _after_ loaders have completed. Don't clean up if we short\n // circuited because fetchControllers would have been aborted and\n // reassigned to new controllers for the next navigation\n if (pendingNavigationController) {\n pendingNavigationController.signal.removeEventListener(\n \"abort\",\n abortPendingFetchRevalidations\n );\n }\n\n revalidatingFetchers.forEach((rf) => fetchControllers.delete(rf.key));\n\n // If any loaders returned a redirect Response, start a new REPLACE navigation\n let redirect = findRedirect(loaderResults);\n if (redirect) {\n await startRedirectNavigation(request, redirect.result, true, {\n replace,\n });\n return { shortCircuited: true };\n }\n\n redirect = findRedirect(fetcherResults);\n if (redirect) {\n // If this redirect came from a fetcher make sure we mark it in\n // fetchRedirectIds so it doesn't get revalidated on the next set of\n // loader executions\n fetchRedirectIds.add(redirect.key);\n await startRedirectNavigation(request, redirect.result, true, {\n replace,\n });\n return { shortCircuited: true };\n }\n\n // Process and commit output from loaders\n let { loaderData, errors } = processLoaderData(\n state,\n matches,\n loaderResults,\n pendingActionResult,\n revalidatingFetchers,\n fetcherResults,\n activeDeferreds\n );\n\n // Wire up subscribers to update loaderData as promises settle\n activeDeferreds.forEach((deferredData, routeId) => {\n deferredData.subscribe((aborted) => {\n // Note: No need to updateState here since the TrackedPromise on\n // loaderData is stable across resolve/reject\n // Remove this instance if we were aborted or if promises have settled\n if (aborted || deferredData.done) {\n activeDeferreds.delete(routeId);\n }\n });\n });\n\n // Preserve SSR errors during partial hydration\n if (future.v7_partialHydration && initialHydration && state.errors) {\n errors = { ...state.errors, ...errors };\n }\n\n let updatedFetchers = markFetchRedirectsDone();\n let didAbortFetchLoads = abortStaleFetchLoads(pendingNavigationLoadId);\n let shouldUpdateFetchers =\n updatedFetchers || didAbortFetchLoads || revalidatingFetchers.length > 0;\n\n return {\n matches,\n loaderData,\n errors,\n ...(shouldUpdateFetchers ? { fetchers: new Map(state.fetchers) } : {}),\n };\n }\n\n function getUpdatedActionData(\n pendingActionResult: PendingActionResult | undefined\n ): Record | null | undefined {\n if (pendingActionResult && !isErrorResult(pendingActionResult[1])) {\n // This is cast to `any` currently because `RouteData`uses any and it\n // would be a breaking change to use any.\n // TODO: v7 - change `RouteData` to use `unknown` instead of `any`\n return {\n [pendingActionResult[0]]: pendingActionResult[1].data as any,\n };\n } else if (state.actionData) {\n if (Object.keys(state.actionData).length === 0) {\n return null;\n } else {\n return state.actionData;\n }\n }\n }\n\n function getUpdatedRevalidatingFetchers(\n revalidatingFetchers: RevalidatingFetcher[]\n ) {\n revalidatingFetchers.forEach((rf) => {\n let fetcher = state.fetchers.get(rf.key);\n let revalidatingFetcher = getLoadingFetcher(\n undefined,\n fetcher ? fetcher.data : undefined\n );\n state.fetchers.set(rf.key, revalidatingFetcher);\n });\n return new Map(state.fetchers);\n }\n\n // Trigger a fetcher load/submit for the given fetcher key\n function fetch(\n key: string,\n routeId: string,\n href: string | null,\n opts?: RouterFetchOptions\n ) {\n if (isServer) {\n throw new Error(\n \"router.fetch() was called during the server render, but it shouldn't be. \" +\n \"You are likely calling a useFetcher() method in the body of your component. \" +\n \"Try moving it to a useEffect or a callback.\"\n );\n }\n\n abortFetcher(key);\n\n let flushSync = (opts && opts.flushSync) === true;\n\n let routesToUse = inFlightDataRoutes || dataRoutes;\n let normalizedPath = normalizeTo(\n state.location,\n state.matches,\n basename,\n future.v7_prependBasename,\n href,\n future.v7_relativeSplatPath,\n routeId,\n opts?.relative\n );\n let matches = matchRoutes(routesToUse, normalizedPath, basename);\n\n let fogOfWar = checkFogOfWar(matches, routesToUse, normalizedPath);\n if (fogOfWar.active && fogOfWar.matches) {\n matches = fogOfWar.matches;\n }\n\n if (!matches) {\n setFetcherError(\n key,\n routeId,\n getInternalRouterError(404, { pathname: normalizedPath }),\n { flushSync }\n );\n return;\n }\n\n let { path, submission, error } = normalizeNavigateOptions(\n future.v7_normalizeFormMethod,\n true,\n normalizedPath,\n opts\n );\n\n if (error) {\n setFetcherError(key, routeId, error, { flushSync });\n return;\n }\n\n let match = getTargetMatch(matches, path);\n\n let preventScrollReset = (opts && opts.preventScrollReset) === true;\n\n if (submission && isMutationMethod(submission.formMethod)) {\n handleFetcherAction(\n key,\n routeId,\n path,\n match,\n matches,\n fogOfWar.active,\n flushSync,\n preventScrollReset,\n submission\n );\n return;\n }\n\n // Store off the match so we can call it's shouldRevalidate on subsequent\n // revalidations\n fetchLoadMatches.set(key, { routeId, path });\n handleFetcherLoader(\n key,\n routeId,\n path,\n match,\n matches,\n fogOfWar.active,\n flushSync,\n preventScrollReset,\n submission\n );\n }\n\n // Call the action for the matched fetcher.submit(), and then handle redirects,\n // errors, and revalidation\n async function handleFetcherAction(\n key: string,\n routeId: string,\n path: string,\n match: AgnosticDataRouteMatch,\n requestMatches: AgnosticDataRouteMatch[],\n isFogOfWar: boolean,\n flushSync: boolean,\n preventScrollReset: boolean,\n submission: Submission\n ) {\n interruptActiveLoads();\n fetchLoadMatches.delete(key);\n\n function detectAndHandle405Error(m: AgnosticDataRouteMatch) {\n if (!m.route.action && !m.route.lazy) {\n let error = getInternalRouterError(405, {\n method: submission.formMethod,\n pathname: path,\n routeId: routeId,\n });\n setFetcherError(key, routeId, error, { flushSync });\n return true;\n }\n return false;\n }\n\n if (!isFogOfWar && detectAndHandle405Error(match)) {\n return;\n }\n\n // Put this fetcher into it's submitting state\n let existingFetcher = state.fetchers.get(key);\n updateFetcherState(key, getSubmittingFetcher(submission, existingFetcher), {\n flushSync,\n });\n\n let abortController = new AbortController();\n let fetchRequest = createClientSideRequest(\n init.history,\n path,\n abortController.signal,\n submission\n );\n\n if (isFogOfWar) {\n let discoverResult = await discoverRoutes(\n requestMatches,\n new URL(fetchRequest.url).pathname,\n fetchRequest.signal\n );\n\n if (discoverResult.type === \"aborted\") {\n return;\n } else if (discoverResult.type === \"error\") {\n setFetcherError(key, routeId, discoverResult.error, { flushSync });\n return;\n } else if (!discoverResult.matches) {\n setFetcherError(\n key,\n routeId,\n getInternalRouterError(404, { pathname: path }),\n { flushSync }\n );\n return;\n } else {\n requestMatches = discoverResult.matches;\n match = getTargetMatch(requestMatches, path);\n\n if (detectAndHandle405Error(match)) {\n return;\n }\n }\n }\n\n // Call the action for the fetcher\n fetchControllers.set(key, abortController);\n\n let originatingLoadId = incrementingLoadId;\n let actionResults = await callDataStrategy(\n \"action\",\n state,\n fetchRequest,\n [match],\n requestMatches,\n key\n );\n let actionResult = actionResults[match.route.id];\n\n if (fetchRequest.signal.aborted) {\n // We can delete this so long as we weren't aborted by our own fetcher\n // re-submit which would have put _new_ controller is in fetchControllers\n if (fetchControllers.get(key) === abortController) {\n fetchControllers.delete(key);\n }\n return;\n }\n\n // When using v7_fetcherPersist, we don't want errors bubbling up to the UI\n // or redirects processed for unmounted fetchers so we just revert them to\n // idle\n if (future.v7_fetcherPersist && deletedFetchers.has(key)) {\n if (isRedirectResult(actionResult) || isErrorResult(actionResult)) {\n updateFetcherState(key, getDoneFetcher(undefined));\n return;\n }\n // Let SuccessResult's fall through for revalidation\n } else {\n if (isRedirectResult(actionResult)) {\n fetchControllers.delete(key);\n if (pendingNavigationLoadId > originatingLoadId) {\n // A new navigation was kicked off after our action started, so that\n // should take precedence over this redirect navigation. We already\n // set isRevalidationRequired so all loaders for the new route should\n // fire unless opted out via shouldRevalidate\n updateFetcherState(key, getDoneFetcher(undefined));\n return;\n } else {\n fetchRedirectIds.add(key);\n updateFetcherState(key, getLoadingFetcher(submission));\n return startRedirectNavigation(fetchRequest, actionResult, false, {\n fetcherSubmission: submission,\n preventScrollReset,\n });\n }\n }\n\n // Process any non-redirect errors thrown\n if (isErrorResult(actionResult)) {\n setFetcherError(key, routeId, actionResult.error);\n return;\n }\n }\n\n if (isDeferredResult(actionResult)) {\n throw getInternalRouterError(400, { type: \"defer-action\" });\n }\n\n // Start the data load for current matches, or the next location if we're\n // in the middle of a navigation\n let nextLocation = state.navigation.location || state.location;\n let revalidationRequest = createClientSideRequest(\n init.history,\n nextLocation,\n abortController.signal\n );\n let routesToUse = inFlightDataRoutes || dataRoutes;\n let matches =\n state.navigation.state !== \"idle\"\n ? matchRoutes(routesToUse, state.navigation.location, basename)\n : state.matches;\n\n invariant(matches, \"Didn't find any matches after fetcher action\");\n\n let loadId = ++incrementingLoadId;\n fetchReloadIds.set(key, loadId);\n\n let loadFetcher = getLoadingFetcher(submission, actionResult.data);\n state.fetchers.set(key, loadFetcher);\n\n let [matchesToLoad, revalidatingFetchers] = getMatchesToLoad(\n init.history,\n state,\n matches,\n submission,\n nextLocation,\n false,\n future.v7_skipActionErrorRevalidation,\n isRevalidationRequired,\n cancelledDeferredRoutes,\n cancelledFetcherLoads,\n deletedFetchers,\n fetchLoadMatches,\n fetchRedirectIds,\n routesToUse,\n basename,\n [match.route.id, actionResult]\n );\n\n // Put all revalidating fetchers into the loading state, except for the\n // current fetcher which we want to keep in it's current loading state which\n // contains it's action submission info + action data\n revalidatingFetchers\n .filter((rf) => rf.key !== key)\n .forEach((rf) => {\n let staleKey = rf.key;\n let existingFetcher = state.fetchers.get(staleKey);\n let revalidatingFetcher = getLoadingFetcher(\n undefined,\n existingFetcher ? existingFetcher.data : undefined\n );\n state.fetchers.set(staleKey, revalidatingFetcher);\n abortFetcher(staleKey);\n if (rf.controller) {\n fetchControllers.set(staleKey, rf.controller);\n }\n });\n\n updateState({ fetchers: new Map(state.fetchers) });\n\n let abortPendingFetchRevalidations = () =>\n revalidatingFetchers.forEach((rf) => abortFetcher(rf.key));\n\n abortController.signal.addEventListener(\n \"abort\",\n abortPendingFetchRevalidations\n );\n\n let { loaderResults, fetcherResults } =\n await callLoadersAndMaybeResolveData(\n state,\n matches,\n matchesToLoad,\n revalidatingFetchers,\n revalidationRequest\n );\n\n if (abortController.signal.aborted) {\n return;\n }\n\n abortController.signal.removeEventListener(\n \"abort\",\n abortPendingFetchRevalidations\n );\n\n fetchReloadIds.delete(key);\n fetchControllers.delete(key);\n revalidatingFetchers.forEach((r) => fetchControllers.delete(r.key));\n\n let redirect = findRedirect(loaderResults);\n if (redirect) {\n return startRedirectNavigation(\n revalidationRequest,\n redirect.result,\n false,\n { preventScrollReset }\n );\n }\n\n redirect = findRedirect(fetcherResults);\n if (redirect) {\n // If this redirect came from a fetcher make sure we mark it in\n // fetchRedirectIds so it doesn't get revalidated on the next set of\n // loader executions\n fetchRedirectIds.add(redirect.key);\n return startRedirectNavigation(\n revalidationRequest,\n redirect.result,\n false,\n { preventScrollReset }\n );\n }\n\n // Process and commit output from loaders\n let { loaderData, errors } = processLoaderData(\n state,\n matches,\n loaderResults,\n undefined,\n revalidatingFetchers,\n fetcherResults,\n activeDeferreds\n );\n\n // Since we let revalidations complete even if the submitting fetcher was\n // deleted, only put it back to idle if it hasn't been deleted\n if (state.fetchers.has(key)) {\n let doneFetcher = getDoneFetcher(actionResult.data);\n state.fetchers.set(key, doneFetcher);\n }\n\n abortStaleFetchLoads(loadId);\n\n // If we are currently in a navigation loading state and this fetcher is\n // more recent than the navigation, we want the newer data so abort the\n // navigation and complete it with the fetcher data\n if (\n state.navigation.state === \"loading\" &&\n loadId > pendingNavigationLoadId\n ) {\n invariant(pendingAction, \"Expected pending action\");\n pendingNavigationController && pendingNavigationController.abort();\n\n completeNavigation(state.navigation.location, {\n matches,\n loaderData,\n errors,\n fetchers: new Map(state.fetchers),\n });\n } else {\n // otherwise just update with the fetcher data, preserving any existing\n // loaderData for loaders that did not need to reload. We have to\n // manually merge here since we aren't going through completeNavigation\n updateState({\n errors,\n loaderData: mergeLoaderData(\n state.loaderData,\n loaderData,\n matches,\n errors\n ),\n fetchers: new Map(state.fetchers),\n });\n isRevalidationRequired = false;\n }\n }\n\n // Call the matched loader for fetcher.load(), handling redirects, errors, etc.\n async function handleFetcherLoader(\n key: string,\n routeId: string,\n path: string,\n match: AgnosticDataRouteMatch,\n matches: AgnosticDataRouteMatch[],\n isFogOfWar: boolean,\n flushSync: boolean,\n preventScrollReset: boolean,\n submission?: Submission\n ) {\n let existingFetcher = state.fetchers.get(key);\n updateFetcherState(\n key,\n getLoadingFetcher(\n submission,\n existingFetcher ? existingFetcher.data : undefined\n ),\n { flushSync }\n );\n\n let abortController = new AbortController();\n let fetchRequest = createClientSideRequest(\n init.history,\n path,\n abortController.signal\n );\n\n if (isFogOfWar) {\n let discoverResult = await discoverRoutes(\n matches,\n new URL(fetchRequest.url).pathname,\n fetchRequest.signal\n );\n\n if (discoverResult.type === \"aborted\") {\n return;\n } else if (discoverResult.type === \"error\") {\n setFetcherError(key, routeId, discoverResult.error, { flushSync });\n return;\n } else if (!discoverResult.matches) {\n setFetcherError(\n key,\n routeId,\n getInternalRouterError(404, { pathname: path }),\n { flushSync }\n );\n return;\n } else {\n matches = discoverResult.matches;\n match = getTargetMatch(matches, path);\n }\n }\n\n // Call the loader for this fetcher route match\n fetchControllers.set(key, abortController);\n\n let originatingLoadId = incrementingLoadId;\n let results = await callDataStrategy(\n \"loader\",\n state,\n fetchRequest,\n [match],\n matches,\n key\n );\n let result = results[match.route.id];\n\n // Deferred isn't supported for fetcher loads, await everything and treat it\n // as a normal load. resolveDeferredData will return undefined if this\n // fetcher gets aborted, so we just leave result untouched and short circuit\n // below if that happens\n if (isDeferredResult(result)) {\n result =\n (await resolveDeferredData(result, fetchRequest.signal, true)) ||\n result;\n }\n\n // We can delete this so long as we weren't aborted by our our own fetcher\n // re-load which would have put _new_ controller is in fetchControllers\n if (fetchControllers.get(key) === abortController) {\n fetchControllers.delete(key);\n }\n\n if (fetchRequest.signal.aborted) {\n return;\n }\n\n // We don't want errors bubbling up or redirects followed for unmounted\n // fetchers, so short circuit here if it was removed from the UI\n if (deletedFetchers.has(key)) {\n updateFetcherState(key, getDoneFetcher(undefined));\n return;\n }\n\n // If the loader threw a redirect Response, start a new REPLACE navigation\n if (isRedirectResult(result)) {\n if (pendingNavigationLoadId > originatingLoadId) {\n // A new navigation was kicked off after our loader started, so that\n // should take precedence over this redirect navigation\n updateFetcherState(key, getDoneFetcher(undefined));\n return;\n } else {\n fetchRedirectIds.add(key);\n await startRedirectNavigation(fetchRequest, result, false, {\n preventScrollReset,\n });\n return;\n }\n }\n\n // Process any non-redirect errors thrown\n if (isErrorResult(result)) {\n setFetcherError(key, routeId, result.error);\n return;\n }\n\n invariant(!isDeferredResult(result), \"Unhandled fetcher deferred data\");\n\n // Put the fetcher back into an idle state\n updateFetcherState(key, getDoneFetcher(result.data));\n }\n\n /**\n * Utility function to handle redirects returned from an action or loader.\n * Normally, a redirect \"replaces\" the navigation that triggered it. So, for\n * example:\n *\n * - user is on /a\n * - user clicks a link to /b\n * - loader for /b redirects to /c\n *\n * In a non-JS app the browser would track the in-flight navigation to /b and\n * then replace it with /c when it encountered the redirect response. In\n * the end it would only ever update the URL bar with /c.\n *\n * In client-side routing using pushState/replaceState, we aim to emulate\n * this behavior and we also do not update history until the end of the\n * navigation (including processed redirects). This means that we never\n * actually touch history until we've processed redirects, so we just use\n * the history action from the original navigation (PUSH or REPLACE).\n */\n async function startRedirectNavigation(\n request: Request,\n redirect: RedirectResult,\n isNavigation: boolean,\n {\n submission,\n fetcherSubmission,\n preventScrollReset,\n replace,\n }: {\n submission?: Submission;\n fetcherSubmission?: Submission;\n preventScrollReset?: boolean;\n replace?: boolean;\n } = {}\n ) {\n if (redirect.response.headers.has(\"X-Remix-Revalidate\")) {\n isRevalidationRequired = true;\n }\n\n let location = redirect.response.headers.get(\"Location\");\n invariant(location, \"Expected a Location header on the redirect Response\");\n location = normalizeRedirectLocation(\n location,\n new URL(request.url),\n basename\n );\n let redirectLocation = createLocation(state.location, location, {\n _isRedirect: true,\n });\n\n if (isBrowser) {\n let isDocumentReload = false;\n\n if (redirect.response.headers.has(\"X-Remix-Reload-Document\")) {\n // Hard reload if the response contained X-Remix-Reload-Document\n isDocumentReload = true;\n } else if (ABSOLUTE_URL_REGEX.test(location)) {\n const url = init.history.createURL(location);\n isDocumentReload =\n // Hard reload if it's an absolute URL to a new origin\n url.origin !== routerWindow.location.origin ||\n // Hard reload if it's an absolute URL that does not match our basename\n stripBasename(url.pathname, basename) == null;\n }\n\n if (isDocumentReload) {\n if (replace) {\n routerWindow.location.replace(location);\n } else {\n routerWindow.location.assign(location);\n }\n return;\n }\n }\n\n // There's no need to abort on redirects, since we don't detect the\n // redirect until the action/loaders have settled\n pendingNavigationController = null;\n\n let redirectHistoryAction =\n replace === true || redirect.response.headers.has(\"X-Remix-Replace\")\n ? HistoryAction.Replace\n : HistoryAction.Push;\n\n // Use the incoming submission if provided, fallback on the active one in\n // state.navigation\n let { formMethod, formAction, formEncType } = state.navigation;\n if (\n !submission &&\n !fetcherSubmission &&\n formMethod &&\n formAction &&\n formEncType\n ) {\n submission = getSubmissionFromNavigation(state.navigation);\n }\n\n // If this was a 307/308 submission we want to preserve the HTTP method and\n // re-submit the GET/POST/PUT/PATCH/DELETE as a submission navigation to the\n // redirected location\n let activeSubmission = submission || fetcherSubmission;\n if (\n redirectPreserveMethodStatusCodes.has(redirect.response.status) &&\n activeSubmission &&\n isMutationMethod(activeSubmission.formMethod)\n ) {\n await startNavigation(redirectHistoryAction, redirectLocation, {\n submission: {\n ...activeSubmission,\n formAction: location,\n },\n // Preserve these flags across redirects\n preventScrollReset: preventScrollReset || pendingPreventScrollReset,\n enableViewTransition: isNavigation\n ? pendingViewTransitionEnabled\n : undefined,\n });\n } else {\n // If we have a navigation submission, we will preserve it through the\n // redirect navigation\n let overrideNavigation = getLoadingNavigation(\n redirectLocation,\n submission\n );\n await startNavigation(redirectHistoryAction, redirectLocation, {\n overrideNavigation,\n // Send fetcher submissions through for shouldRevalidate\n fetcherSubmission,\n // Preserve these flags across redirects\n preventScrollReset: preventScrollReset || pendingPreventScrollReset,\n enableViewTransition: isNavigation\n ? pendingViewTransitionEnabled\n : undefined,\n });\n }\n }\n\n // Utility wrapper for calling dataStrategy client-side without having to\n // pass around the manifest, mapRouteProperties, etc.\n async function callDataStrategy(\n type: \"loader\" | \"action\",\n state: RouterState,\n request: Request,\n matchesToLoad: AgnosticDataRouteMatch[],\n matches: AgnosticDataRouteMatch[],\n fetcherKey: string | null\n ): Promise> {\n let results: Record;\n let dataResults: Record = {};\n try {\n results = await callDataStrategyImpl(\n dataStrategyImpl,\n type,\n state,\n request,\n matchesToLoad,\n matches,\n fetcherKey,\n manifest,\n mapRouteProperties\n );\n } catch (e) {\n // If the outer dataStrategy method throws, just return the error for all\n // matches - and it'll naturally bubble to the root\n matchesToLoad.forEach((m) => {\n dataResults[m.route.id] = {\n type: ResultType.error,\n error: e,\n };\n });\n return dataResults;\n }\n\n for (let [routeId, result] of Object.entries(results)) {\n if (isRedirectDataStrategyResultResult(result)) {\n let response = result.result as Response;\n dataResults[routeId] = {\n type: ResultType.redirect,\n response: normalizeRelativeRoutingRedirectResponse(\n response,\n request,\n routeId,\n matches,\n basename,\n future.v7_relativeSplatPath\n ),\n };\n } else {\n dataResults[routeId] = await convertDataStrategyResultToDataResult(\n result\n );\n }\n }\n\n return dataResults;\n }\n\n async function callLoadersAndMaybeResolveData(\n state: RouterState,\n matches: AgnosticDataRouteMatch[],\n matchesToLoad: AgnosticDataRouteMatch[],\n fetchersToLoad: RevalidatingFetcher[],\n request: Request\n ) {\n let currentMatches = state.matches;\n\n // Kick off loaders and fetchers in parallel\n let loaderResultsPromise = callDataStrategy(\n \"loader\",\n state,\n request,\n matchesToLoad,\n matches,\n null\n );\n\n let fetcherResultsPromise = Promise.all(\n fetchersToLoad.map(async (f) => {\n if (f.matches && f.match && f.controller) {\n let results = await callDataStrategy(\n \"loader\",\n state,\n createClientSideRequest(init.history, f.path, f.controller.signal),\n [f.match],\n f.matches,\n f.key\n );\n let result = results[f.match.route.id];\n // Fetcher results are keyed by fetcher key from here on out, not routeId\n return { [f.key]: result };\n } else {\n return Promise.resolve({\n [f.key]: {\n type: ResultType.error,\n error: getInternalRouterError(404, {\n pathname: f.path,\n }),\n } as ErrorResult,\n });\n }\n })\n );\n\n let loaderResults = await loaderResultsPromise;\n let fetcherResults = (await fetcherResultsPromise).reduce(\n (acc, r) => Object.assign(acc, r),\n {}\n );\n\n await Promise.all([\n resolveNavigationDeferredResults(\n matches,\n loaderResults,\n request.signal,\n currentMatches,\n state.loaderData\n ),\n resolveFetcherDeferredResults(matches, fetcherResults, fetchersToLoad),\n ]);\n\n return {\n loaderResults,\n fetcherResults,\n };\n }\n\n function interruptActiveLoads() {\n // Every interruption triggers a revalidation\n isRevalidationRequired = true;\n\n // Cancel pending route-level deferreds and mark cancelled routes for\n // revalidation\n cancelledDeferredRoutes.push(...cancelActiveDeferreds());\n\n // Abort in-flight fetcher loads\n fetchLoadMatches.forEach((_, key) => {\n if (fetchControllers.has(key)) {\n cancelledFetcherLoads.add(key);\n }\n abortFetcher(key);\n });\n }\n\n function updateFetcherState(\n key: string,\n fetcher: Fetcher,\n opts: { flushSync?: boolean } = {}\n ) {\n state.fetchers.set(key, fetcher);\n updateState(\n { fetchers: new Map(state.fetchers) },\n { flushSync: (opts && opts.flushSync) === true }\n );\n }\n\n function setFetcherError(\n key: string,\n routeId: string,\n error: any,\n opts: { flushSync?: boolean } = {}\n ) {\n let boundaryMatch = findNearestBoundary(state.matches, routeId);\n deleteFetcher(key);\n updateState(\n {\n errors: {\n [boundaryMatch.route.id]: error,\n },\n fetchers: new Map(state.fetchers),\n },\n { flushSync: (opts && opts.flushSync) === true }\n );\n }\n\n function getFetcher(key: string): Fetcher {\n activeFetchers.set(key, (activeFetchers.get(key) || 0) + 1);\n // If this fetcher was previously marked for deletion, unmark it since we\n // have a new instance\n if (deletedFetchers.has(key)) {\n deletedFetchers.delete(key);\n }\n return state.fetchers.get(key) || IDLE_FETCHER;\n }\n\n function deleteFetcher(key: string): void {\n let fetcher = state.fetchers.get(key);\n // Don't abort the controller if this is a deletion of a fetcher.submit()\n // in it's loading phase since - we don't want to abort the corresponding\n // revalidation and want them to complete and land\n if (\n fetchControllers.has(key) &&\n !(fetcher && fetcher.state === \"loading\" && fetchReloadIds.has(key))\n ) {\n abortFetcher(key);\n }\n fetchLoadMatches.delete(key);\n fetchReloadIds.delete(key);\n fetchRedirectIds.delete(key);\n\n // If we opted into the flag we can clear this now since we're calling\n // deleteFetcher() at the end of updateState() and we've already handed the\n // deleted fetcher keys off to the data layer.\n // If not, we're eagerly calling deleteFetcher() and we need to keep this\n // Set populated until the next updateState call, and we'll clear\n // `deletedFetchers` then\n if (future.v7_fetcherPersist) {\n deletedFetchers.delete(key);\n }\n\n cancelledFetcherLoads.delete(key);\n state.fetchers.delete(key);\n }\n\n function deleteFetcherAndUpdateState(key: string): void {\n let count = (activeFetchers.get(key) || 0) - 1;\n if (count <= 0) {\n activeFetchers.delete(key);\n deletedFetchers.add(key);\n if (!future.v7_fetcherPersist) {\n deleteFetcher(key);\n }\n } else {\n activeFetchers.set(key, count);\n }\n\n updateState({ fetchers: new Map(state.fetchers) });\n }\n\n function abortFetcher(key: string) {\n let controller = fetchControllers.get(key);\n if (controller) {\n controller.abort();\n fetchControllers.delete(key);\n }\n }\n\n function markFetchersDone(keys: string[]) {\n for (let key of keys) {\n let fetcher = getFetcher(key);\n let doneFetcher = getDoneFetcher(fetcher.data);\n state.fetchers.set(key, doneFetcher);\n }\n }\n\n function markFetchRedirectsDone(): boolean {\n let doneKeys = [];\n let updatedFetchers = false;\n for (let key of fetchRedirectIds) {\n let fetcher = state.fetchers.get(key);\n invariant(fetcher, `Expected fetcher: ${key}`);\n if (fetcher.state === \"loading\") {\n fetchRedirectIds.delete(key);\n doneKeys.push(key);\n updatedFetchers = true;\n }\n }\n markFetchersDone(doneKeys);\n return updatedFetchers;\n }\n\n function abortStaleFetchLoads(landedId: number): boolean {\n let yeetedKeys = [];\n for (let [key, id] of fetchReloadIds) {\n if (id < landedId) {\n let fetcher = state.fetchers.get(key);\n invariant(fetcher, `Expected fetcher: ${key}`);\n if (fetcher.state === \"loading\") {\n abortFetcher(key);\n fetchReloadIds.delete(key);\n yeetedKeys.push(key);\n }\n }\n }\n markFetchersDone(yeetedKeys);\n return yeetedKeys.length > 0;\n }\n\n function getBlocker(key: string, fn: BlockerFunction) {\n let blocker: Blocker = state.blockers.get(key) || IDLE_BLOCKER;\n\n if (blockerFunctions.get(key) !== fn) {\n blockerFunctions.set(key, fn);\n }\n\n return blocker;\n }\n\n function deleteBlocker(key: string) {\n state.blockers.delete(key);\n blockerFunctions.delete(key);\n }\n\n // Utility function to update blockers, ensuring valid state transitions\n function updateBlocker(key: string, newBlocker: Blocker) {\n let blocker = state.blockers.get(key) || IDLE_BLOCKER;\n\n // Poor mans state machine :)\n // https://mermaid.live/edit#pako:eNqVkc9OwzAMxl8l8nnjAYrEtDIOHEBIgwvKJTReGy3_lDpIqO27k6awMG0XcrLlnz87nwdonESogKXXBuE79rq75XZO3-yHds0RJVuv70YrPlUrCEe2HfrORS3rubqZfuhtpg5C9wk5tZ4VKcRUq88q9Z8RS0-48cE1iHJkL0ugbHuFLus9L6spZy8nX9MP2CNdomVaposqu3fGayT8T8-jJQwhepo_UtpgBQaDEUom04dZhAN1aJBDlUKJBxE1ceB2Smj0Mln-IBW5AFU2dwUiktt_2Qaq2dBfaKdEup85UV7Yd-dKjlnkabl2Pvr0DTkTreM\n invariant(\n (blocker.state === \"unblocked\" && newBlocker.state === \"blocked\") ||\n (blocker.state === \"blocked\" && newBlocker.state === \"blocked\") ||\n (blocker.state === \"blocked\" && newBlocker.state === \"proceeding\") ||\n (blocker.state === \"blocked\" && newBlocker.state === \"unblocked\") ||\n (blocker.state === \"proceeding\" && newBlocker.state === \"unblocked\"),\n `Invalid blocker state transition: ${blocker.state} -> ${newBlocker.state}`\n );\n\n let blockers = new Map(state.blockers);\n blockers.set(key, newBlocker);\n updateState({ blockers });\n }\n\n function shouldBlockNavigation({\n currentLocation,\n nextLocation,\n historyAction,\n }: {\n currentLocation: Location;\n nextLocation: Location;\n historyAction: HistoryAction;\n }): string | undefined {\n if (blockerFunctions.size === 0) {\n return;\n }\n\n // We ony support a single active blocker at the moment since we don't have\n // any compelling use cases for multi-blocker yet\n if (blockerFunctions.size > 1) {\n warning(false, \"A router only supports one blocker at a time\");\n }\n\n let entries = Array.from(blockerFunctions.entries());\n let [blockerKey, blockerFunction] = entries[entries.length - 1];\n let blocker = state.blockers.get(blockerKey);\n\n if (blocker && blocker.state === \"proceeding\") {\n // If the blocker is currently proceeding, we don't need to re-check\n // it and can let this navigation continue\n return;\n }\n\n // At this point, we know we're unblocked/blocked so we need to check the\n // user-provided blocker function\n if (blockerFunction({ currentLocation, nextLocation, historyAction })) {\n return blockerKey;\n }\n }\n\n function handleNavigational404(pathname: string) {\n let error = getInternalRouterError(404, { pathname });\n let routesToUse = inFlightDataRoutes || dataRoutes;\n let { matches, route } = getShortCircuitMatches(routesToUse);\n\n // Cancel all pending deferred on 404s since we don't keep any routes\n cancelActiveDeferreds();\n\n return { notFoundMatches: matches, route, error };\n }\n\n function cancelActiveDeferreds(\n predicate?: (routeId: string) => boolean\n ): string[] {\n let cancelledRouteIds: string[] = [];\n activeDeferreds.forEach((dfd, routeId) => {\n if (!predicate || predicate(routeId)) {\n // Cancel the deferred - but do not remove from activeDeferreds here -\n // we rely on the subscribers to do that so our tests can assert proper\n // cleanup via _internalActiveDeferreds\n dfd.cancel();\n cancelledRouteIds.push(routeId);\n activeDeferreds.delete(routeId);\n }\n });\n return cancelledRouteIds;\n }\n\n // Opt in to capturing and reporting scroll positions during navigations,\n // used by the component\n function enableScrollRestoration(\n positions: Record,\n getPosition: GetScrollPositionFunction,\n getKey?: GetScrollRestorationKeyFunction\n ) {\n savedScrollPositions = positions;\n getScrollPosition = getPosition;\n getScrollRestorationKey = getKey || null;\n\n // Perform initial hydration scroll restoration, since we miss the boat on\n // the initial updateState() because we've not yet rendered \n // and therefore have no savedScrollPositions available\n if (!initialScrollRestored && state.navigation === IDLE_NAVIGATION) {\n initialScrollRestored = true;\n let y = getSavedScrollPosition(state.location, state.matches);\n if (y != null) {\n updateState({ restoreScrollPosition: y });\n }\n }\n\n return () => {\n savedScrollPositions = null;\n getScrollPosition = null;\n getScrollRestorationKey = null;\n };\n }\n\n function getScrollKey(location: Location, matches: AgnosticDataRouteMatch[]) {\n if (getScrollRestorationKey) {\n let key = getScrollRestorationKey(\n location,\n matches.map((m) => convertRouteMatchToUiMatch(m, state.loaderData))\n );\n return key || location.key;\n }\n return location.key;\n }\n\n function saveScrollPosition(\n location: Location,\n matches: AgnosticDataRouteMatch[]\n ): void {\n if (savedScrollPositions && getScrollPosition) {\n let key = getScrollKey(location, matches);\n savedScrollPositions[key] = getScrollPosition();\n }\n }\n\n function getSavedScrollPosition(\n location: Location,\n matches: AgnosticDataRouteMatch[]\n ): number | null {\n if (savedScrollPositions) {\n let key = getScrollKey(location, matches);\n let y = savedScrollPositions[key];\n if (typeof y === \"number\") {\n return y;\n }\n }\n return null;\n }\n\n function checkFogOfWar(\n matches: AgnosticDataRouteMatch[] | null,\n routesToUse: AgnosticDataRouteObject[],\n pathname: string\n ): { active: boolean; matches: AgnosticDataRouteMatch[] | null } {\n if (patchRoutesOnNavigationImpl) {\n if (!matches) {\n let fogMatches = matchRoutesImpl(\n routesToUse,\n pathname,\n basename,\n true\n );\n\n return { active: true, matches: fogMatches || [] };\n } else {\n if (Object.keys(matches[0].params).length > 0) {\n // If we matched a dynamic param or a splat, it might only be because\n // we haven't yet discovered other routes that would match with a\n // higher score. Call patchRoutesOnNavigation just to be sure\n let partialMatches = matchRoutesImpl(\n routesToUse,\n pathname,\n basename,\n true\n );\n return { active: true, matches: partialMatches };\n }\n }\n }\n\n return { active: false, matches: null };\n }\n\n type DiscoverRoutesSuccessResult = {\n type: \"success\";\n matches: AgnosticDataRouteMatch[] | null;\n };\n type DiscoverRoutesErrorResult = {\n type: \"error\";\n error: any;\n partialMatches: AgnosticDataRouteMatch[];\n };\n type DiscoverRoutesAbortedResult = { type: \"aborted\" };\n type DiscoverRoutesResult =\n | DiscoverRoutesSuccessResult\n | DiscoverRoutesErrorResult\n | DiscoverRoutesAbortedResult;\n\n async function discoverRoutes(\n matches: AgnosticDataRouteMatch[],\n pathname: string,\n signal: AbortSignal\n ): Promise {\n if (!patchRoutesOnNavigationImpl) {\n return { type: \"success\", matches };\n }\n\n let partialMatches: AgnosticDataRouteMatch[] | null = matches;\n while (true) {\n let isNonHMR = inFlightDataRoutes == null;\n let routesToUse = inFlightDataRoutes || dataRoutes;\n let localManifest = manifest;\n try {\n await patchRoutesOnNavigationImpl({\n signal,\n path: pathname,\n matches: partialMatches,\n patch: (routeId, children) => {\n if (signal.aborted) return;\n patchRoutesImpl(\n routeId,\n children,\n routesToUse,\n localManifest,\n mapRouteProperties\n );\n },\n });\n } catch (e) {\n return { type: \"error\", error: e, partialMatches };\n } finally {\n // If we are not in the middle of an HMR revalidation and we changed the\n // routes, provide a new identity so when we `updateState` at the end of\n // this navigation/fetch `router.routes` will be a new identity and\n // trigger a re-run of memoized `router.routes` dependencies.\n // HMR will already update the identity and reflow when it lands\n // `inFlightDataRoutes` in `completeNavigation`\n if (isNonHMR && !signal.aborted) {\n dataRoutes = [...dataRoutes];\n }\n }\n\n if (signal.aborted) {\n return { type: \"aborted\" };\n }\n\n let newMatches = matchRoutes(routesToUse, pathname, basename);\n if (newMatches) {\n return { type: \"success\", matches: newMatches };\n }\n\n let newPartialMatches = matchRoutesImpl(\n routesToUse,\n pathname,\n basename,\n true\n );\n\n // Avoid loops if the second pass results in the same partial matches\n if (\n !newPartialMatches ||\n (partialMatches.length === newPartialMatches.length &&\n partialMatches.every(\n (m, i) => m.route.id === newPartialMatches![i].route.id\n ))\n ) {\n return { type: \"success\", matches: null };\n }\n\n partialMatches = newPartialMatches;\n }\n }\n\n function _internalSetRoutes(newRoutes: AgnosticDataRouteObject[]) {\n manifest = {};\n inFlightDataRoutes = convertRoutesToDataRoutes(\n newRoutes,\n mapRouteProperties,\n undefined,\n manifest\n );\n }\n\n function patchRoutes(\n routeId: string | null,\n children: AgnosticRouteObject[]\n ): void {\n let isNonHMR = inFlightDataRoutes == null;\n let routesToUse = inFlightDataRoutes || dataRoutes;\n patchRoutesImpl(\n routeId,\n children,\n routesToUse,\n manifest,\n mapRouteProperties\n );\n\n // If we are not in the middle of an HMR revalidation and we changed the\n // routes, provide a new identity and trigger a reflow via `updateState`\n // to re-run memoized `router.routes` dependencies.\n // HMR will already update the identity and reflow when it lands\n // `inFlightDataRoutes` in `completeNavigation`\n if (isNonHMR) {\n dataRoutes = [...dataRoutes];\n updateState({});\n }\n }\n\n router = {\n get basename() {\n return basename;\n },\n get future() {\n return future;\n },\n get state() {\n return state;\n },\n get routes() {\n return dataRoutes;\n },\n get window() {\n return routerWindow;\n },\n initialize,\n subscribe,\n enableScrollRestoration,\n navigate,\n fetch,\n revalidate,\n // Passthrough to history-aware createHref used by useHref so we get proper\n // hash-aware URLs in DOM paths\n createHref: (to: To) => init.history.createHref(to),\n encodeLocation: (to: To) => init.history.encodeLocation(to),\n getFetcher,\n deleteFetcher: deleteFetcherAndUpdateState,\n dispose,\n getBlocker,\n deleteBlocker,\n patchRoutes,\n _internalFetchControllers: fetchControllers,\n _internalActiveDeferreds: activeDeferreds,\n // TODO: Remove setRoutes, it's temporary to avoid dealing with\n // updating the tree while validating the update algorithm.\n _internalSetRoutes,\n };\n\n return router;\n}\n//#endregion\n\n////////////////////////////////////////////////////////////////////////////////\n//#region createStaticHandler\n////////////////////////////////////////////////////////////////////////////////\n\nexport const UNSAFE_DEFERRED_SYMBOL = Symbol(\"deferred\");\n\n/**\n * Future flags to toggle new feature behavior\n */\nexport interface StaticHandlerFutureConfig {\n v7_relativeSplatPath: boolean;\n v7_throwAbortReason: boolean;\n}\n\nexport interface CreateStaticHandlerOptions {\n basename?: string;\n /**\n * @deprecated Use `mapRouteProperties` instead\n */\n detectErrorBoundary?: DetectErrorBoundaryFunction;\n mapRouteProperties?: MapRoutePropertiesFunction;\n future?: Partial;\n}\n\nexport function createStaticHandler(\n routes: AgnosticRouteObject[],\n opts?: CreateStaticHandlerOptions\n): StaticHandler {\n invariant(\n routes.length > 0,\n \"You must provide a non-empty routes array to createStaticHandler\"\n );\n\n let manifest: RouteManifest = {};\n let basename = (opts ? opts.basename : null) || \"/\";\n let mapRouteProperties: MapRoutePropertiesFunction;\n if (opts?.mapRouteProperties) {\n mapRouteProperties = opts.mapRouteProperties;\n } else if (opts?.detectErrorBoundary) {\n // If they are still using the deprecated version, wrap it with the new API\n let detectErrorBoundary = opts.detectErrorBoundary;\n mapRouteProperties = (route) => ({\n hasErrorBoundary: detectErrorBoundary(route),\n });\n } else {\n mapRouteProperties = defaultMapRouteProperties;\n }\n // Config driven behavior flags\n let future: StaticHandlerFutureConfig = {\n v7_relativeSplatPath: false,\n v7_throwAbortReason: false,\n ...(opts ? opts.future : null),\n };\n\n let dataRoutes = convertRoutesToDataRoutes(\n routes,\n mapRouteProperties,\n undefined,\n manifest\n );\n\n /**\n * The query() method is intended for document requests, in which we want to\n * call an optional action and potentially multiple loaders for all nested\n * routes. It returns a StaticHandlerContext object, which is very similar\n * to the router state (location, loaderData, actionData, errors, etc.) and\n * also adds SSR-specific information such as the statusCode and headers\n * from action/loaders Responses.\n *\n * It _should_ never throw and should report all errors through the\n * returned context.errors object, properly associating errors to their error\n * boundary. Additionally, it tracks _deepestRenderedBoundaryId which can be\n * used to emulate React error boundaries during SSr by performing a second\n * pass only down to the boundaryId.\n *\n * The one exception where we do not return a StaticHandlerContext is when a\n * redirect response is returned or thrown from any action/loader. We\n * propagate that out and return the raw Response so the HTTP server can\n * return it directly.\n *\n * - `opts.requestContext` is an optional server context that will be passed\n * to actions/loaders in the `context` parameter\n * - `opts.skipLoaderErrorBubbling` is an optional parameter that will prevent\n * the bubbling of errors which allows single-fetch-type implementations\n * where the client will handle the bubbling and we may need to return data\n * for the handling route\n */\n async function query(\n request: Request,\n {\n requestContext,\n skipLoaderErrorBubbling,\n dataStrategy,\n }: {\n requestContext?: unknown;\n skipLoaderErrorBubbling?: boolean;\n dataStrategy?: DataStrategyFunction;\n } = {}\n ): Promise {\n let url = new URL(request.url);\n let method = request.method;\n let location = createLocation(\"\", createPath(url), null, \"default\");\n let matches = matchRoutes(dataRoutes, location, basename);\n\n // SSR supports HEAD requests while SPA doesn't\n if (!isValidMethod(method) && method !== \"HEAD\") {\n let error = getInternalRouterError(405, { method });\n let { matches: methodNotAllowedMatches, route } =\n getShortCircuitMatches(dataRoutes);\n return {\n basename,\n location,\n matches: methodNotAllowedMatches,\n loaderData: {},\n actionData: null,\n errors: {\n [route.id]: error,\n },\n statusCode: error.status,\n loaderHeaders: {},\n actionHeaders: {},\n activeDeferreds: null,\n };\n } else if (!matches) {\n let error = getInternalRouterError(404, { pathname: location.pathname });\n let { matches: notFoundMatches, route } =\n getShortCircuitMatches(dataRoutes);\n return {\n basename,\n location,\n matches: notFoundMatches,\n loaderData: {},\n actionData: null,\n errors: {\n [route.id]: error,\n },\n statusCode: error.status,\n loaderHeaders: {},\n actionHeaders: {},\n activeDeferreds: null,\n };\n }\n\n let result = await queryImpl(\n request,\n location,\n matches,\n requestContext,\n dataStrategy || null,\n skipLoaderErrorBubbling === true,\n null\n );\n if (isResponse(result)) {\n return result;\n }\n\n // When returning StaticHandlerContext, we patch back in the location here\n // since we need it for React Context. But this helps keep our submit and\n // loadRouteData operating on a Request instead of a Location\n return { location, basename, ...result };\n }\n\n /**\n * The queryRoute() method is intended for targeted route requests, either\n * for fetch ?_data requests or resource route requests. In this case, we\n * are only ever calling a single action or loader, and we are returning the\n * returned value directly. In most cases, this will be a Response returned\n * from the action/loader, but it may be a primitive or other value as well -\n * and in such cases the calling context should handle that accordingly.\n *\n * We do respect the throw/return differentiation, so if an action/loader\n * throws, then this method will throw the value. This is important so we\n * can do proper boundary identification in Remix where a thrown Response\n * must go to the Catch Boundary but a returned Response is happy-path.\n *\n * One thing to note is that any Router-initiated Errors that make sense\n * to associate with a status code will be thrown as an ErrorResponse\n * instance which include the raw Error, such that the calling context can\n * serialize the error as they see fit while including the proper response\n * code. Examples here are 404 and 405 errors that occur prior to reaching\n * any user-defined loaders.\n *\n * - `opts.routeId` allows you to specify the specific route handler to call.\n * If not provided the handler will determine the proper route by matching\n * against `request.url`\n * - `opts.requestContext` is an optional server context that will be passed\n * to actions/loaders in the `context` parameter\n */\n async function queryRoute(\n request: Request,\n {\n routeId,\n requestContext,\n dataStrategy,\n }: {\n requestContext?: unknown;\n routeId?: string;\n dataStrategy?: DataStrategyFunction;\n } = {}\n ): Promise {\n let url = new URL(request.url);\n let method = request.method;\n let location = createLocation(\"\", createPath(url), null, \"default\");\n let matches = matchRoutes(dataRoutes, location, basename);\n\n // SSR supports HEAD requests while SPA doesn't\n if (!isValidMethod(method) && method !== \"HEAD\" && method !== \"OPTIONS\") {\n throw getInternalRouterError(405, { method });\n } else if (!matches) {\n throw getInternalRouterError(404, { pathname: location.pathname });\n }\n\n let match = routeId\n ? matches.find((m) => m.route.id === routeId)\n : getTargetMatch(matches, location);\n\n if (routeId && !match) {\n throw getInternalRouterError(403, {\n pathname: location.pathname,\n routeId,\n });\n } else if (!match) {\n // This should never hit I don't think?\n throw getInternalRouterError(404, { pathname: location.pathname });\n }\n\n let result = await queryImpl(\n request,\n location,\n matches,\n requestContext,\n dataStrategy || null,\n false,\n match\n );\n\n if (isResponse(result)) {\n return result;\n }\n\n let error = result.errors ? Object.values(result.errors)[0] : undefined;\n if (error !== undefined) {\n // If we got back result.errors, that means the loader/action threw\n // _something_ that wasn't a Response, but it's not guaranteed/required\n // to be an `instanceof Error` either, so we have to use throw here to\n // preserve the \"error\" state outside of queryImpl.\n throw error;\n }\n\n // Pick off the right state value to return\n if (result.actionData) {\n return Object.values(result.actionData)[0];\n }\n\n if (result.loaderData) {\n let data = Object.values(result.loaderData)[0];\n if (result.activeDeferreds?.[match.route.id]) {\n data[UNSAFE_DEFERRED_SYMBOL] = result.activeDeferreds[match.route.id];\n }\n return data;\n }\n\n return undefined;\n }\n\n async function queryImpl(\n request: Request,\n location: Location,\n matches: AgnosticDataRouteMatch[],\n requestContext: unknown,\n dataStrategy: DataStrategyFunction | null,\n skipLoaderErrorBubbling: boolean,\n routeMatch: AgnosticDataRouteMatch | null\n ): Promise | Response> {\n invariant(\n request.signal,\n \"query()/queryRoute() requests must contain an AbortController signal\"\n );\n\n try {\n if (isMutationMethod(request.method.toLowerCase())) {\n let result = await submit(\n request,\n matches,\n routeMatch || getTargetMatch(matches, location),\n requestContext,\n dataStrategy,\n skipLoaderErrorBubbling,\n routeMatch != null\n );\n return result;\n }\n\n let result = await loadRouteData(\n request,\n matches,\n requestContext,\n dataStrategy,\n skipLoaderErrorBubbling,\n routeMatch\n );\n return isResponse(result)\n ? result\n : {\n ...result,\n actionData: null,\n actionHeaders: {},\n };\n } catch (e) {\n // If the user threw/returned a Response in callLoaderOrAction for a\n // `queryRoute` call, we throw the `DataStrategyResult` to bail out early\n // and then return or throw the raw Response here accordingly\n if (isDataStrategyResult(e) && isResponse(e.result)) {\n if (e.type === ResultType.error) {\n throw e.result;\n }\n return e.result;\n }\n // Redirects are always returned since they don't propagate to catch\n // boundaries\n if (isRedirectResponse(e)) {\n return e;\n }\n throw e;\n }\n }\n\n async function submit(\n request: Request,\n matches: AgnosticDataRouteMatch[],\n actionMatch: AgnosticDataRouteMatch,\n requestContext: unknown,\n dataStrategy: DataStrategyFunction | null,\n skipLoaderErrorBubbling: boolean,\n isRouteRequest: boolean\n ): Promise | Response> {\n let result: DataResult;\n\n if (!actionMatch.route.action && !actionMatch.route.lazy) {\n let error = getInternalRouterError(405, {\n method: request.method,\n pathname: new URL(request.url).pathname,\n routeId: actionMatch.route.id,\n });\n if (isRouteRequest) {\n throw error;\n }\n result = {\n type: ResultType.error,\n error,\n };\n } else {\n let results = await callDataStrategy(\n \"action\",\n request,\n [actionMatch],\n matches,\n isRouteRequest,\n requestContext,\n dataStrategy\n );\n result = results[actionMatch.route.id];\n\n if (request.signal.aborted) {\n throwStaticHandlerAbortedError(request, isRouteRequest, future);\n }\n }\n\n if (isRedirectResult(result)) {\n // Uhhhh - this should never happen, we should always throw these from\n // callLoaderOrAction, but the type narrowing here keeps TS happy and we\n // can get back on the \"throw all redirect responses\" train here should\n // this ever happen :/\n throw new Response(null, {\n status: result.response.status,\n headers: {\n Location: result.response.headers.get(\"Location\")!,\n },\n });\n }\n\n if (isDeferredResult(result)) {\n let error = getInternalRouterError(400, { type: \"defer-action\" });\n if (isRouteRequest) {\n throw error;\n }\n result = {\n type: ResultType.error,\n error,\n };\n }\n\n if (isRouteRequest) {\n // Note: This should only be non-Response values if we get here, since\n // isRouteRequest should throw any Response received in callLoaderOrAction\n if (isErrorResult(result)) {\n throw result.error;\n }\n\n return {\n matches: [actionMatch],\n loaderData: {},\n actionData: { [actionMatch.route.id]: result.data },\n errors: null,\n // Note: statusCode + headers are unused here since queryRoute will\n // return the raw Response or value\n statusCode: 200,\n loaderHeaders: {},\n actionHeaders: {},\n activeDeferreds: null,\n };\n }\n\n // Create a GET request for the loaders\n let loaderRequest = new Request(request.url, {\n headers: request.headers,\n redirect: request.redirect,\n signal: request.signal,\n });\n\n if (isErrorResult(result)) {\n // Store off the pending error - we use it to determine which loaders\n // to call and will commit it when we complete the navigation\n let boundaryMatch = skipLoaderErrorBubbling\n ? actionMatch\n : findNearestBoundary(matches, actionMatch.route.id);\n\n let context = await loadRouteData(\n loaderRequest,\n matches,\n requestContext,\n dataStrategy,\n skipLoaderErrorBubbling,\n null,\n [boundaryMatch.route.id, result]\n );\n\n // action status codes take precedence over loader status codes\n return {\n ...context,\n statusCode: isRouteErrorResponse(result.error)\n ? result.error.status\n : result.statusCode != null\n ? result.statusCode\n : 500,\n actionData: null,\n actionHeaders: {\n ...(result.headers ? { [actionMatch.route.id]: result.headers } : {}),\n },\n };\n }\n\n let context = await loadRouteData(\n loaderRequest,\n matches,\n requestContext,\n dataStrategy,\n skipLoaderErrorBubbling,\n null\n );\n\n return {\n ...context,\n actionData: {\n [actionMatch.route.id]: result.data,\n },\n // action status codes take precedence over loader status codes\n ...(result.statusCode ? { statusCode: result.statusCode } : {}),\n actionHeaders: result.headers\n ? { [actionMatch.route.id]: result.headers }\n : {},\n };\n }\n\n async function loadRouteData(\n request: Request,\n matches: AgnosticDataRouteMatch[],\n requestContext: unknown,\n dataStrategy: DataStrategyFunction | null,\n skipLoaderErrorBubbling: boolean,\n routeMatch: AgnosticDataRouteMatch | null,\n pendingActionResult?: PendingActionResult\n ): Promise<\n | Omit<\n StaticHandlerContext,\n \"location\" | \"basename\" | \"actionData\" | \"actionHeaders\"\n >\n | Response\n > {\n let isRouteRequest = routeMatch != null;\n\n // Short circuit if we have no loaders to run (queryRoute())\n if (\n isRouteRequest &&\n !routeMatch?.route.loader &&\n !routeMatch?.route.lazy\n ) {\n throw getInternalRouterError(400, {\n method: request.method,\n pathname: new URL(request.url).pathname,\n routeId: routeMatch?.route.id,\n });\n }\n\n let requestMatches = routeMatch\n ? [routeMatch]\n : pendingActionResult && isErrorResult(pendingActionResult[1])\n ? getLoaderMatchesUntilBoundary(matches, pendingActionResult[0])\n : matches;\n let matchesToLoad = requestMatches.filter(\n (m) => m.route.loader || m.route.lazy\n );\n\n // Short circuit if we have no loaders to run (query())\n if (matchesToLoad.length === 0) {\n return {\n matches,\n // Add a null for all matched routes for proper revalidation on the client\n loaderData: matches.reduce(\n (acc, m) => Object.assign(acc, { [m.route.id]: null }),\n {}\n ),\n errors:\n pendingActionResult && isErrorResult(pendingActionResult[1])\n ? {\n [pendingActionResult[0]]: pendingActionResult[1].error,\n }\n : null,\n statusCode: 200,\n loaderHeaders: {},\n activeDeferreds: null,\n };\n }\n\n let results = await callDataStrategy(\n \"loader\",\n request,\n matchesToLoad,\n matches,\n isRouteRequest,\n requestContext,\n dataStrategy\n );\n\n if (request.signal.aborted) {\n throwStaticHandlerAbortedError(request, isRouteRequest, future);\n }\n\n // Process and commit output from loaders\n let activeDeferreds = new Map();\n let context = processRouteLoaderData(\n matches,\n results,\n pendingActionResult,\n activeDeferreds,\n skipLoaderErrorBubbling\n );\n\n // Add a null for any non-loader matches for proper revalidation on the client\n let executedLoaders = new Set(\n matchesToLoad.map((match) => match.route.id)\n );\n matches.forEach((match) => {\n if (!executedLoaders.has(match.route.id)) {\n context.loaderData[match.route.id] = null;\n }\n });\n\n return {\n ...context,\n matches,\n activeDeferreds:\n activeDeferreds.size > 0\n ? Object.fromEntries(activeDeferreds.entries())\n : null,\n };\n }\n\n // Utility wrapper for calling dataStrategy server-side without having to\n // pass around the manifest, mapRouteProperties, etc.\n async function callDataStrategy(\n type: \"loader\" | \"action\",\n request: Request,\n matchesToLoad: AgnosticDataRouteMatch[],\n matches: AgnosticDataRouteMatch[],\n isRouteRequest: boolean,\n requestContext: unknown,\n dataStrategy: DataStrategyFunction | null\n ): Promise> {\n let results = await callDataStrategyImpl(\n dataStrategy || defaultDataStrategy,\n type,\n null,\n request,\n matchesToLoad,\n matches,\n null,\n manifest,\n mapRouteProperties,\n requestContext\n );\n\n let dataResults: Record = {};\n await Promise.all(\n matches.map(async (match) => {\n if (!(match.route.id in results)) {\n return;\n }\n let result = results[match.route.id];\n if (isRedirectDataStrategyResultResult(result)) {\n let response = result.result as Response;\n // Throw redirects and let the server handle them with an HTTP redirect\n throw normalizeRelativeRoutingRedirectResponse(\n response,\n request,\n match.route.id,\n matches,\n basename,\n future.v7_relativeSplatPath\n );\n }\n if (isResponse(result.result) && isRouteRequest) {\n // For SSR single-route requests, we want to hand Responses back\n // directly without unwrapping\n throw result;\n }\n\n dataResults[match.route.id] =\n await convertDataStrategyResultToDataResult(result);\n })\n );\n return dataResults;\n }\n\n return {\n dataRoutes,\n query,\n queryRoute,\n };\n}\n\n//#endregion\n\n////////////////////////////////////////////////////////////////////////////////\n//#region Helpers\n////////////////////////////////////////////////////////////////////////////////\n\n/**\n * Given an existing StaticHandlerContext and an error thrown at render time,\n * provide an updated StaticHandlerContext suitable for a second SSR render\n */\nexport function getStaticContextFromError(\n routes: AgnosticDataRouteObject[],\n context: StaticHandlerContext,\n error: any\n) {\n let newContext: StaticHandlerContext = {\n ...context,\n statusCode: isRouteErrorResponse(error) ? error.status : 500,\n errors: {\n [context._deepestRenderedBoundaryId || routes[0].id]: error,\n },\n };\n return newContext;\n}\n\nfunction throwStaticHandlerAbortedError(\n request: Request,\n isRouteRequest: boolean,\n future: StaticHandlerFutureConfig\n) {\n if (future.v7_throwAbortReason && request.signal.reason !== undefined) {\n throw request.signal.reason;\n }\n\n let method = isRouteRequest ? \"queryRoute\" : \"query\";\n throw new Error(`${method}() call aborted: ${request.method} ${request.url}`);\n}\n\nfunction isSubmissionNavigation(\n opts: BaseNavigateOrFetchOptions\n): opts is SubmissionNavigateOptions {\n return (\n opts != null &&\n ((\"formData\" in opts && opts.formData != null) ||\n (\"body\" in opts && opts.body !== undefined))\n );\n}\n\nfunction normalizeTo(\n location: Path,\n matches: AgnosticDataRouteMatch[],\n basename: string,\n prependBasename: boolean,\n to: To | null,\n v7_relativeSplatPath: boolean,\n fromRouteId?: string,\n relative?: RelativeRoutingType\n) {\n let contextualMatches: AgnosticDataRouteMatch[];\n let activeRouteMatch: AgnosticDataRouteMatch | undefined;\n if (fromRouteId) {\n // Grab matches up to the calling route so our route-relative logic is\n // relative to the correct source route\n contextualMatches = [];\n for (let match of matches) {\n contextualMatches.push(match);\n if (match.route.id === fromRouteId) {\n activeRouteMatch = match;\n break;\n }\n }\n } else {\n contextualMatches = matches;\n activeRouteMatch = matches[matches.length - 1];\n }\n\n // Resolve the relative path\n let path = resolveTo(\n to ? to : \".\",\n getResolveToMatches(contextualMatches, v7_relativeSplatPath),\n stripBasename(location.pathname, basename) || location.pathname,\n relative === \"path\"\n );\n\n // When `to` is not specified we inherit search/hash from the current\n // location, unlike when to=\".\" and we just inherit the path.\n // See https://github.com/remix-run/remix/issues/927\n if (to == null) {\n path.search = location.search;\n path.hash = location.hash;\n }\n\n // Account for `?index` params when routing to the current location\n if ((to == null || to === \"\" || to === \".\") && activeRouteMatch) {\n let nakedIndex = hasNakedIndexQuery(path.search);\n if (activeRouteMatch.route.index && !nakedIndex) {\n // Add one when we're targeting an index route\n path.search = path.search\n ? path.search.replace(/^\\?/, \"?index&\")\n : \"?index\";\n } else if (!activeRouteMatch.route.index && nakedIndex) {\n // Remove existing ones when we're not\n let params = new URLSearchParams(path.search);\n let indexValues = params.getAll(\"index\");\n params.delete(\"index\");\n indexValues.filter((v) => v).forEach((v) => params.append(\"index\", v));\n let qs = params.toString();\n path.search = qs ? `?${qs}` : \"\";\n }\n }\n\n // If we're operating within a basename, prepend it to the pathname. If\n // this is a root navigation, then just use the raw basename which allows\n // the basename to have full control over the presence of a trailing slash\n // on root actions\n if (prependBasename && basename !== \"/\") {\n path.pathname =\n path.pathname === \"/\" ? basename : joinPaths([basename, path.pathname]);\n }\n\n return createPath(path);\n}\n\n// Normalize navigation options by converting formMethod=GET formData objects to\n// URLSearchParams so they behave identically to links with query params\nfunction normalizeNavigateOptions(\n normalizeFormMethod: boolean,\n isFetcher: boolean,\n path: string,\n opts?: BaseNavigateOrFetchOptions\n): {\n path: string;\n submission?: Submission;\n error?: ErrorResponseImpl;\n} {\n // Return location verbatim on non-submission navigations\n if (!opts || !isSubmissionNavigation(opts)) {\n return { path };\n }\n\n if (opts.formMethod && !isValidMethod(opts.formMethod)) {\n return {\n path,\n error: getInternalRouterError(405, { method: opts.formMethod }),\n };\n }\n\n let getInvalidBodyError = () => ({\n path,\n error: getInternalRouterError(400, { type: \"invalid-body\" }),\n });\n\n // Create a Submission on non-GET navigations\n let rawFormMethod = opts.formMethod || \"get\";\n let formMethod = normalizeFormMethod\n ? (rawFormMethod.toUpperCase() as V7_FormMethod)\n : (rawFormMethod.toLowerCase() as FormMethod);\n let formAction = stripHashFromPath(path);\n\n if (opts.body !== undefined) {\n if (opts.formEncType === \"text/plain\") {\n // text only support POST/PUT/PATCH/DELETE submissions\n if (!isMutationMethod(formMethod)) {\n return getInvalidBodyError();\n }\n\n let text =\n typeof opts.body === \"string\"\n ? opts.body\n : opts.body instanceof FormData ||\n opts.body instanceof URLSearchParams\n ? // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#plain-text-form-data\n Array.from(opts.body.entries()).reduce(\n (acc, [name, value]) => `${acc}${name}=${value}\\n`,\n \"\"\n )\n : String(opts.body);\n\n return {\n path,\n submission: {\n formMethod,\n formAction,\n formEncType: opts.formEncType,\n formData: undefined,\n json: undefined,\n text,\n },\n };\n } else if (opts.formEncType === \"application/json\") {\n // json only supports POST/PUT/PATCH/DELETE submissions\n if (!isMutationMethod(formMethod)) {\n return getInvalidBodyError();\n }\n\n try {\n let json =\n typeof opts.body === \"string\" ? JSON.parse(opts.body) : opts.body;\n\n return {\n path,\n submission: {\n formMethod,\n formAction,\n formEncType: opts.formEncType,\n formData: undefined,\n json,\n text: undefined,\n },\n };\n } catch (e) {\n return getInvalidBodyError();\n }\n }\n }\n\n invariant(\n typeof FormData === \"function\",\n \"FormData is not available in this environment\"\n );\n\n let searchParams: URLSearchParams;\n let formData: FormData;\n\n if (opts.formData) {\n searchParams = convertFormDataToSearchParams(opts.formData);\n formData = opts.formData;\n } else if (opts.body instanceof FormData) {\n searchParams = convertFormDataToSearchParams(opts.body);\n formData = opts.body;\n } else if (opts.body instanceof URLSearchParams) {\n searchParams = opts.body;\n formData = convertSearchParamsToFormData(searchParams);\n } else if (opts.body == null) {\n searchParams = new URLSearchParams();\n formData = new FormData();\n } else {\n try {\n searchParams = new URLSearchParams(opts.body);\n formData = convertSearchParamsToFormData(searchParams);\n } catch (e) {\n return getInvalidBodyError();\n }\n }\n\n let submission: Submission = {\n formMethod,\n formAction,\n formEncType:\n (opts && opts.formEncType) || \"application/x-www-form-urlencoded\",\n formData,\n json: undefined,\n text: undefined,\n };\n\n if (isMutationMethod(submission.formMethod)) {\n return { path, submission };\n }\n\n // Flatten submission onto URLSearchParams for GET submissions\n let parsedPath = parsePath(path);\n // On GET navigation submissions we can drop the ?index param from the\n // resulting location since all loaders will run. But fetcher GET submissions\n // only run a single loader so we need to preserve any incoming ?index params\n if (isFetcher && parsedPath.search && hasNakedIndexQuery(parsedPath.search)) {\n searchParams.append(\"index\", \"\");\n }\n parsedPath.search = `?${searchParams}`;\n\n return { path: createPath(parsedPath), submission };\n}\n\n// Filter out all routes at/below any caught error as they aren't going to\n// render so we don't need to load them\nfunction getLoaderMatchesUntilBoundary(\n matches: AgnosticDataRouteMatch[],\n boundaryId: string,\n includeBoundary = false\n) {\n let index = matches.findIndex((m) => m.route.id === boundaryId);\n if (index >= 0) {\n return matches.slice(0, includeBoundary ? index + 1 : index);\n }\n return matches;\n}\n\nfunction getMatchesToLoad(\n history: History,\n state: RouterState,\n matches: AgnosticDataRouteMatch[],\n submission: Submission | undefined,\n location: Location,\n initialHydration: boolean,\n skipActionErrorRevalidation: boolean,\n isRevalidationRequired: boolean,\n cancelledDeferredRoutes: string[],\n cancelledFetcherLoads: Set,\n deletedFetchers: Set,\n fetchLoadMatches: Map,\n fetchRedirectIds: Set,\n routesToUse: AgnosticDataRouteObject[],\n basename: string | undefined,\n pendingActionResult?: PendingActionResult\n): [AgnosticDataRouteMatch[], RevalidatingFetcher[]] {\n let actionResult = pendingActionResult\n ? isErrorResult(pendingActionResult[1])\n ? pendingActionResult[1].error\n : pendingActionResult[1].data\n : undefined;\n let currentUrl = history.createURL(state.location);\n let nextUrl = history.createURL(location);\n\n // Pick navigation matches that are net-new or qualify for revalidation\n let boundaryMatches = matches;\n if (initialHydration && state.errors) {\n // On initial hydration, only consider matches up to _and including_ the boundary.\n // This is inclusive to handle cases where a server loader ran successfully,\n // a child server loader bubbled up to this route, but this route has\n // `clientLoader.hydrate` so we want to still run the `clientLoader` so that\n // we have a complete version of `loaderData`\n boundaryMatches = getLoaderMatchesUntilBoundary(\n matches,\n Object.keys(state.errors)[0],\n true\n );\n } else if (pendingActionResult && isErrorResult(pendingActionResult[1])) {\n // If an action threw an error, we call loaders up to, but not including the\n // boundary\n boundaryMatches = getLoaderMatchesUntilBoundary(\n matches,\n pendingActionResult[0]\n );\n }\n\n // Don't revalidate loaders by default after action 4xx/5xx responses\n // when the flag is enabled. They can still opt-into revalidation via\n // `shouldRevalidate` via `actionResult`\n let actionStatus = pendingActionResult\n ? pendingActionResult[1].statusCode\n : undefined;\n let shouldSkipRevalidation =\n skipActionErrorRevalidation && actionStatus && actionStatus >= 400;\n\n let navigationMatches = boundaryMatches.filter((match, index) => {\n let { route } = match;\n if (route.lazy) {\n // We haven't loaded this route yet so we don't know if it's got a loader!\n return true;\n }\n\n if (route.loader == null) {\n return false;\n }\n\n if (initialHydration) {\n return shouldLoadRouteOnHydration(route, state.loaderData, state.errors);\n }\n\n // Always call the loader on new route instances and pending defer cancellations\n if (\n isNewLoader(state.loaderData, state.matches[index], match) ||\n cancelledDeferredRoutes.some((id) => id === match.route.id)\n ) {\n return true;\n }\n\n // This is the default implementation for when we revalidate. If the route\n // provides it's own implementation, then we give them full control but\n // provide this value so they can leverage it if needed after they check\n // their own specific use cases\n let currentRouteMatch = state.matches[index];\n let nextRouteMatch = match;\n\n return shouldRevalidateLoader(match, {\n currentUrl,\n currentParams: currentRouteMatch.params,\n nextUrl,\n nextParams: nextRouteMatch.params,\n ...submission,\n actionResult,\n actionStatus,\n defaultShouldRevalidate: shouldSkipRevalidation\n ? false\n : // Forced revalidation due to submission, useRevalidator, or X-Remix-Revalidate\n isRevalidationRequired ||\n currentUrl.pathname + currentUrl.search ===\n nextUrl.pathname + nextUrl.search ||\n // Search params affect all loaders\n currentUrl.search !== nextUrl.search ||\n isNewRouteInstance(currentRouteMatch, nextRouteMatch),\n });\n });\n\n // Pick fetcher.loads that need to be revalidated\n let revalidatingFetchers: RevalidatingFetcher[] = [];\n fetchLoadMatches.forEach((f, key) => {\n // Don't revalidate:\n // - on initial hydration (shouldn't be any fetchers then anyway)\n // - if fetcher won't be present in the subsequent render\n // - no longer matches the URL (v7_fetcherPersist=false)\n // - was unmounted but persisted due to v7_fetcherPersist=true\n if (\n initialHydration ||\n !matches.some((m) => m.route.id === f.routeId) ||\n deletedFetchers.has(key)\n ) {\n return;\n }\n\n let fetcherMatches = matchRoutes(routesToUse, f.path, basename);\n\n // If the fetcher path no longer matches, push it in with null matches so\n // we can trigger a 404 in callLoadersAndMaybeResolveData. Note this is\n // currently only a use-case for Remix HMR where the route tree can change\n // at runtime and remove a route previously loaded via a fetcher\n if (!fetcherMatches) {\n revalidatingFetchers.push({\n key,\n routeId: f.routeId,\n path: f.path,\n matches: null,\n match: null,\n controller: null,\n });\n return;\n }\n\n // Revalidating fetchers are decoupled from the route matches since they\n // load from a static href. They revalidate based on explicit revalidation\n // (submission, useRevalidator, or X-Remix-Revalidate)\n let fetcher = state.fetchers.get(key);\n let fetcherMatch = getTargetMatch(fetcherMatches, f.path);\n\n let shouldRevalidate = false;\n if (fetchRedirectIds.has(key)) {\n // Never trigger a revalidation of an actively redirecting fetcher\n shouldRevalidate = false;\n } else if (cancelledFetcherLoads.has(key)) {\n // Always mark for revalidation if the fetcher was cancelled\n cancelledFetcherLoads.delete(key);\n shouldRevalidate = true;\n } else if (\n fetcher &&\n fetcher.state !== \"idle\" &&\n fetcher.data === undefined\n ) {\n // If the fetcher hasn't ever completed loading yet, then this isn't a\n // revalidation, it would just be a brand new load if an explicit\n // revalidation is required\n shouldRevalidate = isRevalidationRequired;\n } else {\n // Otherwise fall back on any user-defined shouldRevalidate, defaulting\n // to explicit revalidations only\n shouldRevalidate = shouldRevalidateLoader(fetcherMatch, {\n currentUrl,\n currentParams: state.matches[state.matches.length - 1].params,\n nextUrl,\n nextParams: matches[matches.length - 1].params,\n ...submission,\n actionResult,\n actionStatus,\n defaultShouldRevalidate: shouldSkipRevalidation\n ? false\n : isRevalidationRequired,\n });\n }\n\n if (shouldRevalidate) {\n revalidatingFetchers.push({\n key,\n routeId: f.routeId,\n path: f.path,\n matches: fetcherMatches,\n match: fetcherMatch,\n controller: new AbortController(),\n });\n }\n });\n\n return [navigationMatches, revalidatingFetchers];\n}\n\nfunction shouldLoadRouteOnHydration(\n route: AgnosticDataRouteObject,\n loaderData: RouteData | null | undefined,\n errors: RouteData | null | undefined\n) {\n // We dunno if we have a loader - gotta find out!\n if (route.lazy) {\n return true;\n }\n\n // No loader, nothing to initialize\n if (!route.loader) {\n return false;\n }\n\n let hasData = loaderData != null && loaderData[route.id] !== undefined;\n let hasError = errors != null && errors[route.id] !== undefined;\n\n // Don't run if we error'd during SSR\n if (!hasData && hasError) {\n return false;\n }\n\n // Explicitly opting-in to running on hydration\n if (typeof route.loader === \"function\" && route.loader.hydrate === true) {\n return true;\n }\n\n // Otherwise, run if we're not yet initialized with anything\n return !hasData && !hasError;\n}\n\nfunction isNewLoader(\n currentLoaderData: RouteData,\n currentMatch: AgnosticDataRouteMatch,\n match: AgnosticDataRouteMatch\n) {\n let isNew =\n // [a] -> [a, b]\n !currentMatch ||\n // [a, b] -> [a, c]\n match.route.id !== currentMatch.route.id;\n\n // Handle the case that we don't have data for a re-used route, potentially\n // from a prior error or from a cancelled pending deferred\n let isMissingData = currentLoaderData[match.route.id] === undefined;\n\n // Always load if this is a net-new route or we don't yet have data\n return isNew || isMissingData;\n}\n\nfunction isNewRouteInstance(\n currentMatch: AgnosticDataRouteMatch,\n match: AgnosticDataRouteMatch\n) {\n let currentPath = currentMatch.route.path;\n return (\n // param change for this match, /users/123 -> /users/456\n currentMatch.pathname !== match.pathname ||\n // splat param changed, which is not present in match.path\n // e.g. /files/images/avatar.jpg -> files/finances.xls\n (currentPath != null &&\n currentPath.endsWith(\"*\") &&\n currentMatch.params[\"*\"] !== match.params[\"*\"])\n );\n}\n\nfunction shouldRevalidateLoader(\n loaderMatch: AgnosticDataRouteMatch,\n arg: ShouldRevalidateFunctionArgs\n) {\n if (loaderMatch.route.shouldRevalidate) {\n let routeChoice = loaderMatch.route.shouldRevalidate(arg);\n if (typeof routeChoice === \"boolean\") {\n return routeChoice;\n }\n }\n\n return arg.defaultShouldRevalidate;\n}\n\nfunction patchRoutesImpl(\n routeId: string | null,\n children: AgnosticRouteObject[],\n routesToUse: AgnosticDataRouteObject[],\n manifest: RouteManifest,\n mapRouteProperties: MapRoutePropertiesFunction\n) {\n let childrenToPatch: AgnosticDataRouteObject[];\n if (routeId) {\n let route = manifest[routeId];\n invariant(\n route,\n `No route found to patch children into: routeId = ${routeId}`\n );\n if (!route.children) {\n route.children = [];\n }\n childrenToPatch = route.children;\n } else {\n childrenToPatch = routesToUse;\n }\n\n // Don't patch in routes we already know about so that `patch` is idempotent\n // to simplify user-land code. This is useful because we re-call the\n // `patchRoutesOnNavigation` function for matched routes with params.\n let uniqueChildren = children.filter(\n (newRoute) =>\n !childrenToPatch.some((existingRoute) =>\n isSameRoute(newRoute, existingRoute)\n )\n );\n\n let newRoutes = convertRoutesToDataRoutes(\n uniqueChildren,\n mapRouteProperties,\n [routeId || \"_\", \"patch\", String(childrenToPatch?.length || \"0\")],\n manifest\n );\n\n childrenToPatch.push(...newRoutes);\n}\n\nfunction isSameRoute(\n newRoute: AgnosticRouteObject,\n existingRoute: AgnosticRouteObject\n): boolean {\n // Most optimal check is by id\n if (\n \"id\" in newRoute &&\n \"id\" in existingRoute &&\n newRoute.id === existingRoute.id\n ) {\n return true;\n }\n\n // Second is by pathing differences\n if (\n !(\n newRoute.index === existingRoute.index &&\n newRoute.path === existingRoute.path &&\n newRoute.caseSensitive === existingRoute.caseSensitive\n )\n ) {\n return false;\n }\n\n // Pathless layout routes are trickier since we need to check children.\n // If they have no children then they're the same as far as we can tell\n if (\n (!newRoute.children || newRoute.children.length === 0) &&\n (!existingRoute.children || existingRoute.children.length === 0)\n ) {\n return true;\n }\n\n // Otherwise, we look to see if every child in the new route is already\n // represented in the existing route's children\n return newRoute.children!.every((aChild, i) =>\n existingRoute.children?.some((bChild) => isSameRoute(aChild, bChild))\n );\n}\n\n/**\n * Execute route.lazy() methods to lazily load route modules (loader, action,\n * shouldRevalidate) and update the routeManifest in place which shares objects\n * with dataRoutes so those get updated as well.\n */\nasync function loadLazyRouteModule(\n route: AgnosticDataRouteObject,\n mapRouteProperties: MapRoutePropertiesFunction,\n manifest: RouteManifest\n) {\n if (!route.lazy) {\n return;\n }\n\n let lazyRoute = await route.lazy();\n\n // If the lazy route function was executed and removed by another parallel\n // call then we can return - first lazy() to finish wins because the return\n // value of lazy is expected to be static\n if (!route.lazy) {\n return;\n }\n\n let routeToUpdate = manifest[route.id];\n invariant(routeToUpdate, \"No route found in manifest\");\n\n // Update the route in place. This should be safe because there's no way\n // we could yet be sitting on this route as we can't get there without\n // resolving lazy() first.\n //\n // This is different than the HMR \"update\" use-case where we may actively be\n // on the route being updated. The main concern boils down to \"does this\n // mutation affect any ongoing navigations or any current state.matches\n // values?\". If not, it should be safe to update in place.\n let routeUpdates: Record = {};\n for (let lazyRouteProperty in lazyRoute) {\n let staticRouteValue =\n routeToUpdate[lazyRouteProperty as keyof typeof routeToUpdate];\n\n let isPropertyStaticallyDefined =\n staticRouteValue !== undefined &&\n // This property isn't static since it should always be updated based\n // on the route updates\n lazyRouteProperty !== \"hasErrorBoundary\";\n\n warning(\n !isPropertyStaticallyDefined,\n `Route \"${routeToUpdate.id}\" has a static property \"${lazyRouteProperty}\" ` +\n `defined but its lazy function is also returning a value for this property. ` +\n `The lazy route property \"${lazyRouteProperty}\" will be ignored.`\n );\n\n if (\n !isPropertyStaticallyDefined &&\n !immutableRouteKeys.has(lazyRouteProperty as ImmutableRouteKey)\n ) {\n routeUpdates[lazyRouteProperty] =\n lazyRoute[lazyRouteProperty as keyof typeof lazyRoute];\n }\n }\n\n // Mutate the route with the provided updates. Do this first so we pass\n // the updated version to mapRouteProperties\n Object.assign(routeToUpdate, routeUpdates);\n\n // Mutate the `hasErrorBoundary` property on the route based on the route\n // updates and remove the `lazy` function so we don't resolve the lazy\n // route again.\n Object.assign(routeToUpdate, {\n // To keep things framework agnostic, we use the provided\n // `mapRouteProperties` (or wrapped `detectErrorBoundary`) function to\n // set the framework-aware properties (`element`/`hasErrorBoundary`) since\n // the logic will differ between frameworks.\n ...mapRouteProperties(routeToUpdate),\n lazy: undefined,\n });\n}\n\n// Default implementation of `dataStrategy` which fetches all loaders in parallel\nasync function defaultDataStrategy({\n matches,\n}: DataStrategyFunctionArgs): ReturnType {\n let matchesToLoad = matches.filter((m) => m.shouldLoad);\n let results = await Promise.all(matchesToLoad.map((m) => m.resolve()));\n return results.reduce(\n (acc, result, i) =>\n Object.assign(acc, { [matchesToLoad[i].route.id]: result }),\n {}\n );\n}\n\nasync function callDataStrategyImpl(\n dataStrategyImpl: DataStrategyFunction,\n type: \"loader\" | \"action\",\n state: RouterState | null,\n request: Request,\n matchesToLoad: AgnosticDataRouteMatch[],\n matches: AgnosticDataRouteMatch[],\n fetcherKey: string | null,\n manifest: RouteManifest,\n mapRouteProperties: MapRoutePropertiesFunction,\n requestContext?: unknown\n): Promise> {\n let loadRouteDefinitionsPromises = matches.map((m) =>\n m.route.lazy\n ? loadLazyRouteModule(m.route, mapRouteProperties, manifest)\n : undefined\n );\n\n let dsMatches = matches.map((match, i) => {\n let loadRoutePromise = loadRouteDefinitionsPromises[i];\n let shouldLoad = matchesToLoad.some((m) => m.route.id === match.route.id);\n // `resolve` encapsulates route.lazy(), executing the loader/action,\n // and mapping return values/thrown errors to a `DataStrategyResult`. Users\n // can pass a callback to take fine-grained control over the execution\n // of the loader/action\n let resolve: DataStrategyMatch[\"resolve\"] = async (handlerOverride) => {\n if (\n handlerOverride &&\n request.method === \"GET\" &&\n (match.route.lazy || match.route.loader)\n ) {\n shouldLoad = true;\n }\n return shouldLoad\n ? callLoaderOrAction(\n type,\n request,\n match,\n loadRoutePromise,\n handlerOverride,\n requestContext\n )\n : Promise.resolve({ type: ResultType.data, result: undefined });\n };\n\n return {\n ...match,\n shouldLoad,\n resolve,\n };\n });\n\n // Send all matches here to allow for a middleware-type implementation.\n // handler will be a no-op for unneeded routes and we filter those results\n // back out below.\n let results = await dataStrategyImpl({\n matches: dsMatches,\n request,\n params: matches[0].params,\n fetcherKey,\n context: requestContext,\n });\n\n // Wait for all routes to load here but 'swallow the error since we want\n // it to bubble up from the `await loadRoutePromise` in `callLoaderOrAction` -\n // called from `match.resolve()`\n try {\n await Promise.all(loadRouteDefinitionsPromises);\n } catch (e) {\n // No-op\n }\n\n return results;\n}\n\n// Default logic for calling a loader/action is the user has no specified a dataStrategy\nasync function callLoaderOrAction(\n type: \"loader\" | \"action\",\n request: Request,\n match: AgnosticDataRouteMatch,\n loadRoutePromise: Promise | undefined,\n handlerOverride: Parameters[0],\n staticContext?: unknown\n): Promise {\n let result: DataStrategyResult;\n let onReject: (() => void) | undefined;\n\n let runHandler = (\n handler: AgnosticRouteObject[\"loader\"] | AgnosticRouteObject[\"action\"]\n ): Promise => {\n // Setup a promise we can race against so that abort signals short circuit\n let reject: () => void;\n // This will never resolve so safe to type it as Promise to\n // satisfy the function return value\n let abortPromise = new Promise((_, r) => (reject = r));\n onReject = () => reject();\n request.signal.addEventListener(\"abort\", onReject);\n\n let actualHandler = (ctx?: unknown) => {\n if (typeof handler !== \"function\") {\n return Promise.reject(\n new Error(\n `You cannot call the handler for a route which defines a boolean ` +\n `\"${type}\" [routeId: ${match.route.id}]`\n )\n );\n }\n return handler(\n {\n request,\n params: match.params,\n context: staticContext,\n },\n ...(ctx !== undefined ? [ctx] : [])\n );\n };\n\n let handlerPromise: Promise = (async () => {\n try {\n let val = await (handlerOverride\n ? handlerOverride((ctx: unknown) => actualHandler(ctx))\n : actualHandler());\n return { type: \"data\", result: val };\n } catch (e) {\n return { type: \"error\", result: e };\n }\n })();\n\n return Promise.race([handlerPromise, abortPromise]);\n };\n\n try {\n let handler = match.route[type];\n\n // If we have a route.lazy promise, await that first\n if (loadRoutePromise) {\n if (handler) {\n // Run statically defined handler in parallel with lazy()\n let handlerError;\n let [value] = await Promise.all([\n // If the handler throws, don't let it immediately bubble out,\n // since we need to let the lazy() execution finish so we know if this\n // route has a boundary that can handle the error\n runHandler(handler).catch((e) => {\n handlerError = e;\n }),\n loadRoutePromise,\n ]);\n if (handlerError !== undefined) {\n throw handlerError;\n }\n result = value!;\n } else {\n // Load lazy route module, then run any returned handler\n await loadRoutePromise;\n\n handler = match.route[type];\n if (handler) {\n // Handler still runs even if we got interrupted to maintain consistency\n // with un-abortable behavior of handler execution on non-lazy or\n // previously-lazy-loaded routes\n result = await runHandler(handler);\n } else if (type === \"action\") {\n let url = new URL(request.url);\n let pathname = url.pathname + url.search;\n throw getInternalRouterError(405, {\n method: request.method,\n pathname,\n routeId: match.route.id,\n });\n } else {\n // lazy() route has no loader to run. Short circuit here so we don't\n // hit the invariant below that errors on returning undefined.\n return { type: ResultType.data, result: undefined };\n }\n }\n } else if (!handler) {\n let url = new URL(request.url);\n let pathname = url.pathname + url.search;\n throw getInternalRouterError(404, {\n pathname,\n });\n } else {\n result = await runHandler(handler);\n }\n\n invariant(\n result.result !== undefined,\n `You defined ${type === \"action\" ? \"an action\" : \"a loader\"} for route ` +\n `\"${match.route.id}\" but didn't return anything from your \\`${type}\\` ` +\n `function. Please return a value or \\`null\\`.`\n );\n } catch (e) {\n // We should already be catching and converting normal handler executions to\n // DataStrategyResults and returning them, so anything that throws here is an\n // unexpected error we still need to wrap\n return { type: ResultType.error, result: e };\n } finally {\n if (onReject) {\n request.signal.removeEventListener(\"abort\", onReject);\n }\n }\n\n return result;\n}\n\nasync function convertDataStrategyResultToDataResult(\n dataStrategyResult: DataStrategyResult\n): Promise {\n let { result, type } = dataStrategyResult;\n\n if (isResponse(result)) {\n let data: any;\n\n try {\n let contentType = result.headers.get(\"Content-Type\");\n // Check between word boundaries instead of startsWith() due to the last\n // paragraph of https://httpwg.org/specs/rfc9110.html#field.content-type\n if (contentType && /\\bapplication\\/json\\b/.test(contentType)) {\n if (result.body == null) {\n data = null;\n } else {\n data = await result.json();\n }\n } else {\n data = await result.text();\n }\n } catch (e) {\n return { type: ResultType.error, error: e };\n }\n\n if (type === ResultType.error) {\n return {\n type: ResultType.error,\n error: new ErrorResponseImpl(result.status, result.statusText, data),\n statusCode: result.status,\n headers: result.headers,\n };\n }\n\n return {\n type: ResultType.data,\n data,\n statusCode: result.status,\n headers: result.headers,\n };\n }\n\n if (type === ResultType.error) {\n if (isDataWithResponseInit(result)) {\n if (result.data instanceof Error) {\n return {\n type: ResultType.error,\n error: result.data,\n statusCode: result.init?.status,\n headers: result.init?.headers\n ? new Headers(result.init.headers)\n : undefined,\n };\n }\n\n // Convert thrown data() to ErrorResponse instances\n return {\n type: ResultType.error,\n error: new ErrorResponseImpl(\n result.init?.status || 500,\n undefined,\n result.data\n ),\n statusCode: isRouteErrorResponse(result) ? result.status : undefined,\n headers: result.init?.headers\n ? new Headers(result.init.headers)\n : undefined,\n };\n }\n return {\n type: ResultType.error,\n error: result,\n statusCode: isRouteErrorResponse(result) ? result.status : undefined,\n };\n }\n\n if (isDeferredData(result)) {\n return {\n type: ResultType.deferred,\n deferredData: result,\n statusCode: result.init?.status,\n headers: result.init?.headers && new Headers(result.init.headers),\n };\n }\n\n if (isDataWithResponseInit(result)) {\n return {\n type: ResultType.data,\n data: result.data,\n statusCode: result.init?.status,\n headers: result.init?.headers\n ? new Headers(result.init.headers)\n : undefined,\n };\n }\n\n return { type: ResultType.data, data: result };\n}\n\n// Support relative routing in internal redirects\nfunction normalizeRelativeRoutingRedirectResponse(\n response: Response,\n request: Request,\n routeId: string,\n matches: AgnosticDataRouteMatch[],\n basename: string,\n v7_relativeSplatPath: boolean\n) {\n let location = response.headers.get(\"Location\");\n invariant(\n location,\n \"Redirects returned/thrown from loaders/actions must have a Location header\"\n );\n\n if (!ABSOLUTE_URL_REGEX.test(location)) {\n let trimmedMatches = matches.slice(\n 0,\n matches.findIndex((m) => m.route.id === routeId) + 1\n );\n location = normalizeTo(\n new URL(request.url),\n trimmedMatches,\n basename,\n true,\n location,\n v7_relativeSplatPath\n );\n response.headers.set(\"Location\", location);\n }\n\n return response;\n}\n\nfunction normalizeRedirectLocation(\n location: string,\n currentUrl: URL,\n basename: string\n): string {\n if (ABSOLUTE_URL_REGEX.test(location)) {\n // Strip off the protocol+origin for same-origin + same-basename absolute redirects\n let normalizedLocation = location;\n let url = normalizedLocation.startsWith(\"//\")\n ? new URL(currentUrl.protocol + normalizedLocation)\n : new URL(normalizedLocation);\n let isSameBasename = stripBasename(url.pathname, basename) != null;\n if (url.origin === currentUrl.origin && isSameBasename) {\n return url.pathname + url.search + url.hash;\n }\n }\n return location;\n}\n\n// Utility method for creating the Request instances for loaders/actions during\n// client-side navigations and fetches. During SSR we will always have a\n// Request instance from the static handler (query/queryRoute)\nfunction createClientSideRequest(\n history: History,\n location: string | Location,\n signal: AbortSignal,\n submission?: Submission\n): Request {\n let url = history.createURL(stripHashFromPath(location)).toString();\n let init: RequestInit = { signal };\n\n if (submission && isMutationMethod(submission.formMethod)) {\n let { formMethod, formEncType } = submission;\n // Didn't think we needed this but it turns out unlike other methods, patch\n // won't be properly normalized to uppercase and results in a 405 error.\n // See: https://fetch.spec.whatwg.org/#concept-method\n init.method = formMethod.toUpperCase();\n\n if (formEncType === \"application/json\") {\n init.headers = new Headers({ \"Content-Type\": formEncType });\n init.body = JSON.stringify(submission.json);\n } else if (formEncType === \"text/plain\") {\n // Content-Type is inferred (https://fetch.spec.whatwg.org/#dom-request)\n init.body = submission.text;\n } else if (\n formEncType === \"application/x-www-form-urlencoded\" &&\n submission.formData\n ) {\n // Content-Type is inferred (https://fetch.spec.whatwg.org/#dom-request)\n init.body = convertFormDataToSearchParams(submission.formData);\n } else {\n // Content-Type is inferred (https://fetch.spec.whatwg.org/#dom-request)\n init.body = submission.formData;\n }\n }\n\n return new Request(url, init);\n}\n\nfunction convertFormDataToSearchParams(formData: FormData): URLSearchParams {\n let searchParams = new URLSearchParams();\n\n for (let [key, value] of formData.entries()) {\n // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#converting-an-entry-list-to-a-list-of-name-value-pairs\n searchParams.append(key, typeof value === \"string\" ? value : value.name);\n }\n\n return searchParams;\n}\n\nfunction convertSearchParamsToFormData(\n searchParams: URLSearchParams\n): FormData {\n let formData = new FormData();\n for (let [key, value] of searchParams.entries()) {\n formData.append(key, value);\n }\n return formData;\n}\n\nfunction processRouteLoaderData(\n matches: AgnosticDataRouteMatch[],\n results: Record,\n pendingActionResult: PendingActionResult | undefined,\n activeDeferreds: Map,\n skipLoaderErrorBubbling: boolean\n): {\n loaderData: RouterState[\"loaderData\"];\n errors: RouterState[\"errors\"] | null;\n statusCode: number;\n loaderHeaders: Record;\n} {\n // Fill in loaderData/errors from our loaders\n let loaderData: RouterState[\"loaderData\"] = {};\n let errors: RouterState[\"errors\"] | null = null;\n let statusCode: number | undefined;\n let foundError = false;\n let loaderHeaders: Record = {};\n let pendingError =\n pendingActionResult && isErrorResult(pendingActionResult[1])\n ? pendingActionResult[1].error\n : undefined;\n\n // Process loader results into state.loaderData/state.errors\n matches.forEach((match) => {\n if (!(match.route.id in results)) {\n return;\n }\n let id = match.route.id;\n let result = results[id];\n invariant(\n !isRedirectResult(result),\n \"Cannot handle redirect results in processLoaderData\"\n );\n if (isErrorResult(result)) {\n let error = result.error;\n // If we have a pending action error, we report it at the highest-route\n // that throws a loader error, and then clear it out to indicate that\n // it was consumed\n if (pendingError !== undefined) {\n error = pendingError;\n pendingError = undefined;\n }\n\n errors = errors || {};\n\n if (skipLoaderErrorBubbling) {\n errors[id] = error;\n } else {\n // Look upwards from the matched route for the closest ancestor error\n // boundary, defaulting to the root match. Prefer higher error values\n // if lower errors bubble to the same boundary\n let boundaryMatch = findNearestBoundary(matches, id);\n if (errors[boundaryMatch.route.id] == null) {\n errors[boundaryMatch.route.id] = error;\n }\n }\n\n // Clear our any prior loaderData for the throwing route\n loaderData[id] = undefined;\n\n // Once we find our first (highest) error, we set the status code and\n // prevent deeper status codes from overriding\n if (!foundError) {\n foundError = true;\n statusCode = isRouteErrorResponse(result.error)\n ? result.error.status\n : 500;\n }\n if (result.headers) {\n loaderHeaders[id] = result.headers;\n }\n } else {\n if (isDeferredResult(result)) {\n activeDeferreds.set(id, result.deferredData);\n loaderData[id] = result.deferredData.data;\n // Error status codes always override success status codes, but if all\n // loaders are successful we take the deepest status code.\n if (\n result.statusCode != null &&\n result.statusCode !== 200 &&\n !foundError\n ) {\n statusCode = result.statusCode;\n }\n if (result.headers) {\n loaderHeaders[id] = result.headers;\n }\n } else {\n loaderData[id] = result.data;\n // Error status codes always override success status codes, but if all\n // loaders are successful we take the deepest status code.\n if (result.statusCode && result.statusCode !== 200 && !foundError) {\n statusCode = result.statusCode;\n }\n if (result.headers) {\n loaderHeaders[id] = result.headers;\n }\n }\n }\n });\n\n // If we didn't consume the pending action error (i.e., all loaders\n // resolved), then consume it here. Also clear out any loaderData for the\n // throwing route\n if (pendingError !== undefined && pendingActionResult) {\n errors = { [pendingActionResult[0]]: pendingError };\n loaderData[pendingActionResult[0]] = undefined;\n }\n\n return {\n loaderData,\n errors,\n statusCode: statusCode || 200,\n loaderHeaders,\n };\n}\n\nfunction processLoaderData(\n state: RouterState,\n matches: AgnosticDataRouteMatch[],\n results: Record,\n pendingActionResult: PendingActionResult | undefined,\n revalidatingFetchers: RevalidatingFetcher[],\n fetcherResults: Record,\n activeDeferreds: Map\n): {\n loaderData: RouterState[\"loaderData\"];\n errors?: RouterState[\"errors\"];\n} {\n let { loaderData, errors } = processRouteLoaderData(\n matches,\n results,\n pendingActionResult,\n activeDeferreds,\n false // This method is only called client side so we always want to bubble\n );\n\n // Process results from our revalidating fetchers\n revalidatingFetchers.forEach((rf) => {\n let { key, match, controller } = rf;\n let result = fetcherResults[key];\n invariant(result, \"Did not find corresponding fetcher result\");\n\n // Process fetcher non-redirect errors\n if (controller && controller.signal.aborted) {\n // Nothing to do for aborted fetchers\n return;\n } else if (isErrorResult(result)) {\n let boundaryMatch = findNearestBoundary(state.matches, match?.route.id);\n if (!(errors && errors[boundaryMatch.route.id])) {\n errors = {\n ...errors,\n [boundaryMatch.route.id]: result.error,\n };\n }\n state.fetchers.delete(key);\n } else if (isRedirectResult(result)) {\n // Should never get here, redirects should get processed above, but we\n // keep this to type narrow to a success result in the else\n invariant(false, \"Unhandled fetcher revalidation redirect\");\n } else if (isDeferredResult(result)) {\n // Should never get here, deferred data should be awaited for fetchers\n // in resolveDeferredResults\n invariant(false, \"Unhandled fetcher deferred data\");\n } else {\n let doneFetcher = getDoneFetcher(result.data);\n state.fetchers.set(key, doneFetcher);\n }\n });\n\n return { loaderData, errors };\n}\n\nfunction mergeLoaderData(\n loaderData: RouteData,\n newLoaderData: RouteData,\n matches: AgnosticDataRouteMatch[],\n errors: RouteData | null | undefined\n): RouteData {\n let mergedLoaderData = { ...newLoaderData };\n for (let match of matches) {\n let id = match.route.id;\n if (newLoaderData.hasOwnProperty(id)) {\n if (newLoaderData[id] !== undefined) {\n mergedLoaderData[id] = newLoaderData[id];\n } else {\n // No-op - this is so we ignore existing data if we have a key in the\n // incoming object with an undefined value, which is how we unset a prior\n // loaderData if we encounter a loader error\n }\n } else if (loaderData[id] !== undefined && match.route.loader) {\n // Preserve existing keys not included in newLoaderData and where a loader\n // wasn't removed by HMR\n mergedLoaderData[id] = loaderData[id];\n }\n\n if (errors && errors.hasOwnProperty(id)) {\n // Don't keep any loader data below the boundary\n break;\n }\n }\n return mergedLoaderData;\n}\n\nfunction getActionDataForCommit(\n pendingActionResult: PendingActionResult | undefined\n) {\n if (!pendingActionResult) {\n return {};\n }\n return isErrorResult(pendingActionResult[1])\n ? {\n // Clear out prior actionData on errors\n actionData: {},\n }\n : {\n actionData: {\n [pendingActionResult[0]]: pendingActionResult[1].data,\n },\n };\n}\n\n// Find the nearest error boundary, looking upwards from the leaf route (or the\n// route specified by routeId) for the closest ancestor error boundary,\n// defaulting to the root match\nfunction findNearestBoundary(\n matches: AgnosticDataRouteMatch[],\n routeId?: string\n): AgnosticDataRouteMatch {\n let eligibleMatches = routeId\n ? matches.slice(0, matches.findIndex((m) => m.route.id === routeId) + 1)\n : [...matches];\n return (\n eligibleMatches.reverse().find((m) => m.route.hasErrorBoundary === true) ||\n matches[0]\n );\n}\n\nfunction getShortCircuitMatches(routes: AgnosticDataRouteObject[]): {\n matches: AgnosticDataRouteMatch[];\n route: AgnosticDataRouteObject;\n} {\n // Prefer a root layout route if present, otherwise shim in a route object\n let route =\n routes.length === 1\n ? routes[0]\n : routes.find((r) => r.index || !r.path || r.path === \"/\") || {\n id: `__shim-error-route__`,\n };\n\n return {\n matches: [\n {\n params: {},\n pathname: \"\",\n pathnameBase: \"\",\n route,\n },\n ],\n route,\n };\n}\n\nfunction getInternalRouterError(\n status: number,\n {\n pathname,\n routeId,\n method,\n type,\n message,\n }: {\n pathname?: string;\n routeId?: string;\n method?: string;\n type?: \"defer-action\" | \"invalid-body\";\n message?: string;\n } = {}\n) {\n let statusText = \"Unknown Server Error\";\n let errorMessage = \"Unknown @remix-run/router error\";\n\n if (status === 400) {\n statusText = \"Bad Request\";\n if (method && pathname && routeId) {\n errorMessage =\n `You made a ${method} request to \"${pathname}\" but ` +\n `did not provide a \\`loader\\` for route \"${routeId}\", ` +\n `so there is no way to handle the request.`;\n } else if (type === \"defer-action\") {\n errorMessage = \"defer() is not supported in actions\";\n } else if (type === \"invalid-body\") {\n errorMessage = \"Unable to encode submission body\";\n }\n } else if (status === 403) {\n statusText = \"Forbidden\";\n errorMessage = `Route \"${routeId}\" does not match URL \"${pathname}\"`;\n } else if (status === 404) {\n statusText = \"Not Found\";\n errorMessage = `No route matches URL \"${pathname}\"`;\n } else if (status === 405) {\n statusText = \"Method Not Allowed\";\n if (method && pathname && routeId) {\n errorMessage =\n `You made a ${method.toUpperCase()} request to \"${pathname}\" but ` +\n `did not provide an \\`action\\` for route \"${routeId}\", ` +\n `so there is no way to handle the request.`;\n } else if (method) {\n errorMessage = `Invalid request method \"${method.toUpperCase()}\"`;\n }\n }\n\n return new ErrorResponseImpl(\n status || 500,\n statusText,\n new Error(errorMessage),\n true\n );\n}\n\n// Find any returned redirect errors, starting from the lowest match\nfunction findRedirect(\n results: Record\n): { key: string; result: RedirectResult } | undefined {\n let entries = Object.entries(results);\n for (let i = entries.length - 1; i >= 0; i--) {\n let [key, result] = entries[i];\n if (isRedirectResult(result)) {\n return { key, result };\n }\n }\n}\n\nfunction stripHashFromPath(path: To) {\n let parsedPath = typeof path === \"string\" ? parsePath(path) : path;\n return createPath({ ...parsedPath, hash: \"\" });\n}\n\nfunction isHashChangeOnly(a: Location, b: Location): boolean {\n if (a.pathname !== b.pathname || a.search !== b.search) {\n return false;\n }\n\n if (a.hash === \"\") {\n // /page -> /page#hash\n return b.hash !== \"\";\n } else if (a.hash === b.hash) {\n // /page#hash -> /page#hash\n return true;\n } else if (b.hash !== \"\") {\n // /page#hash -> /page#other\n return true;\n }\n\n // If the hash is removed the browser will re-perform a request to the server\n // /page#hash -> /page\n return false;\n}\n\nfunction isPromise(val: unknown): val is Promise {\n return typeof val === \"object\" && val != null && \"then\" in val;\n}\n\nfunction isDataStrategyResult(result: unknown): result is DataStrategyResult {\n return (\n result != null &&\n typeof result === \"object\" &&\n \"type\" in result &&\n \"result\" in result &&\n (result.type === ResultType.data || result.type === ResultType.error)\n );\n}\n\nfunction isRedirectDataStrategyResultResult(result: DataStrategyResult) {\n return (\n isResponse(result.result) && redirectStatusCodes.has(result.result.status)\n );\n}\n\nfunction isDeferredResult(result: DataResult): result is DeferredResult {\n return result.type === ResultType.deferred;\n}\n\nfunction isErrorResult(result: DataResult): result is ErrorResult {\n return result.type === ResultType.error;\n}\n\nfunction isRedirectResult(result?: DataResult): result is RedirectResult {\n return (result && result.type) === ResultType.redirect;\n}\n\nexport function isDataWithResponseInit(\n value: any\n): value is DataWithResponseInit {\n return (\n typeof value === \"object\" &&\n value != null &&\n \"type\" in value &&\n \"data\" in value &&\n \"init\" in value &&\n value.type === \"DataWithResponseInit\"\n );\n}\n\nexport function isDeferredData(value: any): value is DeferredData {\n let deferred: DeferredData = value;\n return (\n deferred &&\n typeof deferred === \"object\" &&\n typeof deferred.data === \"object\" &&\n typeof deferred.subscribe === \"function\" &&\n typeof deferred.cancel === \"function\" &&\n typeof deferred.resolveData === \"function\"\n );\n}\n\nfunction isResponse(value: any): value is Response {\n return (\n value != null &&\n typeof value.status === \"number\" &&\n typeof value.statusText === \"string\" &&\n typeof value.headers === \"object\" &&\n typeof value.body !== \"undefined\"\n );\n}\n\nfunction isRedirectResponse(result: any): result is Response {\n if (!isResponse(result)) {\n return false;\n }\n\n let status = result.status;\n let location = result.headers.get(\"Location\");\n return status >= 300 && status <= 399 && location != null;\n}\n\nfunction isValidMethod(method: string): method is FormMethod | V7_FormMethod {\n return validRequestMethods.has(method.toLowerCase() as FormMethod);\n}\n\nfunction isMutationMethod(\n method: string\n): method is MutationFormMethod | V7_MutationFormMethod {\n return validMutationMethods.has(method.toLowerCase() as MutationFormMethod);\n}\n\nasync function resolveNavigationDeferredResults(\n matches: (AgnosticDataRouteMatch | null)[],\n results: Record,\n signal: AbortSignal,\n currentMatches: AgnosticDataRouteMatch[],\n currentLoaderData: RouteData\n) {\n let entries = Object.entries(results);\n for (let index = 0; index < entries.length; index++) {\n let [routeId, result] = entries[index];\n let match = matches.find((m) => m?.route.id === routeId);\n // If we don't have a match, then we can have a deferred result to do\n // anything with. This is for revalidating fetchers where the route was\n // removed during HMR\n if (!match) {\n continue;\n }\n\n let currentMatch = currentMatches.find(\n (m) => m.route.id === match!.route.id\n );\n let isRevalidatingLoader =\n currentMatch != null &&\n !isNewRouteInstance(currentMatch, match) &&\n (currentLoaderData && currentLoaderData[match.route.id]) !== undefined;\n\n if (isDeferredResult(result) && isRevalidatingLoader) {\n // Note: we do not have to touch activeDeferreds here since we race them\n // against the signal in resolveDeferredData and they'll get aborted\n // there if needed\n await resolveDeferredData(result, signal, false).then((result) => {\n if (result) {\n results[routeId] = result;\n }\n });\n }\n }\n}\n\nasync function resolveFetcherDeferredResults(\n matches: (AgnosticDataRouteMatch | null)[],\n results: Record,\n revalidatingFetchers: RevalidatingFetcher[]\n) {\n for (let index = 0; index < revalidatingFetchers.length; index++) {\n let { key, routeId, controller } = revalidatingFetchers[index];\n let result = results[key];\n let match = matches.find((m) => m?.route.id === routeId);\n // If we don't have a match, then we can have a deferred result to do\n // anything with. This is for revalidating fetchers where the route was\n // removed during HMR\n if (!match) {\n continue;\n }\n\n if (isDeferredResult(result)) {\n // Note: we do not have to touch activeDeferreds here since we race them\n // against the signal in resolveDeferredData and they'll get aborted\n // there if needed\n invariant(\n controller,\n \"Expected an AbortController for revalidating fetcher deferred result\"\n );\n await resolveDeferredData(result, controller.signal, true).then(\n (result) => {\n if (result) {\n results[key] = result;\n }\n }\n );\n }\n }\n}\n\nasync function resolveDeferredData(\n result: DeferredResult,\n signal: AbortSignal,\n unwrap = false\n): Promise {\n let aborted = await result.deferredData.resolveData(signal);\n if (aborted) {\n return;\n }\n\n if (unwrap) {\n try {\n return {\n type: ResultType.data,\n data: result.deferredData.unwrappedData,\n };\n } catch (e) {\n // Handle any TrackedPromise._error values encountered while unwrapping\n return {\n type: ResultType.error,\n error: e,\n };\n }\n }\n\n return {\n type: ResultType.data,\n data: result.deferredData.data,\n };\n}\n\nfunction hasNakedIndexQuery(search: string): boolean {\n return new URLSearchParams(search).getAll(\"index\").some((v) => v === \"\");\n}\n\nfunction getTargetMatch(\n matches: AgnosticDataRouteMatch[],\n location: Location | string\n) {\n let search =\n typeof location === \"string\" ? parsePath(location).search : location.search;\n if (\n matches[matches.length - 1].route.index &&\n hasNakedIndexQuery(search || \"\")\n ) {\n // Return the leaf index route when index is present\n return matches[matches.length - 1];\n }\n // Otherwise grab the deepest \"path contributing\" match (ignoring index and\n // pathless layout routes)\n let pathMatches = getPathContributingMatches(matches);\n return pathMatches[pathMatches.length - 1];\n}\n\nfunction getSubmissionFromNavigation(\n navigation: Navigation\n): Submission | undefined {\n let { formMethod, formAction, formEncType, text, formData, json } =\n navigation;\n if (!formMethod || !formAction || !formEncType) {\n return;\n }\n\n if (text != null) {\n return {\n formMethod,\n formAction,\n formEncType,\n formData: undefined,\n json: undefined,\n text,\n };\n } else if (formData != null) {\n return {\n formMethod,\n formAction,\n formEncType,\n formData,\n json: undefined,\n text: undefined,\n };\n } else if (json !== undefined) {\n return {\n formMethod,\n formAction,\n formEncType,\n formData: undefined,\n json,\n text: undefined,\n };\n }\n}\n\nfunction getLoadingNavigation(\n location: Location,\n submission?: Submission\n): NavigationStates[\"Loading\"] {\n if (submission) {\n let navigation: NavigationStates[\"Loading\"] = {\n state: \"loading\",\n location,\n formMethod: submission.formMethod,\n formAction: submission.formAction,\n formEncType: submission.formEncType,\n formData: submission.formData,\n json: submission.json,\n text: submission.text,\n };\n return navigation;\n } else {\n let navigation: NavigationStates[\"Loading\"] = {\n state: \"loading\",\n location,\n formMethod: undefined,\n formAction: undefined,\n formEncType: undefined,\n formData: undefined,\n json: undefined,\n text: undefined,\n };\n return navigation;\n }\n}\n\nfunction getSubmittingNavigation(\n location: Location,\n submission: Submission\n): NavigationStates[\"Submitting\"] {\n let navigation: NavigationStates[\"Submitting\"] = {\n state: \"submitting\",\n location,\n formMethod: submission.formMethod,\n formAction: submission.formAction,\n formEncType: submission.formEncType,\n formData: submission.formData,\n json: submission.json,\n text: submission.text,\n };\n return navigation;\n}\n\nfunction getLoadingFetcher(\n submission?: Submission,\n data?: Fetcher[\"data\"]\n): FetcherStates[\"Loading\"] {\n if (submission) {\n let fetcher: FetcherStates[\"Loading\"] = {\n state: \"loading\",\n formMethod: submission.formMethod,\n formAction: submission.formAction,\n formEncType: submission.formEncType,\n formData: submission.formData,\n json: submission.json,\n text: submission.text,\n data,\n };\n return fetcher;\n } else {\n let fetcher: FetcherStates[\"Loading\"] = {\n state: \"loading\",\n formMethod: undefined,\n formAction: undefined,\n formEncType: undefined,\n formData: undefined,\n json: undefined,\n text: undefined,\n data,\n };\n return fetcher;\n }\n}\n\nfunction getSubmittingFetcher(\n submission: Submission,\n existingFetcher?: Fetcher\n): FetcherStates[\"Submitting\"] {\n let fetcher: FetcherStates[\"Submitting\"] = {\n state: \"submitting\",\n formMethod: submission.formMethod,\n formAction: submission.formAction,\n formEncType: submission.formEncType,\n formData: submission.formData,\n json: submission.json,\n text: submission.text,\n data: existingFetcher ? existingFetcher.data : undefined,\n };\n return fetcher;\n}\n\nfunction getDoneFetcher(data: Fetcher[\"data\"]): FetcherStates[\"Idle\"] {\n let fetcher: FetcherStates[\"Idle\"] = {\n state: \"idle\",\n formMethod: undefined,\n formAction: undefined,\n formEncType: undefined,\n formData: undefined,\n json: undefined,\n text: undefined,\n data,\n };\n return fetcher;\n}\n\nfunction restoreAppliedTransitions(\n _window: Window,\n transitions: Map>\n) {\n try {\n let sessionPositions = _window.sessionStorage.getItem(\n TRANSITIONS_STORAGE_KEY\n );\n if (sessionPositions) {\n let json = JSON.parse(sessionPositions);\n for (let [k, v] of Object.entries(json || {})) {\n if (v && Array.isArray(v)) {\n transitions.set(k, new Set(v || []));\n }\n }\n }\n } catch (e) {\n // no-op, use default empty object\n }\n}\n\nfunction persistAppliedTransitions(\n _window: Window,\n transitions: Map>\n) {\n if (transitions.size > 0) {\n let json: Record = {};\n for (let [k, v] of transitions) {\n json[k] = [...v];\n }\n try {\n _window.sessionStorage.setItem(\n TRANSITIONS_STORAGE_KEY,\n JSON.stringify(json)\n );\n } catch (error) {\n warning(\n false,\n `Failed to save applied view transitions in sessionStorage (${error}).`\n );\n }\n }\n}\n//#endregion\n","import * as React from \"react\";\nimport type {\n AgnosticIndexRouteObject,\n AgnosticNonIndexRouteObject,\n AgnosticRouteMatch,\n History,\n LazyRouteFunction,\n Location,\n Action as NavigationType,\n RelativeRoutingType,\n Router,\n StaticHandlerContext,\n To,\n TrackedPromise,\n} from \"@remix-run/router\";\n\n// Create react-specific types from the agnostic types in @remix-run/router to\n// export from react-router\nexport interface IndexRouteObject {\n caseSensitive?: AgnosticIndexRouteObject[\"caseSensitive\"];\n path?: AgnosticIndexRouteObject[\"path\"];\n id?: AgnosticIndexRouteObject[\"id\"];\n loader?: AgnosticIndexRouteObject[\"loader\"];\n action?: AgnosticIndexRouteObject[\"action\"];\n hasErrorBoundary?: AgnosticIndexRouteObject[\"hasErrorBoundary\"];\n shouldRevalidate?: AgnosticIndexRouteObject[\"shouldRevalidate\"];\n handle?: AgnosticIndexRouteObject[\"handle\"];\n index: true;\n children?: undefined;\n element?: React.ReactNode | null;\n hydrateFallbackElement?: React.ReactNode | null;\n errorElement?: React.ReactNode | null;\n Component?: React.ComponentType | null;\n HydrateFallback?: React.ComponentType | null;\n ErrorBoundary?: React.ComponentType | null;\n lazy?: LazyRouteFunction;\n}\n\nexport interface NonIndexRouteObject {\n caseSensitive?: AgnosticNonIndexRouteObject[\"caseSensitive\"];\n path?: AgnosticNonIndexRouteObject[\"path\"];\n id?: AgnosticNonIndexRouteObject[\"id\"];\n loader?: AgnosticNonIndexRouteObject[\"loader\"];\n action?: AgnosticNonIndexRouteObject[\"action\"];\n hasErrorBoundary?: AgnosticNonIndexRouteObject[\"hasErrorBoundary\"];\n shouldRevalidate?: AgnosticNonIndexRouteObject[\"shouldRevalidate\"];\n handle?: AgnosticNonIndexRouteObject[\"handle\"];\n index?: false;\n children?: RouteObject[];\n element?: React.ReactNode | null;\n hydrateFallbackElement?: React.ReactNode | null;\n errorElement?: React.ReactNode | null;\n Component?: React.ComponentType | null;\n HydrateFallback?: React.ComponentType | null;\n ErrorBoundary?: React.ComponentType | null;\n lazy?: LazyRouteFunction;\n}\n\nexport type RouteObject = IndexRouteObject | NonIndexRouteObject;\n\nexport type DataRouteObject = RouteObject & {\n children?: DataRouteObject[];\n id: string;\n};\n\nexport interface RouteMatch<\n ParamKey extends string = string,\n RouteObjectType extends RouteObject = RouteObject\n> extends AgnosticRouteMatch {}\n\nexport interface DataRouteMatch extends RouteMatch {}\n\nexport interface DataRouterContextObject\n // Omit `future` since those can be pulled from the `router`\n // `NavigationContext` needs future since it doesn't have a `router` in all cases\n extends Omit {\n router: Router;\n staticContext?: StaticHandlerContext;\n}\n\nexport const DataRouterContext =\n React.createContext(null);\nif (__DEV__) {\n DataRouterContext.displayName = \"DataRouter\";\n}\n\nexport const DataRouterStateContext = React.createContext<\n Router[\"state\"] | null\n>(null);\nif (__DEV__) {\n DataRouterStateContext.displayName = \"DataRouterState\";\n}\n\nexport const AwaitContext = React.createContext(null);\nif (__DEV__) {\n AwaitContext.displayName = \"Await\";\n}\n\nexport interface NavigateOptions {\n replace?: boolean;\n state?: any;\n preventScrollReset?: boolean;\n relative?: RelativeRoutingType;\n flushSync?: boolean;\n viewTransition?: boolean;\n}\n\n/**\n * A Navigator is a \"location changer\"; it's how you get to different locations.\n *\n * Every history instance conforms to the Navigator interface, but the\n * distinction is useful primarily when it comes to the low-level `` API\n * where both the location and a navigator must be provided separately in order\n * to avoid \"tearing\" that may occur in a suspense-enabled app if the action\n * and/or location were to be read directly from the history instance.\n */\nexport interface Navigator {\n createHref: History[\"createHref\"];\n // Optional for backwards-compat with Router/HistoryRouter usage (edge case)\n encodeLocation?: History[\"encodeLocation\"];\n go: History[\"go\"];\n push(to: To, state?: any, opts?: NavigateOptions): void;\n replace(to: To, state?: any, opts?: NavigateOptions): void;\n}\n\ninterface NavigationContextObject {\n basename: string;\n navigator: Navigator;\n static: boolean;\n future: {\n v7_relativeSplatPath: boolean;\n };\n}\n\nexport const NavigationContext = React.createContext(\n null!\n);\n\nif (__DEV__) {\n NavigationContext.displayName = \"Navigation\";\n}\n\ninterface LocationContextObject {\n location: Location;\n navigationType: NavigationType;\n}\n\nexport const LocationContext = React.createContext(\n null!\n);\n\nif (__DEV__) {\n LocationContext.displayName = \"Location\";\n}\n\nexport interface RouteContextObject {\n outlet: React.ReactElement | null;\n matches: RouteMatch[];\n isDataRoute: boolean;\n}\n\nexport const RouteContext = React.createContext({\n outlet: null,\n matches: [],\n isDataRoute: false,\n});\n\nif (__DEV__) {\n RouteContext.displayName = \"Route\";\n}\n\nexport const RouteErrorContext = React.createContext(null);\n\nif (__DEV__) {\n RouteErrorContext.displayName = \"RouteError\";\n}\n","import * as React from \"react\";\nimport type {\n AgnosticRouteMatch,\n Blocker,\n BlockerFunction,\n Location,\n ParamParseKey,\n Params,\n Path,\n PathMatch,\n PathPattern,\n RelativeRoutingType,\n Router as RemixRouter,\n RevalidationState,\n To,\n UIMatch,\n} from \"@remix-run/router\";\nimport {\n IDLE_BLOCKER,\n Action as NavigationType,\n UNSAFE_convertRouteMatchToUiMatch as convertRouteMatchToUiMatch,\n UNSAFE_decodePath as decodePath,\n UNSAFE_getResolveToMatches as getResolveToMatches,\n UNSAFE_invariant as invariant,\n isRouteErrorResponse,\n joinPaths,\n matchPath,\n matchRoutes,\n parsePath,\n resolveTo,\n stripBasename,\n UNSAFE_warning as warning,\n} from \"@remix-run/router\";\n\nimport type {\n DataRouteMatch,\n NavigateOptions,\n RouteContextObject,\n RouteMatch,\n RouteObject,\n} from \"./context\";\nimport {\n AwaitContext,\n DataRouterContext,\n DataRouterStateContext,\n LocationContext,\n NavigationContext,\n RouteContext,\n RouteErrorContext,\n} from \"./context\";\n\n/**\n * Returns the full href for the given \"to\" value. This is useful for building\n * custom links that are also accessible and preserve right-click behavior.\n *\n * @see https://reactrouter.com/v6/hooks/use-href\n */\nexport function useHref(\n to: To,\n { relative }: { relative?: RelativeRoutingType } = {}\n): string {\n invariant(\n useInRouterContext(),\n // TODO: This error is probably because they somehow have 2 versions of the\n // router loaded. We can help them understand how to avoid that.\n `useHref() may be used only in the context of a component.`\n );\n\n let { basename, navigator } = React.useContext(NavigationContext);\n let { hash, pathname, search } = useResolvedPath(to, { relative });\n\n let joinedPathname = pathname;\n\n // If we're operating within a basename, prepend it to the pathname prior\n // to creating the href. If this is a root navigation, then just use the raw\n // basename which allows the basename to have full control over the presence\n // of a trailing slash on root links\n if (basename !== \"/\") {\n joinedPathname =\n pathname === \"/\" ? basename : joinPaths([basename, pathname]);\n }\n\n return navigator.createHref({ pathname: joinedPathname, search, hash });\n}\n\n/**\n * Returns true if this component is a descendant of a ``.\n *\n * @see https://reactrouter.com/v6/hooks/use-in-router-context\n */\nexport function useInRouterContext(): boolean {\n return React.useContext(LocationContext) != null;\n}\n\n/**\n * Returns the current location object, which represents the current URL in web\n * browsers.\n *\n * Note: If you're using this it may mean you're doing some of your own\n * \"routing\" in your app, and we'd like to know what your use case is. We may\n * be able to provide something higher-level to better suit your needs.\n *\n * @see https://reactrouter.com/v6/hooks/use-location\n */\nexport function useLocation(): Location {\n invariant(\n useInRouterContext(),\n // TODO: This error is probably because they somehow have 2 versions of the\n // router loaded. We can help them understand how to avoid that.\n `useLocation() may be used only in the context of a component.`\n );\n\n return React.useContext(LocationContext).location;\n}\n\n/**\n * Returns the current navigation action which describes how the router came to\n * the current location, either by a pop, push, or replace on the history stack.\n *\n * @see https://reactrouter.com/v6/hooks/use-navigation-type\n */\nexport function useNavigationType(): NavigationType {\n return React.useContext(LocationContext).navigationType;\n}\n\n/**\n * Returns a PathMatch object if the given pattern matches the current URL.\n * This is useful for components that need to know \"active\" state, e.g.\n * ``.\n *\n * @see https://reactrouter.com/v6/hooks/use-match\n */\nexport function useMatch<\n ParamKey extends ParamParseKey,\n Path extends string\n>(pattern: PathPattern | Path): PathMatch | null {\n invariant(\n useInRouterContext(),\n // TODO: This error is probably because they somehow have 2 versions of the\n // router loaded. We can help them understand how to avoid that.\n `useMatch() may be used only in the context of a component.`\n );\n\n let { pathname } = useLocation();\n return React.useMemo(\n () => matchPath(pattern, decodePath(pathname)),\n [pathname, pattern]\n );\n}\n\n/**\n * The interface for the navigate() function returned from useNavigate().\n */\nexport interface NavigateFunction {\n (to: To, options?: NavigateOptions): void;\n (delta: number): void;\n}\n\nconst navigateEffectWarning =\n `You should call navigate() in a React.useEffect(), not when ` +\n `your component is first rendered.`;\n\n// Mute warnings for calls to useNavigate in SSR environments\nfunction useIsomorphicLayoutEffect(\n cb: Parameters[0]\n) {\n let isStatic = React.useContext(NavigationContext).static;\n if (!isStatic) {\n // We should be able to get rid of this once react 18.3 is released\n // See: https://github.com/facebook/react/pull/26395\n // eslint-disable-next-line react-hooks/rules-of-hooks\n React.useLayoutEffect(cb);\n }\n}\n\n/**\n * Returns an imperative method for changing the location. Used by ``s, but\n * may also be used by other elements to change the location.\n *\n * @see https://reactrouter.com/v6/hooks/use-navigate\n */\nexport function useNavigate(): NavigateFunction {\n let { isDataRoute } = React.useContext(RouteContext);\n // Conditional usage is OK here because the usage of a data router is static\n // eslint-disable-next-line react-hooks/rules-of-hooks\n return isDataRoute ? useNavigateStable() : useNavigateUnstable();\n}\n\nfunction useNavigateUnstable(): NavigateFunction {\n invariant(\n useInRouterContext(),\n // TODO: This error is probably because they somehow have 2 versions of the\n // router loaded. We can help them understand how to avoid that.\n `useNavigate() may be used only in the context of a component.`\n );\n\n let dataRouterContext = React.useContext(DataRouterContext);\n let { basename, future, navigator } = React.useContext(NavigationContext);\n let { matches } = React.useContext(RouteContext);\n let { pathname: locationPathname } = useLocation();\n\n let routePathnamesJson = JSON.stringify(\n getResolveToMatches(matches, future.v7_relativeSplatPath)\n );\n\n let activeRef = React.useRef(false);\n useIsomorphicLayoutEffect(() => {\n activeRef.current = true;\n });\n\n let navigate: NavigateFunction = React.useCallback(\n (to: To | number, options: NavigateOptions = {}) => {\n warning(activeRef.current, navigateEffectWarning);\n\n // Short circuit here since if this happens on first render the navigate\n // is useless because we haven't wired up our history listener yet\n if (!activeRef.current) return;\n\n if (typeof to === \"number\") {\n navigator.go(to);\n return;\n }\n\n let path = resolveTo(\n to,\n JSON.parse(routePathnamesJson),\n locationPathname,\n options.relative === \"path\"\n );\n\n // If we're operating within a basename, prepend it to the pathname prior\n // to handing off to history (but only if we're not in a data router,\n // otherwise it'll prepend the basename inside of the router).\n // If this is a root navigation, then we navigate to the raw basename\n // which allows the basename to have full control over the presence of a\n // trailing slash on root links\n if (dataRouterContext == null && basename !== \"/\") {\n path.pathname =\n path.pathname === \"/\"\n ? basename\n : joinPaths([basename, path.pathname]);\n }\n\n (!!options.replace ? navigator.replace : navigator.push)(\n path,\n options.state,\n options\n );\n },\n [\n basename,\n navigator,\n routePathnamesJson,\n locationPathname,\n dataRouterContext,\n ]\n );\n\n return navigate;\n}\n\nconst OutletContext = React.createContext(null);\n\n/**\n * Returns the context (if provided) for the child route at this level of the route\n * hierarchy.\n * @see https://reactrouter.com/v6/hooks/use-outlet-context\n */\nexport function useOutletContext(): Context {\n return React.useContext(OutletContext) as Context;\n}\n\n/**\n * Returns the element for the child route at this level of the route\n * hierarchy. Used internally by `` to render child routes.\n *\n * @see https://reactrouter.com/v6/hooks/use-outlet\n */\nexport function useOutlet(context?: unknown): React.ReactElement | null {\n let outlet = React.useContext(RouteContext).outlet;\n if (outlet) {\n return (\n {outlet}\n );\n }\n return outlet;\n}\n\n/**\n * Returns an object of key/value pairs of the dynamic params from the current\n * URL that were matched by the route path.\n *\n * @see https://reactrouter.com/v6/hooks/use-params\n */\nexport function useParams<\n ParamsOrKey extends string | Record = string\n>(): Readonly<\n [ParamsOrKey] extends [string] ? Params : Partial\n> {\n let { matches } = React.useContext(RouteContext);\n let routeMatch = matches[matches.length - 1];\n return routeMatch ? (routeMatch.params as any) : {};\n}\n\n/**\n * Resolves the pathname of the given `to` value against the current location.\n *\n * @see https://reactrouter.com/v6/hooks/use-resolved-path\n */\nexport function useResolvedPath(\n to: To,\n { relative }: { relative?: RelativeRoutingType } = {}\n): Path {\n let { future } = React.useContext(NavigationContext);\n let { matches } = React.useContext(RouteContext);\n let { pathname: locationPathname } = useLocation();\n let routePathnamesJson = JSON.stringify(\n getResolveToMatches(matches, future.v7_relativeSplatPath)\n );\n\n return React.useMemo(\n () =>\n resolveTo(\n to,\n JSON.parse(routePathnamesJson),\n locationPathname,\n relative === \"path\"\n ),\n [to, routePathnamesJson, locationPathname, relative]\n );\n}\n\n/**\n * Returns the element of the route that matched the current location, prepared\n * with the correct context to render the remainder of the route tree. Route\n * elements in the tree must render an `` to render their child route's\n * element.\n *\n * @see https://reactrouter.com/v6/hooks/use-routes\n */\nexport function useRoutes(\n routes: RouteObject[],\n locationArg?: Partial | string\n): React.ReactElement | null {\n return useRoutesImpl(routes, locationArg);\n}\n\n// Internal implementation with accept optional param for RouterProvider usage\nexport function useRoutesImpl(\n routes: RouteObject[],\n locationArg?: Partial | string,\n dataRouterState?: RemixRouter[\"state\"],\n future?: RemixRouter[\"future\"]\n): React.ReactElement | null {\n invariant(\n useInRouterContext(),\n // TODO: This error is probably because they somehow have 2 versions of the\n // router loaded. We can help them understand how to avoid that.\n `useRoutes() may be used only in the context of a component.`\n );\n\n let { navigator, static: isStatic } = React.useContext(NavigationContext);\n let { matches: parentMatches } = React.useContext(RouteContext);\n let routeMatch = parentMatches[parentMatches.length - 1];\n let parentParams = routeMatch ? routeMatch.params : {};\n let parentPathname = routeMatch ? routeMatch.pathname : \"/\";\n let parentPathnameBase = routeMatch ? routeMatch.pathnameBase : \"/\";\n let parentRoute = routeMatch && routeMatch.route;\n\n if (__DEV__) {\n // You won't get a warning about 2 different under a \n // without a trailing *, but this is a best-effort warning anyway since we\n // cannot even give the warning unless they land at the parent route.\n //\n // Example:\n //\n // \n // {/* This route path MUST end with /* because otherwise\n // it will never match /blog/post/123 */}\n // } />\n // } />\n // \n //\n // function Blog() {\n // return (\n // \n // } />\n // \n // );\n // }\n let parentPath = (parentRoute && parentRoute.path) || \"\";\n warningOnce(\n parentPathname,\n !parentRoute || parentPath.endsWith(\"*\"),\n `You rendered descendant (or called \\`useRoutes()\\`) at ` +\n `\"${parentPathname}\" (under ) but the ` +\n `parent route path has no trailing \"*\". This means if you navigate ` +\n `deeper, the parent won't match anymore and therefore the child ` +\n `routes will never render.\\n\\n` +\n `Please change the parent to .`\n );\n }\n\n let locationFromContext = useLocation();\n\n let location;\n if (locationArg) {\n let parsedLocationArg =\n typeof locationArg === \"string\" ? parsePath(locationArg) : locationArg;\n\n invariant(\n parentPathnameBase === \"/\" ||\n parsedLocationArg.pathname?.startsWith(parentPathnameBase),\n `When overriding the location using \\`\\` or \\`useRoutes(routes, location)\\`, ` +\n `the location pathname must begin with the portion of the URL pathname that was ` +\n `matched by all parent routes. The current pathname base is \"${parentPathnameBase}\" ` +\n `but pathname \"${parsedLocationArg.pathname}\" was given in the \\`location\\` prop.`\n );\n\n location = parsedLocationArg;\n } else {\n location = locationFromContext;\n }\n\n let pathname = location.pathname || \"/\";\n\n let remainingPathname = pathname;\n if (parentPathnameBase !== \"/\") {\n // Determine the remaining pathname by removing the # of URL segments the\n // parentPathnameBase has, instead of removing based on character count.\n // This is because we can't guarantee that incoming/outgoing encodings/\n // decodings will match exactly.\n // We decode paths before matching on a per-segment basis with\n // decodeURIComponent(), but we re-encode pathnames via `new URL()` so they\n // match what `window.location.pathname` would reflect. Those don't 100%\n // align when it comes to encoded URI characters such as % and &.\n //\n // So we may end up with:\n // pathname: \"/descendant/a%25b/match\"\n // parentPathnameBase: \"/descendant/a%b\"\n //\n // And the direct substring removal approach won't work :/\n let parentSegments = parentPathnameBase.replace(/^\\//, \"\").split(\"/\");\n let segments = pathname.replace(/^\\//, \"\").split(\"/\");\n remainingPathname = \"/\" + segments.slice(parentSegments.length).join(\"/\");\n }\n\n let matches =\n !isStatic &&\n dataRouterState &&\n dataRouterState.matches &&\n dataRouterState.matches.length > 0\n ? (dataRouterState.matches as AgnosticRouteMatch[])\n : matchRoutes(routes, { pathname: remainingPathname });\n\n if (__DEV__) {\n warning(\n parentRoute || matches != null,\n `No routes matched location \"${location.pathname}${location.search}${location.hash}\" `\n );\n\n warning(\n matches == null ||\n matches[matches.length - 1].route.element !== undefined ||\n matches[matches.length - 1].route.Component !== undefined ||\n matches[matches.length - 1].route.lazy !== undefined,\n `Matched leaf route at location \"${location.pathname}${location.search}${location.hash}\" ` +\n `does not have an element or Component. This means it will render an with a ` +\n `null value by default resulting in an \"empty\" page.`\n );\n }\n\n let renderedMatches = _renderMatches(\n matches &&\n matches.map((match) =>\n Object.assign({}, match, {\n params: Object.assign({}, parentParams, match.params),\n pathname: joinPaths([\n parentPathnameBase,\n // Re-encode pathnames that were decoded inside matchRoutes\n navigator.encodeLocation\n ? navigator.encodeLocation(match.pathname).pathname\n : match.pathname,\n ]),\n pathnameBase:\n match.pathnameBase === \"/\"\n ? parentPathnameBase\n : joinPaths([\n parentPathnameBase,\n // Re-encode pathnames that were decoded inside matchRoutes\n navigator.encodeLocation\n ? navigator.encodeLocation(match.pathnameBase).pathname\n : match.pathnameBase,\n ]),\n })\n ),\n parentMatches,\n dataRouterState,\n future\n );\n\n // When a user passes in a `locationArg`, the associated routes need to\n // be wrapped in a new `LocationContext.Provider` in order for `useLocation`\n // to use the scoped location instead of the global location.\n if (locationArg && renderedMatches) {\n return (\n \n {renderedMatches}\n \n );\n }\n\n return renderedMatches;\n}\n\nfunction DefaultErrorComponent() {\n let error = useRouteError();\n let message = isRouteErrorResponse(error)\n ? `${error.status} ${error.statusText}`\n : error instanceof Error\n ? error.message\n : JSON.stringify(error);\n let stack = error instanceof Error ? error.stack : null;\n let lightgrey = \"rgba(200,200,200, 0.5)\";\n let preStyles = { padding: \"0.5rem\", backgroundColor: lightgrey };\n let codeStyles = { padding: \"2px 4px\", backgroundColor: lightgrey };\n\n let devInfo = null;\n if (__DEV__) {\n console.error(\n \"Error handled by React Router default ErrorBoundary:\",\n error\n );\n\n devInfo = (\n <>\n

💿 Hey developer 👋

\n

\n You can provide a way better UX than this when your app throws errors\n by providing your own ErrorBoundary or{\" \"}\n errorElement prop on your route.\n

\n \n );\n }\n\n return (\n <>\n

Unexpected Application Error!

\n

{message}

\n {stack ?
{stack}
: null}\n {devInfo}\n \n );\n}\n\nconst defaultErrorElement = ;\n\ntype RenderErrorBoundaryProps = React.PropsWithChildren<{\n location: Location;\n revalidation: RevalidationState;\n error: any;\n component: React.ReactNode;\n routeContext: RouteContextObject;\n}>;\n\ntype RenderErrorBoundaryState = {\n location: Location;\n revalidation: RevalidationState;\n error: any;\n};\n\nexport class RenderErrorBoundary extends React.Component<\n RenderErrorBoundaryProps,\n RenderErrorBoundaryState\n> {\n constructor(props: RenderErrorBoundaryProps) {\n super(props);\n this.state = {\n location: props.location,\n revalidation: props.revalidation,\n error: props.error,\n };\n }\n\n static getDerivedStateFromError(error: any) {\n return { error: error };\n }\n\n static getDerivedStateFromProps(\n props: RenderErrorBoundaryProps,\n state: RenderErrorBoundaryState\n ) {\n // When we get into an error state, the user will likely click \"back\" to the\n // previous page that didn't have an error. Because this wraps the entire\n // application, that will have no effect--the error page continues to display.\n // This gives us a mechanism to recover from the error when the location changes.\n //\n // Whether we're in an error state or not, we update the location in state\n // so that when we are in an error state, it gets reset when a new location\n // comes in and the user recovers from the error.\n if (\n state.location !== props.location ||\n (state.revalidation !== \"idle\" && props.revalidation === \"idle\")\n ) {\n return {\n error: props.error,\n location: props.location,\n revalidation: props.revalidation,\n };\n }\n\n // If we're not changing locations, preserve the location but still surface\n // any new errors that may come through. We retain the existing error, we do\n // this because the error provided from the app state may be cleared without\n // the location changing.\n return {\n error: props.error !== undefined ? props.error : state.error,\n location: state.location,\n revalidation: props.revalidation || state.revalidation,\n };\n }\n\n componentDidCatch(error: any, errorInfo: any) {\n console.error(\n \"React Router caught the following error during render\",\n error,\n errorInfo\n );\n }\n\n render() {\n return this.state.error !== undefined ? (\n \n \n \n ) : (\n this.props.children\n );\n }\n}\n\ninterface RenderedRouteProps {\n routeContext: RouteContextObject;\n match: RouteMatch;\n children: React.ReactNode | null;\n}\n\nfunction RenderedRoute({ routeContext, match, children }: RenderedRouteProps) {\n let dataRouterContext = React.useContext(DataRouterContext);\n\n // Track how deep we got in our render pass to emulate SSR componentDidCatch\n // in a DataStaticRouter\n if (\n dataRouterContext &&\n dataRouterContext.static &&\n dataRouterContext.staticContext &&\n (match.route.errorElement || match.route.ErrorBoundary)\n ) {\n dataRouterContext.staticContext._deepestRenderedBoundaryId = match.route.id;\n }\n\n return (\n \n {children}\n \n );\n}\n\nexport function _renderMatches(\n matches: RouteMatch[] | null,\n parentMatches: RouteMatch[] = [],\n dataRouterState: RemixRouter[\"state\"] | null = null,\n future: RemixRouter[\"future\"] | null = null\n): React.ReactElement | null {\n if (matches == null) {\n if (!dataRouterState) {\n return null;\n }\n\n if (dataRouterState.errors) {\n // Don't bail if we have data router errors so we can render them in the\n // boundary. Use the pre-matched (or shimmed) matches\n matches = dataRouterState.matches as DataRouteMatch[];\n } else if (\n future?.v7_partialHydration &&\n parentMatches.length === 0 &&\n !dataRouterState.initialized &&\n dataRouterState.matches.length > 0\n ) {\n // Don't bail if we're initializing with partial hydration and we have\n // router matches. That means we're actively running `patchRoutesOnNavigation`\n // so we should render down the partial matches to the appropriate\n // `HydrateFallback`. We only do this if `parentMatches` is empty so it\n // only impacts the root matches for `RouterProvider` and no descendant\n // ``\n matches = dataRouterState.matches as DataRouteMatch[];\n } else {\n return null;\n }\n }\n\n let renderedMatches = matches;\n\n // If we have data errors, trim matches to the highest error boundary\n let errors = dataRouterState?.errors;\n if (errors != null) {\n let errorIndex = renderedMatches.findIndex(\n (m) => m.route.id && errors?.[m.route.id] !== undefined\n );\n invariant(\n errorIndex >= 0,\n `Could not find a matching route for errors on route IDs: ${Object.keys(\n errors\n ).join(\",\")}`\n );\n renderedMatches = renderedMatches.slice(\n 0,\n Math.min(renderedMatches.length, errorIndex + 1)\n );\n }\n\n // If we're in a partial hydration mode, detect if we need to render down to\n // a given HydrateFallback while we load the rest of the hydration data\n let renderFallback = false;\n let fallbackIndex = -1;\n if (dataRouterState && future && future.v7_partialHydration) {\n for (let i = 0; i < renderedMatches.length; i++) {\n let match = renderedMatches[i];\n // Track the deepest fallback up until the first route without data\n if (match.route.HydrateFallback || match.route.hydrateFallbackElement) {\n fallbackIndex = i;\n }\n\n if (match.route.id) {\n let { loaderData, errors } = dataRouterState;\n let needsToRunLoader =\n match.route.loader &&\n loaderData[match.route.id] === undefined &&\n (!errors || errors[match.route.id] === undefined);\n if (match.route.lazy || needsToRunLoader) {\n // We found the first route that's not ready to render (waiting on\n // lazy, or has a loader that hasn't run yet). Flag that we need to\n // render a fallback and render up until the appropriate fallback\n renderFallback = true;\n if (fallbackIndex >= 0) {\n renderedMatches = renderedMatches.slice(0, fallbackIndex + 1);\n } else {\n renderedMatches = [renderedMatches[0]];\n }\n break;\n }\n }\n }\n }\n\n return renderedMatches.reduceRight((outlet, match, index) => {\n // Only data routers handle errors/fallbacks\n let error: any;\n let shouldRenderHydrateFallback = false;\n let errorElement: React.ReactNode | null = null;\n let hydrateFallbackElement: React.ReactNode | null = null;\n if (dataRouterState) {\n error = errors && match.route.id ? errors[match.route.id] : undefined;\n errorElement = match.route.errorElement || defaultErrorElement;\n\n if (renderFallback) {\n if (fallbackIndex < 0 && index === 0) {\n warningOnce(\n \"route-fallback\",\n false,\n \"No `HydrateFallback` element provided to render during initial hydration\"\n );\n shouldRenderHydrateFallback = true;\n hydrateFallbackElement = null;\n } else if (fallbackIndex === index) {\n shouldRenderHydrateFallback = true;\n hydrateFallbackElement = match.route.hydrateFallbackElement || null;\n }\n }\n }\n\n let matches = parentMatches.concat(renderedMatches.slice(0, index + 1));\n let getChildren = () => {\n let children: React.ReactNode;\n if (error) {\n children = errorElement;\n } else if (shouldRenderHydrateFallback) {\n children = hydrateFallbackElement;\n } else if (match.route.Component) {\n // Note: This is a de-optimized path since React won't re-use the\n // ReactElement since it's identity changes with each new\n // React.createElement call. We keep this so folks can use\n // `` in `` but generally `Component`\n // usage is only advised in `RouterProvider` when we can convert it to\n // `element` ahead of time.\n children = ;\n } else if (match.route.element) {\n children = match.route.element;\n } else {\n children = outlet;\n }\n return (\n \n );\n };\n // Only wrap in an error boundary within data router usages when we have an\n // ErrorBoundary/errorElement on this route. Otherwise let it bubble up to\n // an ancestor ErrorBoundary/errorElement\n return dataRouterState &&\n (match.route.ErrorBoundary || match.route.errorElement || index === 0) ? (\n \n ) : (\n getChildren()\n );\n }, null as React.ReactElement | null);\n}\n\nenum DataRouterHook {\n UseBlocker = \"useBlocker\",\n UseRevalidator = \"useRevalidator\",\n UseNavigateStable = \"useNavigate\",\n}\n\nenum DataRouterStateHook {\n UseBlocker = \"useBlocker\",\n UseLoaderData = \"useLoaderData\",\n UseActionData = \"useActionData\",\n UseRouteError = \"useRouteError\",\n UseNavigation = \"useNavigation\",\n UseRouteLoaderData = \"useRouteLoaderData\",\n UseMatches = \"useMatches\",\n UseRevalidator = \"useRevalidator\",\n UseNavigateStable = \"useNavigate\",\n UseRouteId = \"useRouteId\",\n}\n\nfunction getDataRouterConsoleError(\n hookName: DataRouterHook | DataRouterStateHook\n) {\n return `${hookName} must be used within a data router. See https://reactrouter.com/v6/routers/picking-a-router.`;\n}\n\nfunction useDataRouterContext(hookName: DataRouterHook) {\n let ctx = React.useContext(DataRouterContext);\n invariant(ctx, getDataRouterConsoleError(hookName));\n return ctx;\n}\n\nfunction useDataRouterState(hookName: DataRouterStateHook) {\n let state = React.useContext(DataRouterStateContext);\n invariant(state, getDataRouterConsoleError(hookName));\n return state;\n}\n\nfunction useRouteContext(hookName: DataRouterStateHook) {\n let route = React.useContext(RouteContext);\n invariant(route, getDataRouterConsoleError(hookName));\n return route;\n}\n\n// Internal version with hookName-aware debugging\nfunction useCurrentRouteId(hookName: DataRouterStateHook) {\n let route = useRouteContext(hookName);\n let thisRoute = route.matches[route.matches.length - 1];\n invariant(\n thisRoute.route.id,\n `${hookName} can only be used on routes that contain a unique \"id\"`\n );\n return thisRoute.route.id;\n}\n\n/**\n * Returns the ID for the nearest contextual route\n */\nexport function useRouteId() {\n return useCurrentRouteId(DataRouterStateHook.UseRouteId);\n}\n\n/**\n * Returns the current navigation, defaulting to an \"idle\" navigation when\n * no navigation is in progress\n */\nexport function useNavigation() {\n let state = useDataRouterState(DataRouterStateHook.UseNavigation);\n return state.navigation;\n}\n\n/**\n * Returns a revalidate function for manually triggering revalidation, as well\n * as the current state of any manual revalidations\n */\nexport function useRevalidator() {\n let dataRouterContext = useDataRouterContext(DataRouterHook.UseRevalidator);\n let state = useDataRouterState(DataRouterStateHook.UseRevalidator);\n return React.useMemo(\n () => ({\n revalidate: dataRouterContext.router.revalidate,\n state: state.revalidation,\n }),\n [dataRouterContext.router.revalidate, state.revalidation]\n );\n}\n\n/**\n * Returns the active route matches, useful for accessing loaderData for\n * parent/child routes or the route \"handle\" property\n */\nexport function useMatches(): UIMatch[] {\n let { matches, loaderData } = useDataRouterState(\n DataRouterStateHook.UseMatches\n );\n return React.useMemo(\n () => matches.map((m) => convertRouteMatchToUiMatch(m, loaderData)),\n [matches, loaderData]\n );\n}\n\n/**\n * Returns the loader data for the nearest ancestor Route loader\n */\nexport function useLoaderData(): unknown {\n let state = useDataRouterState(DataRouterStateHook.UseLoaderData);\n let routeId = useCurrentRouteId(DataRouterStateHook.UseLoaderData);\n\n if (state.errors && state.errors[routeId] != null) {\n console.error(\n `You cannot \\`useLoaderData\\` in an errorElement (routeId: ${routeId})`\n );\n return undefined;\n }\n return state.loaderData[routeId];\n}\n\n/**\n * Returns the loaderData for the given routeId\n */\nexport function useRouteLoaderData(routeId: string): unknown {\n let state = useDataRouterState(DataRouterStateHook.UseRouteLoaderData);\n return state.loaderData[routeId];\n}\n\n/**\n * Returns the action data for the nearest ancestor Route action\n */\nexport function useActionData(): unknown {\n let state = useDataRouterState(DataRouterStateHook.UseActionData);\n let routeId = useCurrentRouteId(DataRouterStateHook.UseLoaderData);\n return state.actionData ? state.actionData[routeId] : undefined;\n}\n\n/**\n * Returns the nearest ancestor Route error, which could be a loader/action\n * error or a render error. This is intended to be called from your\n * ErrorBoundary/errorElement to display a proper error message.\n */\nexport function useRouteError(): unknown {\n let error = React.useContext(RouteErrorContext);\n let state = useDataRouterState(DataRouterStateHook.UseRouteError);\n let routeId = useCurrentRouteId(DataRouterStateHook.UseRouteError);\n\n // If this was a render error, we put it in a RouteError context inside\n // of RenderErrorBoundary\n if (error !== undefined) {\n return error;\n }\n\n // Otherwise look for errors from our data router state\n return state.errors?.[routeId];\n}\n\n/**\n * Returns the happy-path data from the nearest ancestor `` value\n */\nexport function useAsyncValue(): unknown {\n let value = React.useContext(AwaitContext);\n return value?._data;\n}\n\n/**\n * Returns the error from the nearest ancestor `` value\n */\nexport function useAsyncError(): unknown {\n let value = React.useContext(AwaitContext);\n return value?._error;\n}\n\nlet blockerId = 0;\n\n/**\n * Allow the application to block navigations within the SPA and present the\n * user a confirmation dialog to confirm the navigation. Mostly used to avoid\n * using half-filled form data. This does not handle hard-reloads or\n * cross-origin navigations.\n */\nexport function useBlocker(shouldBlock: boolean | BlockerFunction): Blocker {\n let { router, basename } = useDataRouterContext(DataRouterHook.UseBlocker);\n let state = useDataRouterState(DataRouterStateHook.UseBlocker);\n\n let [blockerKey, setBlockerKey] = React.useState(\"\");\n let blockerFunction = React.useCallback(\n (arg) => {\n if (typeof shouldBlock !== \"function\") {\n return !!shouldBlock;\n }\n if (basename === \"/\") {\n return shouldBlock(arg);\n }\n\n // If they provided us a function and we've got an active basename, strip\n // it from the locations we expose to the user to match the behavior of\n // useLocation\n let { currentLocation, nextLocation, historyAction } = arg;\n return shouldBlock({\n currentLocation: {\n ...currentLocation,\n pathname:\n stripBasename(currentLocation.pathname, basename) ||\n currentLocation.pathname,\n },\n nextLocation: {\n ...nextLocation,\n pathname:\n stripBasename(nextLocation.pathname, basename) ||\n nextLocation.pathname,\n },\n historyAction,\n });\n },\n [basename, shouldBlock]\n );\n\n // This effect is in charge of blocker key assignment and deletion (which is\n // tightly coupled to the key)\n React.useEffect(() => {\n let key = String(++blockerId);\n setBlockerKey(key);\n return () => router.deleteBlocker(key);\n }, [router]);\n\n // This effect handles assigning the blockerFunction. This is to handle\n // unstable blocker function identities, and happens only after the prior\n // effect so we don't get an orphaned blockerFunction in the router with a\n // key of \"\". Until then we just have the IDLE_BLOCKER.\n React.useEffect(() => {\n if (blockerKey !== \"\") {\n router.getBlocker(blockerKey, blockerFunction);\n }\n }, [router, blockerKey, blockerFunction]);\n\n // Prefer the blocker from `state` not `router.state` since DataRouterContext\n // is memoized so this ensures we update on blocker state updates\n return blockerKey && state.blockers.has(blockerKey)\n ? state.blockers.get(blockerKey)!\n : IDLE_BLOCKER;\n}\n\n/**\n * Stable version of useNavigate that is used when we are in the context of\n * a RouterProvider.\n */\nfunction useNavigateStable(): NavigateFunction {\n let { router } = useDataRouterContext(DataRouterHook.UseNavigateStable);\n let id = useCurrentRouteId(DataRouterStateHook.UseNavigateStable);\n\n let activeRef = React.useRef(false);\n useIsomorphicLayoutEffect(() => {\n activeRef.current = true;\n });\n\n let navigate: NavigateFunction = React.useCallback(\n (to: To | number, options: NavigateOptions = {}) => {\n warning(activeRef.current, navigateEffectWarning);\n\n // Short circuit here since if this happens on first render the navigate\n // is useless because we haven't wired up our router subscriber yet\n if (!activeRef.current) return;\n\n if (typeof to === \"number\") {\n router.navigate(to);\n } else {\n router.navigate(to, { fromRouteId: id, ...options });\n }\n },\n [router, id]\n );\n\n return navigate;\n}\n\nconst alreadyWarned: Record = {};\n\nfunction warningOnce(key: string, cond: boolean, message: string) {\n if (!cond && !alreadyWarned[key]) {\n alreadyWarned[key] = true;\n warning(false, message);\n }\n}\n","import type { FutureConfig as RouterFutureConfig } from \"@remix-run/router\";\nimport type { FutureConfig as RenderFutureConfig } from \"./components\";\n\nconst alreadyWarned: { [key: string]: boolean } = {};\n\nexport function warnOnce(key: string, message: string): void {\n if (__DEV__ && !alreadyWarned[message]) {\n alreadyWarned[message] = true;\n console.warn(message);\n }\n}\n\nconst logDeprecation = (flag: string, msg: string, link: string) =>\n warnOnce(\n flag,\n `⚠️ React Router Future Flag Warning: ${msg}. ` +\n `You can use the \\`${flag}\\` future flag to opt-in early. ` +\n `For more information, see ${link}.`\n );\n\nexport function logV6DeprecationWarnings(\n renderFuture: Partial | undefined,\n routerFuture?: Omit\n) {\n if (renderFuture?.v7_startTransition === undefined) {\n logDeprecation(\n \"v7_startTransition\",\n \"React Router will begin wrapping state updates in `React.startTransition` in v7\",\n \"https://reactrouter.com/v6/upgrading/future#v7_starttransition\"\n );\n }\n\n if (\n renderFuture?.v7_relativeSplatPath === undefined &&\n (!routerFuture || !routerFuture.v7_relativeSplatPath)\n ) {\n logDeprecation(\n \"v7_relativeSplatPath\",\n \"Relative route resolution within Splat routes is changing in v7\",\n \"https://reactrouter.com/v6/upgrading/future#v7_relativesplatpath\"\n );\n }\n\n if (routerFuture) {\n if (routerFuture.v7_fetcherPersist === undefined) {\n logDeprecation(\n \"v7_fetcherPersist\",\n \"The persistence behavior of fetchers is changing in v7\",\n \"https://reactrouter.com/v6/upgrading/future#v7_fetcherpersist\"\n );\n }\n\n if (routerFuture.v7_normalizeFormMethod === undefined) {\n logDeprecation(\n \"v7_normalizeFormMethod\",\n \"Casing of `formMethod` fields is being normalized to uppercase in v7\",\n \"https://reactrouter.com/v6/upgrading/future#v7_normalizeformmethod\"\n );\n }\n\n if (routerFuture.v7_partialHydration === undefined) {\n logDeprecation(\n \"v7_partialHydration\",\n \"`RouterProvider` hydration behavior is changing in v7\",\n \"https://reactrouter.com/v6/upgrading/future#v7_partialhydration\"\n );\n }\n\n if (routerFuture.v7_skipActionErrorRevalidation === undefined) {\n logDeprecation(\n \"v7_skipActionErrorRevalidation\",\n \"The revalidation behavior after 4xx/5xx `action` responses is changing in v7\",\n \"https://reactrouter.com/v6/upgrading/future#v7_skipactionerrorrevalidation\"\n );\n }\n }\n}\n","import type {\n InitialEntry,\n LazyRouteFunction,\n Location,\n MemoryHistory,\n RelativeRoutingType,\n Router as RemixRouter,\n RouterState,\n RouterSubscriber,\n To,\n TrackedPromise,\n} from \"@remix-run/router\";\nimport {\n AbortedDeferredError,\n Action as NavigationType,\n createMemoryHistory,\n UNSAFE_getResolveToMatches as getResolveToMatches,\n UNSAFE_invariant as invariant,\n parsePath,\n resolveTo,\n stripBasename,\n UNSAFE_warning as warning,\n} from \"@remix-run/router\";\nimport * as React from \"react\";\n\nimport type {\n DataRouteObject,\n IndexRouteObject,\n Navigator,\n NonIndexRouteObject,\n RouteMatch,\n RouteObject,\n} from \"./context\";\nimport {\n AwaitContext,\n DataRouterContext,\n DataRouterStateContext,\n LocationContext,\n NavigationContext,\n RouteContext,\n} from \"./context\";\nimport {\n _renderMatches,\n useAsyncValue,\n useInRouterContext,\n useLocation,\n useNavigate,\n useOutlet,\n useRoutes,\n useRoutesImpl,\n} from \"./hooks\";\nimport { logV6DeprecationWarnings } from \"./deprecations\";\n\nexport interface FutureConfig {\n v7_relativeSplatPath: boolean;\n v7_startTransition: boolean;\n}\n\nexport interface RouterProviderProps {\n fallbackElement?: React.ReactNode;\n router: RemixRouter;\n // Only accept future flags relevant to rendering behavior\n // routing flags should be accessed via router.future\n future?: Partial>;\n}\n\n/**\n Webpack + React 17 fails to compile on any of the following because webpack\n complains that `startTransition` doesn't exist in `React`:\n * import { startTransition } from \"react\"\n * import * as React from from \"react\";\n \"startTransition\" in React ? React.startTransition(() => setState()) : setState()\n * import * as React from from \"react\";\n \"startTransition\" in React ? React[\"startTransition\"](() => setState()) : setState()\n\n Moving it to a constant such as the following solves the Webpack/React 17 issue:\n * import * as React from from \"react\";\n const START_TRANSITION = \"startTransition\";\n START_TRANSITION in React ? React[START_TRANSITION](() => setState()) : setState()\n\n However, that introduces webpack/terser minification issues in production builds\n in React 18 where minification/obfuscation ends up removing the call of\n React.startTransition entirely from the first half of the ternary. Grabbing\n this exported reference once up front resolves that issue.\n\n See https://github.com/remix-run/react-router/issues/10579\n*/\nconst START_TRANSITION = \"startTransition\";\nconst startTransitionImpl = React[START_TRANSITION];\n\n/**\n * Given a Remix Router instance, render the appropriate UI\n */\nexport function RouterProvider({\n fallbackElement,\n router,\n future,\n}: RouterProviderProps): React.ReactElement {\n let [state, setStateImpl] = React.useState(router.state);\n let { v7_startTransition } = future || {};\n\n let setState = React.useCallback(\n (newState: RouterState) => {\n if (v7_startTransition && startTransitionImpl) {\n startTransitionImpl(() => setStateImpl(newState));\n } else {\n setStateImpl(newState);\n }\n },\n [setStateImpl, v7_startTransition]\n );\n\n // Need to use a layout effect here so we are subscribed early enough to\n // pick up on any render-driven redirects/navigations (useEffect/)\n React.useLayoutEffect(() => router.subscribe(setState), [router, setState]);\n\n React.useEffect(() => {\n warning(\n fallbackElement == null || !router.future.v7_partialHydration,\n \"`` is deprecated when using \" +\n \"`v7_partialHydration`, use a `HydrateFallback` component instead\"\n );\n // Only log this once on initial mount\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n let navigator = React.useMemo((): Navigator => {\n return {\n createHref: router.createHref,\n encodeLocation: router.encodeLocation,\n go: (n) => router.navigate(n),\n push: (to, state, opts) =>\n router.navigate(to, {\n state,\n preventScrollReset: opts?.preventScrollReset,\n }),\n replace: (to, state, opts) =>\n router.navigate(to, {\n replace: true,\n state,\n preventScrollReset: opts?.preventScrollReset,\n }),\n };\n }, [router]);\n\n let basename = router.basename || \"/\";\n\n let dataRouterContext = React.useMemo(\n () => ({\n router,\n navigator,\n static: false,\n basename,\n }),\n [router, navigator, basename]\n );\n\n React.useEffect(\n () => logV6DeprecationWarnings(future, router.future),\n [router, future]\n );\n\n // The fragment and {null} here are important! We need them to keep React 18's\n // useId happy when we are server-rendering since we may have a