@ucam/design-system
Version:
University of Cambridge Design System
94 lines (88 loc) • 16.6 kB
JavaScript
;
var tslib = require('tslib');
var React = require('react');
var core = require('@material-ui/core');
var themes = require('../themes/themes.js');
var useTheme = require('../useTheme/useTheme.js');
var useThemeRegister = require('../useThemeRegister/useThemeRegister.js');
var useLocalStorage = require('../../useLocalStorage/useLocalStorage.js');
var HydrationChecker = require('../../HydrationChecker/HydrationChecker.js');
var PropTypes = require('prop-types');
function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
var React__default = /*#__PURE__*/_interopDefaultLegacy(React);
var PropTypes__default = /*#__PURE__*/_interopDefaultLegacy(PropTypes);
/**
* A component that provides the app with theming support
*/
const ThemeProvider = React__default['default'].forwardRef(function ThemeProvider(props, ref) {
const { children, themes: themes$1 = {
Light: themes.light,
Dark: themes.dark
}, defaultThemeName = 'Light', darkThemeName = 'Dark', scope = 'global', disableLocalStorage: propDisableLocalStorage } = props, other = tslib.__rest(props, ["children", "themes", "defaultThemeName", "darkThemeName", "scope", "disableLocalStorage"]);
// Note: false during rehydration run, regardless of browser setting...
// TODO: if we switch to the useHydrated hook and use the standard browser media queries we can possibly avoid extra renders here
const prefersDarkMode = core.useMediaQuery('(prefers-color-scheme: dark)');
// Cannot update the disableLocalStorage value otherwise different hooks would be called on the next line
// React requires that the same hooks be called in the same order
const [localStorageEnabled] = React.useState(!propDisableLocalStorage);
const [chosenTheme, setChosenTheme] = (localStorageEnabled
? () => useLocalStorage('theme')
: () => React.useState(defaultThemeName))();
const [registeredThemes, setRegisteredThemes] = React.useState(new Map(Object.entries(themes$1)));
let themeName;
let theme;
if (chosenTheme !== undefined && registeredThemes.has(chosenTheme)) {
themeName = chosenTheme;
theme = registeredThemes.get(themeName);
}
else {
themeName = prefersDarkMode ? darkThemeName : defaultThemeName;
theme = registeredThemes.get(themeName) || themes.light;
}
const setTheme = React.useCallback((themeName) => {
console.assert(themeName === undefined || registeredThemes.has(themeName), `Unrecognised theme: ${themeName}`);
setChosenTheme(themeName);
}, [setChosenTheme, registeredThemes]);
const selectedTheme = React.useMemo(() => ({
requestedThemeName: chosenTheme,
renderedThemeName: themeName,
theme,
setTheme
}), [chosenTheme, themeName, setTheme]);
const setThemes = React.useCallback((themes) => {
setRegisteredThemes(new Map(Object.entries(themes)));
}, [setRegisteredThemes]);
const themeRegister = React.useMemo(() => [registeredThemes, setThemes], [registeredThemes, setThemes]);
const cssBaselineWrapper = (() => {
switch (scope) {
case 'global':
return React__default['default'].createElement(core.CssBaseline, Object.assign({}, other), children);
case 'local':
return (React__default['default'].createElement(core.ScopedCssBaseline, Object.assign({}, other, { ref: ref }), children));
default:
return React__default['default'].createElement(React__default['default'].Fragment, null, children);
}
})();
return (React__default['default'].createElement(useThemeRegister.ThemeRegisterContext.Provider, { value: themeRegister },
React__default['default'].createElement(useTheme.ThemeUpdateContext.Provider, { value: selectedTheme },
React__default['default'].createElement(core.ThemeProvider, { theme: theme }, cssBaselineWrapper))));
});
ThemeProvider.propTypes = {
children: PropTypes__default['default'].node,
themes: PropTypes__default['default'].objectOf(PropTypes__default['default'].object.isRequired),
defaultThemeName: PropTypes__default['default'].string,
darkThemeName: PropTypes__default['default'].string,
disableLocalStorage: PropTypes__default['default'].bool,
scope: PropTypes__default['default'].oneOf(['global', 'local', false])
};
/**
* A component that provides the app with theming support.
* Also providing a `<HydrationChecker/>`.
*/
const ThemeProviderWithHydrationChecker = React__default['default'].forwardRef((props, ref) => {
return (React__default['default'].createElement(HydrationChecker, null,
React__default['default'].createElement(ThemeProvider, Object.assign({}, props, { ref: ref }))));
});
ThemeProviderWithHydrationChecker.displayName = 'ThemeProvider';
module.exports = ThemeProviderWithHydrationChecker;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ThemeProvider.js","sources":["/@ucam/design-system/src/theme/ThemeProvider/ThemeProvider.tsx"],"sourcesContent":["import React, { useCallback, useMemo, useState } from 'react';\nimport {\n  InternalStandardProps as StandardProps,\n  CssBaseline,\n  ScopedCssBaseline,\n  useMediaQuery,\n  Theme,\n  ThemeProvider as MuiThemeProvider\n} from '@material-ui/core';\nimport { dark, light } from '../themes';\nimport { ThemeUpdate, ThemeUpdateContext } from '../useTheme/useTheme';\nimport { ThemeRegisterContext } from '../useThemeRegister/useThemeRegister';\nimport useLocalStorage from '../../useLocalStorage';\nimport HydrationChecker from '../../HydrationChecker';\nimport PropTypes, { Requireable } from 'prop-types';\n\nexport interface ThemeProviderProps extends StandardProps<React.HTMLAttributes<HTMLDivElement>> {\n  /**\n   * A collection of themes and their names\n   * default: {\n   *   \"Light\": light,\n   *   \"Dark\": dark\n   * }\n   */\n  themes?: {\n    [themeName: string]: Theme;\n  };\n  /**\n   * The name of the default (light) theme\n   * default: \"Light\"\n   */\n  defaultThemeName?: string;\n  /**\n   * The name of the dark theme\n   * default: \"Dark\"\n   */\n  darkThemeName?: string;\n  /**\n   * Store the current theme in local storage, to save it between browser reloads\n   * Note: Uses the same value for the lifetime of the component (cannot be updated)\n   * default: false\n   */\n  disableLocalStorage?: boolean;\n  /**\n   * Global scoped ThemeProviders apply global styles to the body element\n   * Local scoped ThemeProviders apply global styles to a wrapper div\n   * Unscoped ThemeProviders do not apply any global styles\n   * default: \"global\"\n   */\n  scope?: 'global' | 'local' | false;\n}\n\n/**\n * A component that provides the app with theming support\n */\nconst ThemeProvider = React.forwardRef<HTMLDivElement, ThemeProviderProps>(function ThemeProvider(\n  props,\n  ref\n) {\n  const {\n    children,\n    themes = {\n      Light: light,\n      Dark: dark\n    },\n    defaultThemeName = 'Light',\n    darkThemeName = 'Dark',\n    scope = 'global',\n    disableLocalStorage: propDisableLocalStorage,\n    ...other\n  } = props;\n\n  // Note: false during rehydration run, regardless of browser setting...\n  // TODO: if we switch to the useHydrated hook and use the standard browser media queries we can possibly avoid extra renders here\n  const prefersDarkMode = useMediaQuery('(prefers-color-scheme: dark)');\n\n  // Cannot update the disableLocalStorage value otherwise different hooks would be called on the next line\n  // React requires that the same hooks be called in the same order\n  const [localStorageEnabled] = useState(!propDisableLocalStorage);\n  const [chosenTheme, setChosenTheme] = (localStorageEnabled\n    ? () => useLocalStorage('theme')\n    : () => useState<string | undefined>(defaultThemeName))();\n\n  const [registeredThemes, setRegisteredThemes] = useState(new Map(Object.entries(themes)));\n\n  let themeName: string;\n  let theme: Theme;\n  if (chosenTheme !== undefined && registeredThemes.has(chosenTheme)) {\n    themeName = chosenTheme;\n    theme = registeredThemes.get(themeName) as Theme;\n  } else {\n    themeName = prefersDarkMode ? darkThemeName : defaultThemeName;\n    theme = registeredThemes.get(themeName) || light;\n  }\n\n  const setTheme = useCallback(\n    (themeName: string | undefined) => {\n      console.assert(\n        themeName === undefined || registeredThemes.has(themeName),\n        `Unrecognised theme: ${themeName}`\n      );\n      setChosenTheme(themeName);\n    },\n    [setChosenTheme, registeredThemes]\n  );\n\n  const selectedTheme = useMemo<ThemeUpdate>(\n    () => ({\n      requestedThemeName: chosenTheme,\n      renderedThemeName: themeName,\n      theme,\n      setTheme\n    }),\n    [chosenTheme, themeName, setTheme]\n  );\n\n  const setThemes = useCallback(\n    (themes: { [name: string]: Theme }) => {\n      setRegisteredThemes(new Map(Object.entries(themes)));\n    },\n    [setRegisteredThemes]\n  );\n\n  const themeRegister: [Map<string, Theme>, (themes: { [name: string]: Theme }) => void] = useMemo(\n    () => [registeredThemes, setThemes],\n    [registeredThemes, setThemes]\n  );\n\n  const cssBaselineWrapper = (() => {\n    switch (scope) {\n      case 'global':\n        return <CssBaseline {...other}>{children}</CssBaseline>;\n      case 'local':\n        return (\n          <ScopedCssBaseline {...other} ref={ref}>\n            {children}\n          </ScopedCssBaseline>\n        );\n      default:\n        return <>{children}</>;\n    }\n  })();\n\n  return (\n    <ThemeRegisterContext.Provider value={themeRegister}>\n      <ThemeUpdateContext.Provider value={selectedTheme}>\n        <MuiThemeProvider theme={theme}>{cssBaselineWrapper}</MuiThemeProvider>\n      </ThemeUpdateContext.Provider>\n    </ThemeRegisterContext.Provider>\n  );\n});\n\nThemeProvider.propTypes = {\n  children: PropTypes.node,\n  themes: PropTypes.objectOf((PropTypes.object as Requireable<Theme>).isRequired),\n  defaultThemeName: PropTypes.string,\n  darkThemeName: PropTypes.string,\n  disableLocalStorage: PropTypes.bool,\n  scope: PropTypes.oneOf(['global', 'local', false])\n};\n\n/**\n * A component that provides the app with theming support.\n * Also providing a `<HydrationChecker/>`.\n */\nconst ThemeProviderWithHydrationChecker = React.forwardRef<HTMLDivElement, ThemeProviderProps>(\n  (props, ref) => {\n    return (\n      <HydrationChecker>\n        <ThemeProvider {...props} ref={ref} />\n      </HydrationChecker>\n    );\n  }\n);\n\nThemeProviderWithHydrationChecker.displayName = 'ThemeProvider';\n\nexport default ThemeProviderWithHydrationChecker;\n"],"names":["React","themes","light","dark","useMediaQuery","useState","useCallback","useMemo","CssBaseline","ScopedCssBaseline","ThemeRegisterContext","ThemeUpdateContext","MuiThemeProvider","PropTypes"],"mappings":"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"}