From f71ea01e03c7e7fc07f788c25b3dd8283aeed325 Mon Sep 17 00:00:00 2001 From: Ivan Kudryavtsev Date: Tue, 26 Sep 2023 13:19:52 +0300 Subject: [PATCH] feat(plasma-core, plasma-web, plasma-b2c): ModalBase component --- packages/plasma-b2c/api/plasma-b2c.api.md | 9 + .../ModalBase/ModalBase.stories.tsx | 139 +++++++++++++ .../src/components/ModalBase/index.ts | 2 + .../src/components/PopupBase/index.ts | 2 + packages/plasma-b2c/src/index.ts | 1 + packages/plasma-core/api/plasma-core.api.md | 31 ++- .../src/components/ModalBase/ModalBase.tsx | 185 ++++++++++++++++++ .../src/components/ModalBase/index.ts | 2 + .../src/components/PopupBase/PopupBase.tsx | 164 +++++++++------- .../components/PopupBase/PopupBaseContext.tsx | 31 ++- .../src/components/PopupBase/index.ts | 2 + packages/plasma-core/src/index.ts | 1 + packages/plasma-hope/api/plasma-hope.api.md | 14 +- .../src/components/ModalBase/index.ts | 2 + .../src/components/PopupBase/index.ts | 2 + packages/plasma-hope/src/index.ts | 1 + packages/plasma-web/api/plasma-web.api.md | 9 + .../ModalBase/ModalBase.stories.tsx | 139 +++++++++++++ .../src/components/ModalBase/index.ts | 2 + .../PopupBase/PopupBase.stories.tsx | 40 ++-- .../src/components/PopupBase/index.ts | 2 + packages/plasma-web/src/index.ts | 1 + 22 files changed, 678 insertions(+), 103 deletions(-) create mode 100644 packages/plasma-b2c/src/components/ModalBase/ModalBase.stories.tsx create mode 100644 packages/plasma-b2c/src/components/ModalBase/index.ts create mode 100644 packages/plasma-core/src/components/ModalBase/ModalBase.tsx create mode 100644 packages/plasma-core/src/components/ModalBase/index.ts create mode 100644 packages/plasma-hope/src/components/ModalBase/index.ts create mode 100644 packages/plasma-web/src/components/ModalBase/ModalBase.stories.tsx create mode 100644 packages/plasma-web/src/components/ModalBase/index.ts diff --git a/packages/plasma-b2c/api/plasma-b2c.api.md b/packages/plasma-b2c/api/plasma-b2c.api.md index 24815266e3..5c9f9369c8 100644 --- a/packages/plasma-b2c/api/plasma-b2c.api.md +++ b/packages/plasma-b2c/api/plasma-b2c.api.md @@ -144,6 +144,8 @@ import { MaxLinesProps } from '@salutejs/plasma-core'; import { mediaQuery } from '@salutejs/plasma-hope'; import { MediaQueryFunction } from '@salutejs/plasma-hope'; import { Modal } from '@salutejs/plasma-hope'; +import { ModalBase } from '@salutejs/plasma-hope'; +import { ModalBaseProps } from '@salutejs/plasma-hope'; import { ModalProps } from '@salutejs/plasma-hope'; import { ModalsProvider } from '@salutejs/plasma-hope'; import { ModalView } from '@salutejs/plasma-hope'; @@ -169,6 +171,7 @@ import { Popup } from '@salutejs/plasma-hope'; import { PopupBase } from '@salutejs/plasma-hope'; import { PopupBasePlacement } from '@salutejs/plasma-hope'; import { PopupBaseProps } from '@salutejs/plasma-hope'; +import { PopupBaseProvider } from '@salutejs/plasma-hope'; import { PopupProps } from '@salutejs/plasma-hope'; import { PreviewGallery } from '@salutejs/plasma-hope'; import { PreviewGalleryItemProps } from '@salutejs/plasma-hope'; @@ -559,6 +562,10 @@ export { MediaQueryFunction } export { Modal } +export { ModalBase } + +export { ModalBaseProps } + export { ModalProps } export { ModalsProvider } @@ -609,6 +616,8 @@ export { PopupBasePlacement } export { PopupBaseProps } +export { PopupBaseProvider } + export { PopupProps } export { PreviewGallery } diff --git a/packages/plasma-b2c/src/components/ModalBase/ModalBase.stories.tsx b/packages/plasma-b2c/src/components/ModalBase/ModalBase.stories.tsx new file mode 100644 index 0000000000..b32fe0ff19 --- /dev/null +++ b/packages/plasma-b2c/src/components/ModalBase/ModalBase.stories.tsx @@ -0,0 +1,139 @@ +import React from 'react'; +import styled, { createGlobalStyle } from 'styled-components'; +import { Story, Meta } from '@storybook/react'; +import { surfaceSolid02, darkOverlayBlur, overlaySoft } from '@salutejs/plasma-tokens-web'; +import { InSpacingDecorator } from '@salutejs/plasma-sb-utils'; + +import { SSRProvider } from '../SSRProvider'; +import { Button } from '../Button'; +import { PopupBaseProvider } from '../PopupBase'; + +import { ModalBase } from '.'; + +export default { + title: 'Controls/ModalBase', + decorators: [InSpacingDecorator], + argTypes: { + position: { + options: [ + 'center', + 'top', + 'bottom', + 'right', + 'left', + 'top-right', + 'top-left', + 'bottom-right', + 'bottom-left', + ], + control: { + type: 'select', + }, + }, + }, +} as Meta; + +type ModalBaseStoryProps = { + position: string; + offsetX: number; + offsetY: number; + closeOnEsc: boolean; + closeOnOverlayClick: boolean; + withBlur: boolean; +}; + +const StyledButton = styled(Button)` + margin-top: 1rem; + width: 15rem; +`; + +const StyledWrapper = styled.div` + height: 1200px; +`; + +const ModalOverlayVariables = createGlobalStyle` + body { + --plasma-modal-blur-overlay-color: ${darkOverlayBlur}; + --plasma-modal-overlay-color: ${overlaySoft}; + } +`; + +const Content = styled.div` + background: ${surfaceSolid02}; + padding: 1rem; +`; + +export const ModalBaseDemo: Story = ({ position, offsetX, offsetY, ...rest }) => { + const [isOpenA, setIsOpenA] = React.useState(false); + const [isOpenB, setIsOpenB] = React.useState(false); + const [isOpenC, setIsOpenC] = React.useState(false); + + return ( + + + + +
+ setIsOpenA(true)} /> +
+ setIsOpenA(false)} + isOpen={isOpenA} + position={position} + offset={[offsetX, offsetY]} + {...rest} + > + + +
+ setIsOpenB(true)} /> +
+ setIsOpenB(false)} + isOpen={isOpenB} + position="left" + offset={[offsetX, offsetY]} + {...rest} + > + + +
+ setIsOpenC(true)} /> +
+ setIsOpenC(false)} + isOpen={isOpenC} + position="top" + offset={[offsetX, offsetY]} + {...rest} + > + + + <>Content + + +
+
+
+
+
+
+
+ ); +}; + +ModalBaseDemo.args = { + position: 'center', + withBlur: false, + closeOnEsc: true, + closeOnOverlayClick: true, + offsetX: 0, + offsetY: 0, +}; diff --git a/packages/plasma-b2c/src/components/ModalBase/index.ts b/packages/plasma-b2c/src/components/ModalBase/index.ts new file mode 100644 index 0000000000..3d4e2af155 --- /dev/null +++ b/packages/plasma-b2c/src/components/ModalBase/index.ts @@ -0,0 +1,2 @@ +export { ModalBase } from '@salutejs/plasma-hope'; +export type { ModalBaseProps } from '@salutejs/plasma-hope'; diff --git a/packages/plasma-b2c/src/components/PopupBase/index.ts b/packages/plasma-b2c/src/components/PopupBase/index.ts index 312f45f3bf..674b423a6a 100644 --- a/packages/plasma-b2c/src/components/PopupBase/index.ts +++ b/packages/plasma-b2c/src/components/PopupBase/index.ts @@ -1,2 +1,4 @@ +export { PopupBaseProvider } from '@salutejs/plasma-hope'; + export { PopupBase } from '@salutejs/plasma-hope'; export type { PopupBaseProps, PopupBasePlacement } from '@salutejs/plasma-hope'; diff --git a/packages/plasma-b2c/src/index.ts b/packages/plasma-b2c/src/index.ts index adf78596b0..d773f0fa96 100644 --- a/packages/plasma-b2c/src/index.ts +++ b/packages/plasma-b2c/src/index.ts @@ -14,6 +14,7 @@ export * from './components/Image'; export * from './components/Link'; export * from './components/List'; export * from './components/Modal'; +export * from './components/ModalBase'; export * from './components/Notification'; export * from './components/PaginationDots'; export * from './components/Popup'; diff --git a/packages/plasma-core/api/plasma-core.api.md b/packages/plasma-core/api/plasma-core.api.md index 0f3f58f1ae..681098977f 100644 --- a/packages/plasma-core/api/plasma-core.api.md +++ b/packages/plasma-core/api/plasma-core.api.md @@ -25,6 +25,7 @@ import { Keyframes } from 'styled-components'; import { MutableRefObject } from 'react'; import { default as React_2 } from 'react'; import { ReactNode } from 'react'; +import { RefObject } from 'react'; import { spacing } from '@salutejs/plasma-typo'; import { SpacingProps } from '@salutejs/plasma-typo'; import { SpacingProps as SpacingProps_2 } from '@salutejs/plasma-typo/lib/cjs/mixins/applySpacing'; @@ -752,6 +753,23 @@ export interface MaxLinesProps { maxLines?: number; } +// @public +export const ModalBase: FC; + +// @public (undocumented) +export interface ModalBaseProps extends Omit { + closeButtonAriaLabel?: string; + closeOnEsc?: boolean; + closeOnOverlayClick?: boolean; + focusAfterRef?: React_2.RefObject; + initialFocusRef?: React_2.RefObject; + onClose?: () => void; + onEscKeyDown?: (event: KeyboardEvent) => void; + onOverlayClick?: (event: React_2.MouseEvent) => void; + showCloseButton?: boolean; + withBlur?: boolean; +} + // @public (undocumented) export const monthLongName: (val: number) => string; @@ -819,7 +837,9 @@ export interface PinProps { export const Popup: React_2.NamedExoticComponent>; // @public -export const PopupBase: FC; +export const PopupBase: React_2.ForwardRefExoticComponent>; // Warning: (ae-forgotten-export) The symbol "BasicPopupBasePlacement" needs to be exported by the entry point index.d.ts // Warning: (ae-forgotten-export) The symbol "MixedPopupBasePlacement" needs to be exported by the entry point index.d.ts @@ -841,6 +861,11 @@ export interface PopupBaseProps extends React_2.HTMLAttributes { zIndex?: string; } +// @public (undocumented) +export const PopupBaseProvider: React_2.FC<{ + children: ReactNode; +}>; + // @public (undocumented) export interface PopupProps extends HTMLAttributes { children?: ReactNode; @@ -1264,6 +1289,9 @@ export type UseCarouselOptions = Pick any, delay: number, cleanUp?: boolean): (...args: any[]) => void; +// @public +export const useFocusTrap: (active?: boolean, firstFocusSelector?: string | RefObject | undefined, focusAfterNode?: RefObject | undefined) => (instance: HTMLElement | null) => void; + // Warning: (ae-forgotten-export) The symbol "UseForkRefHook" needs to be exported by the entry point index.d.ts // // @public @@ -1381,6 +1409,7 @@ export interface WithSkeletonProps { // Warnings were encountered during analysis: // +// components/PopupBase/PopupBase.d.ts:41:5 - (ae-forgotten-export) The symbol "PopupInfo" needs to be exported by the entry point index.d.ts // components/Toast/Toast.d.ts:4:5 - (ae-forgotten-export) The symbol "ToastRole" needs to be exported by the entry point index.d.ts // components/Toast/useToast.d.ts:2:5 - (ae-forgotten-export) The symbol "ShowToast" needs to be exported by the entry point index.d.ts diff --git a/packages/plasma-core/src/components/ModalBase/ModalBase.tsx b/packages/plasma-core/src/components/ModalBase/ModalBase.tsx new file mode 100644 index 0000000000..e0c9ca2f06 --- /dev/null +++ b/packages/plasma-core/src/components/ModalBase/ModalBase.tsx @@ -0,0 +1,185 @@ +import React, { useCallback, useContext, FC, useEffect } from 'react'; +import styled, { createGlobalStyle, css } from 'styled-components'; + +import { useFocusTrap, useUniqId } from '../../hooks'; +import { PopupBaseContext, PopupInfo } from '../PopupBase/PopupBaseContext'; +import { DEFAULT_Z_INDEX, PopupBase, PopupBaseProps } from '../PopupBase/PopupBase'; + +export interface ModalBaseProps extends Omit { + /** + * Нужно ли применять blur для подложки. + */ + withBlur?: boolean; + /** + * Закрывать модальное окно при нажатии на ESC(по умолчанию true). + */ + closeOnEsc?: boolean; + /** + * Закрывать модальное окно при нажатии вне области модального окна(по умолчанию true), + */ + closeOnOverlayClick?: boolean; + /** + * Обработчик клика при нажатии на ESC(если не передан, то при нажатии используется onClose). + */ + onEscKeyDown?: (event: KeyboardEvent) => void; + /** + * Обработчик клика при нажатии вне области модального окна(если не передан, то при нажатии используется onClose). + */ + onOverlayClick?: (event: React.MouseEvent) => void; + /** + * Первый элемент для фокуса внутри модального окна. + */ + initialFocusRef?: React.RefObject; + /** + * Элемент для фокуса после закрытия модального окна + * (по умолчанию фокус на последнем перед открытием активном элементе). + */ + focusAfterRef?: React.RefObject; + /** + * Общий обработчик клика по кнопке "закрыть". + */ + onClose?: () => void; + /** + * WAI-ARIA атрибут кнопки "закрыть". + */ + closeButtonAriaLabel?: string; + /** + * Отображать кнопку "закрыть"(по умолчанию true). + */ + showCloseButton?: boolean; +} + +const StyledOverlay = styled.div<{ transparent?: boolean; $withBlur?: boolean; clickable?: boolean; zIndex?: string }>` + position: absolute; + + width: 100%; + height: 100%; + + top: 0; + left: 0; + + ${({ zIndex }) => css` + z-index: ${zIndex || DEFAULT_Z_INDEX}; + `} + + ${({ $withBlur }) => { + return css` + --background-color: ${$withBlur + ? 'var(--plasma-modal-blur-overlay-color)' + : 'var(--plasma-modal-overlay-color)'}; + --backdrop-filter: ${$withBlur ? 'blur(1rem)' : 'none'}; + `; + }}; + + background-color: ${({ transparent }) => (transparent ? 'transparent' : 'var(--background-color)')}; + backdrop-filter: var(--backdrop-filter); + cursor: ${({ clickable }) => (clickable ? 'pointer' : 'default')}; +`; + +const NoScroll = createGlobalStyle` + body { + overflow-y: hidden; + } +`; + +const ESCAPE_KEYCODE = 27; + +/** + * ModalBase. + * Управляет показом/скрытием, подложкой и анимацией визуальной части модального окна. + */ +export const ModalBase: FC = ({ + id, + isOpen, + position, + onClose, + onOverlayClick, + onEscKeyDown, + closeOnEsc = true, + closeOnOverlayClick = true, + withBlur, + initialFocusRef, + focusAfterRef, + children, + ...rest +}) => { + const uniqId = useUniqId(); + const innerId = id || uniqId; + const controller = useContext(PopupBaseContext); + + const trapRef = useFocusTrap(true, initialFocusRef, focusAfterRef); + + const onOverlayClickCallback = useCallback( + (event: React.MouseEvent) => { + if (!closeOnOverlayClick) { + return; + } + + if (onOverlayClick) { + onOverlayClick(event); + return; + } + + onClose?.(); + }, + [closeOnOverlayClick, onOverlayClick, onClose], + ); + + // Вызов обработчика текущего окна + const onOverlayKeyDown = useCallback( + (event: React.MouseEvent) => { + controller.callCurrentModalClose(event); + }, + [controller.items], + ); + + // При ESC закрывает текущее окно, если это возможно + const onKeyDown = useCallback( + (event: KeyboardEvent) => { + if (closeOnEsc && event.keyCode === ESCAPE_KEYCODE && controller.getIdLastModal() === innerId) { + if (onEscKeyDown) { + onEscKeyDown(event); + return; + } + + onClose?.(); + } + }, + [onClose, onEscKeyDown, controller.items, closeOnEsc], + ); + + useEffect(() => { + window.addEventListener('keydown', onKeyDown); + return () => { + window.removeEventListener('keydown', onKeyDown); + }; + }, [onClose, onEscKeyDown, controller.items, closeOnEsc]); + + const modalInfo: PopupInfo = { + id: innerId, + isModal: true, + onOverlayClick: onOverlayClickCallback, + }; + + if (!isOpen) { + return null; + } + + return ( + <> + + + + {children} + + + ); +}; diff --git a/packages/plasma-core/src/components/ModalBase/index.ts b/packages/plasma-core/src/components/ModalBase/index.ts new file mode 100644 index 0000000000..f6c7678e33 --- /dev/null +++ b/packages/plasma-core/src/components/ModalBase/index.ts @@ -0,0 +1,2 @@ +export { ModalBase } from './ModalBase'; +export type { ModalBaseProps } from './ModalBase'; diff --git a/packages/plasma-core/src/components/PopupBase/PopupBase.tsx b/packages/plasma-core/src/components/PopupBase/PopupBase.tsx index 897231432d..5ffc3cb0a5 100644 --- a/packages/plasma-core/src/components/PopupBase/PopupBase.tsx +++ b/packages/plasma-core/src/components/PopupBase/PopupBase.tsx @@ -1,10 +1,10 @@ -import React, { useEffect, useRef, useState, useContext, FC } from 'react'; +import React, { useEffect, useRef, useState, useContext } from 'react'; import ReactDOM from 'react-dom'; import styled, { Keyframes, css } from 'styled-components'; import { useUniqId } from '../../hooks'; -import { PopupBaseContext, POPOVER_PORTAL_ID } from './PopupBaseContext'; +import { PopupBaseContext, POPOVER_PORTAL_ID, PopupInfo } from './PopupBaseContext'; type BasicPopupBasePlacement = 'center' | 'top' | 'bottom' | 'right' | 'left'; type MixedPopupBasePlacement = 'top-right' | 'top-left' | 'bottom-right' | 'bottom-left'; @@ -144,79 +144,91 @@ const PopupBaseRoot = styled.div` * Базовый PopupBase. * Управляет показом/скрытием и анимацией(?) высплывающего окна. */ -export const PopupBase: FC = ({ - id, - isOpen, - position, - offset, - frame = 'document', - children, - role, - zIndex, - showAnimation, - hideAnimation, - ...rest -}) => { - const uniqId = useUniqId(); - const innerId = id || uniqId; - - const portalRef = useRef(null); - - const controller = useContext(PopupBaseContext); - - const [, forceRender] = useState(false); - - useEffect(() => { - let portal = document.getElementById(POPOVER_PORTAL_ID); - - if (frame !== 'document' && frame && frame.current) { - portal = frame.current; +export const PopupBase = React.forwardRef( + ( + { + id, + isOpen, + position, + offset, + frame = 'document', + children, + role, + zIndex, + showAnimation, + hideAnimation, + popupInfo, + ...rest + }, + ref, + ) => { + const uniqId = useUniqId(); + const innerId = id || uniqId; + + const portalRef = useRef(null); + + const controller = useContext(PopupBaseContext); + + const [, forceRender] = useState(false); + + useEffect(() => { + let portal = document.getElementById(POPOVER_PORTAL_ID); + + if (frame !== 'document' && frame && frame.current) { + portal = frame.current; + } + + if (!portal) { + portal = document.createElement('div'); + portal.setAttribute('id', POPOVER_PORTAL_ID); + document.body.appendChild(portal); + } + + portalRef.current = portal; + + /** + * Изменение стейта нужно для того, чтобы PopupBase + * отобразился после записи DOM элемента в portalRef.current + */ + forceRender(true); + + return () => { + controller.unregister(innerId); + }; + }, [controller, innerId, zIndex]); + + useEffect(() => { + if (isOpen) { + controller.register({ id: innerId, ...popupInfo }); + } else { + controller.unregister(innerId); + } + }, [isOpen]); + + if (!isOpen) { + return null; } - if (!portal) { - portal = document.createElement('div'); - portal.setAttribute('id', POPOVER_PORTAL_ID); - document.body.appendChild(portal); - } - - portalRef.current = portal; - - /** - * Изменение стейта нужно для того, чтобы PopupBase - * отобразился после записи DOM элемента в portalRef.current - */ - forceRender(true); - - return () => { - controller.unregister(innerId); - }; - }, [controller, innerId, zIndex]); - - if (isOpen) { - controller.register(innerId); - } else { - controller.unregister(innerId); - return null; - } - - return ( - <> - {portalRef.current && - ReactDOM.createPortal( - - - {children} - - , - portalRef.current, - )} - - ); -}; + return ( + <> + {portalRef.current && + ReactDOM.createPortal( + + + {children} + + , + portalRef.current, + )} + + ); + }, +); diff --git a/packages/plasma-core/src/components/PopupBase/PopupBaseContext.tsx b/packages/plasma-core/src/components/PopupBase/PopupBaseContext.tsx index 5b519ff416..e0bedb081f 100644 --- a/packages/plasma-core/src/components/PopupBase/PopupBaseContext.tsx +++ b/packages/plasma-core/src/components/PopupBase/PopupBaseContext.tsx @@ -1,17 +1,40 @@ import React, { ReactNode, useEffect } from 'react'; +export interface PopupInfo { + id: string; + isModal?: true; + onOverlayClick?: (event: React.MouseEvent) => void; +} + /** * Хранилище модальных окон. */ class PopupBaseController { - public items: string[] = []; + public items: PopupInfo[] = []; - public register(id: string) { - return this.items.push(id); + public register(info: PopupInfo) { + return this.items.push(info); } public unregister(id: string) { - this.items.splice(this.items.indexOf(id), 1); + const index = this.items.findIndex((item: PopupInfo) => id === item.id); + if (index === -1) { + return; + } + this.items.splice(index, 1); + } + + getLastModal() { + const modals = this.items.filter((item: PopupInfo) => item.isModal); + return modals && modals[modals.length - 1]; + } + + public getIdLastModal() { + return this.getLastModal()?.id; + } + + public callCurrentModalClose(event: React.MouseEvent) { + this.getLastModal()?.onOverlayClick?.(event); } } diff --git a/packages/plasma-core/src/components/PopupBase/index.ts b/packages/plasma-core/src/components/PopupBase/index.ts index e9ec8717f5..131f3df0d0 100644 --- a/packages/plasma-core/src/components/PopupBase/index.ts +++ b/packages/plasma-core/src/components/PopupBase/index.ts @@ -1,2 +1,4 @@ +export { PopupBaseProvider } from './PopupBaseContext'; + export { PopupBase } from './PopupBase'; export type { PopupBaseProps, PopupBasePlacement } from './PopupBase'; diff --git a/packages/plasma-core/src/index.ts b/packages/plasma-core/src/index.ts index d80b6d93e6..9540f0622e 100644 --- a/packages/plasma-core/src/index.ts +++ b/packages/plasma-core/src/index.ts @@ -6,6 +6,7 @@ export * from './components/Fade'; export * from './components/Field'; export * from './components/Image'; export * from './components/Input'; +export * from './components/ModalBase'; export * from './components/PaginationDots'; export * from './components/Popup'; export * from './components/PopupBase'; diff --git a/packages/plasma-hope/api/plasma-hope.api.md b/packages/plasma-hope/api/plasma-hope.api.md index 9a36eed62c..4627a0dc2f 100644 --- a/packages/plasma-hope/api/plasma-hope.api.md +++ b/packages/plasma-hope/api/plasma-hope.api.md @@ -96,6 +96,8 @@ import { KeyboardEvent as KeyboardEvent_2 } from 'react'; import { LineSkeleton } from '@salutejs/plasma-core'; import { LineSkeletonProps } from '@salutejs/plasma-core'; import { MaxLinesProps } from '@salutejs/plasma-core'; +import { ModalBase } from '@salutejs/plasma-core'; +import { ModalBaseProps } from '@salutejs/plasma-core'; import { monthLongName } from '@salutejs/plasma-core'; import { monthShortName } from '@salutejs/plasma-core'; import { MutableRefObject } from 'react'; @@ -115,6 +117,7 @@ import { Popup } from '@salutejs/plasma-core'; import { PopupBase } from '@salutejs/plasma-core'; import { PopupBasePlacement } from '@salutejs/plasma-core'; import { PopupBaseProps } from '@salutejs/plasma-core'; +import { PopupBaseProvider } from '@salutejs/plasma-core'; import { PopupProps } from '@salutejs/plasma-core'; import type { PriceProps as PriceProps_2 } from '@salutejs/plasma-core'; import { PropsWithChildren } from 'react'; @@ -126,7 +129,6 @@ import { ReactNode } from 'react'; import { RectSkeleton } from '@salutejs/plasma-core'; import { RectSkeletonProps } from '@salutejs/plasma-core'; import { RefAttributes } from 'react'; -import { RefObject } from 'react'; import { Roundness } from '@salutejs/plasma-core'; import { RoundnessProps } from '@salutejs/plasma-core'; import { ScrollSnapProps } from '@salutejs/plasma-core'; @@ -169,6 +171,7 @@ import { transformStyles } from '@salutejs/plasma-core'; import { TypographyTypes } from '@salutejs/plasma-core'; import { Underline } from '@salutejs/plasma-core'; import { useDebouncedFunction } from '@salutejs/plasma-core'; +import { useFocusTrap } from '@salutejs/plasma-core'; import { useForkRef } from '@salutejs/plasma-core'; import { useIsomorphicLayoutEffect } from '@salutejs/plasma-core'; import { useResizeObserver } from '@salutejs/plasma-core'; @@ -909,6 +912,10 @@ export type MediaQueryFunction = (content: FlattenSimpleInterpolation | string) // @public export const Modal: FC; +export { ModalBase } + +export { ModalBaseProps } + // @public (undocumented) export interface ModalProps extends ModalViewProps { closeOnEsc?: boolean; @@ -1005,6 +1012,8 @@ export { PopupBasePlacement } export { PopupBaseProps } +export { PopupBaseProvider } + export { PopupProps } // @public @@ -1359,8 +1368,7 @@ export const useAutoResize: (active: boolean, ref export { useDebouncedFunction } -// @public -export const useFocusTrap: (active?: boolean, firstFocusSelector?: string | RefObject | undefined, focusAfterNode?: RefObject | undefined) => (instance: HTMLElement | null) => void; +export { useFocusTrap } export { useForkRef } diff --git a/packages/plasma-hope/src/components/ModalBase/index.ts b/packages/plasma-hope/src/components/ModalBase/index.ts new file mode 100644 index 0000000000..639a988486 --- /dev/null +++ b/packages/plasma-hope/src/components/ModalBase/index.ts @@ -0,0 +1,2 @@ +export { ModalBase } from '@salutejs/plasma-core'; +export type { ModalBaseProps } from '@salutejs/plasma-core'; diff --git a/packages/plasma-hope/src/components/PopupBase/index.ts b/packages/plasma-hope/src/components/PopupBase/index.ts index be803c11df..854ec76fc3 100644 --- a/packages/plasma-hope/src/components/PopupBase/index.ts +++ b/packages/plasma-hope/src/components/PopupBase/index.ts @@ -1,2 +1,4 @@ +export { PopupBaseProvider } from '@salutejs/plasma-core'; + export { PopupBase } from '@salutejs/plasma-core'; export type { PopupBaseProps, PopupBasePlacement } from '@salutejs/plasma-core'; diff --git a/packages/plasma-hope/src/index.ts b/packages/plasma-hope/src/index.ts index 1fb3c72fc0..8ec604f333 100644 --- a/packages/plasma-hope/src/index.ts +++ b/packages/plasma-hope/src/index.ts @@ -14,6 +14,7 @@ export * from './components/Image'; export * from './components/Link'; export * from './components/List'; export * from './components/Modal'; +export * from './components/ModalBase'; export * from './components/Notification'; export * from './components/PaginationDots'; export * from './components/Popup'; diff --git a/packages/plasma-web/api/plasma-web.api.md b/packages/plasma-web/api/plasma-web.api.md index 8d7d5586b4..8c0cf6e737 100644 --- a/packages/plasma-web/api/plasma-web.api.md +++ b/packages/plasma-web/api/plasma-web.api.md @@ -144,6 +144,8 @@ import { MaxLinesProps } from '@salutejs/plasma-core'; import { mediaQuery } from '@salutejs/plasma-hope'; import { MediaQueryFunction } from '@salutejs/plasma-hope'; import { Modal } from '@salutejs/plasma-hope'; +import { ModalBase } from '@salutejs/plasma-hope'; +import { ModalBaseProps } from '@salutejs/plasma-hope'; import { ModalProps } from '@salutejs/plasma-hope'; import { ModalsProvider } from '@salutejs/plasma-hope'; import { ModalView } from '@salutejs/plasma-hope'; @@ -169,6 +171,7 @@ import { Popup } from '@salutejs/plasma-hope'; import { PopupBase } from '@salutejs/plasma-hope'; import { PopupBasePlacement } from '@salutejs/plasma-hope'; import { PopupBaseProps } from '@salutejs/plasma-hope'; +import { PopupBaseProvider } from '@salutejs/plasma-hope'; import { PopupProps } from '@salutejs/plasma-hope'; import { PreviewGallery } from '@salutejs/plasma-hope'; import { PreviewGalleryItemProps } from '@salutejs/plasma-hope'; @@ -559,6 +562,10 @@ export { MediaQueryFunction } export { Modal } +export { ModalBase } + +export { ModalBaseProps } + export { ModalProps } export { ModalsProvider } @@ -609,6 +616,8 @@ export { PopupBasePlacement } export { PopupBaseProps } +export { PopupBaseProvider } + export { PopupProps } export { PreviewGallery } diff --git a/packages/plasma-web/src/components/ModalBase/ModalBase.stories.tsx b/packages/plasma-web/src/components/ModalBase/ModalBase.stories.tsx new file mode 100644 index 0000000000..eddcea2bee --- /dev/null +++ b/packages/plasma-web/src/components/ModalBase/ModalBase.stories.tsx @@ -0,0 +1,139 @@ +import React from 'react'; +import styled, { createGlobalStyle } from 'styled-components'; +import { Story, Meta } from '@storybook/react'; +import { surfaceSolid02, darkOverlayBlur, overlaySoft } from '@salutejs/plasma-tokens-web'; + +import { SSRProvider } from '../SSRProvider'; +import { InSpacingDecorator } from '../../helpers'; +import { Button } from '../Button'; +import { PopupBaseProvider } from '../PopupBase'; + +import { ModalBase } from '.'; + +export default { + title: 'Controls/ModalBase', + decorators: [InSpacingDecorator], + argTypes: { + position: { + options: [ + 'center', + 'top', + 'bottom', + 'right', + 'left', + 'top-right', + 'top-left', + 'bottom-right', + 'bottom-left', + ], + control: { + type: 'select', + }, + }, + }, +} as Meta; + +type ModalBaseStoryProps = { + position: string; + offsetX: number; + offsetY: number; + closeOnEsc: boolean; + closeOnOverlayClick: boolean; + withBlur: boolean; +}; + +const StyledButton = styled(Button)` + margin-top: 1rem; + width: 15rem; +`; + +const StyledWrapper = styled.div` + height: 1200px; +`; + +const ModalOverlayVariables = createGlobalStyle` + body { + --plasma-modal-blur-overlay-color: ${darkOverlayBlur}; + --plasma-modal-overlay-color: ${overlaySoft}; + } +`; + +const Content = styled.div` + background: ${surfaceSolid02}; + padding: 1rem; +`; + +export const ModalBaseDemo: Story = ({ position, offsetX, offsetY, ...rest }) => { + const [isOpenA, setIsOpenA] = React.useState(false); + const [isOpenB, setIsOpenB] = React.useState(false); + const [isOpenC, setIsOpenC] = React.useState(false); + + return ( + + + + +
+ setIsOpenA(true)} /> +
+ setIsOpenA(false)} + isOpen={isOpenA} + position={position} + offset={[offsetX, offsetY]} + {...rest} + > + + +
+ setIsOpenB(true)} /> +
+ setIsOpenB(false)} + isOpen={isOpenB} + position="left" + offset={[offsetX, offsetY]} + {...rest} + > + + +
+ setIsOpenC(true)} /> +
+ setIsOpenC(false)} + isOpen={isOpenC} + position="top" + offset={[offsetX, offsetY]} + {...rest} + > + + + <>Content + + +
+
+
+
+
+
+
+ ); +}; + +ModalBaseDemo.args = { + position: 'center', + withBlur: false, + closeOnEsc: true, + closeOnOverlayClick: true, + offsetX: 0, + offsetY: 0, +}; diff --git a/packages/plasma-web/src/components/ModalBase/index.ts b/packages/plasma-web/src/components/ModalBase/index.ts new file mode 100644 index 0000000000..3d4e2af155 --- /dev/null +++ b/packages/plasma-web/src/components/ModalBase/index.ts @@ -0,0 +1,2 @@ +export { ModalBase } from '@salutejs/plasma-hope'; +export type { ModalBaseProps } from '@salutejs/plasma-hope'; diff --git a/packages/plasma-web/src/components/PopupBase/PopupBase.stories.tsx b/packages/plasma-web/src/components/PopupBase/PopupBase.stories.tsx index 994d2eb9e2..82a69e8949 100644 --- a/packages/plasma-web/src/components/PopupBase/PopupBase.stories.tsx +++ b/packages/plasma-web/src/components/PopupBase/PopupBase.stories.tsx @@ -7,7 +7,7 @@ import { SSRProvider } from '../SSRProvider'; import { InSpacingDecorator } from '../../helpers'; import { Button } from '../Button'; -import { PopupBase } from '.'; +import { PopupBase, PopupBaseProvider } from '.'; export default { title: 'Controls/PopupBase', @@ -91,25 +91,27 @@ export const PopupBaseDemo: Story = ({ position, offsetX, o return ( -
- setIsOpenA(true)} /> - setIsOpenB(true)} /> -
- -
- - <>Content + +
+ setIsOpenA(true)} /> + setIsOpenB(true)} />
- - - <>Frame - - -
- - <>Content -
-
+ +
+ + <>Content +
+
+ + <>Frame + + +
+ + <>Content +
+
+
); diff --git a/packages/plasma-web/src/components/PopupBase/index.ts b/packages/plasma-web/src/components/PopupBase/index.ts index 312f45f3bf..674b423a6a 100644 --- a/packages/plasma-web/src/components/PopupBase/index.ts +++ b/packages/plasma-web/src/components/PopupBase/index.ts @@ -1,2 +1,4 @@ +export { PopupBaseProvider } from '@salutejs/plasma-hope'; + export { PopupBase } from '@salutejs/plasma-hope'; export type { PopupBaseProps, PopupBasePlacement } from '@salutejs/plasma-hope'; diff --git a/packages/plasma-web/src/index.ts b/packages/plasma-web/src/index.ts index 1dc6f57822..1a6210d93a 100644 --- a/packages/plasma-web/src/index.ts +++ b/packages/plasma-web/src/index.ts @@ -14,6 +14,7 @@ export * from './components/Image'; export * from './components/Link'; export * from './components/List'; export * from './components/Modal'; +export * from './components/ModalBase'; export * from './components/Notification'; export * from './components/PaginationDots'; export * from './components/Popup';