Skip to content

Commit

Permalink
DropdownMenu V2: add support for rendering in legacy popover slot (#5…
Browse files Browse the repository at this point in the history
…6342)

* DropdownMenuV2: support rendering in slotfill

* Add temporary stories

* CHANGELOG
  • Loading branch information
ciampo authored Nov 23, 2023
1 parent ea60d43 commit 0b6c4a0
Show file tree
Hide file tree
Showing 6 changed files with 141 additions and 1 deletion.
1 change: 1 addition & 0 deletions packages/components/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@

### Internal

- `DropdownMenuV2`: add support for rendering in legacy popover slot ([#56342](https://github.com/WordPress/gutenberg/pull/56342)).
- `Slot`: add `style` prop to `bubblesVirtually` version ([#56428](https://github.com/WordPress/gutenberg/pull/56428))

## 25.12.0 (2023-11-16)
Expand Down
14 changes: 14 additions & 0 deletions packages/components/src/dropdown-menu-v2-ariakit/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,11 @@ import { SVG, Circle } from '@wordpress/primitives';
import { useContextSystem, contextConnect } from '../context';
import type { WordPressComponentProps } from '../context';
import Icon from '../icon';
import { useSlot } from '../slot-fill';
import {
SLOT_NAME as POPOVER_DEFAULT_SLOT_NAME,
slotNameContext,
} from '../popover';
import type {
DropdownMenuContext as DropdownMenuContextType,
DropdownMenuProps,
Expand Down Expand Up @@ -182,6 +187,9 @@ const UnconnectedDropdownMenu = (
shift,
modal = true,

// Other props
slotName: slotNameProp = POPOVER_DEFAULT_SLOT_NAME,

// From internal components context
variant,

Expand Down Expand Up @@ -270,6 +278,11 @@ const UnconnectedDropdownMenu = (
[ computedDirection ]
);

// Render the portal in the default slot used by the legacy Popover component.
const slotName = useContext( slotNameContext ) || slotNameProp;
const slot = useSlot( slotName );
const portalContainer = slot.ref?.current;

return (
<>
{ /* Menu trigger */ }
Expand Down Expand Up @@ -305,6 +318,7 @@ const UnconnectedDropdownMenu = (
wrapperProps={ wrapperProps }
hideOnEscape={ hideOnEscape }
unmountOnHide
portalElement={ portalContainer }
>
<DropdownMenuContext.Provider value={ contextValue }>
{ children }
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ import {
import Icon from '../../icon';
import Button from '../../button';
import Modal from '../../modal';
import Popover from '../../popover';
import { createSlotFill, Provider as SlotFillProvider } from '../../slot-fill';
import { ContextSystemProvider } from '../../context';

Expand Down Expand Up @@ -501,3 +502,68 @@ InsideModal.parameters = {
source: { type: 'code' },
},
};

export const WithDefaultSlotFill: StoryFn< typeof DropdownMenu > = (
props
) => (
<SlotFillProvider>
{ /* @ts-expect-error Slot is not currently typed on Popover */ }
<Popover.Slot />
<DropdownMenu { ...props }>
<DropdownMenuItem>Top level item</DropdownMenuItem>
<DropdownMenu
trigger={ <DropdownMenuItem>Open submenu</DropdownMenuItem> }
>
<DropdownMenuItem>Nested item</DropdownMenuItem>
</DropdownMenu>
</DropdownMenu>
</SlotFillProvider>
);
WithDefaultSlotFill.args = {
...Default.args,
};

export const WithNamedSlotFill: StoryFn< typeof DropdownMenu > = ( props ) => (
<SlotFillProvider>
{ /* @ts-expect-error Slot is not currently typed on Popover */ }
<Popover.Slot name="dropdown-menu-slot-with-name" />
<DropdownMenu { ...props }>
<DropdownMenuItem>Top level item</DropdownMenuItem>
<DropdownMenu
trigger={ <DropdownMenuItem>Open submenu</DropdownMenuItem> }
>
<DropdownMenuItem>Nested item</DropdownMenuItem>
</DropdownMenu>
</DropdownMenu>
</SlotFillProvider>
);
WithNamedSlotFill.args = {
...Default.args,
slotName: 'dropdown-menu-slot-with-name',
};

// @ts-expect-error __unstableSlotNameProvider is not currently typed on Popover
const SlotNameProvider = Popover.__unstableSlotNameProvider;
export const WithSlotFillViaContext: StoryFn< typeof DropdownMenu > = (
props
) => (
<SlotFillProvider>
<SlotNameProvider value="dropdown-menu-slot-via-context">
{ /* @ts-expect-error Slot is not currently typed on Popover */ }
<Popover.Slot name="dropdown-menu-slot-via-context" />
<DropdownMenu { ...props }>
<DropdownMenuItem>Top level item</DropdownMenuItem>
<DropdownMenu
trigger={
<DropdownMenuItem>Open submenu</DropdownMenuItem>
}
>
<DropdownMenuItem>Nested item</DropdownMenuItem>
</DropdownMenu>
</DropdownMenu>
</SlotNameProvider>
</SlotFillProvider>
);
WithSlotFillViaContext.args = {
...Default.args,
};
7 changes: 7 additions & 0 deletions packages/components/src/dropdown-menu-v2-ariakit/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,13 @@ export interface DropdownMenuProps {
| ( (
event: KeyboardEvent | React.KeyboardEvent< Element >
) => boolean );
/**
* The name of the Slot in which the popover should be rendered. It should
* be also passed to the corresponding `PopoverSlot` component.
*
* @default 'Popover'
*/
slotName?: string;
}

export interface DropdownMenuGroupProps {
Expand Down
52 changes: 52 additions & 0 deletions packages/components/src/dropdown-menu/stories/index.story.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@ import type { Meta, StoryFn } from '@storybook/react';
import { DropdownMenu } from '..';
import MenuItem from '../../menu-item';
import MenuGroup from '../../menu-group';
import Popover from '../../popover';
import { Provider as SlotFillProvider } from '../../slot-fill';

/**
* WordPress dependencies
Expand Down Expand Up @@ -116,3 +118,53 @@ WithChildren.args = {
</>
),
};

export const WithDefaultSlotFill: StoryFn< typeof DropdownMenu > = (
props
) => (
<SlotFillProvider>
<div style={ { height: 150 } }>
{ /* @ts-expect-error Slot is not currently typed on Popover */ }
<Popover.Slot />
<DropdownMenu { ...props } />
</div>
</SlotFillProvider>
);
WithDefaultSlotFill.args = {
...Default.args,
};

export const WithNamedSlotFill: StoryFn< typeof DropdownMenu > = ( props ) => (
<SlotFillProvider>
<div style={ { height: 150 } }>
{ /* @ts-expect-error Slot is not currently typed on Popover */ }
<Popover.Slot name="dropdown-menu-slot-with-name" />
<DropdownMenu { ...props } />
</div>
</SlotFillProvider>
);
WithNamedSlotFill.args = {
...Default.args,
popoverProps: {
__unstableSlotName: 'dropdown-menu-slot-with-name',
},
};

// @ts-expect-error __unstableSlotNameProvider is not currently typed on Popover
const SlotNameProvider = Popover.__unstableSlotNameProvider;
export const WithSlotFillViaContext: StoryFn< typeof DropdownMenu > = (
props
) => (
<SlotFillProvider>
<SlotNameProvider value="dropdown-menu-slot-via-context">
{ /* @ts-expect-error Slot is not currently typed on Popover */ }
<Popover.Slot name="dropdown-menu-slot-via-context" />
<div style={ { height: 150 } }>
<DropdownMenu { ...props } />
</div>
</SlotNameProvider>
</SlotFillProvider>
);
WithSlotFillViaContext.args = {
...Default.args,
};
2 changes: 1 addition & 1 deletion packages/components/src/popover/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -97,7 +97,7 @@ const ArrowTriangle = () => (
</SVG>
);

const slotNameContext = createContext< string | undefined >( undefined );
export const slotNameContext = createContext< string | undefined >( undefined );

const fallbackContainerClassname = 'components-popover__fallback-container';
const getPopoverFallbackContainer = () => {
Expand Down

0 comments on commit 0b6c4a0

Please sign in to comment.