import type { GetStaticProps } from 'next'; import Head from 'next/head'; import { useRouter } from 'next/router'; import { useCallback, type ReactNode } from 'react'; import { useIntl } from 'react-intl'; import { getLayout, Heading, Link, LinksWidget, Page, PageBody, PageHeader, PageSidebar, SearchForm, Spinner, type SearchFormSubmit, } from '../components'; import { convertWPThematicPreviewToPageLink, convertWPTopicPreviewToPageLink, fetchThematicsCount, fetchThematicsList, fetchTopicsCount, fetchTopicsList, } from '../services/graphql'; import styles from '../styles/pages/blog.module.scss'; import type { GraphQLConnection, NextPageWithLayout, WPThematicPreview, WPTopicPreview, } from '../types'; import { CONFIG } from '../utils/config'; import { ROUTES } from '../utils/constants'; import { getLinksItemData, getSchemaFrom, getWebPageGraph, } from '../utils/helpers'; import { loadTranslation, type Messages } from '../utils/helpers/server'; import { useBreadcrumbs, useThematicsList, useTopicsList, } from '../utils/hooks'; const link = (chunks: ReactNode) => {chunks}; type Error404PageProps = { data: { thematics: GraphQLConnection; topics: GraphQLConnection; }; translation: Messages; }; /** * Error 404 page. */ const Error404Page: NextPageWithLayout = ({ data }) => { const router = useRouter(); const intl = useIntl(); const { isLoading: areThematicsLoading, thematics } = useThematicsList({ fallback: data.thematics, input: { first: data.thematics.pageInfo.total }, }); const { isLoading: areTopicsLoading, topics } = useTopicsList({ fallback: data.topics, input: { first: data.topics.pageInfo.total }, }); const messages = { loading: { thematicsList: intl.formatMessage({ defaultMessage: 'Thematics are loading...', description: 'Error404Page: loading thematics message', id: '6IAJYx', }), topicsList: intl.formatMessage({ defaultMessage: 'Topics are loading...', description: 'Error404Page: loading topics message', id: 'HnMf0i', }), }, page: { title: intl.formatMessage({ defaultMessage: 'Page not found', description: 'Error404Page: page title', id: 'KnWeKh', }), }, seo: { title: intl.formatMessage( { defaultMessage: 'Error 404: Page not found - {websiteName}', description: 'Error404Page: SEO - Page title', id: 'pNIIU1', }, { websiteName: CONFIG.name } ), metaDesc: intl.formatMessage({ defaultMessage: 'Page not found.', description: 'Error404Page: SEO - Meta description', id: 'yKoGqg', }), }, widgets: { thematicsListTitle: intl.formatMessage({ defaultMessage: 'Thematics', description: 'Error404Page: thematics list widget title', id: 'HohQPh', }), topicsListTitle: intl.formatMessage({ defaultMessage: 'Topics', description: 'Error404Page: topics list widget title', id: 'GVpTIl', }), }, }; const { items: breadcrumbItems, schema: breadcrumbSchema } = useBreadcrumbs( messages.page.title ); const jsonLd = getSchemaFrom([ getWebPageGraph({ breadcrumb: breadcrumbSchema, description: messages.seo.metaDesc, slug: ROUTES.NOT_FOUND, title: messages.page.title, }), ]); const searchSubmitHandler: SearchFormSubmit = useCallback( async ({ query }) => { if (!query) return { messages: { error: intl.formatMessage({ defaultMessage: 'Query must be longer than one character.', description: 'Error404Page: invalid query message', id: '3u29G5', }), }, validator: (value) => value.query.length > 1, }; await router.push({ pathname: ROUTES.SEARCH, query: { s: query } }); return undefined; }, [intl, router] ); return ( {messages.seo.title} {/*eslint-disable-next-line react/jsx-no-literals -- Name allowed */}