From fe9656efabe6ca5b130fc1949efab62b3ee6802c 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 | 12 ++ .../ModalBase/ModalBase.stories.tsx | 139 ++++++++++++++ .../src/components/ModalBase/index.ts | 2 + .../PopupBase/PopupBase.stories.tsx | 42 +++-- .../src/components/PopupBase/index.ts | 2 + packages/plasma-b2c/src/index.ts | 1 + packages/plasma-core/api/plasma-core.api.md | 29 ++- .../src/components/ModalBase/ModalBase.tsx | 177 ++++++++++++++++++ .../src/components/ModalBase/index.ts | 2 + .../src/components/PopupBase/PopupBase.tsx | 128 ++++++------- .../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 | 12 ++ .../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 | 12 ++ .../ModalBase/ModalBase.stories.tsx | 139 ++++++++++++++ .../src/components/ModalBase/index.ts | 2 + .../PopupBase/PopupBase.stories.tsx | 42 +++-- .../src/components/PopupBase/index.ts | 2 + packages/plasma-web/src/index.ts | 1 + 23 files changed, 675 insertions(+), 108 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 30a570ab87..a8301a6417 100644 --- a/packages/plasma-b2c/api/plasma-b2c.api.md +++ b/packages/plasma-b2c/api/plasma-b2c.api.md @@ -146,6 +146,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'; @@ -172,8 +174,10 @@ import { PopoverPlacement } from '@salutejs/plasma-hope'; import { PopoverProps } from '@salutejs/plasma-hope'; import { Popup } from '@salutejs/plasma-hope'; import { PopupBase } from '@salutejs/plasma-hope'; +import { PopupBaseContext } 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'; @@ -591,6 +595,10 @@ export { MediaQueryFunction } export { Modal } +export { ModalBase } + +export { ModalBaseProps } + export { ModalProps } export { ModalsProvider } @@ -643,10 +651,14 @@ export { Popup } export { PopupBase } +export { PopupBaseContext } + 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..b45c90b3d9 --- /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: { + placement: { + options: [ + 'center', + 'top', + 'bottom', + 'right', + 'left', + 'top-right', + 'top-left', + 'bottom-right', + 'bottom-left', + ], + control: { + type: 'select', + }, + }, + }, +} as Meta; + +type ModalBaseStoryProps = { + placement: 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 = ({ placement, 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} + placement={placement} + offset={[offsetX, offsetY]} + {...rest} + > + + +
+ setIsOpenB(true)} /> +
+ setIsOpenB(false)} + isOpen={isOpenB} + placement="left" + offset={[offsetX, offsetY]} + {...rest} + > + + +
+ setIsOpenC(true)} /> +
+ setIsOpenC(false)} + isOpen={isOpenC} + placement="top" + offset={[offsetX, offsetY]} + {...rest} + > + + + <>Content + + +
+
+
+
+
+
+
+ ); +}; + +ModalBaseDemo.args = { + placement: '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/PopupBase.stories.tsx b/packages/plasma-b2c/src/components/PopupBase/PopupBase.stories.tsx index a2f3220db9..a83ab16898 100644 --- a/packages/plasma-b2c/src/components/PopupBase/PopupBase.stories.tsx +++ b/packages/plasma-b2c/src/components/PopupBase/PopupBase.stories.tsx @@ -7,7 +7,7 @@ import { surfaceSolid03, surfaceSolid02 } from '@salutejs/plasma-tokens-web'; import { SSRProvider } from '../SSRProvider'; import { Button } from '../Button'; -import { PopupBase } from '.'; +import { PopupBase, PopupBaseProvider } from '.'; export default { title: 'Controls/PopupBase', @@ -73,25 +73,27 @@ export const PopupBaseDemo: Story = ({ placement, offsetX, return ( -
- setIsOpenA(true)} /> - setIsOpenB(true)} /> -
- - - - <>Content - - - - <>Frame - - - - - <>Content - - + +
+ setIsOpenA(true)} /> + setIsOpenB(true)} /> +
+ + + + <>Content + + + + <>Frame + + + + + <>Content + + +
); diff --git a/packages/plasma-b2c/src/components/PopupBase/index.ts b/packages/plasma-b2c/src/components/PopupBase/index.ts index 312f45f3bf..8a274eead0 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, PopupBaseContext } 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 648424960b..42433d06fc 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 031e167033..b514cf302e 100644 --- a/packages/plasma-core/api/plasma-core.api.md +++ b/packages/plasma-core/api/plasma-core.api.md @@ -752,6 +752,21 @@ export interface MaxLinesProps { maxLines?: number; } +// @public +export const ModalBase: FC; + +// @public (undocumented) +export interface ModalBaseProps extends Omit { + closeOnEsc?: boolean; + closeOnOverlayClick?: boolean; + focusAfterRef?: React_2.RefObject; + initialFocusRef?: React_2.RefObject; + onClose?: () => void; + onEscKeyDown?: (event: KeyboardEvent) => void; + onOverlayClick?: (event: React_2.MouseEvent) => void; + withBlur?: boolean; +} + // @public (undocumented) export const monthLongName: (val: number) => string; @@ -842,7 +857,12 @@ export interface PopoverProps extends HTMLAttributes { export const Popup: React_2.NamedExoticComponent>; // @public -export const PopupBase: FC; +export const PopupBase: React_2.ForwardRefExoticComponent>; + +// Warning: (ae-forgotten-export) The symbol "PopupBaseController" needs to be exported by the entry point index.d.ts +// +// @public (undocumented) +export const PopupBaseContext: React_2.Context; // 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 @@ -859,9 +879,16 @@ export interface PopupBaseProps extends React_2.HTMLAttributes { offset?: [number | string, number | string]; // (undocumented) placement?: PopupBasePlacement; + // Warning: (ae-forgotten-export) The symbol "PopupInfo" needs to be exported by the entry point index.d.ts + popupInfo?: PopupInfo; zIndex?: string; } +// @public (undocumented) +export const PopupBaseProvider: React_2.FC<{ + children: ReactNode; +}>; + // @public (undocumented) export interface PopupProps extends HTMLAttributes { children?: ReactNode; 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..1fa14101d3 --- /dev/null +++ b/packages/plasma-core/src/components/ModalBase/ModalBase.tsx @@ -0,0 +1,177 @@ +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; +} + +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, + placement, + 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 2da0686ae2..fea575f1c2 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, { 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'; @@ -37,6 +37,10 @@ export interface PopupBaseProps extends React.HTMLAttributes { * Значение z-index для PopupBase. */ zIndex?: string; + /** + * Дополнительная информация для программного взаимодействия с окном через контекст. + */ + popupInfo?: PopupInfo; } interface HidingProps { @@ -132,70 +136,68 @@ const PopupBaseRoot = styled.div` * Базовый PopupBase. * Управляет показом/скрытием и анимацией(?) высплывающего окна. */ -export const PopupBase: FC = ({ - id, - isOpen, - placement, - offset, - frame = 'document', - children, - role, - zIndex, - ...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; - } - if (!portal) { - portal = document.createElement('div'); - portal.setAttribute('id', POPOVER_PORTAL_ID); - document.body.appendChild(portal); - } +export const PopupBase = React.forwardRef( + ({ id, isOpen, placement, offset, frame = 'document', children, role, zIndex, popupInfo, ...rest }, ref) => { + const uniqId = useUniqId(); + const innerId = id || uniqId; - portalRef.current = portal; + const portalRef = useRef(null); - /** - * Изменение стейта нужно для того, чтобы PopupBase - * отобразился после записи DOM элемента в portalRef.current - */ - forceRender(true); + const controller = useContext(PopupBaseContext); - return () => { - controller.unregister(innerId); - }; - }, [controller, innerId, zIndex]); + const [, forceRender] = useState(false); - if (isOpen) { - controller.register(innerId); - } else { - controller.unregister(innerId); - return null; - } + useEffect(() => { + let portal = document.getElementById(POPOVER_PORTAL_ID); - return ( - <> - {portalRef.current && - ReactDOM.createPortal( - - - {children} - - , - portalRef.current, - )} - - ); -}; + 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; + } + + 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..a8736de25a 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, PopupBaseContext } 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 2774fbe9f9..48de7019c2 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 38fe634d0d..bb3da6f144 100644 --- a/packages/plasma-hope/api/plasma-hope.api.md +++ b/packages/plasma-hope/api/plasma-hope.api.md @@ -94,6 +94,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'; @@ -114,8 +116,10 @@ import { PopoverPlacement } from '@salutejs/plasma-core'; import { PopoverProps } from '@salutejs/plasma-core'; import { Popup } from '@salutejs/plasma-core'; import { PopupBase } from '@salutejs/plasma-core'; +import { PopupBaseContext } 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'; @@ -898,6 +902,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; @@ -996,10 +1004,14 @@ export { Popup } export { PopupBase } +export { PopupBaseContext } + export { PopupBasePlacement } export { PopupBaseProps } +export { PopupBaseProvider } + export { PopupProps } // @public 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..9636e9486c 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, PopupBaseContext } 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 4e857428c4..dc90fd1178 100644 --- a/packages/plasma-hope/src/index.ts +++ b/packages/plasma-hope/src/index.ts @@ -12,6 +12,7 @@ export * from './components/Grid'; export * from './components/Image'; 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 e10f1c54dc..68ed111577 100644 --- a/packages/plasma-web/api/plasma-web.api.md +++ b/packages/plasma-web/api/plasma-web.api.md @@ -146,6 +146,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'; @@ -172,8 +174,10 @@ import { PopoverPlacement } from '@salutejs/plasma-hope'; import { PopoverProps } from '@salutejs/plasma-hope'; import { Popup } from '@salutejs/plasma-hope'; import { PopupBase } from '@salutejs/plasma-hope'; +import { PopupBaseContext } 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'; @@ -591,6 +595,10 @@ export { MediaQueryFunction } export { Modal } +export { ModalBase } + +export { ModalBaseProps } + export { ModalProps } export { ModalsProvider } @@ -643,10 +651,14 @@ export { Popup } export { PopupBase } +export { PopupBaseContext } + 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..41454833a4 --- /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: { + placement: { + options: [ + 'center', + 'top', + 'bottom', + 'right', + 'left', + 'top-right', + 'top-left', + 'bottom-right', + 'bottom-left', + ], + control: { + type: 'select', + }, + }, + }, +} as Meta; + +type ModalBaseStoryProps = { + placement: 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 = ({ placement, 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} + placement={placement} + offset={[offsetX, offsetY]} + {...rest} + > + + +
+ setIsOpenB(true)} /> +
+ setIsOpenB(false)} + isOpen={isOpenB} + placement="left" + offset={[offsetX, offsetY]} + {...rest} + > + + +
+ setIsOpenC(true)} /> +
+ setIsOpenC(false)} + isOpen={isOpenC} + placement="top" + offset={[offsetX, offsetY]} + {...rest} + > + + + <>Content + + +
+
+
+
+
+
+
+ ); +}; + +ModalBaseDemo.args = { + placement: '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 c12cd1c9a5..d97b42026b 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', @@ -73,25 +73,27 @@ export const PopupBaseDemo: Story = ({ placement, offsetX, return ( -
- setIsOpenA(true)} /> - setIsOpenB(true)} /> -
- - - - <>Content - - - - <>Frame - - - - - <>Content - - + +
+ setIsOpenA(true)} /> + setIsOpenB(true)} /> +
+ + + + <>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..8a274eead0 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, PopupBaseContext } 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 1051adfec3..33024424cd 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';