Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add centralized dialog management #2489

Merged
merged 33 commits into from
Sep 16, 2024
Merged
Show file tree
Hide file tree
Changes from 15 commits
Commits
Show all changes
33 commits
Select commit Hold shift + click to select a range
9751e15
feat: add centralized dialog management
MartinCupela Sep 4, 2024
eb6ab99
Merge branch 'release-v11' into feat/dialogs-manager
MartinCupela Sep 5, 2024
143555b
feat: use own anchor root for DialogAnchor
MartinCupela Sep 5, 2024
c3ed317
feat: forward custom DialogsManager id via DialogsManagerProvider
MartinCupela Sep 5, 2024
71c7685
feat: apply dialogs manager to message lists only
MartinCupela Sep 5, 2024
8e48860
fix: do not forward prop mine to MessageActionsBox root div
MartinCupela Sep 5, 2024
223ddea
test: fix MessageActions tests
MartinCupela Sep 5, 2024
d45459b
test: fix tests rendering message actions
MartinCupela Sep 5, 2024
ba89493
feat: assign static id to DialogsManager inside MessageList
MartinCupela Sep 5, 2024
993358f
docs: fix todo comment
MartinCupela Sep 5, 2024
b557e25
feat: handle focus within dialog
MartinCupela Sep 5, 2024
0a14548
feat: prevent rendering dialog contents if not open
MartinCupela Sep 6, 2024
07eb261
refactor: do not register event listeners by MessageActions component
MartinCupela Sep 6, 2024
d9f3709
test: open MessageActionsBox first in MessageActionsBox.test.js
MartinCupela Sep 6, 2024
af6b94b
feat: control ReactionsSelector dialog display
MartinCupela Sep 6, 2024
8d1c98a
fix: close MessageActionsBox on click inside
MartinCupela Sep 9, 2024
9f071e4
test: add DialogManager tests
MartinCupela Sep 9, 2024
9d1ee9d
refactor: unmound popper element if closed in useDialogAnchor
MartinCupela Sep 9, 2024
86b4cb0
Merge branch 'feat/dialogs-manager'
MartinCupela Sep 9, 2024
d1e4df4
refactor: use StateStore to handle DialogsManager subscriptions
MartinCupela Sep 10, 2024
f76cc75
refactor: rename DialogsManager to DialogManager
MartinCupela Sep 11, 2024
cb52d15
refactor: rename DialogsManager.open param single to closeRest
MartinCupela Sep 11, 2024
45c5eb7
refactor: rename DialogsManager.state.dialogs to DialogsManager.state…
MartinCupela Sep 11, 2024
401af81
refactor: move useStateStore to src/store/hooks
MartinCupela Sep 11, 2024
4f120a4
refactor: remove openDialogCount from DialogManager
MartinCupela Sep 11, 2024
a344533
refactor: apply suggestions about declaring state selector
MartinCupela Sep 11, 2024
1447022
Merge branch 'master' into feat/dialog-manager
MartinCupela Sep 11, 2024
a212d27
fix: remove unsupported onClick prop from ReactionListProps
MartinCupela Sep 11, 2024
9932001
docs: remove references to removed props
MartinCupela Sep 11, 2024
7b32ea3
docs: add dialog management guide and migration guide
MartinCupela Sep 11, 2024
f649140
chore(docs): bump stream-chat-css to v5.0.0-rc.6
MartinCupela Sep 13, 2024
7a67c03
refactor: merge Dialog.toggleOpen and Dialog.toggleOpenSingle into Di…
MartinCupela Sep 16, 2024
580b2bf
Merge branch 'master' into feat/dialog-manager
MartinCupela Sep 16, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion docusaurus/docs/React/guides/theming/message-ui.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -387,7 +387,7 @@ const CustomMessageUi = () => {

Message grouping is being managed automatically by the SDK and each parent element (which holds our message UI) receives an appropriate class name based on which we can adjust our rules to display metadata elements only when it's appropriate to make our UI look less busy.

{/_ TODO: link to grouping logic (maybe how to adjust it if needed) _/}
[//]: # 'TODO: link to grouping logic (maybe how to adjust it if needed)'

```css
.custom-message-ui__metadata {
Expand Down
8 changes: 4 additions & 4 deletions src/components/Channel/Channel.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,9 @@ import type { UnreadMessagesNotificationProps } from '../MessageList';
import { hasMoreMessagesProbably, UnreadMessagesSeparator } from '../MessageList';
import { useChannelContainerClasses } from './hooks/useChannelContainerClasses';
import { findInMsgSetByDate, findInMsgSetById, makeAddNotifications } from './utils';
import { DateSeparator } from '../DateSeparator';
import { EventComponent } from '../EventComponent';
import { defaultReactionOptions, ReactionOptions } from '../Reactions';
import { getChannel } from '../../utils';

import type { MessageProps } from '../Message/types';
Expand All @@ -96,9 +99,6 @@ import {
getVideoAttachmentConfiguration,
} from '../Attachment/attachment-sizing';
import type { URLEnrichmentConfig } from '../MessageInput/hooks/useLinkPreviews';
import { defaultReactionOptions, ReactionOptions } from '../Reactions';
import { EventComponent } from '../EventComponent';
import { DateSeparator } from '../DateSeparator';

type ChannelPropsForwardedToComponentContext<
StreamChatGenerics extends DefaultStreamChatGenerics = DefaultStreamChatGenerics
Expand Down Expand Up @@ -1241,7 +1241,7 @@ const ChannelInner = <
],
);

const componentContextValue: ComponentContextValue<StreamChatGenerics> = useMemo(
const componentContextValue = useMemo<ComponentContextValue<StreamChatGenerics>>(
() => ({
Attachment: props.Attachment || DefaultAttachment,
AttachmentPreviewList: props.AttachmentPreviewList,
Expand Down
119 changes: 119 additions & 0 deletions src/components/Dialog/DialogAnchor.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,119 @@
import clsx from 'clsx';
import { Placement } from '@popperjs/core';
import React, { ComponentProps, PropsWithChildren, useEffect, useState } from 'react';
import { FocusScope } from '@react-aria/focus';
import { usePopper } from 'react-popper';
import { DialogPortalEntry } from './DialogPortal';
import { useDialog, useDialogIsOpen } from './hooks';

export interface DialogAnchorOptions {
open: boolean;
placement: Placement;
referenceElement: HTMLElement | null;
}

export function useDialogAnchor<T extends HTMLElement>({
open,
placement,
referenceElement,
}: DialogAnchorOptions) {
const [popperElement, setPopperElement] = useState<T | null>(null);
const { attributes, styles, update } = usePopper(referenceElement, popperElement, {
modifiers: [
{
name: 'eventListeners',
options: {
// It's not safe to update popper position on resize and scroll, since popper's
// reference element might not be visible at the time.
resize: false,
scroll: false,
},
},
],
placement,
});

useEffect(() => {
if (open && popperElement) {
// Since the popper's reference element might not be (and usually is not) visible
// all the time, it's safer to force popper update before showing it.
// update is non-null only if popperElement is non-null
update?.();
}
}, [open, popperElement, update]);

return {
attributes,
setPopperElement,
styles,
};
}

type DialogAnchorProps = PropsWithChildren<Partial<DialogAnchorOptions>> & {
id: string;
focus?: boolean;
trapFocus?: boolean;
} & ComponentProps<'div'>;

export const DialogAnchor = ({
children,
className,
focus = true,
id,
placement = 'auto',
referenceElement = null,
trapFocus,
...restDivProps
}: DialogAnchorProps) => {
const dialog = useDialog({ id });
const open = useDialogIsOpen(id);
const { attributes, setPopperElement, styles } = useDialogAnchor<HTMLDivElement>({
open,
placement,
referenceElement,
});

useEffect(() => {
if (!open) return;
const hideOnEscape = (event: KeyboardEvent) => {
if (event.key !== 'Escape') return;
dialog?.close();
};

document.addEventListener('keyup', hideOnEscape);

return () => {
document.removeEventListener('keyup', hideOnEscape);
};
}, [dialog, open]);

useEffect(() => {
if (!open) {
// setting element reference back to null allows to re-run the usePopper component once the component is re-rendered
setPopperElement(null);
}
}, [open, setPopperElement]);
MartinCupela marked this conversation as resolved.
Show resolved Hide resolved

// prevent rendering the dialog contents if the dialog should not be open / shown
if (!open) {
return null;
}

return (
<DialogPortalEntry dialogId={id}>
<FocusScope autoFocus={focus} contain={trapFocus} restoreFocus>
<div
{...restDivProps}
{...attributes.popper}
className={clsx('str-chat__dialog-contents', className)}
data-testid='str-chat__dialog-contents'
ref={setPopperElement}
style={styles.popper}
tabIndex={0}
>
{children}
</div>
</FocusScope>
</DialogPortalEntry>
);
};
57 changes: 57 additions & 0 deletions src/components/Dialog/DialogPortal.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
import React, { PropsWithChildren, useEffect, useLayoutEffect, useState } from 'react';
import { createPortal } from 'react-dom';
import type { DialogsManager } from './DialogsManager';
import { useDialogIsOpen } from './hooks';
import { useDialogsManager } from '../../context';

export const DialogPortalDestination = () => {
const { dialogsManager } = useDialogsManager();
const [shouldRender, setShouldRender] = useState(!!dialogsManager.openDialogCount);
useEffect(
() =>
dialogsManager.on('openCountChange', {
listener: (dm: DialogsManager) => {
setShouldRender(dm.openDialogCount > 0);
},
}),
[dialogsManager],
);

return (
<div
className='str-chat__dialog-overlay'
data-str-chat__portal-id={dialogsManager.id}
data-testid='str-chat__dialog-overlay'
onClick={() => dialogsManager.closeAll()}
style={
{
'--str-chat__dialog-overlay-height': shouldRender ? '100%' : '0',
} as React.CSSProperties
}
></div>
);
};

type DialogPortalEntryProps = {
dialogId: string;
};

export const DialogPortalEntry = ({
children,
dialogId,
}: PropsWithChildren<DialogPortalEntryProps>) => {
const { dialogsManager } = useDialogsManager();
const dialogIsOpen = useDialogIsOpen(dialogId);
const [portalDestination, setPortalDestination] = useState<Element | null>(null);
useLayoutEffect(() => {
const destination = document.querySelector(
`div[data-str-chat__portal-id="${dialogsManager.id}"]`,
);
if (!destination) return;
setPortalDestination(destination);
}, [dialogsManager, dialogIsOpen]);

if (!portalDestination) return null;

return createPortal(children, portalDestination);
};
178 changes: 178 additions & 0 deletions src/components/Dialog/DialogsManager.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,178 @@
type DialogId = string;

export type GetOrCreateParams = {
id: DialogId;
isOpen?: boolean;
};

export type Dialog = {
close: () => void;
id: DialogId;
isOpen: boolean | undefined;
open: (zIndex?: number) => void;
remove: () => void;
toggle: () => void;
toggleSingle: () => void;
};

type DialogEvent = { type: 'close' | 'open' | 'openCountChange' };

const dialogsManagerEvents = ['openCountChange'] as const;
type DialogsManagerEvent = { type: typeof dialogsManagerEvents[number] };

type DialogEventHandler = (dialog: Dialog) => void;
type DialogsManagerEventHandler = (dialogsManager: DialogsManager) => void;

type DialogInitOptions = {
id?: string;
};

const noop = (): void => undefined;

MartinCupela marked this conversation as resolved.
Show resolved Hide resolved
export class DialogsManager {
MartinCupela marked this conversation as resolved.
Show resolved Hide resolved
id: string;
openDialogCount = 0;
dialogs: Record<DialogId, Dialog> = {};
private dialogEventListeners: Record<
DialogId,
Partial<Record<DialogEvent['type'], DialogEventHandler[]>>
> = {};
private dialogsManagerEventListeners: Record<
DialogsManagerEvent['type'],
DialogsManagerEventHandler[]
> = { openCountChange: [] };
MartinCupela marked this conversation as resolved.
Show resolved Hide resolved

constructor({ id }: DialogInitOptions = {}) {
this.id = id ?? new Date().getTime().toString();
}

getOrCreate({ id, isOpen = false }: GetOrCreateParams) {
let dialog = this.dialogs[id];
if (!dialog) {
dialog = {
close: () => {
this.close(id);
},
id,
isOpen,
open: () => {
this.open({ id });

Check warning on line 59 in src/components/Dialog/DialogsManager.ts

View check run for this annotation

Codecov / codecov/patch

src/components/Dialog/DialogsManager.ts#L59

Added line #L59 was not covered by tests
},
remove: () => {
this.remove(id);

Check warning on line 62 in src/components/Dialog/DialogsManager.ts

View check run for this annotation

Codecov / codecov/patch

src/components/Dialog/DialogsManager.ts#L62

Added line #L62 was not covered by tests
},
toggle: () => {
this.toggleOpen({ id });

Check warning on line 65 in src/components/Dialog/DialogsManager.ts

View check run for this annotation

Codecov / codecov/patch

src/components/Dialog/DialogsManager.ts#L65

Added line #L65 was not covered by tests
},
toggleSingle: () => {
this.toggleOpenSingle({ id });
},
};
this.dialogs[id] = dialog;
}
return dialog;
}

on(
eventType: DialogEvent['type'] | DialogsManagerEvent['type'],
{ id, listener }: { listener: DialogEventHandler | DialogsManagerEventHandler; id?: DialogId },
) {
if (dialogsManagerEvents.includes(eventType as DialogsManagerEvent['type'])) {
this.dialogsManagerEventListeners[eventType as DialogsManagerEvent['type']].push(
listener as DialogsManagerEventHandler,
);
return () => {
this.off(eventType, { listener });
};
}
if (!id) return noop;

if (!this.dialogEventListeners[id]) {
this.dialogEventListeners[id] = { close: [], open: [] };
}
this.dialogEventListeners[id][eventType] = [
...(this.dialogEventListeners[id][eventType] ?? []),
listener as DialogEventHandler,
];
return () => {
this.off(eventType, { id, listener });
};
}

off(
eventType: DialogEvent['type'] | DialogsManagerEvent['type'],
{ id, listener }: { listener: DialogEventHandler | DialogsManagerEventHandler; id?: DialogId },
) {
if (dialogsManagerEvents.includes(eventType as DialogsManagerEvent['type'])) {
const eventListeners = this.dialogsManagerEventListeners[
eventType as DialogsManagerEvent['type']
];
eventListeners?.filter((l) => l !== listener);
return;
}

if (!id) return;

const eventListeners = this.dialogEventListeners[id]?.[eventType];
if (!eventListeners) return;
this.dialogEventListeners[id][eventType] = eventListeners.filter((l) => l !== listener);
}

open(params: GetOrCreateParams, single?: boolean) {
MartinCupela marked this conversation as resolved.
Show resolved Hide resolved
const dialog = this.getOrCreate(params);
if (dialog.isOpen) return;
if (single) {
this.closeAll();
}
this.dialogs[params.id].isOpen = true;
this.openDialogCount++;
this.dialogsManagerEventListeners.openCountChange.forEach((listener) => listener(this));
this.dialogEventListeners[params.id].open?.forEach((listener) => listener(dialog));
}

close(id: DialogId) {
const dialog = this.dialogs[id];
if (!dialog?.isOpen) return;
dialog.isOpen = false;
MartinCupela marked this conversation as resolved.
Show resolved Hide resolved
this.openDialogCount--;
arnautov-anton marked this conversation as resolved.
Show resolved Hide resolved
this.dialogEventListeners[id].close?.forEach((listener) => listener(dialog));
this.dialogsManagerEventListeners.openCountChange.forEach((listener) => listener(this));
}

closeAll() {
Object.values(this.dialogs).forEach((dialog) => dialog.close());
}

toggleOpen(params: GetOrCreateParams) {
if (this.dialogs[params.id].isOpen) {
this.close(params.id);

Check warning on line 148 in src/components/Dialog/DialogsManager.ts

View check run for this annotation

Codecov / codecov/patch

src/components/Dialog/DialogsManager.ts#L148

Added line #L148 was not covered by tests
} else {
this.open(params);

Check warning on line 150 in src/components/Dialog/DialogsManager.ts

View check run for this annotation

Codecov / codecov/patch

src/components/Dialog/DialogsManager.ts#L150

Added line #L150 was not covered by tests
}
}

toggleOpenSingle(params: GetOrCreateParams) {
if (this.dialogs[params.id].isOpen) {
this.close(params.id);
} else {
this.open(params, true);
}
}

remove(id: DialogId) {
const dialogs = { ...this.dialogs };
if (!dialogs[id]) return;

const countListeners =
!!this.dialogEventListeners[id] &&
Object.values(this.dialogEventListeners[id]).reduce((acc, listeners) => {
acc += listeners.length;
return acc;
}, 0);

if (!countListeners) {
delete this.dialogEventListeners[id];
delete dialogs[id];
}
}
}
1 change: 1 addition & 0 deletions src/components/Dialog/hooks/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export * from './useDialog';
Loading
Loading