feat(SW-2863): Move contentstack router to trpc package * Add exports to packages and lint rule to prevent relative imports * Add env to trpc package * Add eslint to trpc package * Apply lint rules * Use direct imports from trpc package * Add lint-staged config to trpc * Move lang enum to common * Restructure trpc package folder structure * WIP first step * update internal imports in trpc * Fix most errors in scandic-web Just 100 left... * Move Props type out of trpc * Fix CategorizedFilters types * Move more schemas in hotel router * Fix deps * fix getNonContentstackUrls * Fix import error * Fix entry error handling * Fix generateMetadata metrics * Fix alertType enum * Fix duplicated types * lint:fix * Merge branch 'master' into feat/sw-2863-move-contentstack-router-to-trpc-package * Fix broken imports * Merge branch 'master' into feat/sw-2863-move-contentstack-router-to-trpc-package Approved-by: Linus Flood
142 lines
4.5 KiB
TypeScript
142 lines
4.5 KiB
TypeScript
"use client"
|
|
import { useEffect, useRef, useState } from "react"
|
|
import { type DateRange, DayPicker } from "react-day-picker"
|
|
import { useIntl } from "react-intl"
|
|
|
|
import { Lang } from "@scandic-hotels/common/constants/language"
|
|
import { dt } from "@scandic-hotels/common/dt"
|
|
import { MaterialIcon } from "@scandic-hotels/design-system/Icons/MaterialIcon"
|
|
import { Typography } from "@scandic-hotels/design-system/Typography"
|
|
|
|
import Button from "@/components/TempDesignSystem/Button"
|
|
import Body from "@/components/TempDesignSystem/Text/Body"
|
|
import useLang from "@/hooks/useLang"
|
|
|
|
import { locales } from "../locales"
|
|
|
|
import styles from "./mobile.module.css"
|
|
import classNames from "react-day-picker/style.module.css"
|
|
|
|
import type { DatePickerRangeProps } from "@/types/components/datepicker"
|
|
|
|
export default function DatePickerRangeMobile({
|
|
close,
|
|
handleOnSelect,
|
|
selectedRange,
|
|
}: DatePickerRangeProps) {
|
|
const lang = useLang()
|
|
const intl = useIntl()
|
|
|
|
/** English is default language and doesn't need to be imported */
|
|
const locale = lang === Lang.en ? undefined : locales[lang]
|
|
const monthsRef = useRef<HTMLDivElement | null>(null)
|
|
const [autoScrollEnabled, setAutoScrollEnabled] = useState(true)
|
|
|
|
const currentDate = dt().toDate()
|
|
const startOfCurrentMonth = dt(currentDate).set("date", 1).toDate()
|
|
const yesterday = dt(currentDate).subtract(1, "day").toDate()
|
|
useEffect(() => {
|
|
if (!monthsRef.current || !selectedRange?.from || !autoScrollEnabled) return
|
|
|
|
const selectedDay = monthsRef.current.querySelector(
|
|
'td[aria-selected="true"]:not([data-outside="true"])'
|
|
)
|
|
|
|
const targetMonth = selectedDay?.closest(`.${styles.month}`)
|
|
|
|
if (targetMonth) {
|
|
targetMonth.scrollIntoView({ block: "start" })
|
|
}
|
|
}, [selectedRange, autoScrollEnabled])
|
|
|
|
function handleSelectWrapper(
|
|
dateRange: DateRange | undefined,
|
|
selectedDay: Date
|
|
) {
|
|
setAutoScrollEnabled(false)
|
|
handleOnSelect(dateRange, selectedDay)
|
|
}
|
|
|
|
// Max future date allowed to book kept same as of existing prod.
|
|
const endDate = dt().add(395, "day").toDate()
|
|
const endOfLastMonth = dt(endDate).endOf("month").toDate()
|
|
return (
|
|
<div className={styles.container} ref={monthsRef}>
|
|
<header className={styles.header}>
|
|
<button className={styles.close} onClick={close} type="button">
|
|
<MaterialIcon icon="close" />
|
|
</button>
|
|
</header>
|
|
<DayPicker
|
|
classNames={{
|
|
...classNames,
|
|
caption_label: `${classNames.caption_label} ${styles.captionLabel}`,
|
|
day: `${classNames.day} ${styles.day}`,
|
|
day_button: `${classNames.day_button} ${styles.dayButton}`,
|
|
month: styles.month,
|
|
month_caption: `${classNames.month_caption} ${styles.monthCaption}`,
|
|
months: styles.months,
|
|
range_end: styles.rangeEnd,
|
|
range_middle: styles.rangeMiddle,
|
|
range_start: styles.rangeStart,
|
|
root: `${classNames.root} ${styles.root}`,
|
|
week: styles.week,
|
|
weekday: `${classNames.weekday} ${styles.weekDay}`,
|
|
}}
|
|
disabled={[
|
|
{ from: startOfCurrentMonth, to: yesterday },
|
|
{ from: endDate, to: endOfLastMonth },
|
|
]}
|
|
endMonth={endDate}
|
|
excludeDisabled
|
|
formatters={{
|
|
formatWeekdayName(weekday) {
|
|
return dt(weekday).locale(lang).format("ddd")
|
|
},
|
|
}}
|
|
hideNavigation
|
|
lang={lang}
|
|
locale={locale}
|
|
mode="range"
|
|
/** Showing full year or what's left of it */
|
|
numberOfMonths={13}
|
|
onSelect={(dateRange, selectedDay) =>
|
|
handleSelectWrapper(dateRange, selectedDay)
|
|
}
|
|
required
|
|
selected={selectedRange}
|
|
startMonth={currentDate}
|
|
weekStartsOn={1}
|
|
components={{
|
|
MonthCaption(props) {
|
|
return (
|
|
<div className={props.className}>
|
|
<Typography variant="Title/Subtitle/md">
|
|
<h3>{props.children}</h3>
|
|
</Typography>
|
|
</div>
|
|
)
|
|
},
|
|
}}
|
|
/>
|
|
<footer className={styles.footer}>
|
|
<Button
|
|
className={styles.button}
|
|
intent="tertiary"
|
|
onPress={close}
|
|
size="large"
|
|
theme="base"
|
|
>
|
|
<Body color="white" textTransform="bold" asChild>
|
|
<span>
|
|
{intl.formatMessage({
|
|
defaultMessage: "Select dates",
|
|
})}
|
|
</span>
|
|
</Body>
|
|
</Button>
|
|
</footer>
|
|
</div>
|
|
)
|
|
}
|