/** * Copyright (c) 2021 OpenLens Authors * * Permission is hereby granted, free of charge, to any person obtaining a copy of * this software and associated documentation files (the "Software"), to deal in * the Software without restriction, including without limitation the rights to * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of * the Software, and to permit persons to whom the Software is furnished to do so, * subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ import styles from "./top-bar.module.scss"; import React, { useEffect, useMemo, useRef } from "react"; import { observer } from "mobx-react"; import type { IComputedValue } from "mobx"; import { Icon } from "../../icon"; import { webContents, getCurrentWindow } from "@electron/remote"; import { observable } from "mobx"; import { broadcastMessage, ipcRendererOn } from "../../../../common/ipc"; import { watchHistoryState } from "../../../remote-helpers/history-updater"; import { isActiveRoute, navigate } from "../../../navigation"; import { catalogRoute, catalogURL } from "../../../../common/routes"; import { IpcMainWindowEvents } from "../../../../main/window-manager"; import { isLinux, isWindows } from "../../../../common/vars"; import { cssNames } from "../../../utils"; import topBarItemsInjectable from "./top-bar-items/top-bar-items.injectable"; import { withInjectables } from "@ogre-tools/injectable-react"; import type { TopBarRegistration } from "./top-bar-registration"; interface Props extends React.HTMLAttributes {} interface Dependencies { items: IComputedValue; } const prevEnabled = observable.box(false); const nextEnabled = observable.box(false); ipcRendererOn("history:can-go-back", (event, state: boolean) => { prevEnabled.set(state); }); ipcRendererOn("history:can-go-forward", (event, state: boolean) => { nextEnabled.set(state); }); const NonInjectedTopBar = (({ items, children, ...rest }: Props & Dependencies) => { const elem = useRef(); const window = useMemo(() => getCurrentWindow(), []); const openContextMenu = () => { broadcastMessage(IpcMainWindowEvents.OPEN_CONTEXT_MENU); }; const goHome = () => { navigate(catalogURL()); }; const goBack = () => { webContents.getAllWebContents().find((webContent) => webContent.getType() === "window")?.goBack(); }; const goForward = () => { webContents.getAllWebContents().find((webContent) => webContent.getType() === "window")?.goForward(); }; const windowSizeToggle = (evt: React.MouseEvent) => { if (elem.current != evt.target) { // Skip clicking on child elements return; } toggleMaximize(); }; const minimizeWindow = () => { window.minimize(); }; const toggleMaximize = () => { if (window.isMaximized()) { window.unmaximize(); } else { window.maximize(); } }; const closeWindow = () => { window.close(); }; useEffect(() => { const disposer = watchHistoryState(); return () => disposer(); }, []); return (
{(isWindows || isLinux) && (
)}
{renderRegisteredItems(items.get())} {children} {(isWindows || isLinux) && (
)}
); }); const renderRegisteredItems = (items: TopBarRegistration[]) => (
{items.map((registration, index) => { if (!registration?.components?.Item) { return null; } return (
); })}
); export const TopBar = withInjectables(observer(NonInjectedTopBar), { getProps: (di, props) => ({ items: di.inject(topBarItemsInjectable), ...props, }), });