From 6d0e7b64ecffbdcca5639bd7692dd9fc4369a05a Mon Sep 17 00:00:00 2001 From: Gururaj J <89023023+Gururajj77@users.noreply.github.com> Date: Fri, 3 May 2024 01:34:02 +0530 Subject: [PATCH] refactor(overflowmenu/next): added typescript types to next/OverflowMenu component (#16280) * refactor(overflowmenu/next): added typescript types * refactor: changes for successful build --- .../src/components/OverflowMenu/next/index.js | 155 ------------- .../components/OverflowMenu/next/index.tsx | 208 ++++++++++++++++++ 2 files changed, 208 insertions(+), 155 deletions(-) delete mode 100644 packages/react/src/components/OverflowMenu/next/index.js create mode 100644 packages/react/src/components/OverflowMenu/next/index.tsx diff --git a/packages/react/src/components/OverflowMenu/next/index.js b/packages/react/src/components/OverflowMenu/next/index.js deleted file mode 100644 index d9abb0707de2..000000000000 --- a/packages/react/src/components/OverflowMenu/next/index.js +++ /dev/null @@ -1,155 +0,0 @@ -/** - * Copyright IBM Corp. 2020, 2023 - * - * This source code is licensed under the Apache-2.0 license found in the - * LICENSE file in the root directory of this source tree. - */ - -import React, { useRef } from 'react'; -import PropTypes from 'prop-types'; -import classNames from 'classnames'; -import { OverflowMenuVertical } from '@carbon/icons-react'; - -import { IconButton } from '../../IconButton'; -import { Menu } from '../../Menu'; - -import { useId } from '../../../internal/useId'; -import { usePrefix } from '../../../internal/usePrefix'; -import { useAttachedMenu } from '../../../internal/useAttachedMenu'; - -const defaultSize = 'md'; - -const OverflowMenu = React.forwardRef(function OverflowMenu( - { - children, - className, - label = 'Options', - renderIcon: IconElement = OverflowMenuVertical, - size = defaultSize, - menuAlignment = 'bottom-start', - tooltipAlignment, - ...rest - }, - forwardRef -) { - const id = useId('overflowmenu'); - const prefix = usePrefix(); - - const triggerRef = useRef(null); - const { - open, - x, - y, - handleClick: hookOnClick, - handleMousedown, - handleClose, - } = useAttachedMenu(triggerRef); - - function handleTriggerClick() { - if (triggerRef.current) { - hookOnClick(); - } - } - - const containerClasses = classNames( - className, - `${prefix}--overflow-menu__container` - ); - - const menuClasses = classNames(`${prefix}--overflow-menu__${menuAlignment}`); - - const triggerClasses = classNames( - `${prefix}--overflow-menu`, - { - [`${prefix}--overflow-menu--open`]: open, - }, - size !== defaultSize && `${prefix}--overflow-menu--${size}` - ); - - return ( -
- - - - - {children} - -
- ); -}); - -OverflowMenu.propTypes = { - /** - * A collection of MenuItems to be rendered within this OverflowMenu. - */ - children: PropTypes.node, - - /** - * Additional CSS class names for the trigger button. - */ - className: PropTypes.string, - - /** - * A label describing the options available. Is used in the trigger tooltip and as the menu's accessible label. - */ - label: PropTypes.string, - - /** - * Experimental property. Specify how the menu should align with the button element - */ - menuAlignment: PropTypes.oneOf([ - 'top-start', - 'top-end', - 'bottom-start', - 'bottom-end', - ]), - - /** - * Otionally provide a custom icon to be rendered on the trigger button. - */ - renderIcon: PropTypes.oneOfType([PropTypes.func, PropTypes.object]), - - /** - * Specify the size of the menu, from a list of available sizes. - */ - size: PropTypes.oneOf(['sm', 'md', 'lg']), - - /** - * Specify how the trigger tooltip should be aligned. - */ - tooltipAlignment: PropTypes.oneOf([ - 'top', - 'top-left', - 'top-right', - 'bottom', - 'bottom-left', - 'bottom-right', - 'left', - 'right', - ]), -}; - -export { OverflowMenu }; diff --git a/packages/react/src/components/OverflowMenu/next/index.tsx b/packages/react/src/components/OverflowMenu/next/index.tsx new file mode 100644 index 000000000000..73d70931d2d9 --- /dev/null +++ b/packages/react/src/components/OverflowMenu/next/index.tsx @@ -0,0 +1,208 @@ +/** + * Copyright IBM Corp. 2020, 2023 + * + * This source code is licensed under the Apache-2.0 license found in the + * LICENSE file in the root directory of this source tree. + */ + +import React, { + type ComponentType, + type FunctionComponent, + useRef, +} from 'react'; +import PropTypes from 'prop-types'; +import classNames from 'classnames'; +import { OverflowMenuVertical } from '@carbon/icons-react'; + +import { IconButton } from '../../IconButton'; +import { Menu } from '../../Menu'; + +import { useId } from '../../../internal/useId'; +import { usePrefix } from '../../../internal/usePrefix'; +import { useAttachedMenu } from '../../../internal/useAttachedMenu'; + +const defaultSize = 'md'; + +interface OverflowMenuProps { + /** + * A collection of MenuItems to be rendered within this OverflowMenu. + */ + children?: React.ReactNode; + + /** + * Additional CSS class names for the trigger button. + */ + className?: string; + + /** + * A label describing the options available. Is used in the trigger tooltip and as the menu's accessible label. + */ + label?: string; + + /** + * Experimental property. Specify how the menu should align with the button element + */ + menuAlignment?: 'top-start' | 'top-end' | 'bottom-start' | 'bottom-end'; + + /** + * Optionally provide a custom icon to be rendered on the trigger button. + */ + renderIcon?: ComponentType | FunctionComponent; + + /** + * Specify the size of the menu, from a list of available sizes. + */ + size?: 'sm' | 'md' | 'lg'; + + /** + * Specify how the trigger tooltip should be aligned. + */ + tooltipAlignment?: + | 'top' + | 'top-left' + | 'top-right' + | 'bottom' + | 'bottom-left' + | 'bottom-right' + | 'left' + | 'right'; +} + +const OverflowMenu = React.forwardRef( + function OverflowMenu( + { + children, + className, + label = 'Options', + renderIcon: IconElement = OverflowMenuVertical, + size = defaultSize, + menuAlignment = 'bottom-start', + tooltipAlignment, + ...rest + }, + forwardRef + ) { + const id = useId('overflowmenu'); + const prefix = usePrefix(); + + const triggerRef = useRef(null); + const { + open, + x, + y, + handleClick: hookOnClick, + handleMousedown, + handleClose, + } = useAttachedMenu(triggerRef); + + function handleTriggerClick() { + if (triggerRef.current) { + hookOnClick(); + } + } + + const containerClasses = classNames( + className, + `${prefix}--overflow-menu__container` + ); + + const menuClasses = classNames( + `${prefix}--overflow-menu__${menuAlignment}` + ); + + const triggerClasses = classNames( + `${prefix}--overflow-menu`, + { + [`${prefix}--overflow-menu--open`]: open, + }, + size !== defaultSize && `${prefix}--overflow-menu--${size}` + ); + + return ( +
+ + + + + {children} + +
+ ); + } +); +OverflowMenu.propTypes = { + /** + * A collection of MenuItems to be rendered within this OverflowMenu. + */ + children: PropTypes.node, + + /** + * Additional CSS class names for the trigger button. + */ + className: PropTypes.string, + + /** + * A label describing the options available. Is used in the trigger tooltip and as the menu's accessible label. + */ + label: PropTypes.string, + + /** + * Experimental property. Specify how the menu should align with the button element + */ + menuAlignment: PropTypes.oneOf([ + 'top-start', + 'top-end', + 'bottom-start', + 'bottom-end', + ]), + + /** + * Optionally provide a custom icon to be rendered on the trigger button. + */ + // @ts-expect-error: PropTypes are not expressive enough to cover this case + renderIcon: PropTypes.oneOfType([PropTypes.func, PropTypes.object]), + + /** + * Specify the size of the menu, from a list of available sizes. + */ + size: PropTypes.oneOf(['sm', 'md', 'lg']), + + /** + * Specify how the trigger tooltip should be aligned. + */ + tooltipAlignment: PropTypes.oneOf([ + 'top', + 'top-left', + 'top-right', + 'bottom', + 'bottom-left', + 'bottom-right', + 'left', + 'right', + ]), +}; + +export { OverflowMenu };