From bfaf7adab97e6979d52de7d7b46e11cb58b5acda Mon Sep 17 00:00:00 2001 From: Marco Ciampini Date: Tue, 19 Sep 2023 17:16:52 +0200 Subject: [PATCH 01/11] Basic dropdown implementation --- .../src/dropdown-menu-v2-ariakit/index.tsx | 65 ++++++++ .../stories/index.story.tsx | 72 +++++++++ .../src/dropdown-menu-v2-ariakit/styles.ts | 147 ++++++++++++++++++ 3 files changed, 284 insertions(+) create mode 100644 packages/components/src/dropdown-menu-v2-ariakit/index.tsx create mode 100644 packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx create mode 100644 packages/components/src/dropdown-menu-v2-ariakit/styles.ts diff --git a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx new file mode 100644 index 00000000000000..dbe0137e28cf65 --- /dev/null +++ b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx @@ -0,0 +1,65 @@ +/** + * External dependencies + */ +import * as Ariakit from '@ariakit/react'; + +/** + * WordPress dependencies + */ +import { forwardRef } from '@wordpress/element'; + +/** + * Internal dependencies + */ +import { + StyledAriakitMenu, + StyledAriakitMenuItem, + toggleButton, +} from './styles'; +import { useCx } from '../utils'; + +export interface DropdownMenuItemProps extends Ariakit.MenuItemProps {} + +export const DropdownMenuItem = forwardRef< + HTMLDivElement, + DropdownMenuItemProps +>( function DropdownMenuItem( props, ref ) { + return ; +} ); + +export interface DropdownMenuProps extends Ariakit.MenuButtonProps { + trigger: React.ReactNode; +} + +export const DropdownMenu = forwardRef< HTMLDivElement, DropdownMenuProps >( + function DropdownMenu( { trigger, children, ...props }, ref ) { + const cx = useCx(); + const menu = Ariakit.useMenuStore(); + + const menuButtonClassName = cx( + ! menu.parent && toggleButton, + props.className + ); + + return ( + + + ) : undefined + } + > + { trigger } + + + + { children } + + + ); + } +); diff --git a/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx b/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx new file mode 100644 index 00000000000000..3ae65695d03e29 --- /dev/null +++ b/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx @@ -0,0 +1,72 @@ +/** + * External dependencies + */ +import type { Meta, StoryFn } from '@storybook/react'; + +/** + * WordPress dependencies + */ +import { menu } from '@wordpress/icons'; + +/** + * Internal dependencies + */ +import { DropdownMenu, DropdownMenuItem } from '..'; +import Icon from '../../icon'; + +const meta: Meta< typeof DropdownMenu > = { + title: 'Components (Experimental)/DropdownMenu v2 ariakit', + component: DropdownMenu, + subcomponents: { + // @ts-expect-error - See https://github.com/storybookjs/storybook/issues/23170 + DropdownMenuItem, + }, + argTypes: { + children: { control: { type: null } }, + }, + parameters: { + actions: { argTypesRegex: '^on.*' }, + controls: { expanded: true }, + docs: { + canvas: { sourceState: 'shown' }, + source: { excludeDecorators: true }, + }, + }, + decorators: [ + // Layout wrapper + ( Story ) => ( +
+ +
+ ), + ], +}; +export default meta; + +const Template: StoryFn< typeof DropdownMenu > = ( props ) => ( + +); +export const Default = Template.bind( {} ); +Default.args = { + trigger: , + children: ( + <> + Undo + Redo + + Search the Web... + Find... + Find Next + Find Previous + + + Start Speaking + Stop Speaking + + + ), +}; diff --git a/packages/components/src/dropdown-menu-v2-ariakit/styles.ts b/packages/components/src/dropdown-menu-v2-ariakit/styles.ts new file mode 100644 index 00000000000000..0867678203b8c0 --- /dev/null +++ b/packages/components/src/dropdown-menu-v2-ariakit/styles.ts @@ -0,0 +1,147 @@ +/** + * External dependencies + */ +import * as Ariakit from '@ariakit/react'; +import styled from '@emotion/styled'; +import { css } from '@emotion/react'; + +export const toggleButton = css` + display: flex; + height: 2.5rem; + touch-action: none; + user-select: none; + align-items: center; + justify-content: center; + gap: 0.25rem; + white-space: nowrap; + border-radius: 0.5rem; + border-style: none; + background-color: hsl( 204 20% 100% ); + padding-left: 1rem; + padding-right: 1rem; + font-size: 1rem; + line-height: 1.5rem; + color: hsl( 204 10% 10% ); + text-decoration-line: none; + outline-width: 2px; + outline-offset: 2px; + outline-color: hsl( 204 100% 40% ); + box-shadow: inset 0 0 0 1px rgba( 0, 0, 0, 0.1 ), + inset 0 -1px 0 rgba( 0, 0, 0, 0.1 ), 0 1px 1px rgba( 0, 0, 0, 0.1 ); + font-weight: 500; + + &:hover { + background-color: hsl( 204 20% 96% ); + } + + &[aria-disabled='true'] { + opacity: 0.5; + } + + &[aria-expanded='true'] { + background-color: hsl( 204 20% 96% ); + } + + &[data-focus-visible] { + outline-style: solid; + } + + &:active, + &[data-active] { + transform: scale( 0.98 ); + } + + &:active[aria-expanded='true'], + &[data-active][aria-expanded='true'] { + transform: scale( 1 ); + } + + @media ( min-width: 640px ) { + gap: 0.5rem; + } +`; + +// :is(.dark .button) { +// background-color: hsl(204 20% 100% / 0.05); +// color: hsl(204 20% 100%); +// box-shadow: +// inset 0 0 0 1px rgba(255, 255, 255, 0.1), +// inset 0 -1px 0 1px rgba(0, 0, 0, 0.2), +// inset 0 1px 0 rgba(255, 255, 255, 0.05); +// } + +// :is(.dark .button:hover) { +// background-color: hsl(204 20% 100% / 0.1); +// } + +// :is(.dark .button)[aria-expanded="true"] { +// background-color: hsl(204 20% 100% / 0.1); +// } + +export const StyledAriakitMenu = styled( Ariakit.Menu )` + position: relative; + z-index: 50; + display: flex; + max-height: var( --popover-available-height ); + min-width: 180px; + flex-direction: column; + overscroll-behavior: contain; + border-radius: 0.5rem; + border-width: 1px; + border-style: solid; + border-color: hsl( 204 20% 88% ); + background-color: hsl( 204 20% 100% ); + padding: 0.5rem; + color: hsl( 204 10% 10% ); + box-shadow: 0 10px 15px -3px rgb( 0 0 0 / 0.1 ), + 0 4px 6px -4px rgb( 0 0 0 / 0.1 ); + outline: none !important; + overflow: visible; +`; + +// :is(.dark .menu) { +// border-color: hsl(204 3% 26%); +// background-color: hsl(204 3% 18%); +// color: hsl(204 20% 100%); +// box-shadow: +// 0 10px 15px -3px rgb(0 0 0 / 0.25), +// 0 4px 6px -4px rgb(0 0 0 / 0.1); +// } + +export const StyledAriakitMenuItem = styled( Ariakit.MenuItem )` + display: flex; + cursor: default; + scroll-margin: 0.5rem; + align-items: center; + gap: 0.5rem; + border-radius: 0.25rem; + padding: 0.5rem; + outline: none !important; + + &[aria-disabled='true'] { + opacity: 0.25; + } + + &[data-active-item] { + background-color: hsl( 204 100% 40% ); + color: hsl( 204 20% 100% ); + } + + &:active, + &[data-active] { + background-color: hsl( 204 100% 32% ); + } + + ${ StyledAriakitMenu }:not(:focus) &:not(:focus)[aria-expanded="true"] { + background-color: hsl( 204 10% 10% / 0.1 ); + color: currentColor; + } +`; + +// :is(.dark .menu:not(:focus) .menu-item:not(:focus)[aria-expanded="true"]) { +// background-color: hsl(204 20% 100% / 0.1); +// } + +export const StyledMenuButtonLabel = styled.span` + flex: 1 1 0%; +`; From a985cf4efe414e5172bb4c60c89e17a5993214ce Mon Sep 17 00:00:00 2001 From: Marco Ciampini Date: Tue, 19 Sep 2023 17:48:52 +0200 Subject: [PATCH 02/11] Export new dropdown components as private APIs via lock/unlock --- packages/components/src/dropdown-menu/index.tsx | 2 +- packages/components/src/private-apis.ts | 6 ++++++ 2 files changed, 7 insertions(+), 1 deletion(-) diff --git a/packages/components/src/dropdown-menu/index.tsx b/packages/components/src/dropdown-menu/index.tsx index 9105555927f47c..b5ccd92d68b907 100644 --- a/packages/components/src/dropdown-menu/index.tsx +++ b/packages/components/src/dropdown-menu/index.tsx @@ -22,7 +22,7 @@ import type { } from './types'; function mergeProps< - T extends { className?: string; [ key: string ]: unknown }, + T extends { className?: string; [ key: string ]: unknown } >( defaultProps: Partial< T > = {}, props: T = {} as T ) { const mergedProps: T = { ...defaultProps, diff --git a/packages/components/src/private-apis.ts b/packages/components/src/private-apis.ts index 6e17abde0c627e..966be9887a30a3 100644 --- a/packages/components/src/private-apis.ts +++ b/packages/components/src/private-apis.ts @@ -22,6 +22,10 @@ import { DropdownSubMenu as DropdownSubMenuV2, DropdownSubMenuTrigger as DropdownSubMenuTriggerV2, } from './dropdown-menu-v2'; +import { + DropdownMenu as DropdownMenuV2Ariakit, + DropdownMenuItem as DropdownMenuItemV2Ariakit, +} from './dropdown-menu-v2-ariakit'; import { ComponentsContext } from './ui/context/context-system-provider'; import Theme from './theme'; @@ -49,4 +53,6 @@ lock( privateApis, { DropdownSubMenuTriggerV2, ProgressBar, Theme, + DropdownMenuV2Ariakit, + DropdownMenuItemV2Ariakit, } ); From 19194e305e63084ce9eeee94ea5532e89380cf55 Mon Sep 17 00:00:00 2001 From: Marco Ciampini Date: Tue, 19 Sep 2023 21:26:58 +0200 Subject: [PATCH 03/11] add modal storybook example --- .../stories/index.story.tsx | 37 +++++++++++++++++++ 1 file changed, 37 insertions(+) diff --git a/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx b/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx index 3ae65695d03e29..7b8085b22f0953 100644 --- a/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx +++ b/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx @@ -7,12 +7,14 @@ import type { Meta, StoryFn } from '@storybook/react'; * WordPress dependencies */ import { menu } from '@wordpress/icons'; +import { useState } from '@wordpress/element'; /** * Internal dependencies */ import { DropdownMenu, DropdownMenuItem } from '..'; import Icon from '../../icon'; +import Modal from '../../modal'; const meta: Meta< typeof DropdownMenu > = { title: 'Components (Experimental)/DropdownMenu v2 ariakit', @@ -70,3 +72,38 @@ Default.args = { ), }; + +export const WithModal: StoryFn< typeof DropdownMenu > = ( props ) => { + const [ isModalOpen, setModalOpen ] = useState( false ); + return ( + <> + + setModalOpen( true ) }> + Open modal + + Redo + + Search the Web... + Find... + Find Next + Find Previous + + + Start Speaking + Stop Speaking + + + { isModalOpen && ( + setModalOpen( false ) }> + Yo! + + + ) } + + ); +}; +WithModal.args = { + trigger: , +}; From a101e42d92806b4e97700bcf120a2647a83d7381 Mon Sep 17 00:00:00 2001 From: Marco Ciampini Date: Tue, 19 Sep 2023 21:28:10 +0200 Subject: [PATCH 04/11] Make dropdownmenu modal --- packages/components/src/dropdown-menu-v2-ariakit/index.tsx | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx index dbe0137e28cf65..b332c7b2082aca 100644 --- a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx +++ b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx @@ -56,7 +56,11 @@ export const DropdownMenu = forwardRef< HTMLDivElement, DropdownMenuProps >( { trigger } - + { children } From 378f98010d6100ca6e4450f6a2cabacbf48a0480 Mon Sep 17 00:00:00 2001 From: Marco Ciampini Date: Tue, 19 Sep 2023 22:15:15 +0200 Subject: [PATCH 05/11] WIP --- .../block-settings-menu-controls/index.js | 12 +- .../block-settings-dropdown.js | 391 ++++++++++-------- .../src/dropdown-menu-v2-ariakit/index.tsx | 4 +- .../stories/index.story.tsx | 112 ++++- .../src/components/pattern-convert-button.js | 17 +- 5 files changed, 343 insertions(+), 193 deletions(-) diff --git a/packages/block-editor/src/components/block-settings-menu-controls/index.js b/packages/block-editor/src/components/block-settings-menu-controls/index.js index 53b3835fad1a1b..cb4be30d3b2869 100644 --- a/packages/block-editor/src/components/block-settings-menu-controls/index.js +++ b/packages/block-editor/src/components/block-settings-menu-controls/index.js @@ -72,8 +72,8 @@ const BlockSettingsMenuControlsSlot = ( { } return ( - - { showConvertToGroupButton && ( + <> + { /* { showConvertToGroupButton && ( - ) } + ) } */ } { fills } - { fillProps?.canMove && ! fillProps?.onlyBlock && ( + { /* { fillProps?.canMove && ! fillProps?.onlyBlock && ( - ) } - + ) } */ } + ); } } diff --git a/packages/block-editor/src/components/block-settings-menu/block-settings-dropdown.js b/packages/block-editor/src/components/block-settings-menu/block-settings-dropdown.js index bb8e00d0c11db9..df18990b8e464c 100644 --- a/packages/block-editor/src/components/block-settings-menu/block-settings-dropdown.js +++ b/packages/block-editor/src/components/block-settings-menu/block-settings-dropdown.js @@ -6,7 +6,11 @@ import { serialize, store as blocksStore, } from '@wordpress/blocks'; -import { DropdownMenu, MenuGroup, MenuItem } from '@wordpress/components'; +import { + // DropdownMenu, MenuGroup, MenuItem + Icon, + privateApis as componentsPrivateApis, +} from '@wordpress/components'; import { useDispatch, useSelect } from '@wordpress/data'; import { moreVertical } from '@wordpress/icons'; import { @@ -34,6 +38,10 @@ import { store as blockEditorStore } from '../../store'; import { unlock } from '../../lock-unlock'; import { useShowHoveredOrFocusedGestures } from '../block-toolbar/utils'; +const { DropdownMenuV2Ariakit, DropdownMenuItemV2Ariakit } = unlock( + componentsPrivateApis +); + const POPOVER_PROPS = { className: 'block-editor-block-settings-menu__popover', placement: 'bottom-start', @@ -44,7 +52,11 @@ function CopyMenuItem( { blocks, onCopy, label } ) { const copyMenuItemBlocksLabel = blocks.length > 1 ? __( 'Copy blocks' ) : __( 'Copy' ); const copyMenuItemLabel = label ? label : copyMenuItemBlocksLabel; - return { copyMenuItemLabel }; + return ( + + { copyMenuItemLabel } + + ); } export function BlockSettingsDropdown( { @@ -234,153 +246,28 @@ export function BlockSettingsDropdown( { onMoveTo, blocks, } ) => ( - } > - { ( { onClose } ) => ( + + Hello Brooke + + { canCopyStyles && ( <> - - <__unstableBlockSettingsMenuFirstItem.Slot - fillProps={ { onClose } } - /> - { ! parentBlockIsSelected && - !! firstParentClientId && ( - - } - onClick={ () => - selectBlock( - firstParentClientId - ) - } - > - { sprintf( - /* translators: %s: Name of the block's parent. */ - __( - 'Select parent block (%s)' - ), - parentBlockType.title - ) } - - ) } - { count === 1 && ( - - ) } - - { canDuplicate && ( - - { __( 'Duplicate' ) } - - ) } - { canInsertDefaultBlock && ( - <> - - { __( 'Add before' ) } - - - { __( 'Add after' ) } - - - ) } - - { canCopyStyles && ( - - - - { __( 'Paste styles' ) } - - - ) } + + + { __( 'Paste styles' ) } + + - { typeof children === 'function' - ? children( { onClose } ) - : Children.map( ( child ) => - cloneElement( child, { onClose } ) - ) } - { canRemove && ( - - - { removeBlockLabel } - - - ) } + + + } + hideOnClick={ false } + > + { __( 'I am a link' ) } + ) } - + + // + // { ( { onClose } ) => ( + // <> + // + // <__unstableBlockSettingsMenuFirstItem.Slot + // fillProps={ { onClose } } + // /> + // { ! parentBlockIsSelected && + // !! firstParentClientId && ( + // + // } + // onClick={ () => + // selectBlock( + // firstParentClientId + // ) + // } + // > + // { sprintf( + // /* translators: %s: Name of the block's parent. */ + // __( + // 'Select parent block (%s)' + // ), + // parentBlockType.title + // ) } + // + // ) } + // { count === 1 && ( + // + // ) } + // + // { canDuplicate && ( + // + // { __( 'Duplicate' ) } + // + // ) } + // { canInsertDefaultBlock && ( + // <> + // + // { __( 'Add before' ) } + // + // + // { __( 'Add after' ) } + // + // + // ) } + // + // { canCopyStyles && ( + // + // + // + // { __( 'Paste styles' ) } + // + // + // ) } + // + // { typeof children === 'function' + // ? children( { onClose } ) + // : Children.map( ( child ) => + // cloneElement( child, { onClose } ) + // ) } + // { canRemove && ( + // + // + // { removeBlockLabel } + // + // + // ) } + // + // ) } + // ) } ); diff --git a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx index b332c7b2082aca..768cc8fa26a5be 100644 --- a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx +++ b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx @@ -1,6 +1,7 @@ /** * External dependencies */ +// eslint-disable-next-line no-restricted-imports import * as Ariakit from '@ariakit/react'; /** @@ -24,7 +25,8 @@ export const DropdownMenuItem = forwardRef< HTMLDivElement, DropdownMenuItemProps >( function DropdownMenuItem( props, ref ) { - return ; + const store = Ariakit.useMenuContext(); + return ; } ); export interface DropdownMenuProps extends Ariakit.MenuButtonProps { diff --git a/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx b/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx index 7b8085b22f0953..16580d03cb03ac 100644 --- a/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx +++ b/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx @@ -1,13 +1,15 @@ /** * External dependencies */ +// eslint-disable-next-line no-restricted-imports +import * as Ariakit from '@ariakit/react'; import type { Meta, StoryFn } from '@storybook/react'; /** * WordPress dependencies */ import { menu } from '@wordpress/icons'; -import { useState } from '@wordpress/element'; +import { useState, useMemo } from '@wordpress/element'; /** * Internal dependencies @@ -15,6 +17,7 @@ import { useState } from '@wordpress/element'; import { DropdownMenu, DropdownMenuItem } from '..'; import Icon from '../../icon'; import Modal from '../../modal'; +import { createSlotFill, Provider as SlotFillProvider } from '../../slot-fill'; const meta: Meta< typeof DropdownMenu > = { title: 'Components (Experimental)/DropdownMenu v2 ariakit', @@ -73,7 +76,9 @@ Default.args = { ), }; -export const WithModal: StoryFn< typeof DropdownMenu > = ( props ) => { +export const WithModalAsSiblingOfMenu: StoryFn< typeof DropdownMenu > = ( + props +) => { const [ isModalOpen, setModalOpen ] = useState( false ); return ( <> @@ -81,21 +86,10 @@ export const WithModal: StoryFn< typeof DropdownMenu > = ( props ) => { setModalOpen( true ) }> Open modal - Redo - - Search the Web... - Find... - Find Next - Find Previous - - - Start Speaking - Stop Speaking - { isModalOpen && ( setModalOpen( false ) }> - Yo! + Modal's contents @@ -104,6 +98,94 @@ export const WithModal: StoryFn< typeof DropdownMenu > = ( props ) => { ); }; -WithModal.args = { +WithModalAsSiblingOfMenu.args = { + trigger: , +}; + +export const WithModalAsSiblingOfMenuItem: StoryFn< typeof DropdownMenu > = ( + props +) => { + const [ isModalOpen, setModalOpen ] = useState( false ); + return ( + + setModalOpen( true ) }> + Open modal + + { isModalOpen && ( + setModalOpen( false ) }> + Yo! + + + ) } + + ); +}; +WithModalAsSiblingOfMenuItem.args = { + trigger: , +}; + +const ExampleSlotFill = createSlotFill( 'Example' ); + +const Slot = () => { + const ariakitMenuStore = Ariakit.useMenuContext(); + + // Forwarding the content of the slot so that it can be used by the fill + const fillProps = useMemo( + () => ( { + forwardedContext: [ + [ Ariakit.MenuProvider, { store: ariakitMenuStore } ], + ], + } ), + [ ariakitMenuStore ] + ); + + return ; +}; + +type ForwardedContextTuple< P = {} > = [ + React.ComponentType< React.PropsWithChildren< P > >, + P, +]; + +const Fill = ( { children }: { children: React.ReactNode } ) => { + const innerMarkup = <>{ children }; + + return ( + + { ( fillProps: { forwardedContext?: ForwardedContextTuple[] } ) => { + const { forwardedContext = [] } = fillProps; + + return forwardedContext.reduce( + ( inner: JSX.Element, [ Provider, props ] ) => ( + { inner } + ), + innerMarkup + ); + } } + + ); +}; + +export const AddItemsViaSlotFill: StoryFn< typeof DropdownMenu > = ( + props +) => { + return ( + + + Item + + + + + + Item from fill + + + + ); +}; +AddItemsViaSlotFill.args = { trigger: , }; diff --git a/packages/patterns/src/components/pattern-convert-button.js b/packages/patterns/src/components/pattern-convert-button.js index 002dbbd8c0181d..f6177e3ec15992 100644 --- a/packages/patterns/src/components/pattern-convert-button.js +++ b/packages/patterns/src/components/pattern-convert-button.js @@ -9,8 +9,8 @@ import { } from '@wordpress/blocks'; import { store as blockEditorStore } from '@wordpress/block-editor'; import { useState, useCallback } from '@wordpress/element'; -import { MenuItem } from '@wordpress/components'; -import { symbol } from '@wordpress/icons'; +import { privateApis as componentsPrivateApis } from '@wordpress/components'; +// import { symbol } from '@wordpress/icons'; import { useSelect, useDispatch } from '@wordpress/data'; import { store as coreStore } from '@wordpress/core-data'; import { __, sprintf } from '@wordpress/i18n'; @@ -23,6 +23,8 @@ import CreatePatternModal from './create-pattern-modal'; import { unlock } from '../lock-unlock'; import { PATTERN_SYNC_TYPES } from '../constants'; +const { DropdownMenuItemV2Ariakit } = unlock( componentsPrivateApis ); + /** * Menu control to convert block(s) to a pattern block. * @@ -127,14 +129,17 @@ export default function PatternConvertButton( { clientIds, rootClientId } ) { }; return ( <> - setIsModalOpen( true ) } + { + setIsModalOpen( true ); + event.preventDefault(); + } } aria-expanded={ isModalOpen } aria-haspopup="dialog" > { __( 'Create pattern' ) } - + { isModalOpen && ( Date: Mon, 25 Sep 2023 17:53:38 +0200 Subject: [PATCH 06/11] Format --- .../components/src/dropdown-menu-v2-ariakit/styles.ts | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/packages/components/src/dropdown-menu-v2-ariakit/styles.ts b/packages/components/src/dropdown-menu-v2-ariakit/styles.ts index 0867678203b8c0..71dc6dc0fc563f 100644 --- a/packages/components/src/dropdown-menu-v2-ariakit/styles.ts +++ b/packages/components/src/dropdown-menu-v2-ariakit/styles.ts @@ -1,6 +1,7 @@ /** * External dependencies */ +// eslint-disable-next-line no-restricted-imports import * as Ariakit from '@ariakit/react'; import styled from '@emotion/styled'; import { css } from '@emotion/react'; @@ -26,8 +27,10 @@ export const toggleButton = css` outline-width: 2px; outline-offset: 2px; outline-color: hsl( 204 100% 40% ); - box-shadow: inset 0 0 0 1px rgba( 0, 0, 0, 0.1 ), - inset 0 -1px 0 rgba( 0, 0, 0, 0.1 ), 0 1px 1px rgba( 0, 0, 0, 0.1 ); + box-shadow: + inset 0 0 0 1px rgba( 0, 0, 0, 0.1 ), + inset 0 -1px 0 rgba( 0, 0, 0, 0.1 ), + 0 1px 1px rgba( 0, 0, 0, 0.1 ); font-weight: 500; &:hover { @@ -93,7 +96,8 @@ export const StyledAriakitMenu = styled( Ariakit.Menu )` background-color: hsl( 204 20% 100% ); padding: 0.5rem; color: hsl( 204 10% 10% ); - box-shadow: 0 10px 15px -3px rgb( 0 0 0 / 0.1 ), + box-shadow: + 0 10px 15px -3px rgb( 0 0 0 / 0.1 ), 0 4px 6px -4px rgb( 0 0 0 / 0.1 ); outline: none !important; overflow: visible; From 19192df3ad366485e427b36c731a3a6381f66fd4 Mon Sep 17 00:00:00 2001 From: Marco Ciampini Date: Mon, 25 Sep 2023 17:54:40 +0200 Subject: [PATCH 07/11] Fix modal closing prematurely --- packages/components/src/dropdown-menu-v2-ariakit/index.tsx | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx index 768cc8fa26a5be..f921b2fb7e1f56 100644 --- a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx +++ b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx @@ -61,6 +61,13 @@ export const DropdownMenu = forwardRef< HTMLDivElement, DropdownMenuProps >( + Array.from( + document.querySelectorAll( + '.components-modal__screen-overlay' + ) + ) + } modal > { children } From b4067db7a5a5a8b4812ae67aaa12a74d1cd31950 Mon Sep 17 00:00:00 2001 From: Marco Ciampini Date: Mon, 25 Sep 2023 17:55:12 +0200 Subject: [PATCH 08/11] Use custom context to pass menu store around, add nested menu to slotfill example --- .../src/dropdown-menu-v2-ariakit/index.tsx | 63 ++++++++++++++----- .../stories/index.story.tsx | 29 ++++++--- 2 files changed, 68 insertions(+), 24 deletions(-) diff --git a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx index f921b2fb7e1f56..4e66519ee15310 100644 --- a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx +++ b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx @@ -7,7 +7,12 @@ import * as Ariakit from '@ariakit/react'; /** * WordPress dependencies */ -import { forwardRef } from '@wordpress/element'; +import { + forwardRef, + createContext, + useContext, + useMemo, +} from '@wordpress/element'; /** * Internal dependencies @@ -19,38 +24,61 @@ import { } from './styles'; import { useCx } from '../utils'; -export interface DropdownMenuItemProps extends Ariakit.MenuItemProps {} +export const DropdownMenuContext = createContext< + { store: Ariakit.MenuStore } | undefined +>( undefined ); + +export interface DropdownMenuItemProps + extends Omit< Ariakit.MenuItemProps, 'store' > {} export const DropdownMenuItem = forwardRef< HTMLDivElement, DropdownMenuItemProps >( function DropdownMenuItem( props, ref ) { - const store = Ariakit.useMenuContext(); - return ; + const dropdownMenuContext = useContext( DropdownMenuContext ); + return ( + + ); } ); export interface DropdownMenuProps extends Ariakit.MenuButtonProps { trigger: React.ReactNode; + children?: React.ReactNode; } export const DropdownMenu = forwardRef< HTMLDivElement, DropdownMenuProps >( - function DropdownMenu( { trigger, children, ...props }, ref ) { + function DropdownMenu( { trigger, children, className, ...props }, ref ) { + const parentContext = useContext( DropdownMenuContext ); + + const dropdownMenuStore = Ariakit.useMenuStore( { + parent: parentContext?.store, + } ); + const cx = useCx(); - const menu = Ariakit.useMenuStore(); + const menuButtonClassName = useMemo( + () => cx( ! dropdownMenuStore.parent && toggleButton, className ), + [ cx, dropdownMenuStore.parent, className ] + ); - const menuButtonClassName = cx( - ! menu.parent && toggleButton, - props.className + const contextValue = useMemo( + () => ( { store: dropdownMenuStore } ), + [ dropdownMenuStore ] ); return ( - + <> + { /* Menu trigger */ } ) : undefined } @@ -58,9 +86,12 @@ export const DropdownMenu = forwardRef< HTMLDivElement, DropdownMenuProps >( { trigger } + + { /* Menu popover */ } Array.from( document.querySelectorAll( @@ -70,9 +101,11 @@ export const DropdownMenu = forwardRef< HTMLDivElement, DropdownMenuProps >( } modal > - { children } + + { children } + - + ); } ); diff --git a/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx b/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx index 16580d03cb03ac..c3e9d479384ee2 100644 --- a/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx +++ b/packages/components/src/dropdown-menu-v2-ariakit/stories/index.story.tsx @@ -1,20 +1,18 @@ /** * External dependencies */ -// eslint-disable-next-line no-restricted-imports -import * as Ariakit from '@ariakit/react'; import type { Meta, StoryFn } from '@storybook/react'; /** * WordPress dependencies */ import { menu } from '@wordpress/icons'; -import { useState, useMemo } from '@wordpress/element'; +import { useState, useMemo, useContext } from '@wordpress/element'; /** * Internal dependencies */ -import { DropdownMenu, DropdownMenuItem } from '..'; +import { DropdownMenu, DropdownMenuItem, DropdownMenuContext } from '..'; import Icon from '../../icon'; import Modal from '../../modal'; import { createSlotFill, Provider as SlotFillProvider } from '../../slot-fill'; @@ -108,7 +106,10 @@ export const WithModalAsSiblingOfMenuItem: StoryFn< typeof DropdownMenu > = ( const [ isModalOpen, setModalOpen ] = useState( false ); return ( - setModalOpen( true ) }> + setModalOpen( true ) } + > Open modal { isModalOpen && ( @@ -129,16 +130,19 @@ WithModalAsSiblingOfMenuItem.args = { const ExampleSlotFill = createSlotFill( 'Example' ); const Slot = () => { - const ariakitMenuStore = Ariakit.useMenuContext(); + const dropdownMenuContext = useContext( DropdownMenuContext ); // Forwarding the content of the slot so that it can be used by the fill const fillProps = useMemo( () => ( { forwardedContext: [ - [ Ariakit.MenuProvider, { store: ariakitMenuStore } ], + [ + DropdownMenuContext.Provider, + { value: dropdownMenuContext }, + ], ], } ), - [ ariakitMenuStore ] + [ dropdownMenuContext ] ); return ; @@ -175,13 +179,20 @@ export const AddItemsViaSlotFill: StoryFn< typeof DropdownMenu > = ( Item - + + + Item from fill + + + Test + + ); From be13403d6a85aa2b931cc0e8b3de2fb7a57840c8 Mon Sep 17 00:00:00 2001 From: Marco Ciampini Date: Tue, 26 Sep 2023 11:35:27 +0200 Subject: [PATCH 09/11] Remove temporary modal fix --- packages/components/src/dropdown-menu-v2-ariakit/index.tsx | 7 ------- 1 file changed, 7 deletions(-) diff --git a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx index 4e66519ee15310..6bf250dbe9a140 100644 --- a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx +++ b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx @@ -92,13 +92,6 @@ export const DropdownMenu = forwardRef< HTMLDivElement, DropdownMenuProps >( store={ dropdownMenuStore } gutter={ dropdownMenuStore.parent ? 16 : 8 } shift={ dropdownMenuStore.parent ? -9 : 0 } - getPersistentElements={ () => - Array.from( - document.querySelectorAll( - '.components-modal__screen-overlay' - ) - ) - } modal > From a41674d54fa578c84df1f5c9dbf82ce63633877a Mon Sep 17 00:00:00 2001 From: Marco Ciampini Date: Tue, 26 Sep 2023 12:06:18 +0200 Subject: [PATCH 10/11] Try nested dropdown menu in slotfill --- .../block-settings-menu/block-settings-dropdown.js | 2 +- .../src/components/pattern-convert-button.js | 14 +++++++------- 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/packages/block-editor/src/components/block-settings-menu/block-settings-dropdown.js b/packages/block-editor/src/components/block-settings-menu/block-settings-dropdown.js index df18990b8e464c..bdc7d50efa8e49 100644 --- a/packages/block-editor/src/components/block-settings-menu/block-settings-dropdown.js +++ b/packages/block-editor/src/components/block-settings-menu/block-settings-dropdown.js @@ -250,7 +250,7 @@ export function BlockSettingsDropdown( { trigger={ } > - Hello Brooke + Test item { canCopyStyles && ( <> diff --git a/packages/patterns/src/components/pattern-convert-button.js b/packages/patterns/src/components/pattern-convert-button.js index f6177e3ec15992..9356eddbfc7b6c 100644 --- a/packages/patterns/src/components/pattern-convert-button.js +++ b/packages/patterns/src/components/pattern-convert-button.js @@ -23,7 +23,9 @@ import CreatePatternModal from './create-pattern-modal'; import { unlock } from '../lock-unlock'; import { PATTERN_SYNC_TYPES } from '../constants'; -const { DropdownMenuItemV2Ariakit } = unlock( componentsPrivateApis ); +const { DropdownMenuV2Ariakit, DropdownMenuItemV2Ariakit } = unlock( + componentsPrivateApis +); /** * Menu control to convert block(s) to a pattern block. @@ -128,13 +130,11 @@ export default function PatternConvertButton( { clientIds, rootClientId } ) { setIsModalOpen( false ); }; return ( - <> + { - setIsModalOpen( true ); - event.preventDefault(); - } } + onClick={ () => setIsModalOpen( true ) } + hideOnClick={ false } aria-expanded={ isModalOpen } aria-haspopup="dialog" > @@ -154,6 +154,6 @@ export default function PatternConvertButton( { clientIds, rootClientId } ) { } } /> ) } - + ); } From 967bb1314059e37ab57b4873c06e0140de8452a0 Mon Sep 17 00:00:00 2001 From: Marco Ciampini Date: Tue, 26 Sep 2023 12:40:22 +0200 Subject: [PATCH 11/11] Use hideOnHoverOutside prop --- packages/components/src/dropdown-menu-v2-ariakit/index.tsx | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx index 6bf250dbe9a140..f5efa980fc3c57 100644 --- a/packages/components/src/dropdown-menu-v2-ariakit/index.tsx +++ b/packages/components/src/dropdown-menu-v2-ariakit/index.tsx @@ -92,6 +92,7 @@ export const DropdownMenu = forwardRef< HTMLDivElement, DropdownMenuProps >( store={ dropdownMenuStore } gutter={ dropdownMenuStore.parent ? 16 : 8 } shift={ dropdownMenuStore.parent ? -9 : 0 } + hideOnHoverOutside={ false } modal >