@blueprintjs/core
Version:
Core styles & components
136 lines (117 loc) • 4.49 kB
text/typescript
/*
* Copyright 2024 Palantir Technologies, Inc. All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import React from "react";
import { Classes } from "../../common";
import { OVERLAY2_REQUIRES_OVERLAY_PROVDER } from "../../common/errors";
import { isNodeEnv } from "../../common/utils";
import type { OverlayInstance } from "../../components";
import { OverlaysContext } from "../../context/overlays/overlaysProvider";
import { useLegacyOverlayStack } from "./useLegacyOverlayStack";
export interface UseOverlayStackReturnValue {
/**
* Removes an existing overlay off the stack.
*
* N.B. This method accepts an id instead of an overlay instance because the latter may be
* null when an overlay is unmounting, and we may stil have some cleanup to do at that time.
* Also, this method is not idempotent: if the overlay is not found on the stack, nothing happens.
*
* @param id identifier of the overlay to be closed
*/
closeOverlay: (id: string) => void;
/**
* @returns the last opened overlay on the stack
*/
getLastOpened: () => OverlayInstance | undefined;
/**
* @param id current overlay identifier
* @returns a list of the current overlay and all overlays which are descendants of it.
*/
getThisOverlayAndDescendants: (id: string) => OverlayInstance[];
/**
* Pushes a new overlay onto the stack.
*/
openOverlay: (overlay: OverlayInstance) => void;
/**
* Resets the overlay stack, to be called after all overlays are closed.
* Warning: this should only be used in unit tests.
*/
resetStack: () => void;
}
/**
* React hook to interact with the global overlay stack.
*
* @see https://blueprintjs.com/docs/#core/hooks/use-overlay-stack
*/
export function useOverlayStack(): UseOverlayStackReturnValue {
// get the overlay stack from application-wide React context
const { stack, hasProvider } = React.useContext(OverlaysContext);
const legacyOverlayStack = useLegacyOverlayStack();
const getLastOpened = React.useCallback(() => {
return stack.current[stack.current.length - 1];
}, [stack]);
const getThisOverlayAndDescendants = React.useCallback(
(id: string) => {
const index = stack.current.findIndex(o => o.id === id);
if (index === -1) {
return [];
}
return stack.current.slice(index);
},
[stack],
);
const resetStack = React.useCallback(() => {
stack.current = [];
}, [stack]);
const openOverlay = React.useCallback(
(overlay: OverlayInstance) => {
stack.current.push(overlay);
if (overlay.props.usePortal && overlay.props.hasBackdrop) {
// add a class to the body to prevent scrolling of content below the overlay
document.body.classList.add(Classes.OVERLAY_OPEN);
}
},
[stack],
);
const closeOverlay = React.useCallback(
(id: string) => {
const otherOverlaysWithBackdrop = stack.current.filter(
o => o.props.usePortal && o.props.hasBackdrop && o.id !== id,
);
const index = stack.current.findIndex(o => o.id === id);
if (index > -1) {
stack.current.splice(index, 1);
}
if (otherOverlaysWithBackdrop.length === 0) {
// remove body class which prevents scrolling of content below overlay
document.body.classList.remove(Classes.OVERLAY_OPEN);
}
},
[stack],
);
if (!hasProvider) {
if (isNodeEnv("development")) {
console.error(OVERLAY2_REQUIRES_OVERLAY_PROVDER);
}
return legacyOverlayStack;
}
return {
closeOverlay,
getLastOpened,
getThisOverlayAndDescendants,
openOverlay,
resetStack,
};
}