Langswitcher as parallel instead of Header. Using promise.all and added some Suspenses

This commit is contained in:
Linus Flood
2024-09-11 19:59:25 +02:00
parent 12178279d3
commit 33ffd781bb
15 changed files with 144 additions and 87 deletions

View File

@@ -1,7 +1,9 @@
import { Suspense } from "react"
import { serverClient } from "@/lib/trpc/server" import { serverClient } from "@/lib/trpc/server"
import BookingWidget from "@/components/BookingWidget" import BookingWidget from "@/components/BookingWidget"
import { getLang } from "@/i18n/serverContext" import LoadingSpinner from "@/components/LoadingSpinner"
export default async function BookingWidgetPage() { export default async function BookingWidgetPage() {
// Get the booking widget show/hide status based on page specific settings // Get the booking widget show/hide status based on page specific settings
@@ -9,10 +11,10 @@ export default async function BookingWidgetPage() {
await serverClient().contentstack.bookingwidget.getToggle() await serverClient().contentstack.bookingwidget.getToggle()
return ( return (
<> <Suspense fallback={<LoadingSpinner />}>
{bookingWidgetToggle && bookingWidgetToggle.hideBookingWidget ? null : ( {bookingWidgetToggle && bookingWidgetToggle.hideBookingWidget ? null : (
<BookingWidget /> <BookingWidget />
)} )}
</> </Suspense>
) )
} }

View File

@@ -0,0 +1 @@
export { default } from "../page"

View File

@@ -0,0 +1 @@
export { default } from "../../page"

View File

@@ -0,0 +1 @@
export { default } from "./page"

View File

@@ -0,0 +1 @@
export { default } from "../../page"

View File

@@ -0,0 +1,21 @@
import { Suspense } from "react"
import { serverClient } from "@/lib/trpc/server"
import LanguageSwitcher from "@/components/LanguageSwitcher"
import { setLang } from "@/i18n/serverContext"
import { LangParams, PageArgs } from "@/types/params"
export default async function HeaderPage({ params }: PageArgs<LangParams>) {
setLang(params.lang)
const languages = await serverClient().contentstack.languageSwitcher.get()
if (!languages) {
return null
}
return (
<Suspense>
<LanguageSwitcher type="desktopHeader" urls={languages.urls} />
</Suspense>
)
}

View File

@@ -21,9 +21,11 @@ import type { LangParams, LayoutArgs } from "@/types/params"
export default async function RootLayout({ export default async function RootLayout({
children, children,
params, params,
languageSwitcher,
bookingwidget, bookingwidget,
}: React.PropsWithChildren< }: React.PropsWithChildren<
LayoutArgs<LangParams> & { LayoutArgs<LangParams> & {
languageSwitcher: React.ReactNode
bookingwidget: React.ReactNode bookingwidget: React.ReactNode
} }
>) { >) {
@@ -52,7 +54,7 @@ export default async function RootLayout({
<body> <body>
<ServerIntlProvider intl={{ defaultLocale, locale, messages }}> <ServerIntlProvider intl={{ defaultLocale, locale, messages }}>
<TrpcProvider> <TrpcProvider>
<Header /> <Header languageSwitcher={languageSwitcher} />
{bookingwidget} {bookingwidget}
{children} {children}
<ToastHandler /> <ToastHandler />

View File

@@ -1,11 +1,17 @@
import { Suspense } from "react"
import Form from "@/components/Forms/BookingWidget" import Form from "@/components/Forms/BookingWidget"
import LoadingSpinner from "../LoadingSpinner"
import styles from "./bookingWidget.module.css" import styles from "./bookingWidget.module.css"
export default async function BookingWidget() { export default async function BookingWidget() {
return ( return (
<section className={styles.container}> <section className={styles.container}>
<Suspense fallback={<LoadingSpinner />}>
<Form /> <Form />
</Suspense>
</section> </section>
) )
} }

View File

@@ -23,13 +23,14 @@ export default async function MainMenu({
menuItems, menuItems,
topLink, topLink,
}: MainMenuProps) { }: MainMenuProps) {
const intl = await getIntl()
const lang = getLang() const lang = getLang()
const myPagesNavigation =
await serverClient().contentstack.myPages.navigation.get()
const user = await serverClient().user.name() const [intl, myPagesNavigation, user, membership] = await Promise.all([
const membership = await serverClient().user.safeMembershipLevel() getIntl(),
serverClient().contentstack.myPages.navigation.get(),
serverClient().user.name(),
serverClient().user.safeMembershipLevel(),
])
return ( return (
<div className={styles.mainMenu}> <div className={styles.mainMenu}>

View File

@@ -1,3 +1,5 @@
import { Suspense } from "react"
import { GiftIcon, SearchIcon } from "@/components/Icons" import { GiftIcon, SearchIcon } from "@/components/Icons"
import LanguageSwitcher from "@/components/LanguageSwitcher" import LanguageSwitcher from "@/components/LanguageSwitcher"
import { getIntl } from "@/i18n" import { getIntl } from "@/i18n"
@@ -8,7 +10,10 @@ import styles from "./topMenu.module.css"
import type { TopMenuProps } from "@/types/components/header/topMenu" import type { TopMenuProps } from "@/types/components/header/topMenu"
export default async function TopMenu({ languageUrls, topLink }: TopMenuProps) { export default async function TopMenu({
languageSwitcher,
topLink,
}: TopMenuProps) {
const intl = await getIntl() const intl = await getIntl()
return ( return (
@@ -21,7 +26,7 @@ export default async function TopMenu({ languageUrls, topLink }: TopMenuProps) {
</HeaderLink> </HeaderLink>
) : null} ) : null}
<div className={styles.options}> <div className={styles.options}>
<LanguageSwitcher type="desktopHeader" urls={languageUrls} /> <Suspense>{languageSwitcher}</Suspense>
<HeaderLink href="#"> <HeaderLink href="#">
<SearchIcon width={20} height={20} color="burgundy" /> <SearchIcon width={20} height={20} color="burgundy" />
{intl.formatMessage({ id: "Find booking" })} {intl.formatMessage({ id: "Find booking" })}

View File

@@ -8,9 +8,15 @@ import TopMenu from "./TopMenu"
import styles from "./header.module.css" import styles from "./header.module.css"
export default async function Header() { export interface HeaderProps {
const languages = await serverClient().contentstack.languageSwitcher.get() languageSwitcher: React.ReactNode
const headerData = await serverClient().contentstack.base.header() }
export default async function Header({ languageSwitcher }: HeaderProps) {
const [languages, headerData] = await Promise.all([
serverClient().contentstack.languageSwitcher.get(),
serverClient().contentstack.base.header(),
])
if (!languages || !headerData) { if (!languages || !headerData) {
return null return null
@@ -19,7 +25,10 @@ export default async function Header() {
return ( return (
<header className={styles.header}> <header className={styles.header}>
<Suspense fallback={<LoadingSpinner />}> <Suspense fallback={<LoadingSpinner />}>
<TopMenu languageUrls={languages.urls} topLink={headerData.topLink} /> <TopMenu
languageSwitcher={languageSwitcher}
topLink={headerData.topLink}
/>
<MainMenu <MainMenu
languageUrls={languages.urls} languageUrls={languages.urls}
menuItems={headerData.menuItems} menuItems={headerData.menuItems}

View File

@@ -1,3 +1,5 @@
import { Suspense } from "react"
import { serverClient } from "@/lib/trpc/server" import { serverClient } from "@/lib/trpc/server"
import { ChevronRightIcon, HouseIcon } from "@/components/Icons" import { ChevronRightIcon, HouseIcon } from "@/components/Icons"
@@ -16,6 +18,7 @@ export default async function Breadcrumbs() {
return ( return (
<nav className={styles.breadcrumbs}> <nav className={styles.breadcrumbs}>
<ul className={styles.list}> <ul className={styles.list}>
<Suspense>
{homeBreadcrumb ? ( {homeBreadcrumb ? (
<li className={styles.listItem}> <li className={styles.listItem}>
<Link <Link
@@ -54,6 +57,7 @@ export default async function Breadcrumbs() {
</li> </li>
) )
})} })}
</Suspense>
</ul> </ul>
</nav> </nav>
) )

View File

@@ -1,4 +1,4 @@
import { Fragment } from "react" import { Fragment, Suspense } from "react"
import { logout } from "@/constants/routes/handleAuth" import { logout } from "@/constants/routes/handleAuth"
import { serverClient } from "@/lib/trpc/server" import { serverClient } from "@/lib/trpc/server"
@@ -14,14 +14,13 @@ import styles from "./sidebar.module.css"
export default async function SidebarMyPages() { export default async function SidebarMyPages() {
const navigation = await serverClient().contentstack.myPages.navigation.get() const navigation = await serverClient().contentstack.myPages.navigation.get()
const { formatMessage } = await getIntl() const { formatMessage } = await getIntl()
if (!navigation) {
return null
}
return ( return (
<aside className={styles.sidebar}> <aside className={styles.sidebar}>
<nav className={styles.nav}> <nav className={styles.nav}>
<Subtitle type="two">{navigation.title}</Subtitle> <Subtitle type="two">{navigation?.title}</Subtitle>
{navigation.menuItems.map((menuItem, idx) => ( <Suspense>
{navigation?.menuItems.map((menuItem, idx) => (
<Fragment key={`${menuItem.display_sign_out_link}-${idx}`}> <Fragment key={`${menuItem.display_sign_out_link}-${idx}`}>
<Divider color="beige" /> <Divider color="beige" />
<ul className={styles.list}> <ul className={styles.list}>
@@ -31,7 +30,10 @@ export default async function SidebarMyPages() {
color="burgundy" color="burgundy"
href={link.originalUrl || link.url} href={link.originalUrl || link.url}
partialMatch partialMatch
size={menuItem.display_sign_out_link ? "small" : "regular"} prefetch={true}
size={
menuItem.display_sign_out_link ? "small" : "regular"
}
variant="sidebar" variant="sidebar"
> >
{link.linkText} {link.linkText}
@@ -54,6 +56,7 @@ export default async function SidebarMyPages() {
</ul> </ul>
</Fragment> </Fragment>
))} ))}
</Suspense>
</nav> </nav>
</aside> </aside>
) )

View File

@@ -2,6 +2,6 @@ import type { Header } from "@/types/header"
import type { LanguageSwitcherData } from "@/types/requests/languageSwitcher" import type { LanguageSwitcherData } from "@/types/requests/languageSwitcher"
export interface TopMenuProps { export interface TopMenuProps {
languageUrls: LanguageSwitcherData languageSwitcher: React.ReactNode
topLink: Header["topLink"] topLink: Header["topLink"]
} }