2024-11-16 06:26:16 +01:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2024-10-27 21:22:06 +01:00
|
|
|
import { Button } from "@/element/button";
|
|
|
|
import {
|
2024-11-16 06:26:16 +01:00
|
|
|
autoUpdate,
|
2024-10-27 21:22:06 +01:00
|
|
|
FloatingPortal,
|
2024-12-03 07:23:44 +01:00
|
|
|
Middleware,
|
2024-10-27 21:22:06 +01:00
|
|
|
offset as offsetMiddleware,
|
2024-11-16 06:26:16 +01:00
|
|
|
useClick,
|
2024-10-27 21:22:06 +01:00
|
|
|
useDismiss,
|
|
|
|
useFloating,
|
|
|
|
useInteractions,
|
2024-11-16 06:26:16 +01:00
|
|
|
type OffsetOptions,
|
2024-10-27 21:22:06 +01:00
|
|
|
type Placement,
|
|
|
|
} from "@floating-ui/react";
|
|
|
|
import clsx from "clsx";
|
|
|
|
import {
|
|
|
|
Children,
|
|
|
|
cloneElement,
|
|
|
|
forwardRef,
|
|
|
|
isValidElement,
|
|
|
|
JSXElementConstructor,
|
|
|
|
memo,
|
|
|
|
ReactElement,
|
|
|
|
ReactNode,
|
|
|
|
useState,
|
|
|
|
} from "react";
|
|
|
|
|
2024-11-22 01:05:04 +01:00
|
|
|
import "./popover.scss";
|
2024-10-27 21:22:06 +01:00
|
|
|
|
|
|
|
interface PopoverProps {
|
|
|
|
children: ReactNode;
|
|
|
|
className?: string;
|
|
|
|
placement?: Placement;
|
2024-11-16 06:26:16 +01:00
|
|
|
offset?: OffsetOptions;
|
|
|
|
onDismiss?: () => void;
|
2024-12-03 07:23:44 +01:00
|
|
|
middleware?: Middleware[];
|
2024-10-27 21:22:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
const isPopoverButton = (
|
|
|
|
element: ReactElement
|
|
|
|
): element is ReactElement<PopoverButtonProps, JSXElementConstructor<PopoverButtonProps>> => {
|
|
|
|
return element.type === PopoverButton;
|
|
|
|
};
|
|
|
|
|
|
|
|
const isPopoverContent = (
|
|
|
|
element: ReactElement
|
|
|
|
): element is ReactElement<PopoverContentProps, JSXElementConstructor<PopoverContentProps>> => {
|
|
|
|
return element.type === PopoverContent;
|
|
|
|
};
|
|
|
|
|
2024-12-03 07:23:44 +01:00
|
|
|
const Popover = memo(
|
|
|
|
({ children, className, placement = "bottom-start", offset = 3, onDismiss, middleware }: PopoverProps) => {
|
|
|
|
const [isOpen, setIsOpen] = useState(false);
|
2024-10-27 21:22:06 +01:00
|
|
|
|
2024-12-03 07:23:44 +01:00
|
|
|
const handleOpenChange = (open: boolean) => {
|
|
|
|
setIsOpen(open);
|
|
|
|
if (!open && onDismiss) {
|
|
|
|
onDismiss();
|
2024-10-27 21:22:06 +01:00
|
|
|
}
|
2024-12-03 07:23:44 +01:00
|
|
|
};
|
2024-10-27 21:22:06 +01:00
|
|
|
|
2024-12-03 07:23:44 +01:00
|
|
|
if (offset === undefined) {
|
|
|
|
offset = 3;
|
2024-10-27 21:22:06 +01:00
|
|
|
}
|
|
|
|
|
2024-12-03 07:23:44 +01:00
|
|
|
middleware ??= [];
|
|
|
|
middleware.push(offsetMiddleware(offset));
|
|
|
|
|
|
|
|
const { refs, floatingStyles, context } = useFloating({
|
|
|
|
placement,
|
|
|
|
open: isOpen,
|
|
|
|
onOpenChange: handleOpenChange,
|
|
|
|
middleware: middleware,
|
|
|
|
whileElementsMounted: autoUpdate,
|
|
|
|
});
|
|
|
|
|
|
|
|
const click = useClick(context);
|
|
|
|
const dismiss = useDismiss(context);
|
|
|
|
const { getReferenceProps, getFloatingProps } = useInteractions([click, dismiss]);
|
|
|
|
|
|
|
|
const renderChildren = Children.map(children, (child) => {
|
|
|
|
if (isValidElement(child)) {
|
|
|
|
if (isPopoverButton(child)) {
|
|
|
|
return cloneElement(child as any, {
|
|
|
|
isActive: isOpen,
|
|
|
|
ref: refs.setReference,
|
|
|
|
getReferenceProps,
|
|
|
|
// Do not overwrite onClick
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isPopoverContent(child)) {
|
|
|
|
return isOpen
|
|
|
|
? cloneElement(child as any, {
|
|
|
|
ref: refs.setFloating,
|
|
|
|
style: floatingStyles,
|
|
|
|
getFloatingProps,
|
|
|
|
})
|
|
|
|
: null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return child;
|
|
|
|
});
|
|
|
|
|
|
|
|
return <div className={clsx("popover", className)}>{renderChildren}</div>;
|
|
|
|
}
|
|
|
|
);
|
2024-10-27 21:22:06 +01:00
|
|
|
|
|
|
|
interface PopoverButtonProps extends React.ButtonHTMLAttributes<HTMLButtonElement> {
|
|
|
|
isActive?: boolean;
|
|
|
|
children: React.ReactNode;
|
|
|
|
getReferenceProps?: () => any;
|
|
|
|
as?: keyof JSX.IntrinsicElements | React.ComponentType<any>;
|
|
|
|
}
|
|
|
|
|
|
|
|
const PopoverButton = forwardRef<HTMLButtonElement | HTMLDivElement, PopoverButtonProps>(
|
2024-11-16 06:26:16 +01:00
|
|
|
(
|
|
|
|
{
|
|
|
|
isActive,
|
|
|
|
children,
|
|
|
|
onClick: userOnClick, // Destructured from props
|
|
|
|
getReferenceProps,
|
|
|
|
className,
|
|
|
|
as: Component = "button",
|
|
|
|
...props // The rest of the props, without onClick
|
|
|
|
},
|
|
|
|
ref
|
|
|
|
) => {
|
|
|
|
const referenceProps = getReferenceProps?.() || {};
|
|
|
|
const popoverOnClick = referenceProps.onClick;
|
|
|
|
|
|
|
|
// Remove onClick from referenceProps to prevent it from overwriting our combinedOnClick
|
|
|
|
const { onClick: refOnClick, ...restReferenceProps } = referenceProps;
|
|
|
|
|
|
|
|
const combinedOnClick = (event: React.MouseEvent) => {
|
|
|
|
if (userOnClick) {
|
|
|
|
userOnClick(event as any); // Our custom onClick logic
|
|
|
|
}
|
|
|
|
if (popoverOnClick) {
|
|
|
|
popoverOnClick(event); // Popover's onClick logic
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-10-27 21:22:06 +01:00
|
|
|
return (
|
|
|
|
<Button
|
|
|
|
ref={ref}
|
|
|
|
className={clsx("popover-button", className, { "is-active": isActive })}
|
2024-11-16 06:26:16 +01:00
|
|
|
{...props} // Spread the rest of the props
|
|
|
|
{...restReferenceProps} // Spread referenceProps without onClick
|
|
|
|
onClick={combinedOnClick} // Assign combined onClick after spreading
|
2024-10-27 21:22:06 +01:00
|
|
|
>
|
|
|
|
{children}
|
|
|
|
</Button>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
interface PopoverContentProps extends React.HTMLAttributes<HTMLDivElement> {
|
|
|
|
children: React.ReactNode;
|
|
|
|
getFloatingProps?: () => any;
|
|
|
|
}
|
|
|
|
|
|
|
|
const PopoverContent = forwardRef<HTMLDivElement, PopoverContentProps>(
|
|
|
|
({ children, className, getFloatingProps, style, ...props }, ref) => {
|
|
|
|
return (
|
|
|
|
<FloatingPortal>
|
|
|
|
<div
|
|
|
|
ref={ref}
|
|
|
|
className={clsx("popover-content", className)}
|
|
|
|
style={style}
|
|
|
|
{...getFloatingProps?.()}
|
|
|
|
{...props}
|
|
|
|
>
|
|
|
|
{children}
|
|
|
|
</div>
|
|
|
|
</FloatingPortal>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
Popover.displayName = "Popover";
|
|
|
|
PopoverButton.displayName = "PopoverButton";
|
|
|
|
PopoverContent.displayName = "PopoverContent";
|
|
|
|
|
|
|
|
export { Popover, PopoverButton, PopoverContent };
|
|
|
|
export type { PopoverButtonProps, PopoverContentProps };
|