Skip to content

Commit

Permalink
feat(component): add animations to modal (#7474)
Browse files Browse the repository at this point in the history
Add opening and closing animations to modal.

The usage of conditional rendering as shown below is not recommended:
```
open ? (
      <Modal
        open={open}
        ...
      />
    ) : null,
```

When the modal is closed, it gets removed from the DOM instantly without running any exit animations that might be defined in the Modal component.
  • Loading branch information
JimmFly committed Jul 22, 2024
1 parent e3c3d1a commit 55db9f9
Show file tree
Hide file tree
Showing 8 changed files with 141 additions and 97 deletions.
43 changes: 39 additions & 4 deletions packages/frontend/component/src/ui/modal/modal.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,8 @@ import * as VisuallyHidden from '@radix-ui/react-visually-hidden';
import { assignInlineVars } from '@vanilla-extract/dynamic';
import clsx from 'clsx';
import type { CSSProperties } from 'react';
import { forwardRef, useCallback } from 'react';
import { forwardRef, useCallback, useEffect } from 'react';
import { type TransitionState, useTransition } from 'react-transition-state';

import type { IconButtonProps } from '../button';
import { IconButton } from '../button';
Expand All @@ -28,7 +29,8 @@ export interface ModalProps extends DialogProps {
* @default false
*/
persistent?: boolean;

// animation for modal open/close
animationTimeout?: number;
portalOptions?: DialogPortalProps;
contentOptions?: DialogContentProps;
overlayOptions?: DialogOverlayProps;
Expand Down Expand Up @@ -57,8 +59,10 @@ export const Modal = forwardRef<HTMLDivElement, ModalProps>(
withoutCloseButton = false,
modal,
persistent,

animationTimeout = 120,
portalOptions,
open: customOpen,
onOpenChange: customOnOpenChange,
contentOptions: {
style: contentStyle,
className: contentClassName,
Expand All @@ -68,6 +72,7 @@ export const Modal = forwardRef<HTMLDivElement, ModalProps>(
} = {},
overlayOptions: {
className: overlayClassName,
style: overlayStyle,
...otherOverlayOptions
} = {},
closeButtonOptions = {},
Expand All @@ -76,11 +81,38 @@ export const Modal = forwardRef<HTMLDivElement, ModalProps>(
},
ref
) => {
const [{ status }, toggle] = useTransition({
timeout: animationTimeout,
onStateChange: useCallback(
({ current }: { current: TransitionState }) => {
// add more status if needed
if (current.status === 'exited') customOnOpenChange?.(false);
if (current.status === 'entered') customOnOpenChange?.(true);
},
[customOnOpenChange]
),
});
useEffect(() => {
toggle(customOpen);
}, [customOpen]);

return (
<Dialog.Root modal={modal} {...props}>
<Dialog.Root
modal={modal}
open={status !== 'exited'}
onOpenChange={toggle}
{...props}
>
<Dialog.Portal {...portalOptions}>
<Dialog.Overlay
className={clsx(styles.modalOverlay, overlayClassName)}
data-state={status}
style={{
...assignInlineVars({
[styles.animationTimeout]: `${animationTimeout}ms`,
}),
...overlayStyle,
}}
{...otherOverlayOptions}
/>
<div data-modal={modal} className={clsx(styles.modalContentWrapper)}>
Expand All @@ -100,14 +132,17 @@ export const Modal = forwardRef<HTMLDivElement, ModalProps>(
[onEscapeKeyDown, persistent]
)}
className={clsx(styles.modalContent, contentClassName)}
data-state={status}
style={{
...assignInlineVars({
[styles.widthVar]: getVar(width, '50vw'),
[styles.heightVar]: getVar(height, 'unset'),
[styles.minHeightVar]: getVar(minHeight, '26px'),
[styles.animationTimeout]: `${animationTimeout}ms`,
}),
...contentStyle,
}}
{...(description ? {} : { 'aria-describedby': undefined })}
{...otherContentOptions}
ref={ref}
>
Expand Down
60 changes: 59 additions & 1 deletion packages/frontend/component/src/ui/modal/styles.css.ts
Original file line number Diff line number Diff line change
@@ -1,13 +1,61 @@
import { cssVar } from '@toeverything/theme';
import { createVar, globalStyle, style } from '@vanilla-extract/css';
import { createVar, globalStyle, keyframes, style } from '@vanilla-extract/css';
export const widthVar = createVar('widthVar');
export const heightVar = createVar('heightVar');
export const minHeightVar = createVar('minHeightVar');
export const animationTimeout = createVar();

const overlayShow = keyframes({
from: {
opacity: 0,
},
to: {
opacity: 1,
},
});
const overlayHide = keyframes({
to: {
opacity: 0,
},
from: {
opacity: 1,
},
});

const contentShow = keyframes({
from: {
opacity: 0,
transform: 'translateY(-2%) scale(0.96)',
},
to: {
opacity: 1,
transform: 'translateY(0) scale(1)',
},
});
const contentHide = keyframes({
to: {
opacity: 0,
transform: 'translateY(-2%) scale(0.96)',
},
from: {
opacity: 1,
transform: 'translateY(0) scale(1)',
},
});

export const modalOverlay = style({
position: 'fixed',
inset: 0,
backgroundColor: cssVar('backgroundModalColor'),
zIndex: cssVar('zIndexModal'),
selectors: {
'&[data-state=entered], &[data-state=entering]': {
animation: `${overlayShow} ${animationTimeout} forwards`,
},
'&[data-state=exited], &[data-state=exiting]': {
animation: `${overlayHide} ${animationTimeout} forwards`,
},
},
});
export const modalContentWrapper = style({
position: 'fixed',
Expand Down Expand Up @@ -39,6 +87,16 @@ export const modalContent = style({
maxHeight: 'calc(100vh - 32px)',
// :focus-visible will set outline
outline: 'none',
selectors: {
'&[data-state=entered], &[data-state=entering]': {
animation: `${contentShow} ${animationTimeout} cubic-bezier(0.42, 0, 0.58, 1)`,
animationFillMode: 'forwards',
},
'&[data-state=exited], &[data-state=exiting]': {
animation: `${contentHide} ${animationTimeout} cubic-bezier(0.42, 0, 0.58, 1)`,
animationFillMode: 'forwards',
},
},
});
export const closeButton = style({
position: 'absolute',
Expand Down
Original file line number Diff line number Diff line change
@@ -1,10 +1,12 @@
import { notify } from '@affine/component';
import { AuthInput, ModalHeader } from '@affine/component/auth-components';
import { Button } from '@affine/component/ui/button';
import { authAtom } from '@affine/core/atoms';
import { useAsyncCallback } from '@affine/core/hooks/affine-async-hooks';
import { Trans, useI18n } from '@affine/i18n';
import { ArrowDownBigIcon } from '@blocksuite/icons/rc';
import { useLiveData, useService } from '@toeverything/infra';
import { useAtomValue } from 'jotai';
import type { FC } from 'react';
import { useCallback, useEffect, useState } from 'react';
import { useSearchParams } from 'react-router-dom';
Expand Down Expand Up @@ -34,6 +36,7 @@ export const SignIn: FC<AuthPanelProps> = ({
const [verifyToken, challenge] = useCaptcha();

const [isValidEmail, setIsValidEmail] = useState(true);
const { openModal } = useAtomValue(authAtom);

useEffect(() => {
const timeout = setInterval(() => {
Expand All @@ -45,7 +48,7 @@ export const SignIn: FC<AuthPanelProps> = ({
};
}, [authService]);
const loginStatus = useLiveData(authService.session.status$);
if (loginStatus === 'authenticated') {
if (loginStatus === 'authenticated' && openModal) {
onSignedIn?.();
}

Expand Down
Original file line number Diff line number Diff line change
@@ -1,47 +1,14 @@
import { cssVar } from '@toeverything/theme';
import { createVar, keyframes, style } from '@vanilla-extract/css';
import { style } from '@vanilla-extract/css';

export const animationTimeout = createVar();

const contentShow = keyframes({
from: {
opacity: 0,
},
to: {
opacity: 1,
},
});
const contentHide = keyframes({
to: {
opacity: 0,
},
from: {
opacity: 1,
},
export const container = style({
maxWidth: 480,
minWidth: 360,
padding: '20px 0',
alignSelf: 'start',
marginTop: '120px',
});

export const overlay = style({
selectors: {
'&.entered, &.entering': {
animation: `${contentShow} ${animationTimeout} forwards`,
},
'&.exited, &.exiting': {
animation: `${contentHide} ${animationTimeout} forwards`,
},
},
});

export const container = style([
overlay,
{
maxWidth: 480,
minWidth: 360,
padding: '20px 0',
alignSelf: 'start',
marginTop: '120px',
},
]);

export const titleContainer = style({
display: 'flex',
width: '100%',
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,17 +12,7 @@ import {
useService,
type Workspace,
} from '@toeverything/infra';
import { assignInlineVars } from '@vanilla-extract/dynamic';
import clsx from 'clsx';
import {
Suspense,
useCallback,
useContext,
useEffect,
useMemo,
useRef,
} from 'react';
import { useTransition } from 'react-transition-state';
import { Suspense, useCallback, useContext, useMemo, useRef } from 'react';

import { BlocksuiteHeaderTitle } from '../../../blocksuite/block-suite-header/title';
import { managerContext } from '../common';
Expand All @@ -37,8 +27,6 @@ import * as styles from './info-modal.css';
import { TagsRow } from './tags-row';
import { TimeRow } from './time-row';

const animationTimeout = 120;

export const InfoModal = ({
open,
onOpenChange,
Expand All @@ -52,14 +40,6 @@ export const InfoModal = ({
}) => {
const titleInputHandleRef = useRef<InlineEditHandle>(null);

const [{ status }, toggle] = useTransition({
timeout: animationTimeout,
});

useEffect(() => {
toggle(open);
}, [open, toggle]);

const manager = usePagePropertiesManager(page);

const handleClose = useCallback(() => {
Expand All @@ -80,20 +60,10 @@ export const InfoModal = ({

return (
<Modal
overlayOptions={{
className: clsx(styles.overlay, status),
style: assignInlineVars({
[styles.animationTimeout]: `${animationTimeout}ms`,
}),
}}
contentOptions={{
className: clsx(styles.container, status),
'aria-describedby': undefined,
style: assignInlineVars({
[styles.animationTimeout]: `${animationTimeout}ms`,
}),
className: styles.container,
}}
open={status !== 'exited'}
open={open}
onOpenChange={onOpenChange}
withoutCloseButton
>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,9 +23,6 @@ export interface EditCollectionModalProps {
}

const contentOptions: DialogContentProps = {
onPointerDownOutside: e => {
e.preventDefault();
},
style: {
padding: 0,
maxWidth: 944,
Expand Down Expand Up @@ -60,6 +57,7 @@ export const EditCollectionModal = ({
width="calc(100% - 64px)"
height="80%"
contentOptions={contentOptions}
persistent
>
{open && init ? (
<EditCollection
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,9 +12,14 @@ export const useSelectPage = ({
init: string[];
onConfirm: (ids: string[]) => void;
}>();
const close = useCallback(() => {
onChange(undefined);
const close = useCallback((open: boolean) => {
if (!open) {
onChange(undefined);
}
}, []);
const handleCancel = useCallback(() => {
close(false);
}, [close]);
return {
node: (
<Modal
Expand All @@ -38,7 +43,7 @@ export const useSelectPage = ({
allPageListConfig={allPageListConfig}
init={value.init}
onConfirm={value.onConfirm}
onCancel={close}
onCancel={handleCancel}
/>
) : null}
</Modal>
Expand All @@ -48,7 +53,7 @@ export const useSelectPage = ({
onChange({
init,
onConfirm: list => {
close();
close(false);
res(list);
},
});
Expand Down
Loading

0 comments on commit 55db9f9

Please sign in to comment.