import type { JSX } from 'solid-js'; import { createComponent, createEffect, mergeProps } from 'solid-js'; import { omitProps } from 'solid-use/props'; import type { DisclosureStateRenderProps } from '../../states/create-disclosure-state'; import { DisclosureStateChild, useDisclosureState, } from '../../states/create-disclosure-state'; import type { HeadlessPropsWithRef, ValidConstructor, } from '../../utils/dynamic-prop'; import { createForwardRef } from '../../utils/dynamic-prop'; import { createARIADisabledState, createARIAExpandedState, createDisabledState, createExpandedState, } from '../../utils/state-props'; import type { OmitAndMerge } from '../../utils/types'; import useEventListener from '../../utils/use-event-listener'; import type { ButtonProps } from '../button'; import { Button } from '../button'; import { usePopoverContext } from './PopoverContext'; import { POPOVER_BUTTON_TAG } from './tags'; export type PopoverButtonProps = HeadlessPropsWithRef< T, OmitAndMerge> >; export function PopoverButton( props: PopoverButtonProps, ): JSX.Element { const context = usePopoverContext('PopoverButton'); const state = useDisclosureState(); const [internalRef, setInternalRef] = createForwardRef(props); const isDisabled = (): boolean | undefined => state.disabled() || props.disabled; createEffect(() => { const current = internalRef(); if (current instanceof HTMLElement) { context.anchor = current; useEventListener(current, 'click', () => { if (!isDisabled()) { state.toggle(); } }); useEventListener(current, 'mouseenter', () => { context.hovering = true; }); useEventListener(current, 'mouseleave', () => { context.hovering = false; }); } }); return createComponent( Button, mergeProps( POPOVER_BUTTON_TAG, { id: context.buttonID, ref: setInternalRef, get 'aria-controls'() { return state.isOpen() && context.panelID; }, }, createDisabledState(isDisabled), createARIADisabledState(isDisabled), createExpandedState(() => state.isOpen()), createARIAExpandedState(() => state.isOpen()), omitProps(props, ['children', 'ref']), { get children() { return createComponent(DisclosureStateChild, { get children() { return props.children; }, }); }, }, ) as ButtonProps, ); }