import type { ReactNode } from "react"; import * as Sentry from "@sentry/react-router"; import Script from "next/script"; import { Links, Meta, Outlet, Scripts } from "react-router"; import type { LinksFunction } from "react-router"; import { ThemeProvider, useTheme } from "next-themes"; // plane imports import { SITE_DESCRIPTION, SITE_NAME } from "@plane/constants"; import { cn } from "@plane/utils"; // types // assets import favicon16 from "@/app/assets/favicon/favicon-16x16.png?url"; import favicon32 from "@/app/assets/favicon/favicon-32x32.png?url"; import faviconIco from "@/app/assets/favicon/favicon.ico?url"; import icon180 from "@/app/assets/icons/icon-180x180.png?url"; import icon512 from "@/app/assets/icons/icon-512x512.png?url"; import ogImage from "@/app/assets/og-image.png?url"; import globalStyles from "@/styles/globals.css?url"; import type { Route } from "./+types/root"; // components import { LogoSpinner } from "@/components/common/logo-spinner"; // local import { CustomErrorComponent } from "./error"; import { AppProvider } from "./provider"; // fonts import "@fontsource-variable/inter"; import interVariableWoff2 from "@fontsource-variable/inter/files/inter-latin-wght-normal.woff2?url"; import "@fontsource/material-symbols-rounded"; import "@fontsource/ibm-plex-mono"; const APP_TITLE = "Plane | Simple, extensible, open-source project management tool."; export const links: LinksFunction = () => [ { rel: "icon", type: "image/png", sizes: "32x32", href: favicon32 }, { rel: "icon", type: "image/png", sizes: "16x16", href: favicon16 }, { rel: "shortcut icon", href: faviconIco }, { rel: "manifest", href: "/site.webmanifest.json" }, { rel: "apple-touch-icon", href: icon512 }, { rel: "apple-touch-icon", sizes: "180x180", href: icon180 }, { rel: "apple-touch-icon", sizes: "512x512", href: icon512 }, { rel: "manifest", href: "/manifest.json" }, { rel: "stylesheet", href: globalStyles }, { rel: "preload", href: interVariableWoff2, as: "font", type: "font/woff2", crossOrigin: "anonymous", }, ]; export function Layout({ children }: { children: ReactNode }) { const isSessionRecorderEnabled = parseInt(process.env.VITE_ENABLE_SESSION_RECORDER || "0"); return ( {/* Meta info for PWA */}
{children} {!!isSessionRecorderEnabled && process.env.VITE_SESSION_RECORDER_KEY && ( )} ); } export const meta: Route.MetaFunction = () => [ { title: APP_TITLE }, { name: "description", content: SITE_DESCRIPTION }, { property: "og:title", content: APP_TITLE }, { property: "og:description", content: "Open-source project management tool to manage work items, cycles, and product roadmaps easily", }, { property: "og:url", content: "https://app.plane.so/" }, { property: "og:image", content: ogImage }, { property: "og:image:width", content: "1200" }, { property: "og:image:height", content: "630" }, { property: "og:image:alt", content: "Plane - Modern project management" }, { name: "keywords", content: "software development, plan, ship, software, accelerate, code management, release management, project management, work item tracking, agile, scrum, kanban, collaboration", }, { name: "twitter:site", content: "@planepowers" }, { name: "twitter:card", content: "summary_large_image" }, { name: "twitter:image", content: ogImage }, { name: "twitter:image:width", content: "1200" }, { name: "twitter:image:height", content: "630" }, { name: "twitter:image:alt", content: "Plane - Modern project management" }, ]; export default function Root() { return (
); } export function HydrateFallback() { const { resolvedTheme } = useTheme(); // if we are on the server or the theme is not resolved, return an empty div if (typeof window === "undefined" || resolvedTheme === undefined) return
; return (
); } export function ErrorBoundary({ error }: Route.ErrorBoundaryProps) { if (error) { Sentry.captureException(error); } return ; }