diff --git a/UNRELEASED.md b/UNRELEASED.md index 7ef5eead645..681ce69e132 100644 --- a/UNRELEASED.md +++ b/UNRELEASED.md @@ -27,6 +27,8 @@ Use [the changelog guidelines](https://git.io/polaris-changelog-guidelines) to f ### Code quality +- Migrated `Frame` to use hooks instead of `withAppProvider` ([#2096](https://github.com/Shopify/polaris-react/pull/2096)) + ### Deprecations ### Development workflow diff --git a/src/components/Frame/Frame.tsx b/src/components/Frame/Frame.tsx index 6df160ce43a..6f6bb7f41f9 100644 --- a/src/components/Frame/Frame.tsx +++ b/src/components/Frame/Frame.tsx @@ -1,18 +1,17 @@ -import React, {createRef} from 'react'; +import React, {useState, useRef, useEffect, useCallback, useMemo} from 'react'; import {MobileCancelMajorMonotone} from '@shopify/polaris-icons'; import {durationSlow} from '@shopify/polaris-tokens'; import {CSSTransition} from '@material-ui/react-transition-group'; import {classNames} from '../../utilities/css'; import {Icon} from '../Icon'; -import {EventListener} from '../EventListener'; -import { - withAppProvider, - WithAppProviderProps, -} from '../../utilities/with-app-provider'; import {Backdrop} from '../Backdrop'; import {TrapFocus} from '../TrapFocus'; import {dataPolarisTopBar, layer} from '../shared'; +import {useForcibleToggle} from '../../utilities/use-toggle'; import {setRootProperty} from '../../utilities/set-root-property'; +import {useI18n} from '../../utilities/i18n'; +import {useMediaQuery} from '../../utilities/media-query'; +import {EventListener} from '../EventListener'; import { FrameContext, ContextualSaveBarProps, @@ -48,14 +47,6 @@ export interface FrameProps { onNavigationDismiss?(): void; } -interface State { - skipFocused?: boolean; - globalRibbonHeight: number; - loadingStack: number; - toastMessages: (ToastPropsWithID)[]; - showContextualSaveBar: boolean; -} - export const GLOBAL_RIBBON_CUSTOM_PROPERTY = '--global-ribbon-height'; export const APP_FRAME_MAIN = 'AppFrameMain'; @@ -66,340 +57,299 @@ const APP_FRAME_NAV = 'AppFrameNav'; const APP_FRAME_TOP_BAR = 'AppFrameTopBar'; const APP_FRAME_LOADING_BAR = 'AppFrameLoadingBar'; -type CombinedProps = FrameProps & WithAppProviderProps; - -class Frame extends React.PureComponent { - state: State = { - skipFocused: false, - globalRibbonHeight: 0, - loadingStack: 0, - toastMessages: [], - showContextualSaveBar: false, - }; - - private contextualSaveBar: ContextualSaveBarProps | null; - private globalRibbonContainer: HTMLDivElement | null = null; - private navigationNode = createRef(); - private skipToMainContentTargetNode = - this.props.skipToContentTarget || React.createRef(); - - componentDidMount() { - this.handleResize(); - if (this.props.globalRibbon) { - return; +export function Frame({ + children, + navigation, + topBar, + globalRibbon, + onNavigationDismiss, + skipToContentTarget, + showMobileNavigation = false, +}: FrameProps) { + const i18n = useI18n(); + const {isNavigationCollapsed} = useMediaQuery(); + + const globalRibbonContainer = useRef(null); + const navigationNode = useRef(null); + const skipToMainContentTargetNode = useRef(null); + + const [, setGlobalRibbonHeight] = useState(0); + const [loadingStack, setLoadingStack] = useState(0); + const [toastMessages, setToastMessages] = useState([]); + const [ + skipFocused, + {forceTrue: forceTrueSkipFocused, forceFalse: forceFalseSkipFocused}, + ] = useForcibleToggle(false); + const [ + showContextualSaveBar, + { + forceTrue: forceTrueShowContextualSaveBar, + forceFalse: forceFalseShowContextualSaveBar, + }, + ] = useForcibleToggle(false); + const [ + contextualSaveBarProps, + setContextualSaveBarProps, + ] = useState(null); + + const findNavigationNode = useCallback(() => navigationNode.current, []); + + const handleNavigationDismiss = useCallback(() => { + if (onNavigationDismiss != null) { + onNavigationDismiss(); } - this.setGlobalRibbonRootProperty(); - } - - componentDidUpdate(prevProps: FrameProps) { - if (this.props.globalRibbon !== prevProps.globalRibbon) { - this.setGlobalRibbonHeight(); + }, [onNavigationDismiss]); + + const handleNavKeydown = useCallback( + (event: React.KeyboardEvent) => { + const {key} = event; + + if (key === 'Escape') { + handleNavigationDismiss(); + } + }, + [handleNavigationDismiss], + ); + + const handleClick = useCallback(() => { + if (skipToContentTarget && skipToContentTarget.current) { + skipToContentTarget.current.focus(); + } else if (skipToMainContentTargetNode.current) { + skipToMainContentTargetNode.current.focus(); } - } - - render() { - const { - skipFocused, - loadingStack, - toastMessages, - showContextualSaveBar, - } = this.state; - const { - children, - navigation, - topBar, - globalRibbon, - showMobileNavigation = false, - skipToContentTarget, - polaris: { - intl, - mediaQuery: {isNavigationCollapsed}, - }, - } = this.props; - - const navClassName = classNames( - styles.Navigation, - showMobileNavigation && styles['Navigation-visible'], - ); + }, [skipToContentTarget]); - const mobileNavHidden = isNavigationCollapsed && !showMobileNavigation; - const mobileNavShowing = isNavigationCollapsed && showMobileNavigation; - const tabIndex = mobileNavShowing ? 0 : -1; - - const navigationMarkup = navigation ? ( - - - - - - ) : null; + const showToast = useCallback((toast: ToastPropsWithID) => { + setToastMessages((toastMessages) => { + const hasToastById = Boolean( + toastMessages.find(({id}) => id === toast.id), + ); - const loadingMarkup = - loadingStack > 0 ? ( -
- -
- ) : null; + return hasToastById ? toastMessages : [...toastMessages, toast]; + }); + }, []); - const contextualSaveBarMarkup = ( - - - + const hideToast = useCallback(({id}: ToastID) => { + setToastMessages((toastMessages) => + toastMessages.filter(({id: toastId}) => id !== toastId), ); - - const topBarMarkup = topBar ? ( -
{ + setLoadingStack((loadingStack) => loadingStack + 1); + }, []); + + const stopLoading = useCallback(() => { + setLoadingStack((loadingStack) => Math.max(0, loadingStack - 1)); + }, []); + + const setContextualSaveBar = useCallback( + (props: ContextualSaveBarProps) => { + setContextualSaveBarProps({...props}); + forceTrueShowContextualSaveBar(); + }, + [forceTrueShowContextualSaveBar], + ); + + const removeContextualSaveBar = useCallback(() => { + setContextualSaveBarProps(null); + forceFalseShowContextualSaveBar(); + }, [forceFalseShowContextualSaveBar]); + + const handleResize = useCallback(() => { + if (globalRibbonContainer.current) { + const globalRibbonHeight = globalRibbonContainer.current.offsetHeight; + setRootProperty( + GLOBAL_RIBBON_CUSTOM_PROPERTY, + `${globalRibbonHeight}px`, + null, + ); + setGlobalRibbonHeight(globalRibbonHeight); + } + }, []); + + useEffect(() => { + handleResize(); + }, [globalRibbon, handleResize]); + + const context = useMemo( + () => ({ + showToast, + hideToast, + startLoading, + stopLoading, + setContextualSaveBar, + removeContextualSaveBar, + }), + [ + hideToast, + removeContextualSaveBar, + setContextualSaveBar, + showToast, + startLoading, + stopLoading, + ], + ); + + const navClassName = classNames( + styles.Navigation, + showMobileNavigation && styles['Navigation-visible'], + ); + + const skipTarget = skipToContentTarget + ? (skipToContentTarget.current && skipToContentTarget.current.id) || '' + : APP_FRAME_MAIN_ANCHOR_TARGET; + + const mobileNavHidden = isNavigationCollapsed && !showMobileNavigation; + const mobileNavShowing = isNavigationCollapsed && showMobileNavigation; + const tabIndex = mobileNavShowing ? 0 : -1; + + const navigationMarkup = navigation ? ( + + - {topBar} + + + + ) : null; + + const loadingMarkup = + loadingStack > 0 ? ( +
+
) : null; - const globalRibbonMarkup = globalRibbon ? ( -
+ + + ); + + const topBarMarkup = topBar ? ( +
+ {topBar} +
+ ) : null; + + const globalRibbonMarkup = globalRibbon ? ( +
+ {globalRibbon} +
+ ) : null; + + const skipClassName = classNames(styles.Skip, skipFocused && styles.focused); + + const skipMarkup = ( + + {i18n.translate('Polaris.Frame.skipToContent')} + +
+ ); + + const navigationAttributes = navigation + ? { + 'data-has-navigation': true, + } + : {}; + + const frameClassName = classNames( + styles.Frame, + navigation && styles.hasNav, + topBar && styles.hasTopBar, + ); + + const navigationOverlayMarkup = + showMobileNavigation && isNavigationCollapsed ? ( + ) : null; - const skipClassName = classNames( - styles.Skip, - skipFocused && styles.focused, - ); - - const skipTarget = skipToContentTarget - ? (skipToContentTarget.current && skipToContentTarget.current.id) || '' - : APP_FRAME_MAIN_ANCHOR_TARGET; - - const skipMarkup = ( -
- + ); + + return ( + + - ); - - const navigationAttributes = navigation - ? { - 'data-has-navigation': true, - } - : {}; - - const frameClassName = classNames( - styles.Frame, - navigation && styles.hasNav, - topBar && styles.hasTopBar, - ); - - const navigationOverlayMarkup = - showMobileNavigation && isNavigationCollapsed ? ( - - ) : null; - - const skipToMainContentTarget = skipToContentTarget ? null : ( - // eslint-disable-next-line jsx-a11y/anchor-is-valid - - ); - - const context = { - showToast: this.showToast, - hideToast: this.hideToast, - startLoading: this.startLoading, - stopLoading: this.stopLoading, - setContextualSaveBar: this.setContextualSaveBar, - removeContextualSaveBar: this.removeContextualSaveBar, - }; - - return ( - -
- {skipMarkup} - {topBarMarkup} - {navigationMarkup} - {contextualSaveBarMarkup} - {loadingMarkup} - {navigationOverlayMarkup} -
- {skipToMainContentTarget} -
{children}
-
- - {globalRibbonMarkup} - -
-
- ); - } - - private setGlobalRibbonHeight = () => { - const {globalRibbonContainer} = this; - if (globalRibbonContainer) { - this.setState( - { - globalRibbonHeight: globalRibbonContainer.offsetHeight, - }, - this.setGlobalRibbonRootProperty, - ); - } - }; - - private setGlobalRibbonRootProperty = () => { - const {globalRibbonHeight} = this.state; - setRootProperty( - GLOBAL_RIBBON_CUSTOM_PROPERTY, - `${globalRibbonHeight}px`, - null, - ); - }; - - private showToast = (toast: ToastPropsWithID) => { - this.setState(({toastMessages}: State) => { - const hasToastById = - toastMessages.find(({id}) => id === toast.id) != null; - return { - toastMessages: hasToastById ? toastMessages : [...toastMessages, toast], - }; - }); - }; - - private hideToast = ({id}: ToastID) => { - this.setState(({toastMessages}: State) => { - return { - toastMessages: toastMessages.filter(({id: toastId}) => id !== toastId), - }; - }); - }; - - private setContextualSaveBar = (props: ContextualSaveBarProps) => { - const {showContextualSaveBar} = this.state; - this.contextualSaveBar = {...props}; - if (showContextualSaveBar === true) { - this.forceUpdate(); - } else { - this.setState({showContextualSaveBar: true}); - } - }; - - private removeContextualSaveBar = () => { - this.contextualSaveBar = null; - this.setState({showContextualSaveBar: false}); - }; - - private startLoading = () => { - this.setState(({loadingStack}: State) => ({ - loadingStack: loadingStack + 1, - })); - }; - - private stopLoading = () => { - this.setState(({loadingStack}: State) => ({ - loadingStack: Math.max(0, loadingStack - 1), - })); - }; - - private handleResize = () => { - if (this.props.globalRibbon) { - this.setGlobalRibbonHeight(); - } - }; - - private handleFocus = () => { - this.setState({skipFocused: true}); - }; - - private handleBlur = () => { - this.setState({skipFocused: false}); - }; - - private handleClick = () => { - this.skipToMainContentTargetNode.current && - this.skipToMainContentTargetNode.current.focus(); - }; - - private handleNavigationDismiss = () => { - const {onNavigationDismiss} = this.props; - if (onNavigationDismiss != null) { - onNavigationDismiss(); - } - }; - - private setGlobalRibbonContainer = (node: HTMLDivElement) => { - this.globalRibbonContainer = node; - }; - - private handleNavKeydown = (event: React.KeyboardEvent) => { - const {key} = event; - - if (key === 'Escape') { - this.handleNavigationDismiss(); - } - }; - - private findNavigationNode = () => { - return this.navigationNode.current; - }; + + ); } const navTransitionClasses = { @@ -409,7 +359,3 @@ const navTransitionClasses = { exit: classNames(styles['Navigation-exit']), exitActive: classNames(styles['Navigation-exitActive']), }; - -// Use named export once withAppProvider is refactored away -// eslint-disable-next-line import/no-default-export -export default withAppProvider()(Frame); diff --git a/src/components/Frame/components/ToastManager/tests/ToastManager.test.tsx b/src/components/Frame/components/ToastManager/tests/ToastManager.test.tsx index 607a528e89f..50d41dc08c1 100644 --- a/src/components/Frame/components/ToastManager/tests/ToastManager.test.tsx +++ b/src/components/Frame/components/ToastManager/tests/ToastManager.test.tsx @@ -2,7 +2,7 @@ import React from 'react'; import {timer} from '@shopify/jest-dom-mocks'; import {mountWithAppProvider} from 'test-utilities/legacy'; import {Toast} from '../../Toast'; -import Frame from '../../../Frame'; +import {Frame} from '../../../Frame'; import {ToastManager} from '..'; window.matchMedia = diff --git a/src/components/Frame/index.ts b/src/components/Frame/index.ts index 5c0e5343228..2cbd6b0a2f0 100644 --- a/src/components/Frame/index.ts +++ b/src/components/Frame/index.ts @@ -1,6 +1,4 @@ -import Frame, {FrameProps} from './Frame'; - -export {Frame, FrameProps}; +export {Frame, FrameProps} from './Frame'; export { DEFAULT_TOAST_DURATION, diff --git a/src/components/Frame/tests/Frame.test.tsx b/src/components/Frame/tests/Frame.test.tsx index 7be9c2da9ad..6d89e7ac6b0 100644 --- a/src/components/Frame/tests/Frame.test.tsx +++ b/src/components/Frame/tests/Frame.test.tsx @@ -7,11 +7,14 @@ import { TrapFocus, ContextualSaveBar as PolarisContextualSavebar, Loading as PolarisLoading, + Toast as PolarisToast, } from 'components'; -import Frame from '../Frame'; +import {Frame} from '../Frame'; import { ContextualSaveBar as FrameContextualSavebar, Loading as FrameLoading, + Toast as FrameToast, + ToastManager, } from '../components'; window.matchMedia = @@ -123,6 +126,52 @@ describe('', () => { expect(cssTransition.prop('in')).toBe(true); }); + it('calls onNavigationDismiss on dismiss button click', () => { + const mockOnNavigationDismiss = jest.fn(); + const navigation =
; + const cssTransition = mountWithAppProvider( + , + {mediaQuery: {isNavigationCollapsed: true}}, + ) + .find(Frame) + .find(TrapFocus) + .find(CSSTransition); + + cssTransition + .find('button') + .first() + .simulate('click'); + animationFrame.runFrame(); + + expect(mockOnNavigationDismiss).toHaveBeenCalledTimes(1); + }); + + it('calls onNavigationDismiss on Escape keypress', () => { + const mockOnNavigationDismiss = jest.fn(); + const navigation =
; + const div = mountWithAppProvider( + , + {mediaQuery: {isNavigationCollapsed: true}}, + ) + .find(Frame) + .find(TrapFocus) + .find(CSSTransition) + .find('div') + .first(); + + div.simulate('keydown', {key: 'Enter'}); + expect(mockOnNavigationDismiss).toHaveBeenCalledTimes(0); + + div.simulate('keydown', {key: 'Escape'}); + expect(mockOnNavigationDismiss).toHaveBeenCalledTimes(1); + }); + it('renders a skip to content link with the proper text', () => { const skipToContentLinkText = mountWithAppProvider() .find('a') @@ -224,6 +273,20 @@ describe('', () => { expect(frame.find(FrameContextualSavebar).exists()).toBe(true); }); + it("unmounts rendered Frame's ContextualSavebar if Polaris' ContextualSavebar is unmounted", () => { + const frame = mountWithAppProvider( + + + , + ); + expect(frame.find(FrameContextualSavebar).exists()).toBe(true); + + frame.setProps({children: null}); + frame.update(); + + expect(frame.find('CSSAnimation.ContextualSaveBar').prop('in')).toBe(false); + }); + it('renders a Frame Loading if Polaris Loading is rendered', () => { const frame = mountWithAppProvider( @@ -232,4 +295,63 @@ describe('', () => { ); expect(frame.find(FrameLoading).exists()).toBe(true); }); + + it('unmounts rendered Frame Loading if Polaris Loading is unmounted', () => { + const frame = mountWithAppProvider( + + + , + ); + + expect(frame.find(FrameLoading).exists()).toBe(true); + + frame.setProps({children: null}); + frame.update(); + + expect(frame.find(FrameLoading).exists()).toBe(false); + }); + + it('renders a Frame Toast if Polaris Toast is rendered', () => { + const props = {content: 'Image uploaded', onDismiss: () => {}}; + const frame = mountWithAppProvider( + + + , + ); + + expect(frame.find(FrameToast).exists()).toBe(true); + }); + + it('only renders a Frame Toast if two Polaris Toasts with same ID are rendered', () => { + const props = { + id: 'mock-id', + content: 'Image uploaded', + onDismiss: () => {}, + }; + const frame = mountWithAppProvider( + + + + , + ); + + expect(frame.find(FrameToast).exists()).toBe(true); + expect(frame.find(FrameToast)).toHaveLength(1); + }); + + it('unmounts rendered Frame Toast if Polaris Toast is unmounted', () => { + const props = {content: 'Image uploaded', onDismiss: () => {}}; + const frame = mountWithAppProvider( + + + , + ); + + expect(frame.find(FrameToast).exists()).toBe(true); + + frame.setProps({children: null}); + frame.update(); + + expect(frame.find(ToastManager).prop('toastMessages')).toHaveLength(0); + }); }); diff --git a/src/utilities/tests/set-root-property.test.ts b/src/utilities/tests/set-root-property.test.ts index e6cdde9e965..94a3863d018 100644 --- a/src/utilities/tests/set-root-property.test.ts +++ b/src/utilities/tests/set-root-property.test.ts @@ -2,11 +2,15 @@ import {documentHasStyle} from 'test-utilities'; import {setRootProperty} from '../set-root-property'; describe('setRootProperty', () => { + it('sets styles on the document element', () => { + setRootProperty('color', 'red', null); + expect(documentHasStyle('color', 'red')).toBe(true); + }); // JSDOM 11.12.0 does not support setting/reading custom properties so we are // unable to assert that we set a custom property // See https://github.com/jsdom/jsdom/issues/1895 // eslint-disable-next-line jest/no-disabled-tests - it.skip('sets styles on the document element', () => { + it.skip('sets custom styles on the document element', () => { setRootProperty('topBar', '#eee', null); expect(documentHasStyle('topBar', '#eee')).toBe(true); });