From 648796a60e55f67fb4308bac463f8de43f83d96c Mon Sep 17 00:00:00 2001 From: Oleksandr Fediashov Date: Tue, 27 Jul 2021 14:04:36 +0200 Subject: [PATCH] chore(Modal|Portal|Popup): use React.forwardRef() (#4253) * chore(Modal|Portal): use React.forwardRef() * remove redundant statics * migrate Popup --- src/addons/Portal/Portal.js | 354 ++++++++-------- src/addons/Portal/PortalInner.js | 44 +- src/addons/Portal/usePortalElement.js | 30 ++ src/addons/Portal/utils/useTrigger.js | 25 ++ src/addons/Portal/utils/validateTrigger.js | 8 +- src/lib/doesNodeContainClick.js | 28 +- src/lib/hooks/useAutoControlledValue.js | 41 +- src/lib/hooks/usePrevious.js | 18 + src/lib/index.js | 2 + src/modules/Modal/Modal.js | 386 +++++++++-------- src/modules/Modal/ModalDimmer.js | 22 +- src/modules/Popup/Popup.js | 421 ++++++++++--------- test/specs/addons/Portal/Portal-test.js | 72 ++-- test/specs/addons/Portal/PortalInner-test.js | 64 ++- test/specs/modules/Modal/Modal-test.js | 8 +- test/specs/modules/Popup/Popup-test.js | 31 +- 16 files changed, 849 insertions(+), 705 deletions(-) create mode 100644 src/addons/Portal/usePortalElement.js create mode 100644 src/addons/Portal/utils/useTrigger.js create mode 100644 src/lib/hooks/usePrevious.js diff --git a/src/addons/Portal/Portal.js b/src/addons/Portal/Portal.js index e987e36f95..4f49874189 100644 --- a/src/addons/Portal/Portal.js +++ b/src/addons/Portal/Portal.js @@ -1,17 +1,16 @@ import EventStack from '@semantic-ui-react/event-stack' -import { handleRef, Ref } from '@fluentui/react-component-ref' import keyboardKey from 'keyboard-key' import _ from 'lodash' import PropTypes from 'prop-types' import React from 'react' import { - ModernAutoControlledComponent as Component, customPropTypes, doesNodeContainClick, makeDebugger, + useAutoControlledValue, } from '../../lib' -import validateTrigger from './utils/validateTrigger' +import useTrigger from './utils/useTrigger' import PortalInner from './PortalInner' const debug = makeDebugger('portal') @@ -23,263 +22,270 @@ const debug = makeDebugger('portal') * @see Dimmer * @see Confirm */ -class Portal extends Component { - contentRef = React.createRef() - triggerRef = React.createRef() - latestDocumentMouseDownEvent = null +function Portal(props) { + const { + children, + closeOnDocumentClick, + closeOnEscape, + closeOnPortalMouseLeave, + closeOnTriggerBlur, + closeOnTriggerClick, + closeOnTriggerMouseLeave, + eventPool, + mountNode, + mouseEnterDelay, + mouseLeaveDelay, + openOnTriggerClick, + openOnTriggerFocus, + openOnTriggerMouseEnter, + } = props + + const [open, setOpen] = useAutoControlledValue({ + state: props.open, + defaultState: props.defaultOpen, + initialState: false, + }) + + const contentRef = React.useRef() + const [triggerRef, trigger] = useTrigger(props.trigger, props.triggerRef) + + const mouseEnterTimer = React.useRef() + const mouseLeaveTimer = React.useRef() + const latestDocumentMouseDownEvent = React.useRef() - componentWillUnmount() { - // Clean up timers - clearTimeout(this.mouseEnterTimer) - clearTimeout(this.mouseLeaveTimer) + // ---------------------------------------- + // Behavior + // ---------------------------------------- + + const openPortal = (e) => { + debug('open()') + + setOpen(true) + _.invoke(props, 'onOpen', e, { ...props, open: true }) + } + + const openPortalWithTimeout = (e, delay) => { + debug('openWithTimeout()', delay) + // React wipes the entire event object and suggests using e.persist() if + // you need the event for async access. However, even with e.persist + // certain required props (e.g. currentTarget) are null so we're forced to clone. + const eventClone = { ...e } + return setTimeout(() => openPortal(eventClone), delay || 0) + } + + const closePortal = (e) => { + debug('close()') + + setOpen(false) + _.invoke(props, 'onClose', e, { ...props, open: false }) + } + + const closePortalWithTimeout = (e, delay) => { + debug('closeWithTimeout()', delay) + // React wipes the entire event object and suggests using e.persist() if + // you need the event for async access. However, even with e.persist + // certain required props (e.g. currentTarget) are null so we're forced to clone. + const eventClone = { ...e } + return setTimeout(() => closePortal(eventClone), delay || 0) } // ---------------------------------------- // Document Event Handlers // ---------------------------------------- - handleDocumentMouseDown = (e) => { - this.latestDocumentMouseDownEvent = e + React.useEffect(() => { + // Clean up timers + clearTimeout(mouseEnterTimer.current) + clearTimeout(mouseLeaveTimer.current) + }, []) + + const handleDocumentMouseDown = (e) => { + latestDocumentMouseDownEvent.current = e } - handleDocumentClick = (e) => { - const { closeOnDocumentClick } = this.props - const currentMouseDownEvent = this.latestDocumentMouseDownEvent - this.latestDocumentMouseDownEvent = null + const handleDocumentClick = (e) => { + const currentMouseDownEvent = latestDocumentMouseDownEvent.current + latestDocumentMouseDownEvent.current = null + + // event happened in trigger (delegate to trigger handlers) + const isInsideTrigger = doesNodeContainClick(triggerRef.current, e) + // event originated in the portal but was ended outside + const isOriginatedFromPortal = + currentMouseDownEvent && doesNodeContainClick(contentRef.current, currentMouseDownEvent) + // event happened in the portal + const isInsidePortal = doesNodeContainClick(contentRef.current, e) if ( - !this.contentRef.current || // no portal - doesNodeContainClick(this.triggerRef.current, e) || // event happened in trigger (delegate to trigger handlers) - (currentMouseDownEvent && - doesNodeContainClick(this.contentRef.current, currentMouseDownEvent)) || // event originated in the portal but was ended outside - doesNodeContainClick(this.contentRef.current, e) // event happened in the portal + !contentRef.current?.contains || // no portal + isInsideTrigger || + isOriginatedFromPortal || + isInsidePortal ) { return } // ignore the click if (closeOnDocumentClick) { debug('handleDocumentClick()') - this.close(e) + closePortal(e) } } - handleEscape = (e) => { - if (!this.props.closeOnEscape) return - if (keyboardKey.getCode(e) !== keyboardKey.Escape) return + const handleEscape = (e) => { + if (!closeOnEscape) { + return + } + if (keyboardKey.getCode(e) !== keyboardKey.Escape) { + return + } debug('handleEscape()') - this.close(e) + closePortal(e) } // ---------------------------------------- // Component Event Handlers // ---------------------------------------- - handlePortalMouseLeave = (e) => { - const { closeOnPortalMouseLeave, mouseLeaveDelay } = this.props - - if (!closeOnPortalMouseLeave) return + const handlePortalMouseLeave = (e) => { + if (!closeOnPortalMouseLeave) { + return + } // Do not close the portal when 'mouseleave' is triggered by children - if (e.target !== this.contentRef.current) return + if (e.target !== contentRef.current) { + return + } debug('handlePortalMouseLeave()') - this.mouseLeaveTimer = this.closeWithTimeout(e, mouseLeaveDelay) + mouseLeaveTimer.current = closePortalWithTimeout(e, mouseLeaveDelay) } - handlePortalMouseEnter = () => { + const handlePortalMouseEnter = () => { // In order to enable mousing from the trigger to the portal, we need to // clear the mouseleave timer that was set when leaving the trigger. - const { closeOnPortalMouseLeave } = this.props - - if (!closeOnPortalMouseLeave) return + if (!closeOnPortalMouseLeave) { + return + } debug('handlePortalMouseEnter()') - clearTimeout(this.mouseLeaveTimer) + clearTimeout(mouseLeaveTimer.current) } - handleTriggerBlur = (e, ...rest) => { - const { trigger, closeOnTriggerBlur } = this.props - + const handleTriggerBlur = (e, ...rest) => { // Call original event handler _.invoke(trigger, 'props.onBlur', e, ...rest) // IE 11 doesn't work with relatedTarget in blur events const target = e.relatedTarget || document.activeElement // do not close if focus is given to the portal - const didFocusPortal = _.invoke(this.contentRef.current, 'contains', target) + const didFocusPortal = _.invoke(contentRef.current, 'contains', target) - if (!closeOnTriggerBlur || didFocusPortal) return + if (!closeOnTriggerBlur || didFocusPortal) { + return + } debug('handleTriggerBlur()') - this.close(e) + closePortal(e) } - handleTriggerClick = (e, ...rest) => { - const { trigger, closeOnTriggerClick, openOnTriggerClick } = this.props - const { open } = this.state - + const handleTriggerClick = (e, ...rest) => { // Call original event handler _.invoke(trigger, 'props.onClick', e, ...rest) if (open && closeOnTriggerClick) { debug('handleTriggerClick() - close') - this.close(e) + closePortal(e) } else if (!open && openOnTriggerClick) { debug('handleTriggerClick() - open') - - this.open(e) + openPortal(e) } } - handleTriggerFocus = (e, ...rest) => { - const { trigger, openOnTriggerFocus } = this.props - + const handleTriggerFocus = (e, ...rest) => { // Call original event handler _.invoke(trigger, 'props.onFocus', e, ...rest) - if (!openOnTriggerFocus) return + if (!openOnTriggerFocus) { + return + } debug('handleTriggerFocus()') - this.open(e) + openPortal(e) } - handleTriggerMouseLeave = (e, ...rest) => { - clearTimeout(this.mouseEnterTimer) - - const { trigger, closeOnTriggerMouseLeave, mouseLeaveDelay } = this.props + const handleTriggerMouseLeave = (e, ...rest) => { + clearTimeout(mouseEnterTimer) // Call original event handler _.invoke(trigger, 'props.onMouseLeave', e, ...rest) - if (!closeOnTriggerMouseLeave) return + if (!closeOnTriggerMouseLeave) { + return + } debug('handleTriggerMouseLeave()') - this.mouseLeaveTimer = this.closeWithTimeout(e, mouseLeaveDelay) + mouseLeaveTimer.current = closePortalWithTimeout(e, mouseLeaveDelay) } - handleTriggerMouseEnter = (e, ...rest) => { - clearTimeout(this.mouseLeaveTimer) - - const { trigger, mouseEnterDelay, openOnTriggerMouseEnter } = this.props + const handleTriggerMouseEnter = (e, ...rest) => { + clearTimeout(mouseLeaveTimer) // Call original event handler _.invoke(trigger, 'props.onMouseEnter', e, ...rest) - if (!openOnTriggerMouseEnter) return + if (!openOnTriggerMouseEnter) { + return + } debug('handleTriggerMouseEnter()') - this.mouseEnterTimer = this.openWithTimeout(e, mouseEnterDelay) - } - - // ---------------------------------------- - // Behavior - // ---------------------------------------- - - open = (e) => { - debug('open()') - - _.invoke(this.props, 'onOpen', e, { ...this.props, open: true }) - this.setState({ open: true }) - } - - openWithTimeout = (e, delay) => { - debug('openWithTimeout()', delay) - // React wipes the entire event object and suggests using e.persist() if - // you need the event for async access. However, even with e.persist - // certain required props (e.g. currentTarget) are null so we're forced to clone. - const eventClone = { ...e } - return setTimeout(() => this.open(eventClone), delay || 0) + mouseEnterTimer.current = openPortalWithTimeout(e, mouseEnterDelay) } - close = (e) => { - debug('close()') - - this.setState({ open: false }) - _.invoke(this.props, 'onClose', e, { ...this.props, open: false }) - } - - closeWithTimeout = (e, delay) => { - debug('closeWithTimeout()', delay) - // React wipes the entire event object and suggests using e.persist() if - // you need the event for async access. However, even with e.persist - // certain required props (e.g. currentTarget) are null so we're forced to clone. - const eventClone = { ...e } - return setTimeout(() => this.close(eventClone), delay || 0) - } - - handleMount = () => { - debug('handleMount()') - _.invoke(this.props, 'onMount', null, this.props) - } - - handleUnmount = () => { - debug('handleUnmount()') - _.invoke(this.props, 'onUnmount', null, this.props) - } - - handleTriggerRef = (c) => { - debug('handleTriggerRef()') - this.triggerRef.current = c - handleRef(this.props.triggerRef, c) - } - - render() { - const { children, eventPool, mountNode, trigger } = this.props - const { open } = this.state - - /* istanbul ignore else */ - if (process.env.NODE_ENV !== 'production') { - validateTrigger(trigger) - } - - return ( - <> - {open && ( - <> - - {children} - - - - - - - - - )} - {trigger && ( - - {React.cloneElement(trigger, { - onBlur: this.handleTriggerBlur, - onClick: this.handleTriggerClick, - onFocus: this.handleTriggerFocus, - onMouseLeave: this.handleTriggerMouseLeave, - onMouseEnter: this.handleTriggerMouseEnter, - })} - - )} - - ) - } + return ( + <> + {open && ( + <> + _.invoke(props, 'onMount', null, props)} + onUnmount={() => _.invoke(props, 'onUnmount', null, props)} + ref={contentRef} + > + {children} + + + + + + + + + )} + {trigger && + React.cloneElement(trigger, { + onBlur: handleTriggerBlur, + onClick: handleTriggerClick, + onFocus: handleTriggerFocus, + onMouseLeave: handleTriggerMouseLeave, + onMouseEnter: handleTriggerMouseEnter, + ref: triggerRef, + })} + + ) } +Portal.displayName = 'Portal' Portal.propTypes = { /** Primary content. */ children: PropTypes.node.isRequired, @@ -379,8 +385,6 @@ Portal.defaultProps = { openOnTriggerClick: true, } -Portal.autoControlledProps = ['open'] - Portal.Inner = PortalInner export default Portal diff --git a/src/addons/Portal/PortalInner.js b/src/addons/Portal/PortalInner.js index 163ac35983..42950165d4 100644 --- a/src/addons/Portal/PortalInner.js +++ b/src/addons/Portal/PortalInner.js @@ -1,40 +1,40 @@ -import { handleRef, Ref } from '@fluentui/react-component-ref' import _ from 'lodash' import PropTypes from 'prop-types' -import React, { Component } from 'react' +import React from 'react' import { createPortal } from 'react-dom' -import { customPropTypes, isBrowser, makeDebugger } from '../../lib' +import { customPropTypes, isBrowser, makeDebugger, useEventCallback } from '../../lib' +import usePortalElement from './usePortalElement' -const debug = makeDebugger('portalInner') +const debug = makeDebugger('PortalInner') /** * An inner component that allows you to render children outside their parent. */ -class PortalInner extends Component { - componentDidMount() { - debug('componentDidMount()') - _.invoke(this.props, 'onMount', null, this.props) - } +const PortalInner = React.forwardRef(function (props, ref) { + const handleMount = useEventCallback(() => _.invoke(props, 'onMount', null, props)) + const handleUnmount = useEventCallback(() => _.invoke(props, 'onUnmount', null, props)) - componentWillUnmount() { - debug('componentWillUnmount()') - _.invoke(this.props, 'onUnmount', null, this.props) - } + const element = usePortalElement(props.children, ref) - handleRef = (c) => { - debug('handleRef', c) - handleRef(this.props.innerRef, c) - } + React.useEffect(() => { + debug('componentDidMount()') + handleMount() - render() { - if (!isBrowser()) return null - const { children, mountNode = document.body } = this.props + return () => { + debug('componentWillUnmount()') + handleUnmount() + } + }, []) - return createPortal({children}, mountNode) + if (!isBrowser()) { + return null } -} + return createPortal(element, props.mountNode || document.body) +}) + +PortalInner.displayName = 'PortalInner' PortalInner.propTypes = { /** Primary content. */ children: PropTypes.node.isRequired, diff --git a/src/addons/Portal/usePortalElement.js b/src/addons/Portal/usePortalElement.js new file mode 100644 index 0000000000..7c5b8fd491 --- /dev/null +++ b/src/addons/Portal/usePortalElement.js @@ -0,0 +1,30 @@ +import React from 'react' +import ReactIs from 'react-is' + +import { useMergedRefs } from '../../lib' + +/** + * Assigns merged ref to an existing element is possible or wraps it with an additional "div". + * + * @param {React.ReactNode} node + * @param {React.Ref} userRef + */ +export default function usePortalElement(node, userRef) { + const ref = useMergedRefs(node.ref, userRef) + + if (React.isValidElement(node)) { + if (ReactIs.isForwardRef(node)) { + return React.cloneElement(node, { ref }) + } + + if (typeof node.type === 'string') { + return React.cloneElement(node, { ref }) + } + } + + return ( +
+ {node} +
+ ) +} diff --git a/src/addons/Portal/utils/useTrigger.js b/src/addons/Portal/utils/useTrigger.js new file mode 100644 index 0000000000..503e231e33 --- /dev/null +++ b/src/addons/Portal/utils/useTrigger.js @@ -0,0 +1,25 @@ +import React from 'react' + +import { useMergedRefs } from '../../../lib' +import validateTrigger from './validateTrigger' + +/** + * @param {React.ReactNode} trigger + * @param {React.Ref} triggerRef + */ +function useTrigger(trigger, triggerRef) { + const ref = useMergedRefs(trigger?.ref, triggerRef) + + if (trigger) { + /* istanbul ignore else */ + if (process.env.NODE_ENV !== 'production') { + validateTrigger(trigger) + } + + return [ref, React.cloneElement(trigger, { ref })] + } + + return [ref, null] +} + +export default useTrigger diff --git a/src/addons/Portal/utils/validateTrigger.js b/src/addons/Portal/utils/validateTrigger.js index 52ed7ab5c5..5dcd4282c1 100644 --- a/src/addons/Portal/utils/validateTrigger.js +++ b/src/addons/Portal/utils/validateTrigger.js @@ -5,11 +5,9 @@ import * as ReactIs from 'react-is' * Asserts that a passed element can be used cloned a props will be applied properly. */ export default function validateTrigger(element) { - if (element) { - React.Children.only(element) + React.Children.only(element) - if (ReactIs.isFragment(element)) { - throw new Error('An "React.Fragment" cannot be used as a `trigger`.') - } + if (ReactIs.isFragment(element)) { + throw new Error('An "React.Fragment" cannot be used as a `trigger`.') } } diff --git a/src/lib/doesNodeContainClick.js b/src/lib/doesNodeContainClick.js index d1ae271216..d400098eeb 100644 --- a/src/lib/doesNodeContainClick.js +++ b/src/lib/doesNodeContainClick.js @@ -10,7 +10,9 @@ import _ from 'lodash' * @returns {boolean} */ const doesNodeContainClick = (node, e) => { - if (_.some([e, node], _.isNil)) return false + if (_.some([e, node], _.isNil)) { + return false + } // if there is an e.target and it is in the document, use a simple node.contains() check if (e.target) { @@ -18,7 +20,10 @@ const doesNodeContainClick = (node, e) => { if (document.querySelector('[data-suir-click-target=true]')) { _.invoke(e.target, 'removeAttribute', 'data-suir-click-target') - return node.contains(e.target) + + if (typeof node.contains === 'function') { + return node.contains(e.target) + } } } @@ -29,18 +34,31 @@ const doesNodeContainClick = (node, e) => { // return early if the event properties aren't available // prevent measuring the node and repainting if we don't need to const { clientX, clientY } = e - if (_.some([clientX, clientY], _.isNil)) return false + + if (_.some([clientX, clientY], _.isNil)) { + return false + } + + if (typeof node.getClientRects !== 'function') { + return false + } // false if the node is not visible const clientRects = node.getClientRects() + // Heads Up! // getClientRects returns a DOMRectList, not an array nor a plain object // We explicitly avoid _.isEmpty and check .length to cover all possible shapes - if (!node.offsetWidth || !node.offsetHeight || !clientRects || !clientRects.length) return false + if (!node.offsetWidth || !node.offsetHeight || !clientRects || !clientRects.length) { + return false + } // false if the node doesn't have a valid bounding rect const { top, bottom, left, right } = _.first(clientRects) - if (_.some([top, bottom, left, right], _.isNil)) return false + + if (_.some([top, bottom, left, right], _.isNil)) { + return false + } // we add a small decimal to the upper bound just to make it inclusive // don't add an whole pixel (1) as the event/node values may be decimal sensitive diff --git a/src/lib/hooks/useAutoControlledValue.js b/src/lib/hooks/useAutoControlledValue.js index f211badf1d..756ed79fb7 100644 --- a/src/lib/hooks/useAutoControlledValue.js +++ b/src/lib/hooks/useAutoControlledValue.js @@ -1,41 +1,5 @@ import * as React from 'react' -/** - * Helper hook to handle previous comparison of controlled/uncontrolled. Prints an error when "isControlled" value - * switches between subsequent renders. - */ -function useIsControlled(controlledValue) { - const [isControlled] = React.useState(controlledValue !== undefined) - - if (process.env.NODE_ENV !== 'production') { - // We don't want these warnings in production even though it is against native behaviour - React.useEffect(() => { - if (isControlled !== (controlledValue !== undefined)) { - const error = new Error() - - const controlWarning = isControlled - ? 'a controlled value to be uncontrolled' - : 'an uncontrolled value to be controlled' - const undefinedWarning = isControlled ? 'defined to an undefined' : 'undefined to a defined' - - // eslint-disable-next-line no-console - console.error( - [ - // Default react error - `A component is changing ${controlWarning}'. This is likely caused by the value changing from `, - `${undefinedWarning} value, which should not happen. Decide between using a controlled or uncontrolled `, - 'input element for the lifetime of the component.', - 'More info: https://reactjs.org/link/controlled-components', - error.stack, - ].join(' '), - ) - } - }, [isControlled, controlledValue]) - } - - return isControlled -} - /** * A hook that allows optional user control, implements an interface similar to `React.useState()`. * Useful for components which allow uncontrolled and controlled behaviours for users. @@ -50,12 +14,11 @@ function useIsControlled(controlledValue) { * @see https://reactjs.org/docs/hooks-state.html */ function useAutoControlledValue(options) { - const isControlled = useIsControlled(options.state) const initialState = typeof options.defaultState === 'undefined' ? options.initialState : options.defaultState - const [internalState, setInternalState] = React.useState(initialState) - const state = isControlled ? options.state : internalState + + const state = typeof options.state === 'undefined' ? internalState : options.state const stateRef = React.useRef(state) React.useEffect(() => { diff --git a/src/lib/hooks/usePrevious.js b/src/lib/hooks/usePrevious.js new file mode 100644 index 0000000000..0178735c75 --- /dev/null +++ b/src/lib/hooks/usePrevious.js @@ -0,0 +1,18 @@ +import * as React from 'react' + +/** + * Hook keeping track of a given value from a previous execution of the component the Hook is used in. + * + * @see https://reactjs.org/docs/hooks-faq.html#how-to-get-the-previous-props-or-state + */ +function usePrevious(value) { + const ref = React.useRef() + + React.useEffect(() => { + ref.current = value + }) + + return ref.current +} + +export default usePrevious diff --git a/src/lib/index.js b/src/lib/index.js index 597ccae801..227cd7a1a8 100644 --- a/src/lib/index.js +++ b/src/lib/index.js @@ -49,4 +49,6 @@ export { makeDebugger } export useAutoControlledValue from './hooks/useAutoControlledValue' export useClassNamesOnNode from './hooks/useClassNamesOnNode' export useEventCallback from './hooks/useEventCallback' +export useIsomorphicLayoutEffect from './hooks/useIsomorphicLayoutEffect' export useMergedRefs from './hooks/useMergedRefs' +export usePrevious from './hooks/usePrevious' diff --git a/src/modules/Modal/Modal.js b/src/modules/Modal/Modal.js index 83268b2d31..cf35c358bb 100644 --- a/src/modules/Modal/Modal.js +++ b/src/modules/Modal/Modal.js @@ -1,12 +1,10 @@ -import { Ref } from '@fluentui/react-component-ref' import cx from 'clsx' import _ from 'lodash' import PropTypes from 'prop-types' -import React, { createRef, isValidElement } from 'react' +import React from 'react' import shallowEqual from 'shallowequal' import { - ModernAutoControlledComponent as Component, childrenUtils, customPropTypes, doesNodeContainClick, @@ -16,6 +14,8 @@ import { isBrowser, makeDebugger, useKeyOnly, + useAutoControlledValue, + useMergedRefs, } from '../../lib' import Icon from '../../elements/Icon' import Portal from '../../addons/Portal' @@ -33,246 +33,262 @@ const debug = makeDebugger('modal') * @see Confirm * @see Portal */ -class Modal extends Component { - legacy = isBrowser() && isLegacy() - ref = createRef() - dimmerRef = createRef() - latestDocumentMouseDownEvent = null - - componentWillUnmount() { - debug('componentWillUnmount()') - this.handlePortalUnmount() - } - +const Modal = React.forwardRef(function (props, ref) { + const { + actions, + basic, + centered, + children, + className, + closeIcon, + closeOnDimmerClick, + closeOnDocumentClick, + content, + dimmer, + eventPool, + header, + size, + style, + trigger, + } = props // Do not access document when server side rendering - getMountNode = () => (isBrowser() ? this.props.mountNode || document.body : null) + const mountNode = isBrowser() ? props.mountNode || document.body : null - handleActionsOverrides = (predefinedProps) => ({ - onActionClick: (e, actionProps) => { - _.invoke(predefinedProps, 'onActionClick', e, actionProps) - _.invoke(this.props, 'onActionClick', e, this.props) - - this.handleClose(e) - }, + const [open, setOpen] = useAutoControlledValue({ + state: props.open, + defaultState: props.defaultOpen, + initialState: false, }) - handleClose = (e) => { + const [legacyStyles, setLegacyStyles] = React.useState({}) + const [scrolling, setScrolling] = React.useState(false) + + const [legacy] = React.useState(() => isBrowser() && isLegacy()) + + const elementRef = useMergedRefs(ref, React.useRef()) + const dimmerRef = React.useRef() + + const animationRequestId = React.useRef() + const latestDocumentMouseDownEvent = React.useRef() + + React.useEffect(() => { + return () => { + cancelAnimationFrame(animationRequestId.current) + latestDocumentMouseDownEvent.current = null + } + }, []) + + // ---------------------------------------- + // Styles calc + // ---------------------------------------- + + const setPositionAndClassNames = () => { + if (elementRef.current) { + const rect = elementRef.current.getBoundingClientRect() + const isFitted = canFit(rect) + + setScrolling(!isFitted) + + // Styles should be computed for IE11 + const computedLegacyStyles = legacy ? getLegacyStyles(isFitted, centered, rect) : {} + + if (!shallowEqual(computedLegacyStyles, computedLegacyStyles)) { + setLegacyStyles(computedLegacyStyles) + } + } + + animationRequestId.current = requestAnimationFrame(setPositionAndClassNames) + } + + // ---------------------------------------- + // Document Event Handlers + // ---------------------------------------- + + const handleClose = (e) => { debug('close()') - this.setState({ open: false }) - _.invoke(this.props, 'onClose', e, { ...this.props, open: false }) + setOpen(false) + _.invoke(props, 'onClose', e, { ...props, open: false }) } - handleDocumentMouseDown = (e) => { - this.latestDocumentMouseDownEvent = e + const handleDocumentMouseDown = (e) => { + latestDocumentMouseDownEvent.current = e } - handleDocumentClick = (e) => { + const handleDocumentClick = (e) => { debug('handleDocumentClick()') - const { closeOnDimmerClick } = this.props - const currentDocumentMouseDownEvent = this.latestDocumentMouseDownEvent - this.latestDocumentMouseDownEvent = null + + const currentDocumentMouseDownEvent = latestDocumentMouseDownEvent.current + latestDocumentMouseDownEvent.current = null if ( !closeOnDimmerClick || - doesNodeContainClick(this.ref.current, currentDocumentMouseDownEvent) || - doesNodeContainClick(this.ref.current, e) + doesNodeContainClick(elementRef.current, currentDocumentMouseDownEvent) || + doesNodeContainClick(elementRef.current, e) ) return - this.setState({ open: false }) - _.invoke(this.props, 'onClose', e, { ...this.props, open: false }) + setOpen(false) + _.invoke(props, 'onClose', e, { ...props, open: false }) } - handleIconOverrides = (predefinedProps) => ({ - onClick: (e) => { - _.invoke(predefinedProps, 'onClick', e) - this.handleClose(e) - }, - }) - - handleOpen = (e) => { + const handleOpen = (e) => { debug('open()') - _.invoke(this.props, 'onOpen', e, { ...this.props, open: true }) - this.setState({ open: true }) + setOpen(true) + _.invoke(props, 'onOpen', e, { ...props, open: true }) } - handlePortalMount = (e) => { - const { eventPool } = this.props + const handlePortalMount = (e) => { debug('handlePortalMount()', { eventPool }) - this.setState({ scrolling: false }) - this.setPositionAndClassNames() + setScrolling(false) + setPositionAndClassNames() - eventStack.sub('mousedown', this.handleDocumentMouseDown, { + eventStack.sub('mousedown', handleDocumentMouseDown, { pool: eventPool, - target: this.dimmerRef.current, + target: dimmerRef.current, }) - eventStack.sub('click', this.handleDocumentClick, { + eventStack.sub('click', handleDocumentClick, { pool: eventPool, - target: this.dimmerRef.current, + target: dimmerRef.current, }) - _.invoke(this.props, 'onMount', e, this.props) + _.invoke(props, 'onMount', e, props) } - handlePortalUnmount = (e) => { - const { eventPool } = this.props + const handlePortalUnmount = (e) => { debug('handlePortalUnmount()', { eventPool }) - cancelAnimationFrame(this.animationRequestId) - eventStack.unsub('mousedown', this.handleDocumentMouseDown, { + cancelAnimationFrame(animationRequestId.current) + eventStack.unsub('mousedown', handleDocumentMouseDown, { pool: eventPool, - target: this.dimmerRef.current, + target: dimmerRef.current, }) - eventStack.unsub('click', this.handleDocumentClick, { + eventStack.unsub('click', handleDocumentClick, { pool: eventPool, - target: this.dimmerRef.current, + target: dimmerRef.current, }) - _.invoke(this.props, 'onUnmount', e, this.props) - } - - setPositionAndClassNames = () => { - const { centered } = this.props - - let scrolling - const newState = {} - - if (this.ref.current) { - const rect = this.ref.current.getBoundingClientRect() - const isFitted = canFit(rect) - - scrolling = !isFitted - // Styles should be computed for IE11 - const legacyStyles = this.legacy ? getLegacyStyles(isFitted, centered, rect) : {} - - if (!shallowEqual(this.state.legacyStyles, legacyStyles)) { - newState.legacyStyles = legacyStyles - } - - if (this.state.scrolling !== scrolling) { - newState.scrolling = scrolling - } - } - - if (!_.isEmpty(newState)) this.setState(newState) - this.animationRequestId = requestAnimationFrame(this.setPositionAndClassNames) + _.invoke(props, 'onUnmount', e, props) } - renderContent = (rest) => { - const { - actions, - basic, - children, - className, - closeIcon, - content, - header, - size, - style, - } = this.props - const { legacyStyles, scrolling } = this.state + // ---------------------------------------- + // Render + // ---------------------------------------- + const renderContent = (rest) => { const classes = cx( 'ui', size, useKeyOnly(basic, 'basic'), - useKeyOnly(this.legacy, 'legacy'), + useKeyOnly(legacy, 'legacy'), useKeyOnly(scrolling, 'scrolling'), 'modal transition visible active', className, ) - const ElementType = getElementType(Modal, this.props) + const ElementType = getElementType(Modal, props) const closeIconName = closeIcon === true ? 'close' : closeIcon - const closeIconJSX = Icon.create(closeIconName, { overrideProps: this.handleIconOverrides }) + const closeIconJSX = Icon.create(closeIconName, { + overrideProps: (predefinedProps) => ({ + onClick: (e) => { + _.invoke(predefinedProps, 'onClick', e) + handleClose(e) + }, + }), + }) return ( - - - {closeIconJSX} - {childrenUtils.isNil(children) ? ( - <> - {ModalHeader.create(header, { autoGenerateKey: false })} - {ModalContent.create(content, { autoGenerateKey: false })} - {ModalActions.create(actions, { overrideProps: this.handleActionsOverrides })} - - ) : ( - children - )} - - + + {closeIconJSX} + {childrenUtils.isNil(children) ? ( + <> + {ModalHeader.create(header, { autoGenerateKey: false })} + {ModalContent.create(content, { autoGenerateKey: false })} + {ModalActions.create(actions, { + overrideProps: (predefinedProps) => ({ + onActionClick: (e, actionProps) => { + _.invoke(predefinedProps, 'onActionClick', e, actionProps) + _.invoke(props, 'onActionClick', e, props) + + handleClose(e) + }, + }), + })} + + ) : ( + children + )} + ) } - render() { - const { centered, closeOnDocumentClick, dimmer, eventPool, trigger } = this.props - const { open, scrolling } = this.state - const mountNode = this.getMountNode() - - // Short circuit when server side rendering - if (!isBrowser()) { - return isValidElement(trigger) ? trigger : null - } - - const unhandled = getUnhandledProps(Modal, this.props) - const portalPropNames = Portal.handledProps - - const rest = _.reduce( - unhandled, - (acc, val, key) => { - if (!_.includes(portalPropNames, key)) acc[key] = val + // Short circuit when server side rendering + if (!isBrowser()) { + return React.isValidElement(trigger) ? trigger : null + } - return acc - }, - {}, - ) - const portalProps = _.pick(unhandled, portalPropNames) - - // Heads up! - // - // The SUI CSS selector to prevent the modal itself from blurring requires an immediate .dimmer child: - // .blurring.dimmed.dimmable>:not(.dimmer) { ... } - // - // The .blurring.dimmed.dimmable is the body, so that all body content inside is blurred. - // We need the immediate child to be the dimmer to :not() blur the modal itself! - // Otherwise, the portal div is also blurred, blurring the modal. - // - // We cannot them wrap the modalJSX in an actual instead, we apply the dimmer classes to the . + const unhandled = getUnhandledProps(Modal, props) + const portalPropNames = Portal.handledProps - return ( - - - {ModalDimmer.create(_.isPlainObject(dimmer) ? dimmer : {}, { - autoGenerateKey: false, - defaultProps: { - blurring: dimmer === 'blurring', - inverted: dimmer === 'inverted', - }, - overrideProps: { - children: this.renderContent(rest), - centered, - mountNode, - scrolling, - }, - })} - - - ) - } -} + const rest = _.reduce( + unhandled, + (acc, val, key) => { + if (!_.includes(portalPropNames, key)) acc[key] = val + return acc + }, + {}, + ) + const portalProps = _.pick(unhandled, portalPropNames) + + // Heads up! + // + // The SUI CSS selector to prevent the modal itself from blurring requires an immediate .dimmer child: + // .blurring.dimmed.dimmable>:not(.dimmer) { ... } + // + // The .blurring.dimmed.dimmable is the body, so that all body content inside is blurred. + // We need the immediate child to be the dimmer to :not() blur the modal itself! + // Otherwise, the portal div is also blurred, blurring the modal. + // + // We cannot them wrap the modalJSX in an actual instead, we apply the dimmer classes to the . + + return ( + + {ModalDimmer.create(_.isPlainObject(dimmer) ? dimmer : {}, { + autoGenerateKey: false, + defaultProps: { + blurring: dimmer === 'blurring', + inverted: dimmer === 'inverted', + }, + overrideProps: { + children: renderContent(rest), + centered, + mountNode, + scrolling, + ref: dimmerRef, + }, + })} + + ) +}) + +Modal.displayName = 'Modal' Modal.propTypes = { /** An element type to render as (string or function). */ as: PropTypes.elementType, @@ -390,8 +406,6 @@ Modal.defaultProps = { eventPool: 'Modal', } -Modal.autoControlledProps = ['open'] - Modal.Actions = ModalActions Modal.Content = ModalContent Modal.Description = ModalDescription diff --git a/src/modules/Modal/ModalDimmer.js b/src/modules/Modal/ModalDimmer.js index 516d7b2bb6..7599a54583 100644 --- a/src/modules/Modal/ModalDimmer.js +++ b/src/modules/Modal/ModalDimmer.js @@ -1,4 +1,3 @@ -import { Ref } from '@fluentui/react-component-ref' import cx from 'clsx' import PropTypes from 'prop-types' import React from 'react' @@ -11,14 +10,15 @@ import { getUnhandledProps, useClassNamesOnNode, useKeyOnly, + useMergedRefs, } from '../../lib' /** * A modal has a dimmer. */ -function ModalDimmer(props) { +const ModalDimmer = React.forwardRef(function (props, ref) { const { blurring, children, className, centered, content, inverted, mountNode, scrolling } = props - const ref = React.useRef() + const elementRef = useMergedRefs(ref, React.useRef()) const classes = cx( 'ui', @@ -37,21 +37,19 @@ function ModalDimmer(props) { const ElementType = getElementType(ModalDimmer, props) useClassNamesOnNode(mountNode, bodyClasses) + React.useEffect(() => { - if (ref.current && ref.current.style) { - ref.current.style.setProperty('display', 'flex', 'important') - } + elementRef.current?.style?.setProperty('display', 'flex', 'important') }, []) return ( - - - {childrenUtils.isNil(children) ? content : children} - - + + {childrenUtils.isNil(children) ? content : children} + ) -} +}) +ModalDimmer.displayName = 'ModalDimmer' ModalDimmer.propTypes = { /** An element type to render as (string or function). */ as: PropTypes.elementType, diff --git a/src/modules/Popup/Popup.js b/src/modules/Popup/Popup.js index 4e5a405106..f6d1111ed1 100644 --- a/src/modules/Popup/Popup.js +++ b/src/modules/Popup/Popup.js @@ -2,12 +2,11 @@ import EventStack from '@semantic-ui-react/event-stack' import cx from 'clsx' import _ from 'lodash' import PropTypes from 'prop-types' -import React, { Component } from 'react' +import React from 'react' import { Popper } from 'react-popper' import shallowEqual from 'shallowequal' import { - eventStack, childrenUtils, createHTMLDivision, customPropTypes, @@ -17,6 +16,9 @@ import { SUI, useKeyOnly, useKeyOrValueAndKey, + useIsomorphicLayoutEffect, + useMergedRefs, + usePrevious, } from '../../lib' import Portal from '../../addons/Portal' import { placementMapping, positions, positionsMapping } from './lib/positions' @@ -27,142 +29,195 @@ import PopupHeader from './PopupHeader' const debug = makeDebugger('popup') /** - * A Popup displays additional information on top of a page. + * Calculates props specific for Portal component. + * + * @param {Object} props */ -export default class Popup extends Component { - state = {} - - open = false - zIndexWasSynced = false +function getPortalProps(props) { + const portalProps = {} + const normalizedOn = _.isArray(props.on) ? props.on : [props.on] - triggerRef = React.createRef() - elementRef = React.createRef() - - static getDerivedStateFromProps(props, state) { - if (state.closed || state.disabled) return {} + if (props.hoverable) { + portalProps.closeOnPortalMouseLeave = true + portalProps.mouseLeaveDelay = 300 + } - const unhandledProps = getUnhandledProps(Popup, props) - const contentRestProps = _.reduce( - unhandledProps, - (acc, val, key) => { - if (!_.includes(Portal.handledProps, key)) acc[key] = val + if (_.includes(normalizedOn, 'hover')) { + portalProps.openOnTriggerClick = false + portalProps.closeOnTriggerClick = false + portalProps.openOnTriggerMouseEnter = true + portalProps.closeOnTriggerMouseLeave = true + // Taken from SUI: https://git.io/vPmCm + portalProps.mouseLeaveDelay = 70 + portalProps.mouseEnterDelay = 50 + } - return acc - }, - {}, - ) - const portalRestProps = _.pick(unhandledProps, Portal.handledProps) + if (_.includes(normalizedOn, 'click')) { + portalProps.openOnTriggerClick = true + portalProps.closeOnTriggerClick = true + portalProps.closeOnDocumentClick = true + } - return { contentRestProps, portalRestProps } + if (_.includes(normalizedOn, 'focus')) { + portalProps.openOnTriggerFocus = true + portalProps.closeOnTriggerBlur = true } - componentDidUpdate(prevProps) { - const depsEqual = shallowEqual(this.props.popperDependencies, prevProps.popperDependencies) + return portalProps +} - if (!depsEqual) { - this.handleUpdate() - } +/** + * Splits props for Portal & Popup. + * + * @param {Object} unhandledProps + * @param {Boolean} closed + * @param {Boolean} disabled + */ +function partitionPortalProps(unhandledProps, closed, disabled) { + if (closed || disabled) { + return {} } - componentWillUnmount() { - clearTimeout(this.timeoutId) - } + const contentRestProps = _.reduce( + unhandledProps, + (acc, val, key) => { + if (!_.includes(Portal.handledProps, key)) acc[key] = val - getPortalProps = () => { - debug('getPortalProps()') - const portalProps = {} + return acc + }, + {}, + ) + const portalRestProps = _.pick(unhandledProps, Portal.handledProps) - const { on, hoverable } = this.props - const normalizedOn = _.isArray(on) ? on : [on] + return { contentRestProps, portalRestProps } +} - if (hoverable) { - portalProps.closeOnPortalMouseLeave = true - portalProps.mouseLeaveDelay = 300 - } - if (_.includes(normalizedOn, 'hover')) { - portalProps.openOnTriggerClick = false - portalProps.closeOnTriggerClick = false - portalProps.openOnTriggerMouseEnter = true - portalProps.closeOnTriggerMouseLeave = true - // Taken from SUI: https://git.io/vPmCm - portalProps.mouseLeaveDelay = 70 - portalProps.mouseEnterDelay = 50 - } - if (_.includes(normalizedOn, 'click')) { - portalProps.openOnTriggerClick = true - portalProps.closeOnTriggerClick = true - portalProps.closeOnDocumentClick = true +/** + * Performs updates when "popperDependencies" are not shallow equal. + * + * @param {Array} popperDependencies + * @param {React.Ref} positionUpdate + */ +function usePositioningEffect(popperDependencies, positionUpdate) { + const previousDependencies = usePrevious(popperDependencies) + + useIsomorphicLayoutEffect(() => { + if (positionUpdate.current) { + positionUpdate.current() } - if (_.includes(normalizedOn, 'focus')) { - portalProps.openOnTriggerFocus = true - portalProps.closeOnTriggerBlur = true + }, [shallowEqual(previousDependencies, popperDependencies)]) +} + +/** + * A Popup displays additional information on top of a page. + */ +const Popup = React.forwardRef(function (props, ref) { + const { + basic, + className, + content, + context, + children, + disabled, + eventsEnabled, + flowing, + header, + inverted, + offset, + pinned, + popper, + popperDependencies, + popperModifiers, + position, + positionFixed, + size, + style, + trigger, + wide, + } = props + + const [closed, setClosed] = React.useState(false) + + const unhandledProps = getUnhandledProps(Popup, props) + const { contentRestProps, portalRestProps } = partitionPortalProps( + unhandledProps, + closed, + disabled, + ) + + const elementRef = useMergedRefs(ref) + const positionUpdate = React.useRef() + const timeoutId = React.useRef() + const triggerRef = React.useRef() + const zIndexWasSynced = React.useRef(false) + + // ---------------------------------------- + // Effects + // ---------------------------------------- + + usePositioningEffect(popperDependencies, positionUpdate) + + React.useEffect(() => { + return () => { + clearTimeout(timeoutId.current) } + }, []) + + // ---------------------------------------- + // Handlers + // ---------------------------------------- + + const handleClose = (e) => { + debug('handleClose()') + _.invoke(props, 'onClose', e, { ...props, open: false }) + } - return portalProps + const handleOpen = (e) => { + debug('handleOpen()') + _.invoke(props, 'onOpen', e, { ...props, open: true }) } - hideOnScroll = (e) => { + const hideOnScroll = (e) => { + debug('hideOnScroll()') + // Do not hide the popup when scroll comes from inside the popup // /~https://github.com/Semantic-Org/Semantic-UI-React/issues/4305 - if (_.isElement(e.target) && this.elementRef.current.contains(e.target)) { + if (_.isElement(e.target) && elementRef.current.contains(e.target)) { return } - debug('hideOnScroll()') - this.setState({ closed: true }) + setClosed(true) - eventStack.unsub('scroll', this.hideOnScroll, { target: window }) - this.timeoutId = setTimeout(() => { - this.setState({ closed: false }) + timeoutId.current = setTimeout(() => { + setClosed(false) }, 50) - this.handleClose(e) + handleClose(e) } - handleClose = (e) => { - debug('handleClose()') - _.invoke(this.props, 'onClose', e, { ...this.props, open: false }) - } - - handleOpen = (e) => { - debug('handleOpen()') - _.invoke(this.props, 'onOpen', e, { ...this.props, open: true }) - } - - handlePortalMount = (e) => { + const handlePortalMount = (e) => { debug('handlePortalMount()') - _.invoke(this.props, 'onMount', e, this.props) + _.invoke(props, 'onMount', e, props) } - handlePortalUnmount = (e) => { + const handlePortalUnmount = (e) => { debug('handlePortalUnmount()') - this.positionUpdate = null - _.invoke(this.props, 'onUnmount', e, this.props) - } - - handleUpdate() { - if (this.positionUpdate) this.positionUpdate() + positionUpdate.current = null + _.invoke(props, 'onUnmount', e, props) } - renderContent = ({ placement: popperPlacement, ref: popperRef, update, style: popperStyle }) => { - const { - basic, - children, - className, - content, - hideOnScroll, - flowing, - header, - inverted, - popper, - size, - style, - wide, - } = this.props - const { contentRestProps } = this.state + // ---------------------------------------- + // Render + // ---------------------------------------- - this.positionUpdate = update + const renderBody = ({ + placement: popperPlacement, + ref: popperRef, + update, + style: popperStyle, + }) => { + positionUpdate.current = update const classes = cx( 'ui', @@ -175,7 +230,8 @@ export default class Popup extends Component { 'popup transition visible', className, ) - const ElementType = getElementType(Popup, this.props) + const ElementType = getElementType(Popup, props) + const styles = { // Heads up! We need default styles to get working correctly `flowing` left: 'auto', @@ -186,7 +242,7 @@ export default class Popup extends Component { } const innerElement = ( - + {childrenUtils.isNil(children) ? ( <> {PopupHeader.create(header, { autoGenerateKey: false })} @@ -195,7 +251,7 @@ export default class Popup extends Component { ) : ( children )} - {hideOnScroll && } + {hideOnScroll && } ) @@ -217,94 +273,79 @@ export default class Popup extends Component { }) } - render() { - const { - context, - disabled, - eventsEnabled, - offset, - pinned, - popper, - popperModifiers, - position, - positionFixed, - trigger, - } = this.props - const { closed, portalRestProps } = this.state - - if (closed || disabled) { - return trigger - } + if (closed || disabled) { + return trigger + } - const modifiers = [ - { name: 'arrow', enabled: false }, - { name: 'eventListeners', options: { scroll: !!eventsEnabled, resize: !!eventsEnabled } }, - { name: 'flip', enabled: !pinned }, - { name: 'preventOverflow', enabled: !!offset }, - { name: 'offset', enabled: !!offset, options: { offset } }, - ...popperModifiers, - - // We are syncing zIndex from `.ui.popup.content` to avoid layering issues as in SUIR we are using an additional - // `div` for Popper.js - // /~https://github.com/Semantic-Org/Semantic-UI-React/issues/4083 - { - name: 'syncZIndex', - enabled: true, - phase: 'beforeRead', - fn: ({ state }) => { - if (this.zIndexWasSynced) { - return - } - - // if zIndex defined in there is no sense to override it - const definedZIndex = popper?.style?.zIndex - - if (_.isUndefined(definedZIndex)) { - // eslint-disable-next-line no-param-reassign - state.elements.popper.style.zIndex = window.getComputedStyle( - state.elements.popper.firstChild, - ).zIndex - } - - this.zIndexWasSynced = true - }, - effect: () => { - return () => { - this.zIndexWasSynced = false - } - }, + const modifiers = [ + { name: 'arrow', enabled: false }, + { name: 'eventListeners', options: { scroll: !!eventsEnabled, resize: !!eventsEnabled } }, + { name: 'flip', enabled: !pinned }, + { name: 'preventOverflow', enabled: !!offset }, + { name: 'offset', enabled: !!offset, options: { offset } }, + ...popperModifiers, + + // We are syncing zIndex from `.ui.popup.content` to avoid layering issues as in SUIR we are using an additional + // `div` for Popper.js + // /~https://github.com/Semantic-Org/Semantic-UI-React/issues/4083 + { + name: 'syncZIndex', + enabled: true, + phase: 'beforeRead', + fn: ({ state }) => { + if (zIndexWasSynced.current) { + return + } + + // if zIndex defined in there is no sense to override it + const definedZIndex = popper?.style?.zIndex + + if (_.isUndefined(definedZIndex)) { + // eslint-disable-next-line no-param-reassign + state.elements.popper.style.zIndex = window.getComputedStyle( + state.elements.popper.firstChild, + ).zIndex + } + + zIndexWasSynced.current = true + }, + effect: () => { + return () => { + zIndexWasSynced.current = false + } }, - ] - debug('popper modifiers:', modifiers) - - const referenceElement = createReferenceProxy(_.isNil(context) ? this.triggerRef : context) - - const mergedPortalProps = { ...this.getPortalProps(), ...portalRestProps } - debug('portal props:', mergedPortalProps) - - return ( - + - - {this.renderContent} - - - ) - } -} + {renderBody} + + + ) +}) +Popup.displayName = 'Popup' Popup.propTypes = { /** An element type to render as (string or function). */ as: PropTypes.elementType, @@ -439,3 +480,5 @@ Popup.defaultProps = { Popup.Content = PopupContent Popup.Header = PopupHeader + +export default Popup diff --git a/test/specs/addons/Portal/Portal-test.js b/test/specs/addons/Portal/Portal-test.js index 383e8bbcfc..05cf68a1ae 100644 --- a/test/specs/addons/Portal/Portal-test.js +++ b/test/specs/addons/Portal/Portal-test.js @@ -1,6 +1,7 @@ import _ from 'lodash' import PropTypes from 'prop-types' import React from 'react' +import { act } from 'react-dom/test-utils' import * as common from 'test/specs/commonTests' import { domEvent, sandbox } from 'test/utils' @@ -147,7 +148,9 @@ describe('Portal', () => { ) wrapper.setProps({ open: false, children:

}) - wrapper.unmount() + act(() => { + wrapper.unmount() + }) onUnmount.should.have.been.calledOnce() }) @@ -159,33 +162,13 @@ describe('Portal', () => { , ) - wrapper.unmount() + act(() => { + wrapper.unmount() + }) onUnmount.should.have.been.calledOnce() }) }) - describe('portalNode', () => { - it('maintains ref to DOM node with host element', () => { - wrapperMount( - -

- , - ) - wrapper.instance().contentRef.current.tagName.should.equal('P') - }) - - it('maintains ref to DOM node with React component', () => { - const EmptyComponent = () =>

- - wrapperMount( - - - , - ) - wrapper.instance().contentRef.current.tagName.should.equal('P') - }) - }) - describe('onOpen', () => { it('is called on trigger click', () => { const onOpen = sandbox.spy() @@ -263,6 +246,25 @@ describe('Portal', () => { }) }) + describe('triggerRef', () => { + it('calls itself and an original ref', () => { + const elementRef = React.createRef() + const triggerRef = React.createRef() + + wrapperMount( + } triggerRef={triggerRef}> +

+ , + ) + const element = wrapper.getDOMNode() + + expect(element.tagName).to.equal('DIV') + + expect(elementRef.current).to.equal(element) + expect(triggerRef.current).to.equal(element) + }) + }) + describe('mountNode', () => { it('passed to PortalInner', () => { const mountNode = document.createElement('div') @@ -712,26 +714,4 @@ describe('Portal', () => { }, 0) }) }) - - describe('triggerRef', () => { - it('maintains ref on the trigger', () => { - const triggerRef = sandbox.spy() - const mountNode = document.createElement('div') - document.body.appendChild(mountNode) - - wrapperMount( - } triggerRef={triggerRef}> -

- , - { attachTo: mountNode }, - ) - const trigger = document.querySelector('#trigger') - - triggerRef.should.have.been.calledOnce() - triggerRef.should.have.been.calledWithMatch(trigger) - - wrapper.detach() - document.body.removeChild(mountNode) - }) - }) }) diff --git a/test/specs/addons/Portal/PortalInner-test.js b/test/specs/addons/Portal/PortalInner-test.js index e53c2724be..47af45d2e4 100644 --- a/test/specs/addons/Portal/PortalInner-test.js +++ b/test/specs/addons/Portal/PortalInner-test.js @@ -1,4 +1,5 @@ -import React, { createRef } from 'react' +import React from 'react' +import { act } from 'react-dom/test-utils' import PortalInner from 'src/addons/Portal/PortalInner' import { isBrowser } from 'src/lib' @@ -20,7 +21,7 @@ describe('PortalInner', () => { isBrowser.override = null }) - it('renders `null` when is SSR', () => { + it('renders `null` when during Server-Side Rendering', () => { mount(

@@ -29,16 +30,59 @@ describe('PortalInner', () => { }) }) - describe('innerRef', () => { - it('returns ref', () => { - const innerRef = createRef() + describe('ref', () => { + it('returns ref a DOM element', () => { + const portalRef = React.createRef() + const elementRef = React.createRef() + const wrapper = mount( - -

+ +

+ , + ) + const domNode = wrapper.getDOMNode() + + expect(elementRef.current).to.equal(domNode) + expect(portalRef.current).to.equal(domNode) + expect(domNode.tagName).to.equal('P') + }) + + it('returns ref a elements that uses ref forwarding', () => { + const CustomComponent = React.forwardRef((props, ref) => { + return

+ }) + + const portalRef = React.createRef() + const elementRef = React.createRef() + + const wrapper = mount( + + + , + ) + const domNode = wrapper.getDOMNode() + + expect(elementRef.current).to.equal(domNode) + expect(portalRef.current).to.equal(domNode) + expect(domNode.tagName).to.equal('P') + }) + + it('returns ref to a create element in other cases', () => { + function CustomComponent(props) { + return

+ } + + const portalRef = React.createRef() + const wrapper = mount( + + , ) + const domNode = wrapper.getDOMNode() - expect(wrapper.getDOMNode()).to.equal(innerRef.current) + expect(portalRef.current).to.equal(domNode) + expect(domNode.tagName).to.equal('DIV') + expect(domNode.dataset.suirPortal).to.equal('true') }) }) @@ -64,7 +108,9 @@ describe('PortalInner', () => { , ) - wrapper.unmount() + act(() => { + wrapper.unmount() + }) onUnmount.should.have.been.calledOnce() }) }) diff --git a/test/specs/modules/Modal/Modal-test.js b/test/specs/modules/Modal/Modal-test.js index c1da3c2dda..d1daa5674d 100644 --- a/test/specs/modules/Modal/Modal-test.js +++ b/test/specs/modules/Modal/Modal-test.js @@ -327,7 +327,7 @@ describe('Modal', () => { wrapper.find('#trigger').simulate('click') onOpen.should.have.been.calledOnce() - onOpen.should.have.been.calledWithMatch({}, { open: true }) + onOpen.should.have.been.calledWithMatch({ type: 'click' }, { open: true }) }) it('is not called on body click', () => { @@ -585,7 +585,11 @@ describe('Modal', () => { it('adds/removes the scrolling class to the body when the window grows/shrinks', (done) => { assertBodyClasses('scrolling', false) - wrapperMount(foo) + wrapperMount( + + + , + ) window.innerHeight = 10 assertWithTimeout( diff --git a/test/specs/modules/Popup/Popup-test.js b/test/specs/modules/Popup/Popup-test.js index 833c3df520..a5a9d96077 100644 --- a/test/specs/modules/Popup/Popup-test.js +++ b/test/specs/modules/Popup/Popup-test.js @@ -420,21 +420,22 @@ describe('Popup', () => { }) describe('popperDependencies', () => { - it('will call "scheduleUpdate" if dependencies changed', () => { - wrapperMount() - const scheduleUpdate = sandbox.spy(wrapper.instance(), 'handleUpdate') - - wrapper.setProps({ popperDependencies: [2, 3, 4] }) - scheduleUpdate.should.have.been.calledOnce() - }) - - it('will skip "scheduleUpdate" if dependencies are same', () => { - wrapperMount() - const scheduleUpdate = sandbox.spy(wrapper.instance(), 'handleUpdate') - - wrapper.setProps({ popperDependencies: [1, 2, 3] }) - scheduleUpdate.should.have.not.been.called() - }) + // TODO: find a way to implement these tests + // it('will call "scheduleUpdate" if dependencies changed', () => { + // wrapperMount() + // const scheduleUpdate = sandbox.spy(wrapper.instance(), 'handleUpdate') + // + // wrapper.setProps({ popperDependencies: [2, 3, 4] }) + // scheduleUpdate.should.have.been.calledOnce() + // }) + // + // it('will skip "scheduleUpdate" if dependencies are same', () => { + // wrapperMount() + // const scheduleUpdate = sandbox.spy(wrapper.instance(), 'handleUpdate') + // + // wrapper.setProps({ popperDependencies: [1, 2, 3] }) + // scheduleUpdate.should.have.not.been.called() + // }) }) describe('size', () => {