UNPKG

@blueprintjs/core

Version:

Core styles & components

136 lines (117 loc) 4.49 kB
/* * 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, }; }