diff --git a/.storybook/theme-picker/ThemeProvider.tsx b/.storybook/theme-picker/ThemeProvider.tsx index e8dd4e2a..0b27911e 100644 --- a/.storybook/theme-picker/ThemeProvider.tsx +++ b/.storybook/theme-picker/ThemeProvider.tsx @@ -1,17 +1,20 @@ import { useAddonState } from '@storybook/client-api'; import { DecoratorFn } from '@storybook/react'; import React from 'react'; -import { ThemeProvider } from 'styled-components'; import themes from '../../src/common/themes/index'; +import { React95Provider } from '../../src/common/React95Provider'; +import { ThemeProvider } from '../../src/common/ThemeProvider'; import { THEMES_ID } from './constants'; export const withThemesProvider: DecoratorFn = story => { const [themeName] = useAddonState(THEMES_ID, 'original'); return ( - - {story()} - + + + {story()} + + ); }; diff --git a/src/IconWrapper/IconWrapper.stories.tsx b/src/IconWrapper/IconWrapper.stories.tsx new file mode 100644 index 00000000..22eda2ae --- /dev/null +++ b/src/IconWrapper/IconWrapper.stories.tsx @@ -0,0 +1,83 @@ +import { ComponentMeta } from '@storybook/react'; +import React from 'react'; +import { IconWrapper } from 'react95'; +import styled from 'styled-components'; + +const icons = [ + '', + '', + '', + '', + '', + '', + '' +]; + +const emojis = ['🎲', '🏠', '🐩', 'πŸ’Ύ', '🚾', 'πŸ“', 'πŸ’­']; + +const Wrapper = styled.div` + padding: 5rem; + background: ${({ theme }) => theme.material}; +`; + +const Row = styled.div` + display: flex; + align-items: center; + gap: 16px; + margin-top: 24px; + img { + width: 48px; + height: 48px; + image-rendering: pixelated; + } + span { + display: inline-block; + font-size: 48px; + line-height: 48px; + } +`; + +export default { + title: 'Other/IconWrapper', + component: IconWrapper, + decorators: [story => {story()}] +} as ComponentMeta; + +export function Default() { + return ( +
+ + {icons.map((icon, i) => ( + + + + ))} + + + {icons.map((icon, i) => ( + + + + ))} + + + {emojis.map((emoji, i) => ( + + {emoji} + + ))} + + + {emojis.map((emoji, i) => ( + + {emoji} + + ))} + +
+ ); +} + +Default.story = { + name: 'default' +}; diff --git a/src/IconWrapper/IconWrapper.tsx b/src/IconWrapper/IconWrapper.tsx new file mode 100644 index 00000000..a4ddc8b4 --- /dev/null +++ b/src/IconWrapper/IconWrapper.tsx @@ -0,0 +1,22 @@ +import React from 'react'; +import styled from 'styled-components'; +import { getDisabledFilterId } from '../common/ThemeProvider'; +import { CommonStyledProps } from '../types'; + +type IconWrapperProps = { + disabled?: boolean; +} & React.HTMLAttributes & + CommonStyledProps; + +const IconWrapper = styled.div` + ${({ theme, disabled }) => ` + display: grid; + filter: ${ + disabled ? `contrast(2) url(#${getDisabledFilterId(theme)})` : 'none' + }; +`} +`; + +IconWrapper.displayName = 'IconWrapper'; + +export { IconWrapper, IconWrapperProps }; diff --git a/src/common/React95Provider.tsx b/src/common/React95Provider.tsx new file mode 100644 index 00000000..7b52a6a7 --- /dev/null +++ b/src/common/React95Provider.tsx @@ -0,0 +1,35 @@ +import React from 'react'; + +type Props = { + children: React.ReactNode; +}; + +const UtilsElementContext = React.createContext(null); + +export function useUtilsElement() { + const utils = React.useContext(UtilsElementContext); + + return utils; +} + +export function React95Provider({ children }: Props) { + const [utils, setUtils] = React.useState(null); + + React.useLayoutEffect(() => { + const filtersWrapper = document.createElement('react95-utils'); + const div = document.createElement('div'); + filtersWrapper.appendChild(div); + document.documentElement.appendChild(filtersWrapper); + setUtils(div); + + return () => { + filtersWrapper.remove(); + }; + }, []); + + return ( + + {children} + + ); +} diff --git a/src/common/ThemeProvider.tsx b/src/common/ThemeProvider.tsx new file mode 100644 index 00000000..60d03dee --- /dev/null +++ b/src/common/ThemeProvider.tsx @@ -0,0 +1,84 @@ +import React from 'react'; +import { ThemeProvider as SCThemeProvider } from 'styled-components'; +import { Theme } from '../types'; +import { useUtilsElement } from './React95Provider'; + +type Props = React.ComponentProps & { + theme: Theme; +}; + +export function getDisabledFilterId(theme: Theme) { + return `disabled-filter--${theme.name}`; +} +export function ThemeProvider({ theme, ...otherProps }: Props) { + const utilsElement = useUtilsElement(); + + React.useLayoutEffect(() => { + if (utilsElement) { + const filterId = getDisabledFilterId(theme); + const filter = document.getElementById(filterId); + if (!filter) { + // const disabledFilter = ` + // + // + // + // + // + // + + // `; + const disabledFilter = ` + + + + + + + + + + + + + + + + + + `; + const svg = document.createElementNS( + 'http://www.w3.org/2000/svg', + 'svg' + ); + svg.innerHTML = disabledFilter; + utilsElement.appendChild(svg); + } + } + }, [theme, utilsElement]); + return ; +} diff --git a/src/index.ts b/src/index.ts index 4e6b18a0..3a4e947e 100644 --- a/src/index.ts +++ b/src/index.ts @@ -15,6 +15,7 @@ export * from './Frame/Frame'; export * from './GroupBox/GroupBox'; export * from './Handle/Handle'; export * from './Hourglass/Hourglass'; +export * from './IconWrapper/IconWrapper'; export * from './MenuList/MenuList'; export * from './Monitor/Monitor'; export * from './NumberInput/NumberInput'; @@ -45,3 +46,8 @@ export * from './legacy/Panel'; export * from './legacy/Progress'; export * from './legacy/TextField'; export * from './legacy/Tree'; + +/* other */ + +export * from './common/ThemeProvider'; +export * from './common/React95Provider';

AltStyle γ«γ‚ˆγ£γ¦ε€‰ζ›γ•γ‚ŒγŸγƒšγƒΌγ‚Έ (->γ‚ͺγƒͺγ‚ΈγƒŠγƒ«) /