mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-12 04:18:54 +00:00
fix(core): fix menu shaking (#8187)
This commit is contained in:
@@ -1,95 +0,0 @@
|
||||
import { useCallback, useState } from 'react';
|
||||
|
||||
import { useRefEffect } from '../../../hooks';
|
||||
|
||||
export const useMenuContentController = ({
|
||||
onOpenChange,
|
||||
side,
|
||||
defaultOpen,
|
||||
sideOffset,
|
||||
open: controlledOpen,
|
||||
}: {
|
||||
defaultOpen?: boolean;
|
||||
side?: 'top' | 'bottom' | 'left' | 'right';
|
||||
onOpenChange?: (open: boolean) => void;
|
||||
open?: boolean;
|
||||
sideOffset?: number;
|
||||
} = {}) => {
|
||||
const [open, setOpen] = useState(defaultOpen ?? false);
|
||||
const actualOpen = controlledOpen ?? open;
|
||||
const contentSide = side ?? 'bottom';
|
||||
const [contentOffset, setContentOffset] = useState<number>(0);
|
||||
|
||||
const handleOpenChange = useCallback(
|
||||
(open: boolean) => {
|
||||
setOpen(open);
|
||||
onOpenChange?.(open);
|
||||
},
|
||||
[onOpenChange]
|
||||
);
|
||||
const contentRef = useRefEffect<HTMLDivElement>(
|
||||
contentElement => {
|
||||
if (!actualOpen) return;
|
||||
|
||||
const wrapperElement = contentElement.parentNode as HTMLDivElement;
|
||||
|
||||
const updateContentOffset = () => {
|
||||
if (!contentElement) return;
|
||||
const contentRect = wrapperElement.getBoundingClientRect();
|
||||
if (contentSide === 'bottom') {
|
||||
setContentOffset(prev => {
|
||||
const viewportHeight = window.innerHeight;
|
||||
const newOffset = Math.min(
|
||||
viewportHeight - (contentRect.bottom - prev),
|
||||
0
|
||||
);
|
||||
return newOffset;
|
||||
});
|
||||
} else if (contentSide === 'top') {
|
||||
setContentOffset(prev => {
|
||||
const newOffset = Math.min(contentRect.top + prev, 0);
|
||||
return newOffset;
|
||||
});
|
||||
} else if (contentSide === 'left') {
|
||||
setContentOffset(prev => {
|
||||
const newOffset = Math.min(contentRect.left + prev, 0);
|
||||
return newOffset;
|
||||
});
|
||||
} else if (contentSide === 'right') {
|
||||
setContentOffset(prev => {
|
||||
const viewportWidth = window.innerWidth;
|
||||
const newOffset = Math.min(
|
||||
viewportWidth - (contentRect.right - prev),
|
||||
0
|
||||
);
|
||||
return newOffset;
|
||||
});
|
||||
}
|
||||
};
|
||||
let animationFrame: number = 0;
|
||||
const requestUpdateContentOffset = () => {
|
||||
cancelAnimationFrame(animationFrame);
|
||||
animationFrame = requestAnimationFrame(updateContentOffset);
|
||||
};
|
||||
|
||||
const observer = new ResizeObserver(requestUpdateContentOffset);
|
||||
observer.observe(wrapperElement);
|
||||
window.addEventListener('resize', requestUpdateContentOffset);
|
||||
requestUpdateContentOffset();
|
||||
return () => {
|
||||
observer.disconnect();
|
||||
window.removeEventListener('resize', requestUpdateContentOffset);
|
||||
cancelAnimationFrame(animationFrame);
|
||||
};
|
||||
},
|
||||
[actualOpen, contentSide]
|
||||
);
|
||||
|
||||
return {
|
||||
handleOpenChange,
|
||||
contentSide,
|
||||
contentOffset: (sideOffset ?? 0) + contentOffset,
|
||||
contentRef,
|
||||
open: actualOpen,
|
||||
};
|
||||
};
|
||||
@@ -4,7 +4,6 @@ import React from 'react';
|
||||
|
||||
import type { MenuProps } from '../menu.types';
|
||||
import * as styles from '../styles.css';
|
||||
import { useMenuContentController } from './controller';
|
||||
import * as desktopStyles from './styles.css';
|
||||
|
||||
export const DesktopMenu = ({
|
||||
@@ -12,36 +11,18 @@ export const DesktopMenu = ({
|
||||
items,
|
||||
noPortal,
|
||||
portalOptions,
|
||||
rootOptions: {
|
||||
onOpenChange,
|
||||
defaultOpen,
|
||||
modal,
|
||||
open: rootOpen,
|
||||
...rootOptions
|
||||
} = {},
|
||||
rootOptions: { defaultOpen, modal, ...rootOptions } = {},
|
||||
contentOptions: {
|
||||
className = '',
|
||||
style: contentStyle = {},
|
||||
side,
|
||||
sideOffset,
|
||||
...otherContentOptions
|
||||
} = {},
|
||||
}: MenuProps) => {
|
||||
const { handleOpenChange, contentSide, contentOffset, contentRef, open } =
|
||||
useMenuContentController({
|
||||
open: rootOpen,
|
||||
defaultOpen,
|
||||
onOpenChange,
|
||||
side,
|
||||
sideOffset: (sideOffset ?? 0) + 5,
|
||||
});
|
||||
const ContentWrapper = noPortal ? React.Fragment : DropdownMenu.Portal;
|
||||
return (
|
||||
<DropdownMenu.Root
|
||||
onOpenChange={handleOpenChange}
|
||||
defaultOpen={defaultOpen}
|
||||
modal={modal ?? false}
|
||||
open={open}
|
||||
{...rootOptions}
|
||||
>
|
||||
<DropdownMenu.Trigger
|
||||
@@ -62,11 +43,7 @@ export const DesktopMenu = ({
|
||||
className
|
||||
)}
|
||||
align="start"
|
||||
ref={contentRef}
|
||||
side={contentSide}
|
||||
style={{ zIndex: 'var(--affine-z-index-popover)', ...contentStyle }}
|
||||
avoidCollisions={false}
|
||||
sideOffset={contentOffset}
|
||||
{...otherContentOptions}
|
||||
>
|
||||
{items}
|
||||
|
||||
@@ -6,22 +6,15 @@ import { useMemo } from 'react';
|
||||
import type { MenuSubProps } from '../menu.types';
|
||||
import * as styles from '../styles.css';
|
||||
import { useMenuItem } from '../use-menu-item';
|
||||
import { useMenuContentController } from './controller';
|
||||
|
||||
export const DesktopMenuSub = ({
|
||||
children: propsChildren,
|
||||
items,
|
||||
portalOptions,
|
||||
subOptions: {
|
||||
defaultOpen,
|
||||
onOpenChange,
|
||||
open: rootOpen,
|
||||
...otherSubOptions
|
||||
} = {},
|
||||
subOptions: { defaultOpen, ...otherSubOptions } = {},
|
||||
triggerOptions,
|
||||
subContentOptions: {
|
||||
className: subContentClassName = '',
|
||||
sideOffset,
|
||||
style: contentStyle,
|
||||
...otherSubContentOptions
|
||||
} = {},
|
||||
@@ -32,35 +25,18 @@ export const DesktopMenuSub = ({
|
||||
suffixIcon: <ArrowRightSmallIcon />,
|
||||
});
|
||||
|
||||
const { handleOpenChange, contentOffset, contentRef, open } =
|
||||
useMenuContentController({
|
||||
defaultOpen,
|
||||
open: rootOpen,
|
||||
onOpenChange,
|
||||
side: 'right',
|
||||
sideOffset: (sideOffset ?? 0) + 12,
|
||||
});
|
||||
|
||||
return (
|
||||
<DropdownMenu.Sub
|
||||
defaultOpen={defaultOpen}
|
||||
onOpenChange={handleOpenChange}
|
||||
open={open}
|
||||
{...otherSubOptions}
|
||||
>
|
||||
<DropdownMenu.Sub defaultOpen={defaultOpen} {...otherSubOptions}>
|
||||
<DropdownMenu.SubTrigger className={className} {...otherProps}>
|
||||
{children}
|
||||
</DropdownMenu.SubTrigger>
|
||||
<DropdownMenu.Portal {...portalOptions}>
|
||||
<DropdownMenu.SubContent
|
||||
sideOffset={contentOffset}
|
||||
ref={contentRef}
|
||||
className={useMemo(
|
||||
() => clsx(styles.menuContent, subContentClassName),
|
||||
[subContentClassName]
|
||||
)}
|
||||
style={{ zIndex: 'var(--affine-z-index-popover)', ...contentStyle }}
|
||||
avoidCollisions={false}
|
||||
{...otherSubContentOptions}
|
||||
>
|
||||
{items}
|
||||
|
||||
@@ -26,7 +26,6 @@ export const workspaceTypeIcon = style({
|
||||
color: cssVar('iconSecondary'),
|
||||
});
|
||||
export const scrollbar = style({
|
||||
transform: 'translateX(8px)',
|
||||
width: '4px',
|
||||
});
|
||||
export const workspaceCard = style({
|
||||
|
||||
Reference in New Issue
Block a user