You've already forked joplin
mirror of
https://github.com/laurent22/joplin.git
synced 2025-07-03 23:50:33 +02:00
52 lines
1.4 KiB
TypeScript
52 lines
1.4 KiB
TypeScript
![]() |
import * as React from 'react';
|
||
|
import { createContext, useCallback, useMemo, useState } from 'react';
|
||
|
import { ModalState } from './types';
|
||
|
|
||
|
export interface FocusControl {
|
||
|
setModalState(dialogId: string, state: ModalState): void;
|
||
|
|
||
|
hasOpenModal: boolean;
|
||
|
hasClosingModal: boolean;
|
||
|
}
|
||
|
|
||
|
export const FocusControlContext = createContext<FocusControl|null>(null);
|
||
|
|
||
|
interface Props {
|
||
|
children: React.ReactNode;
|
||
|
}
|
||
|
|
||
|
const FocusControlProvider: React.FC<Props> = props => {
|
||
|
type ModalStates = Record<string, ModalState>;
|
||
|
const [modalStates, setModalStates] = useState<ModalStates>({});
|
||
|
|
||
|
const setModalOpen = useCallback((dialogId: string, state: ModalState) => {
|
||
|
setModalStates(modalStates => {
|
||
|
modalStates = { ...modalStates };
|
||
|
if (state === ModalState.Closed) {
|
||
|
delete modalStates[dialogId];
|
||
|
} else {
|
||
|
modalStates[dialogId] = state;
|
||
|
}
|
||
|
return modalStates;
|
||
|
});
|
||
|
}, []);
|
||
|
|
||
|
const modalStateValues = Object.values(modalStates);
|
||
|
const hasOpenModal = modalStateValues.includes(ModalState.Open);
|
||
|
const hasClosingModal = modalStateValues.includes(ModalState.Closing);
|
||
|
|
||
|
const focusControl = useMemo((): FocusControl => {
|
||
|
return {
|
||
|
hasOpenModal: hasOpenModal,
|
||
|
hasClosingModal: hasClosingModal,
|
||
|
setModalState: setModalOpen,
|
||
|
};
|
||
|
}, [hasOpenModal, hasClosingModal, setModalOpen]);
|
||
|
|
||
|
return <FocusControlContext.Provider value={focusControl}>
|
||
|
{props.children}
|
||
|
</FocusControlContext.Provider>;
|
||
|
};
|
||
|
|
||
|
export default FocusControlProvider;
|