// Copyright 2024, Command Line Inc. // SPDX-License-Identifier: Apache-2.0 import { Workspace } from "@/app/workspace/workspace"; import { ContextMenuModel } from "@/store/contextmenu"; import { PLATFORM, atoms, createBlock, globalStore, removeFlashError, useSettingsPrefixAtom } from "@/store/global"; import { appHandleKeyDown } from "@/store/keymodel"; import { getElemAsStr } from "@/util/focusutil"; import * as keyutil from "@/util/keyutil"; import * as util from "@/util/util"; import clsx from "clsx"; import Color from "color"; import debug from "debug"; import { Provider, useAtomValue } from "jotai"; import "overlayscrollbars/overlayscrollbars.css"; import { Fragment, useEffect, useState } from "react"; import { DndProvider } from "react-dnd"; import { HTML5Backend } from "react-dnd-html5-backend"; import { AppBackground } from "./app-bg"; import "./app.less"; import { CenteredDiv } from "./element/quickelems"; const dlog = debug("wave:app"); const focusLog = debug("wave:focus"); const App = () => { return ( ); }; function isContentEditableBeingEdited(): boolean { const activeElement = document.activeElement; return ( activeElement && activeElement.getAttribute("contenteditable") !== null && activeElement.getAttribute("contenteditable") !== "false" ); } function canEnablePaste(): boolean { const activeElement = document.activeElement; return activeElement.tagName === "INPUT" || activeElement.tagName === "TEXTAREA" || isContentEditableBeingEdited(); } function canEnableCopy(): boolean { const sel = window.getSelection(); return !util.isBlank(sel?.toString()); } function canEnableCut(): boolean { const sel = window.getSelection(); if (document.activeElement?.classList.contains("xterm-helper-textarea")) { return false; } return !util.isBlank(sel?.toString()) && canEnablePaste(); } async function getClipboardURL(): Promise { try { const clipboardText = await navigator.clipboard.readText(); if (clipboardText == null) { return null; } const url = new URL(clipboardText); return url; } catch (e) { return null; } } async function handleContextMenu(e: React.MouseEvent) { e.preventDefault(); const canPaste = canEnablePaste(); const canCopy = canEnableCopy(); const canCut = canEnableCut(); const clipboardURL = await getClipboardURL(); if (!canPaste && !canCopy && !canCut && !clipboardURL) { return; } let menu: ContextMenuItem[] = []; if (canCut) { menu.push({ label: "Cut", role: "cut" }); } if (canCopy) { menu.push({ label: "Copy", role: "copy" }); } if (canPaste) { menu.push({ label: "Paste", role: "paste" }); } if (clipboardURL) { menu.push({ type: "separator" }); menu.push({ label: "Open Clipboard URL (" + clipboardURL.hostname + ")", click: () => { createBlock({ meta: { view: "web", url: clipboardURL.toString(), }, }); }, }); } ContextMenuModel.showContextMenu(menu, e); } function AppSettingsUpdater() { const windowSettingsAtom = useSettingsPrefixAtom("window"); const windowSettings = useAtomValue(windowSettingsAtom); useEffect(() => { const isTransparentOrBlur = (windowSettings?.["window:transparent"] || windowSettings?.["window:blur"]) ?? false; const opacity = util.boundNumber(windowSettings?.["window:opacity"] ?? 0.8, 0, 1); let baseBgColor = windowSettings?.["window:bgcolor"]; if (isTransparentOrBlur) { document.body.classList.add("is-transparent"); const rootStyles = getComputedStyle(document.documentElement); if (baseBgColor == null) { baseBgColor = rootStyles.getPropertyValue("--main-bg-color").trim(); } const color = new Color(baseBgColor); const rgbaColor = color.alpha(opacity).string(); document.body.style.backgroundColor = rgbaColor; } else { document.body.classList.remove("is-transparent"); document.body.style.opacity = null; } }, [windowSettings]); return null; } function appFocusIn(e: FocusEvent) { focusLog("focusin", getElemAsStr(e.target), "<=", getElemAsStr(e.relatedTarget)); } function appFocusOut(e: FocusEvent) { focusLog("focusout", getElemAsStr(e.target), "=>", getElemAsStr(e.relatedTarget)); } function appSelectionChange(e: Event) { const selection = document.getSelection(); focusLog("selectionchange", getElemAsStr(selection.anchorNode)); } function AppFocusHandler() { return null; // for debugging useEffect(() => { document.addEventListener("focusin", appFocusIn); document.addEventListener("focusout", appFocusOut); document.addEventListener("selectionchange", appSelectionChange); const ivId = setInterval(() => { const activeElement = document.activeElement; if (activeElement instanceof HTMLElement) { focusLog("activeElement", getElemAsStr(activeElement)); } }, 2000); return () => { document.removeEventListener("focusin", appFocusIn); document.removeEventListener("focusout", appFocusOut); document.removeEventListener("selectionchange", appSelectionChange); clearInterval(ivId); }; }); return null; } const AppKeyHandlers = () => { useEffect(() => { const staticKeyDownHandler = keyutil.keydownWrapper(appHandleKeyDown); document.addEventListener("keydown", staticKeyDownHandler); return () => { document.removeEventListener("keydown", staticKeyDownHandler); }; }, []); return null; }; const FlashError = () => { const flashErrors = useAtomValue(atoms.flashErrors); const [hoveredId, setHoveredId] = useState(null); const [ticker, setTicker] = useState(0); useEffect(() => { if (flashErrors.length == 0 || hoveredId != null) { return; } const now = Date.now(); for (let ferr of flashErrors) { if (ferr.expiration == null || ferr.expiration < now) { removeFlashError(ferr.id); } } setTimeout(() => setTicker(ticker + 1), 1000); }, [flashErrors, ticker, hoveredId]); if (flashErrors.length == 0) { return null; } function copyError(id: string) { const ferr = flashErrors.find((f) => f.id === id); if (ferr == null) { return; } let text = ""; if (ferr.title != null) { text += ferr.title; } if (ferr.message != null) { if (text.length > 0) { text += "\n"; } text += ferr.message; } navigator.clipboard.writeText(text); } function convertNewlinesToBreaks(text) { return text.split("\n").map((part, index) => ( {part}
)); } return (
{flashErrors.map((err, idx) => (
copyError(err.id)} onMouseEnter={() => setHoveredId(err.id)} onMouseLeave={() => setHoveredId(null)} title="Click to Copy Error Message" >
{err.title != null ?
{err.title}
: null} {err.message != null ? (
{convertNewlinesToBreaks(err.message)}
) : null}
))}
); }; const AppInner = () => { const prefersReducedMotion = useAtomValue(atoms.prefersReducedMotionAtom); const client = useAtomValue(atoms.client); const windowData = useAtomValue(atoms.waveWindow); const isFullScreen = useAtomValue(atoms.isFullScreen); if (client == null || windowData == null) { return (
invalid configuration, client or window was not loaded
); } return (
); }; export { App };