Newer
Older
CrypticOreWallet / src / global / LocaleContext.tsx
@Drew Lemmy Drew Lemmy on 15 Mar 2021 4 KB feat: skeleton of name transfer modal
// Copyright (c) 2020-2021 Drew Lemmy
// This file is part of KristWeb 2 under GPL-3.0.
// Full details: https://github.com/tmpim/KristWeb2/blob/master/LICENSE.txt
import { FC, createContext, useEffect, useState } from "react";
import { ConfigProvider } from "antd";
import { Locale } from "antd/lib/locale-provider";

import { useTranslation } from "react-i18next";
import { getLanguages } from "@utils/i18n";

import dayjs from "dayjs";

import { Formatter } from "react-timeago";
import buildFormatter from "react-timeago/lib/formatters/buildFormatter";

import Debug from "debug";
const debug = Debug("kristweb:locale-context");

export const TimeagoFormatterContext = createContext<Formatter | undefined>(undefined);

export const LocaleContext: FC = ({ children }): JSX.Element => {
  const { i18n } = useTranslation();
  const langCode = i18n.language;
  const languages = getLanguages();
  const lang = languages?.[langCode];

  // These are wrapped in objects due to some bizarre issues where React was
  // attempting to call the timeagoFormatter at some point??
  const [timeagoFormatter, setTimeagoFormatter] = useState<{ formatter: Formatter }>();
  const [antLocale, setAntLocale] = useState<{ locale: Locale }>();

  // Load the day.js locale if available
  useEffect(() => {
    // See if the language has a dayjs locale set. If not, revert to `en`
    const dayjsLocale = lang?.dayjsLocale;
    if (!dayjsLocale) {
      debug("language %s doesn't have a dayjs locale, reverting to `en`", langCode);
      dayjs.locale("en");
      return;
    }

    // Attempt to import the locale asynchronously. This will usually incur a
    // network request, but it should be cached by the service worker.
    debug("loading dayjs locale %s for language %s", dayjsLocale, langCode);
    // Including only `.js` files here ensures that it doesn't attempt to load
    // the TypeScript typings, which causes build warnings due to a missing
    // loader for those files.
    import(
      /* webpackInclude: /\.js$/ */
      /* webpackMode: "lazy" */
      /* webpackChunkName: "locale-dayjs-[request]" */
      `dayjs/locale/${dayjsLocale}`
    )
      .then(() => {
        debug("got dayjs locale %s", dayjsLocale);
        dayjs.locale(dayjsLocale);
      })
      .catch(console.error);
  }, [lang, langCode, languages]);

  // Load the timeago locale if available
  useEffect(() => {
    // See if the language has a timeago locale set. If not, revert to default
    const timeagoLocale = lang?.timeagoLocale;
    if (!timeagoLocale) {
      debug("language %s doesn't have a timeago locale, reverting to default", langCode);
      setTimeagoFormatter(undefined);
      return;
    }

    // Load the locale
    debug("loading timeago locale %s for language %s", timeagoLocale, langCode);
    import(
      /* webpackInclude: /\.js$/ */
      /* webpackMode: "lazy" */
      /* webpackChunkName: "locale-timeago-[request]" */
      `react-timeago/lib/language-strings/${timeagoLocale}`
    )
      .then(strings => {
        debug("got timeago locale %s", timeagoLocale);
        setTimeagoFormatter({ formatter: buildFormatter(strings.default) });
      })
      .catch(console.error);
  }, [lang, langCode, languages]);

  // Load the antd locale if available
  useEffect(() => {
    // See if the language has an antd locale set. If not, revert to default
    const antLocaleCode = lang?.antLocale;
    if (!antLocaleCode) {
      debug("language %s doesn't have an antd locale, reverting to default", langCode);
      setAntLocale(undefined);
      return;
    }

    // Load the locale
    debug("loading antd locale %s for language %s", antLocaleCode, langCode);
    import(
      /* webpackInclude: /\.js$/ */
      /* webpackMode: "lazy" */
      /* webpackChunkName: "locale-antd-[request]" */
      `antd/lib/locale/${antLocaleCode}`
    )
      .then(locale => {
        debug("got antd locale %s", antLocaleCode);
        setAntLocale({ locale: locale.default });
      })
      .catch(console.error);
  }, [lang, langCode, languages]);

  return <TimeagoFormatterContext.Provider value={timeagoFormatter?.formatter}>
    <ConfigProvider locale={antLocale?.locale}>
      {children}
    </ConfigProvider>
  </TimeagoFormatterContext.Provider>;
};