} from '~/components/primitives/hooks';
import { Portal as RNPPortal } from '~/components/primitives/portal';
import * as Slot from '~/components/primitives/slot';
} from '~/components/primitives/types';
import * as React from 'react';
type GestureResponderEvent,
DropdownMenuCheckboxItemProps,
DropdownMenuOverlayProps,
DropdownMenuRadioGroupProps,
DropdownMenuRadioItemProps,
DropdownMenuSeparatorProps,
DropdownMenuSubTriggerProps,
onOpenChange: (open: boolean) => void;
triggerPosition: LayoutPosition | null;
setTriggerPosition: (triggerPosition: LayoutPosition | null) => void;
contentLayout: LayoutRectangle | null;
setContentLayout: (contentLayout: LayoutRectangle | null) => void;
const RootContext = React.createContext<IRootContext | null>(null);
const Root = React.forwardRef<
SlottableViewProps & { onOpenChange?: (open: boolean) => void }
>(({ asChild, onOpenChange: onOpenChangeProp, ...viewProps }, ref) => {
const nativeID = React.useId();
const [triggerPosition, setTriggerPosition] = React.useState<LayoutPosition | null>(null);
const [contentLayout, setContentLayout] = React.useState<LayoutRectangle | null>(null);
const [open, setOpen] = React.useState(false);
function onOpenChange(open: boolean) {
onOpenChangeProp?.(open);
const Component = asChild ? Slot.View : View;
<Component ref={ref} {...viewProps} />
Root.displayName = 'RootNativeDropdownMenu';
function useRootContext() {
const context = React.useContext(RootContext);
'DropdownMenu compound components cannot be rendered outside the DropdownMenu component'
const Trigger = React.forwardRef<DropdownMenuTriggerRef, SlottablePressableProps>(
({ asChild, onPress: onPressProp, disabled = false, ...props }, ref) => {
const { open, onOpenChange, setTriggerPosition } = useRootContext();
const augmentedRef = useAugmentedRef({
augmentedRef.current?.measure((_x, _y, width, height, pageX, pageY) => {
setTriggerPosition({ width, pageX, pageY: pageY, height });
setTriggerPosition(null);
function onPress(ev: GestureResponderEvent) {
augmentedRef.current?.measure((_x, _y, width, height, pageX, pageY) => {
setTriggerPosition({ width, pageX, pageY: pageY, height });
const Component = asChild ? Slot.Pressable : Pressable;
aria-disabled={disabled ?? undefined}
disabled={disabled ?? undefined}
Trigger.displayName = 'TriggerNativeDropdownMenu';
* @warning when using a custom `<PortalHost />`, you might have to adjust the Content's sideOffset to account for nav elements like headers.
function Portal({ forceMount, hostName, children }: DropdownMenuPortalProps) {
const value = useRootContext();
if (!value.triggerPosition) {
<RNPPortal hostName={hostName} name={`${value.nativeID}_portal`}>
<RootContext.Provider value={value}>{children}</RootContext.Provider>
const Overlay = React.forwardRef<PressableRef, SlottablePressableProps & DropdownMenuOverlayProps>(
({ asChild, forceMount, onPress: OnPressProp, closeOnPress = true, ...props }, ref) => {
const { open, onOpenChange, setContentLayout, setTriggerPosition } = useRootContext();
function onPress(ev: GestureResponderEvent) {
setTriggerPosition(null);
const Component = asChild ? Slot.Pressable : Pressable;
return <Component ref={ref} onPress={onPress} {...props} />;
Overlay.displayName = 'OverlayNativeDropdownMenu';
* @info `position`, `top`, `left`, and `maxWidth` style properties are controlled internally. Opt out of this behavior by setting `disablePositioningStyle` to `true`.
const Content = React.forwardRef<PressableRef, SlottablePressableProps & PositionedContentProps>(
const backHandler = BackHandler.addEventListener('hardwareBackPress', () => {
setTriggerPosition(null);
const positionStyle = useRelativePosition({
function onLayout(event: LayoutChangeEvent) {
setContentLayout(event.nativeEvent.layout);
const Component = asChild ? Slot.Pressable : Pressable;
style={[positionStyle, style]}
Content.displayName = 'ContentNativeDropdownMenu';
const Item = React.forwardRef<PressableRef, SlottablePressableProps & DropdownMenuItemProps>(
{ asChild, textValue, onPress: onPressProp, disabled = false, closeOnPress = true, ...props },
const { onOpenChange, setTriggerPosition, setContentLayout } = useRootContext();
function onPress(ev: GestureResponderEvent) {
setTriggerPosition(null);
const Component = asChild ? Slot.Pressable : Pressable;
aria-valuetext={textValue}
aria-disabled={!!disabled}
accessibilityState={{ disabled: !!disabled }}
Item.displayName = 'ItemNativeDropdownMenu';
const Group = React.forwardRef<ViewRef, SlottableViewProps>(({ asChild, ...props }, ref) => {
const Component = asChild ? Slot.View : View;
return <Component ref={ref} role='group' {...props} />;
Group.displayName = 'GroupNativeDropdownMenu';
const Label = React.forwardRef<TextRef, SlottableTextProps>(({ asChild, ...props }, ref) => {
const Component = asChild ? Slot.Text : Text;
return <Component ref={ref} {...props} />;
Label.displayName = 'LabelNativeDropdownMenu';
value: string | undefined;
onValueChange: (value: string) => void;
const FormItemContext = React.createContext<FormItemContext | null>(null);
const CheckboxItem = React.forwardRef<
SlottablePressableProps & DropdownMenuCheckboxItemProps
const { onOpenChange, setContentLayout, setTriggerPosition, nativeID } = useRootContext();
function onPress(ev: GestureResponderEvent) {
onCheckedChange(!checked);
setTriggerPosition(null);
const Component = asChild ? Slot.Pressable : Pressable;
<FormItemContext.Provider value={{ checked }}>
aria-disabled={!!disabled}
aria-valuetext={textValue}
accessibilityState={{ disabled: !!disabled }}
</FormItemContext.Provider>
CheckboxItem.displayName = 'CheckboxItemNativeDropdownMenu';
function useFormItemContext() {
const context = React.useContext(FormItemContext);
'CheckboxItem or RadioItem compound components cannot be rendered outside of a CheckboxItem or RadioItem component'
const RadioGroup = React.forwardRef<ViewRef, SlottableViewProps & DropdownMenuRadioGroupProps>(
({ asChild, value, onValueChange, ...props }, ref) => {
const Component = asChild ? Slot.View : View;
<FormItemContext.Provider value={{ value, onValueChange }}>
<Component ref={ref} role='radiogroup' {...props} />
</FormItemContext.Provider>
RadioGroup.displayName = 'RadioGroupNativeDropdownMenu';
type BothFormItemContext = Exclude<FormItemContext, { checked: boolean }> & {
const RadioItemContext = React.createContext({} as { itemValue: string });
const RadioItem = React.forwardRef<
SlottablePressableProps & DropdownMenuRadioItemProps
const { onOpenChange, setContentLayout, setTriggerPosition } = useRootContext();
const { value, onValueChange } = useFormItemContext() as BothFormItemContext;
function onPress(ev: GestureResponderEvent) {
onValueChange(itemValue);
setTriggerPosition(null);
const Component = asChild ? Slot.Pressable : Pressable;
<RadioItemContext.Provider value={{ itemValue }}>
aria-checked={value === itemValue}
disabled={disabled ?? false}
disabled: disabled ?? false,
checked: value === itemValue,
aria-valuetext={textValue}
</RadioItemContext.Provider>
RadioItem.displayName = 'RadioItemNativeDropdownMenu';
function useItemIndicatorContext() {
return React.useContext(RadioItemContext);
const ItemIndicator = React.forwardRef<ViewRef, SlottableViewProps & ForceMountable>(
({ asChild, forceMount, ...props }, ref) => {
const { itemValue } = useItemIndicatorContext();
const { checked, value } = useFormItemContext() as BothFormItemContext;
if (itemValue == null && !checked) {
if (value !== itemValue) {
const Component = asChild ? Slot.View : View;
return <Component ref={ref} role='presentation' {...props} />;
ItemIndicator.displayName = 'ItemIndicatorNativeDropdownMenu';
const Separator = React.forwardRef<ViewRef, SlottableViewProps & DropdownMenuSeparatorProps>(
({ asChild, decorative, ...props }, ref) => {
const Component = asChild ? Slot.View : View;
return <Component role={decorative ? 'presentation' : 'separator'} ref={ref} {...props} />;
Separator.displayName = 'SeparatorNativeDropdownMenu';
const SubContext = React.createContext<{
onOpenChange: (value: boolean) => void;
const Sub = React.forwardRef<ViewRef, SlottableViewProps & DropdownMenuSubProps>(
({ asChild, defaultOpen, open: openProp, onOpenChange: onOpenChangeProp, ...props }, ref) => {
const nativeID = React.useId();
const [open = false, onOpenChange] = useControllableState({
defaultProp: defaultOpen,
onChange: onOpenChangeProp,
const Component = asChild ? Slot.View : View;
<Component ref={ref} {...props} />
Sub.displayName = 'SubNativeDropdownMenu';
function useSubContext() {
const context = React.useContext(SubContext);
throw new Error('Sub compound components cannot be rendered outside of a Sub component');
const SubTrigger = React.forwardRef<
SlottablePressableProps & DropdownMenuSubTriggerProps
>(({ asChild, textValue, onPress: onPressProp, disabled = false, ...props }, ref) => {
const { nativeID, open, onOpenChange } = useSubContext();
function onPress(ev: GestureResponderEvent) {
const Component = asChild ? Slot.Pressable : Pressable;
aria-valuetext={textValue}
accessibilityState={{ expanded: open, disabled: !!disabled }}
aria-disabled={!!disabled}
SubTrigger.displayName = 'SubTriggerNativeDropdownMenu';
const SubContent = React.forwardRef<PressableRef, SlottablePressableProps & ForceMountable>(
({ asChild = false, forceMount, ...props }, ref) => {
const { open, nativeID } = useSubContext();
const Component = asChild ? Slot.Pressable : Pressable;
return <Component ref={ref} role='group' aria-labelledby={nativeID} {...props} />;
Content.displayName = 'ContentNativeDropdownMenu';
export type { DropdownMenuTriggerRef };