import ApplicationLogo from '@/Components/ApplicationLogo';
import FlashMessages from '@/Components/FlashMessages';
import Dropdown from '@/Components/Dropdown';
import NavLink from '@/Components/NavLink';
import ResponsiveNavLink from '@/Components/ResponsiveNavLink';
import { useIsMobileNavBreakpoint } from '@/hooks/useMobileNavBreakpoint';
import {
    isAdminAttendanceLogsRoute,
    isAdminCleaningScheduleRoute,
    isAdminSettingsLocationsRoute,
    isAdminStaffRoute,
    isStaffTimeCardRoute,
} from '@/lib/navigation';
import type { PageProps } from '@/types';
import {
    Dialog,
    DialogPanel,
    Transition,
    TransitionChild,
} from '@headlessui/react';
import { Link, usePage } from '@inertiajs/react';
import {
    PropsWithChildren,
    ReactNode,
    useEffect,
    useState,
} from 'react';

export default function Authenticated({
    header,
    children,
}: PropsWithChildren<{ header?: ReactNode }>) {
    const user = usePage<PageProps>().props.auth.user;

    const [showingNavigationDropdown, setShowingNavigationDropdown] =
        useState(false);

    const isMobileNavBreakpoint = useIsMobileNavBreakpoint();

    useEffect(() => {
        if (!isMobileNavBreakpoint) {
            setShowingNavigationDropdown(false);
        }
    }, [isMobileNavBreakpoint]);

    const homeHref =
        user?.role === 'viewer'
            ? route('viewer.home')
            : route('dashboard');

    if (!user) {
        return (
            <div className="flex min-h-screen flex-col items-center justify-center bg-slate-50 px-4">
                <p className="text-center text-base text-slate-600">
                    認証情報を読み込めませんでした。ログインし直してください。
                </p>
                <Link
                    href={route('login')}
                    className="mt-4 min-h-12 text-base font-medium text-indigo-600 underline"
                >
                    ログインへ
                </Link>
            </div>
        );
    }

    const cleaningActive = isAdminCleaningScheduleRoute();
    const settingsLocationsActive = isAdminSettingsLocationsRoute();
    const staffAdminActive = isAdminStaffRoute();
    const attendanceLogsActive = isAdminAttendanceLogsRoute();
    const timeCardActive = isStaffTimeCardRoute();

    return (
        <div className="min-h-screen bg-slate-100">
            <nav className="border-b border-slate-200 bg-white shadow-sm">
                <div className="mx-auto max-w-7xl px-4 sm:px-6 lg:px-8">
                    <div className="flex h-14 items-center justify-between gap-2 sm:h-16">
                        <div className="flex min-w-0 flex-1 items-center gap-2">
                            <div className="flex shrink-0 items-center">
                                <Link
                                    href={homeHref}
                                    className="rounded-md p-1 focus:outline-none focus:ring-2 focus:ring-indigo-500"
                                    aria-label="ホームへ"
                                >
                                    <ApplicationLogo
                                        alt=""
                                        className="block h-7 w-auto max-w-[9rem] object-left sm:h-8 sm:max-w-[10rem]"
                                    />
                                </Link>
                            </div>

                            <div className="hidden min-w-0 flex-1 items-center gap-1 sm:flex sm:gap-2 lg:gap-4">
                                <NavLink
                                    href={route('dashboard')}
                                    active={route().current('dashboard')}
                                >
                                    ホーム
                                </NavLink>
                                <NavLink
                                    href={route('viewer.home')}
                                    active={route().current('viewer.home')}
                                >
                                    清掃予定
                                </NavLink>
                                {user.role === 'staff' && (
                                    <NavLink
                                        href={route('staff.time-card.show')}
                                        active={timeCardActive}
                                    >
                                        タイムカード
                                    </NavLink>
                                )}
                                {user.role === 'admin' && (
                                    <>
                                        <NavLink
                                            href={route(
                                                'admin.cleaning-schedules.index',
                                            )}
                                            active={cleaningActive}
                                        >
                                            予定管理
                                        </NavLink>
                                        <NavLink
                                            href={route(
                                                'admin.settings.locations',
                                            )}
                                            active={settingsLocationsActive}
                                        >
                                            建物・部屋
                                        </NavLink>
                                        <NavLink
                                            href={route('admin.staff.index')}
                                            active={staffAdminActive}
                                        >
                                            スタッフ
                                        </NavLink>
                                        <NavLink
                                            href={route(
                                                'admin.attendance-logs.index',
                                            )}
                                            active={attendanceLogsActive}
                                        >
                                            打刻ログ
                                        </NavLink>
                                    </>
                                )}
                            </div>
                        </div>

                        <div className="hidden sm:flex sm:shrink-0 sm:items-center">
                            <Dropdown>
                                <Dropdown.Trigger>
                                    <span className="inline-flex rounded-md">
                                        <button
                                            type="button"
                                            className="inline-flex max-w-[10rem] items-center rounded-md border border-slate-200 bg-white px-3 py-2 text-sm font-medium text-slate-600 hover:bg-slate-50 focus:outline-none focus:ring-2 focus:ring-indigo-500"
                                        >
                                            <span className="truncate">
                                                {user.name}
                                            </span>
                                            <svg
                                                className="-me-0.5 ms-1 h-4 w-4 shrink-0 text-slate-400"
                                                xmlns="http://www.w3.org/2000/svg"
                                                viewBox="0 0 20 20"
                                                fill="currentColor"
                                                aria-hidden
                                            >
                                                <path
                                                    fillRule="evenodd"
                                                    d="M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a1 1 0 01-1.414 0l-4-4a1 1 0 010-1.414z"
                                                    clipRule="evenodd"
                                                />
                                            </svg>
                                        </button>
                                    </span>
                                </Dropdown.Trigger>

                                <Dropdown.Content>
                                    <Dropdown.Link href={route('profile.edit')}>
                                        アカウント設定
                                    </Dropdown.Link>
                                    <Dropdown.Link
                                        href={route('logout')}
                                        method="post"
                                        as="button"
                                    >
                                        ログアウト
                                    </Dropdown.Link>
                                </Dropdown.Content>
                            </Dropdown>
                        </div>

                        <div className="flex items-center sm:hidden">
                            <button
                                onClick={() =>
                                    setShowingNavigationDropdown(
                                        (p) => !p,
                                    )
                                }
                                className="inline-flex min-h-11 min-w-11 items-center justify-center rounded-md p-2 text-slate-500 hover:bg-slate-100 focus:outline-none focus:ring-2 focus:ring-indigo-500"
                                type="button"
                                aria-expanded={showingNavigationDropdown}
                                aria-label="メニュー"
                            >
                                <svg
                                    className="h-6 w-6"
                                    stroke="currentColor"
                                    fill="none"
                                    viewBox="0 0 24 24"
                                    aria-hidden
                                >
                                    <path
                                        className={
                                            !showingNavigationDropdown
                                                ? 'inline-flex'
                                                : 'hidden'
                                        }
                                        strokeLinecap="round"
                                        strokeLinejoin="round"
                                        strokeWidth="2"
                                        d="M4 6h16M4 12h16M4 18h16"
                                    />
                                    <path
                                        className={
                                            showingNavigationDropdown
                                                ? 'inline-flex'
                                                : 'hidden'
                                        }
                                        strokeLinecap="round"
                                        strokeLinejoin="round"
                                        strokeWidth="2"
                                        d="M6 18L18 6M6 6l12 12"
                                    />
                                </svg>
                            </button>
                        </div>
                    </div>
                </div>
            </nav>

            {isMobileNavBreakpoint && (
                <div className="sm:hidden">
                    <Transition
                        show={showingNavigationDropdown}
                        leave="duration-200"
                    >
                        <Dialog
                            as="div"
                            className="relative z-50"
                            onClose={() =>
                                setShowingNavigationDropdown(false)
                            }
                        >
                            <TransitionChild
                                enter="ease-out duration-200"
                                enterFrom="opacity-0"
                                enterTo="opacity-100"
                                leave="ease-in duration-150"
                                leaveFrom="opacity-100"
                                leaveTo="opacity-0"
                            >
                                <div
                                    className="fixed inset-0 z-40 bg-slate-900/40 touch-manipulation"
                                    aria-hidden
                                />
                            </TransitionChild>

                            <TransitionChild
                                enter="transform transition ease-out duration-300"
                                enterFrom="translate-x-full"
                                enterTo="translate-x-0"
                                leave="transform transition ease-in duration-200"
                                leaveFrom="translate-x-0"
                                leaveTo="translate-x-full"
                            >
                                <DialogPanel className="fixed inset-y-0 end-0 z-50 flex h-full w-[min(100dvw,20rem)] max-w-full flex-col bg-white shadow-xl">
                                <div className="flex shrink-0 items-center justify-between gap-3 border-b border-slate-200 px-4 py-3">
                                    <p className="text-base font-semibold text-slate-900">
                                        メニュー
                                    </p>
                                    <button
                                        type="button"
                                        className="inline-flex min-h-11 min-w-11 items-center justify-center rounded-md text-slate-500 hover:bg-slate-100 focus:outline-none focus:ring-2 focus:ring-indigo-500"
                                        onClick={() =>
                                            setShowingNavigationDropdown(false)
                                        }
                                        aria-label="メニューを閉じる"
                                    >
                                        <svg
                                            className="h-6 w-6"
                                            fill="none"
                                            stroke="currentColor"
                                            viewBox="0 0 24 24"
                                            aria-hidden
                                        >
                                            <path
                                                strokeLinecap="round"
                                                strokeLinejoin="round"
                                                strokeWidth="2"
                                                d="M6 18L18 6M6 6l12 12"
                                            />
                                        </svg>
                                    </button>
                                </div>

                                <div className="min-h-0 flex-1 overflow-y-auto overscroll-contain">
                                    <div className="space-y-1 px-2 py-3">
                                        <ResponsiveNavLink
                                            href={route('dashboard')}
                                            active={route().current(
                                                'dashboard',
                                            )}
                                            onClick={() =>
                                                setShowingNavigationDropdown(
                                                    false,
                                                )
                                            }
                                        >
                                            ホーム
                                        </ResponsiveNavLink>
                                        <ResponsiveNavLink
                                            href={route('viewer.home')}
                                            active={route().current(
                                                'viewer.home',
                                            )}
                                            onClick={() =>
                                                setShowingNavigationDropdown(
                                                    false,
                                                )
                                            }
                                        >
                                            清掃予定
                                        </ResponsiveNavLink>
                                        {user.role === 'staff' && (
                                            <ResponsiveNavLink
                                                href={route(
                                                    'staff.time-card.show',
                                                )}
                                                active={timeCardActive}
                                                onClick={() =>
                                                    setShowingNavigationDropdown(
                                                        false,
                                                    )
                                                }
                                            >
                                                タイムカード
                                            </ResponsiveNavLink>
                                        )}
                                        {user.role === 'admin' && (
                                            <>
                                                <ResponsiveNavLink
                                                    href={route(
                                                        'admin.cleaning-schedules.index',
                                                    )}
                                                    active={cleaningActive}
                                                    onClick={() =>
                                                        setShowingNavigationDropdown(
                                                            false,
                                                        )
                                                    }
                                                >
                                                    予定管理
                                                </ResponsiveNavLink>
                                                <ResponsiveNavLink
                                                    href={route(
                                                        'admin.settings.locations',
                                                    )}
                                                    active={
                                                        settingsLocationsActive
                                                    }
                                                    onClick={() =>
                                                        setShowingNavigationDropdown(
                                                            false,
                                                        )
                                                    }
                                                >
                                                    建物・部屋
                                                </ResponsiveNavLink>
                                                <ResponsiveNavLink
                                                    href={route(
                                                        'admin.staff.index',
                                                    )}
                                                    active={staffAdminActive}
                                                    onClick={() =>
                                                        setShowingNavigationDropdown(
                                                            false,
                                                        )
                                                    }
                                                >
                                                    スタッフ
                                                </ResponsiveNavLink>
                                                <ResponsiveNavLink
                                                    href={route(
                                                        'admin.attendance-logs.index',
                                                    )}
                                                    active={attendanceLogsActive}
                                                    onClick={() =>
                                                        setShowingNavigationDropdown(
                                                            false,
                                                        )
                                                    }
                                                >
                                                    打刻ログ
                                                </ResponsiveNavLink>
                                            </>
                                        )}
                                    </div>

                                    <div className="border-t border-slate-200 px-4 pb-4 pt-4">
                                        <p className="text-sm font-medium text-slate-800">
                                            {user.name}
                                        </p>
                                        <p className="truncate text-xs text-slate-500">
                                            {user.email}
                                        </p>
                                        <div className="mt-2 space-y-1">
                                            <ResponsiveNavLink
                                                href={route('profile.edit')}
                                                onClick={() =>
                                                    setShowingNavigationDropdown(
                                                        false,
                                                    )
                                                }
                                            >
                                                アカウント設定
                                            </ResponsiveNavLink>
                                            <ResponsiveNavLink
                                                method="post"
                                                href={route('logout')}
                                                as="button"
                                                onClick={() =>
                                                    setShowingNavigationDropdown(
                                                        false,
                                                    )
                                                }
                                            >
                                                ログアウト
                                            </ResponsiveNavLink>
                                        </div>
                                    </div>
                                </div>
                            </DialogPanel>
                        </TransitionChild>
                        </Dialog>
                    </Transition>
                </div>
            )}

            {header && (
                <header className="border-b border-slate-200 bg-white">
                    <div className="mx-auto max-w-7xl px-4 py-4 sm:px-6 sm:py-5">
                        {header}
                    </div>
                </header>
            )}

            <main>
                <FlashMessages />
                {children}
            </main>
        </div>
    );
}
