Merge branch 'main' into feature/auto-hide-tab-bar

This commit is contained in:
Ritik Ranjan 2024-12-21 09:19:02 +05:30 committed by GitHub
commit 2af5c024f0
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
41 changed files with 701 additions and 252 deletions

View File

@ -23,4 +23,4 @@ jobs:
uses: upsidr/merge-gatekeeper@v1 uses: upsidr/merge-gatekeeper@v1
with: with:
token: ${{ secrets.GITHUB_TOKEN }} token: ${{ secrets.GITHUB_TOKEN }}
ignored: Test Onboarding, Analyze (go), Analyze (javascript-typescript), License Compliance ignored: Test Onboarding, Analyze (go), Analyze (javascript-typescript), License Compliance, CodeRabbit

View File

@ -58,5 +58,5 @@ func shellCmdInner() string {
} }
} }
// none found // none found
return "bin/bash\n" return "/bin/bash\n"
} }

View File

@ -45,6 +45,8 @@ wsh editconfig
| term:localshellopts | string[] | set to pass additional parameters to the term:localshellpath (example: `["-NoLogo"]` for PowerShell will remove the copyright notice) | | term:localshellopts | string[] | set to pass additional parameters to the term:localshellpath (example: `["-NoLogo"]` for PowerShell will remove the copyright notice) |
| term:copyonselect | bool | set to false to disable terminal copy-on-select | | term:copyonselect | bool | set to false to disable terminal copy-on-select |
| term:scrollback | int | size of terminal scrollback buffer, max is 10000 | | term:scrollback | int | size of terminal scrollback buffer, max is 10000 |
| term:theme | string | preset name of terminal theme to apply by default (default is "default-dark") |
| term:transparency | float64 | set the background transparency of terminal theme (default 0.5, 0 = not transparent, 1.0 = fully transparent) |
| editor:minimapenabled | bool | set to false to disable editor minimap | | editor:minimapenabled | bool | set to false to disable editor minimap |
| editor:stickyscrollenabled | bool | enables monaco editor's stickyScroll feature (pinning headers of current context, e.g. class names, method names, etc.), defaults to false | | editor:stickyscrollenabled | bool | enables monaco editor's stickyScroll feature (pinning headers of current context, e.g. class names, method names, etc.), defaults to false |
| editor:wordwrap | bool | set to true to enable word wrapping in the editor (defaults to false) | | editor:wordwrap | bool | set to true to enable word wrapping in the editor (defaults to false) |
@ -75,9 +77,11 @@ wsh editconfig
| window:autohidetabbar | bool | show and hide the tab bar automatically when the mouse moves near the top of the window | window:autohidetabbar | bool | show and hide the tab bar automatically when the mouse moves near the top of the window
| window:nativetitlebar | bool | set to use the OS-native title bar, rather than the overlay (Windows and Linux only, requires app restart) | | window:nativetitlebar | bool | set to use the OS-native title bar, rather than the overlay (Windows and Linux only, requires app restart) |
| window:disablehardwareacceleration | bool | set to disable Chromium hardware acceleration to resolve graphical bugs (requires app restart) | | window:disablehardwareacceleration | bool | set to disable Chromium hardware acceleration to resolve graphical bugs (requires app restart) |
| window:savelastwindow | bool | when `true`, the last window that is closed is preserved and is reopened the next time the app is launched (defaults to `true`) |
| window:confirmonclose | bool | when `true`, a prompt will ask a user to confirm that they want to close a window if it has an unsaved workspace with more than one tab (defaults to `true`) |
| telemetry:enabled | bool | set to enable/disable telemetry | | telemetry:enabled | bool | set to enable/disable telemetry |
For reference this is the current default configuration (v0.9.3): For reference, this is the current default configuration (v0.10.4):
```json ```json
{ {
@ -89,6 +93,7 @@ For reference this is the current default configuration (v0.9.3):
"autoupdate:installonquit": true, "autoupdate:installonquit": true,
"autoupdate:intervalms": 3600000, "autoupdate:intervalms": 3600000,
"conn:askbeforewshinstall": true, "conn:askbeforewshinstall": true,
"conn:wshenabled": true,
"editor:minimapenabled": true, "editor:minimapenabled": true,
"web:defaulturl": "https://github.com/wavetermdev/waveterm", "web:defaulturl": "https://github.com/wavetermdev/waveterm",
"web:defaultsearch": "https://www.google.com/search?q={query}", "web:defaultsearch": "https://www.google.com/search?q={query}",
@ -99,6 +104,8 @@ For reference this is the current default configuration (v0.9.3):
"window:magnifiedblocksize": 0.9, "window:magnifiedblocksize": 0.9,
"window:magnifiedblockblurprimarypx": 10, "window:magnifiedblockblurprimarypx": 10,
"window:magnifiedblockblursecondarypx": 2, "window:magnifiedblockblursecondarypx": 2,
"window:confirmclose": true,
"window:savelastwindow": true,
"telemetry:enabled": true, "telemetry:enabled": true,
"term:copyonselect": true "term:copyonselect": true
} }

View File

@ -71,6 +71,14 @@ replace "Cmd" with "Alt" (note that "Ctrl" is "Ctrl" on both Mac, Windows, and L
| ---------------- | ------------- | | ---------------- | ------------- |
| <Kbd k="Cmd:l"/> | Clear AI Chat | | <Kbd k="Cmd:l"/> | Clear AI Chat |
## Terminal Keybindings
| Key | Function |
| ----------------------- | -------------- |
| <Kbd k="Ctrl:Shift:c"/> | Copy |
| <Kbd k="Ctrl:Shift:v"/> | Paste |
| <Kbd k="Cmd:k"/> | Clear Terminal |
## Customizeable Systemwide Global Hotkey ## Customizeable Systemwide Global Hotkey
Wave allows setting a custom global hotkey to focus your most recent window from anywhere in your computer. For more information on this, see [the config docs](./config#customizable-systemwide-global-hotkey). Wave allows setting a custom global hotkey to focus your most recent window from anywhere in your computer. For more information on this, see [the config docs](./config#customizable-systemwide-global-hotkey).

View File

@ -6,6 +6,36 @@ sidebar_position: 200
# Release Notes # Release Notes
### v0.10.4 &mdash; Dec 20, 2024
Quick update with bug fixes and new configuration options
- Added "window:confirmclose" and "window:savelastwindow" configuration options
- [bugfix] Fixed broken scroll bar in the AI widget
- [bugfix] Fixed default path for wsh shell detection (used in remote connections)
- Dependency updates
### v0.10.3 &mdash; Dec 19, 2024
Quick update to v0.10 with new features and bug fixes.
- Global hotkey support [docs](https://docs.waveterm.dev/config#customizable-systemwide-global-hotkey)
- Added configuration to override the font size for markdown, AI-chat, and preview editor [docs](https://docs.waveterm.dev/config)
- Added ability to set independent zoom level for the web view (right click block header)
- New `wsh wavepath` command to open the config directory, data directory, and log file
- [bugfix] Fixed crash when /etc/sshd_config contained an unsupported Match directive (most common on Fedora)
- [bugfix] Workspaces are now more consistent across windows, closes associated window when Workspaces are deleted
- [bugfix] Fixed zsh on WSL
- [bugfix] Fixed long-standing bug around control sequences sometimes showing up in terminal output when switching tabs
- Lots of new examples in the docs for shell overrides, presets, widgets, and connections
- Other bug fixes and UI updates
(note, v0.10.2 and v0.10.3's release notes have been merged together)
### v0.10.1 &mdash; Dec 12, 2024
Quick update to fix the workspace app menu actions. Also fixes workspace switching to always open a new window when invoked from a non-workspace window. This reduces the chance of losing a non-workspace window's tabs accidentally.
### v0.10.0 &mdash; Dec 11, 2024 ### v0.10.0 &mdash; Dec 11, 2024
Wave Terminal v0.10.0 introduces workspaces, making it easier to manage multiple work environments. We've added powerful new command execution capabilities with `wsh run`, allowing you to launch and control commands in dedicated blocks. This release also brings significant improvements to SSH with a new connections configuration system for managing your remote environments. Wave Terminal v0.10.0 introduces workspaces, making it easier to manage multiple work environments. We've added powerful new command execution capabilities with `wsh run`, allowing you to launch and control commands in dedicated blocks. This release also brings significant improvements to SSH with a new connections configuration system for managing your remote environments.

View File

@ -38,9 +38,9 @@ function convertKey(platform: Platform, key: string): [any, string, boolean] {
return ["⇧", "Shift", true]; return ["⇧", "Shift", true];
} }
if (key == "Escape") { if (key == "Escape") {
return ["Esc", null, false]; return ["Esc", "Escape", false];
} }
return [key, null, false]; return [key.length > 1 ? key : key.toUpperCase(), key, false];
} }
// Custom KBD component // Custom KBD component
@ -50,7 +50,7 @@ const KbdInternal = ({ k }: { k: string }) => {
const keyElems = keys.map((key, i) => { const keyElems = keys.map((key, i) => {
const [displayKey, title, symbol] = convertKey(platform, key); const [displayKey, title, symbol] = convertKey(platform, key);
return ( return (
<kbd key={i} title={title} className={symbol ? "symbol" : null}> <kbd key={i} title={title} aria-label={title} className={symbol ? "symbol" : null}>
{displayKey} {displayKey}
</kbd> </kbd>
); );

View File

@ -229,8 +229,11 @@ export class WaveBrowserWindow extends BaseWindow {
e.preventDefault(); e.preventDefault();
fireAndForget(async () => { fireAndForget(async () => {
const numWindows = waveWindowMap.size; const numWindows = waveWindowMap.size;
if (numWindows > 1) { const fullConfig = await FileService.GetFullConfig();
console.log("numWindows > 1", numWindows); if (numWindows > 1 || !fullConfig.settings["window:savelastwindow"]) {
console.log("numWindows > 1 or user does not want last window saved", numWindows);
if (fullConfig.settings["window:confirmclose"]) {
console.log("confirmclose", this.waveWindowId);
const workspace = await WorkspaceService.GetWorkspace(this.workspaceId); const workspace = await WorkspaceService.GetWorkspace(this.workspaceId);
console.log("workspace", workspace); console.log("workspace", workspace);
if (isNonEmptyUnsavedWorkspace(workspace)) { if (isNonEmptyUnsavedWorkspace(workspace)) {
@ -239,13 +242,15 @@ export class WaveBrowserWindow extends BaseWindow {
type: "question", type: "question",
buttons: ["Cancel", "Close Window"], buttons: ["Cancel", "Close Window"],
title: "Confirm", title: "Confirm",
message: "Window has unsaved tabs, closing window will delete existing tabs.\n\nContinue?", message:
"Window has unsaved tabs, closing window will delete existing tabs.\n\nContinue?",
}); });
if (choice === 0) { if (choice === 0) {
console.log("user cancelled close window", this.waveWindowId); console.log("user cancelled close window", this.waveWindowId);
return; return;
} }
} }
}
console.log("deleteAllowed = true", this.waveWindowId); console.log("deleteAllowed = true", this.waveWindowId);
this.deleteAllowed = true; this.deleteAllowed = true;
} }
@ -270,11 +275,6 @@ export class WaveBrowserWindow extends BaseWindow {
this.destroy(); this.destroy();
return; return;
} }
const numWindows = waveWindowMap.size;
if (numWindows == 0) {
console.log("win no windows left", this.waveWindowId);
return;
}
if (this.deleteAllowed) { if (this.deleteAllowed) {
console.log("win removing window from backend DB", this.waveWindowId); console.log("win removing window from backend DB", this.waveWindowId);
fireAndForget(() => WindowService.CloseWindow(this.waveWindowId, true)); fireAndForget(() => WindowService.CloseWindow(this.waveWindowId, true));

View File

@ -24,6 +24,10 @@
overflow: hidden; overflow: hidden;
} }
*:last-child {
margin-bottom: 0 !important;
}
.heading:not(.heading ~ .heading) { .heading:not(.heading ~ .heading) {
margin-top: 0 !important; margin-top: 0 !important;
} }

View File

@ -9,7 +9,7 @@ import {
resolveSrcSet, resolveSrcSet,
transformBlocks, transformBlocks,
} from "@/app/element/markdown-util"; } from "@/app/element/markdown-util";
import { useAtomValueSafe } from "@/util/util"; import { boundNumber, useAtomValueSafe } from "@/util/util";
import { clsx } from "clsx"; import { clsx } from "clsx";
import { Atom } from "jotai"; import { Atom } from "jotai";
import { OverlayScrollbarsComponent, OverlayScrollbarsComponentRef } from "overlayscrollbars-react"; import { OverlayScrollbarsComponent, OverlayScrollbarsComponentRef } from "overlayscrollbars-react";
@ -380,10 +380,10 @@ const Markdown = ({
const mergedStyle = { ...style }; const mergedStyle = { ...style };
if (fontSizeOverride != null) { if (fontSizeOverride != null) {
mergedStyle["--markdown-font-size"] = `${fontSizeOverride}px`; mergedStyle["--markdown-font-size"] = `${boundNumber(fontSizeOverride, 6, 64)}px`;
} }
if (fixedFontSizeOverride != null) { if (fixedFontSizeOverride != null) {
mergedStyle["--markdown-fixed-font-size"] = `${fixedFontSizeOverride}px`; mergedStyle["--markdown-fixed-font-size"] = `${boundNumber(fixedFontSizeOverride, 6, 64)}px`;
} }
return ( return (
<div className={clsx("markdown", className)} style={mergedStyle}> <div className={clsx("markdown", className)} style={mergedStyle}>

View File

@ -0,0 +1,43 @@
.search-container {
display: flex;
flex-direction: row;
background-color: var(--modal-bg-color);
border: 1px solid var(--accent-color);
border-radius: var(--modal-border-radius);
box-shadow: var(--modal-box-shadow);
color: var(--main-text-color);
padding: 5px 5px 5px 10px;
gap: 5px;
width: 50%;
max-width: 300px;
min-width: 200px;
input {
flex: 1 1 auto;
border: none;
font-size: 14px;
height: 100%;
padding: 0;
border-radius: 0;
}
.search-results {
font-size: 12px;
margin: auto 0;
color: var(--secondary-text-color);
&.hidden {
display: none;
}
}
.right-buttons {
display: flex;
gap: 5px;
padding-left: 5px;
border-left: 1px solid var(--modal-border-color);
button {
font-size: 12px;
}
}
}

View File

@ -0,0 +1,127 @@
// Copyright 2024, Command Line Inc.
// SPDX-License-Identifier: Apache-2.0
import type { Meta, StoryObj } from "@storybook/react";
import { useSetAtom } from "jotai";
import { useEffect } from "react";
import { Popover } from "./popover";
import { Search, useSearch } from "./search";
const meta: Meta<typeof Search> = {
title: "Elements/Search",
component: Search,
args: {},
};
export default meta;
type Story = StoryObj<typeof Popover>;
export const DefaultSearch: Story = {
render: (args) => {
const props = useSearch();
const setIsOpen = useSetAtom(props.isOpenAtom);
useEffect(() => {
setIsOpen(true);
}, []);
return (
<div
className="viewbox"
ref={props.anchorRef as React.RefObject<HTMLDivElement>}
style={{
border: "2px solid black",
width: "100%",
height: "200px",
background: "var(--main-bg-color)",
}}
>
<Search {...args} {...props} />
</div>
);
},
args: {},
};
export const Results10: Story = {
render: (args) => {
const props = useSearch();
const setIsOpen = useSetAtom(props.isOpenAtom);
const setNumResults = useSetAtom(props.numResultsAtom);
useEffect(() => {
setIsOpen(true);
setNumResults(10);
}, []);
return (
<div
className="viewbox"
ref={props.anchorRef as React.RefObject<HTMLDivElement>}
style={{
border: "2px solid black",
width: "100%",
height: "200px",
background: "var(--main-bg-color)",
}}
>
<Search {...args} {...props} />
</div>
);
},
args: {},
};
export const InputAndResults10: Story = {
render: (args) => {
const props = useSearch();
const setIsOpen = useSetAtom(props.isOpenAtom);
const setNumResults = useSetAtom(props.numResultsAtom);
const setSearch = useSetAtom(props.searchAtom);
useEffect(() => {
setIsOpen(true);
setNumResults(10);
setSearch("search term");
}, []);
return (
<div
className="viewbox"
ref={props.anchorRef as React.RefObject<HTMLDivElement>}
style={{
border: "2px solid black",
width: "100%",
height: "200px",
background: "var(--main-bg-color)",
}}
>
<Search {...args} {...props} />
</div>
);
},
args: {},
};
export const LongInputAndResults10: Story = {
render: (args) => {
const props = useSearch();
const setIsOpen = useSetAtom(props.isOpenAtom);
const setNumResults = useSetAtom(props.numResultsAtom);
const setSearch = useSetAtom(props.searchAtom);
useEffect(() => {
setIsOpen(true);
setNumResults(10);
setSearch("search term ".repeat(10).trimEnd());
}, []);
return (
<div
className="viewbox"
ref={props.anchorRef as React.RefObject<HTMLDivElement>}
style={{
border: "2px solid black",
width: "100%",
height: "200px",
background: "var(--main-bg-color)",
}}
>
<Search {...args} {...props} />
</div>
);
},
args: {},
};

View File

@ -0,0 +1,115 @@
import { autoUpdate, FloatingPortal, Middleware, offset, useDismiss, useFloating } from "@floating-ui/react";
import clsx from "clsx";
import { atom, PrimitiveAtom, useAtom, useAtomValue } from "jotai";
import { memo, useCallback, useRef, useState } from "react";
import { IconButton } from "./iconbutton";
import { Input } from "./input";
import "./search.scss";
type SearchProps = {
searchAtom: PrimitiveAtom<string>;
indexAtom: PrimitiveAtom<number>;
numResultsAtom: PrimitiveAtom<number>;
isOpenAtom: PrimitiveAtom<boolean>;
anchorRef?: React.RefObject<HTMLElement>;
offsetX?: number;
offsetY?: number;
};
const SearchComponent = ({
searchAtom,
indexAtom,
numResultsAtom,
isOpenAtom,
anchorRef,
offsetX = 10,
offsetY = 10,
}: SearchProps) => {
const [isOpen, setIsOpen] = useAtom(isOpenAtom);
const [search, setSearch] = useAtom(searchAtom);
const [index, setIndex] = useAtom(indexAtom);
const numResults = useAtomValue(numResultsAtom);
const handleOpenChange = useCallback((open: boolean) => {
setIsOpen(open);
}, []);
const middleware: Middleware[] = [];
middleware.push(
offset(({ rects }) => ({
mainAxis: -rects.floating.height - offsetY,
crossAxis: -offsetX,
}))
);
const { refs, floatingStyles, context } = useFloating({
placement: "top-end",
open: isOpen,
onOpenChange: handleOpenChange,
whileElementsMounted: autoUpdate,
middleware,
elements: {
reference: anchorRef!.current,
},
});
const dismiss = useDismiss(context);
const prevDecl: IconButtonDecl = {
elemtype: "iconbutton",
icon: "chevron-up",
title: "Previous Result",
disabled: index === 0,
click: () => setIndex(index - 1),
};
const nextDecl: IconButtonDecl = {
elemtype: "iconbutton",
icon: "chevron-down",
title: "Next Result",
disabled: !numResults || index === numResults - 1,
click: () => setIndex(index + 1),
};
const closeDecl: IconButtonDecl = {
elemtype: "iconbutton",
icon: "xmark-large",
title: "Close",
click: () => setIsOpen(false),
};
return (
<>
{isOpen && (
<FloatingPortal>
<div className="search-container" style={{ ...floatingStyles }} {...dismiss} ref={refs.setFloating}>
<Input placeholder="Search" value={search} onChange={setSearch} />
<div
className={clsx("search-results", { hidden: numResults === 0 })}
aria-live="polite"
aria-label="Search Results"
>
{index + 1}/{numResults}
</div>
<div className="right-buttons">
<IconButton decl={prevDecl} />
<IconButton decl={nextDecl} />
<IconButton decl={closeDecl} />
</div>
</div>
</FloatingPortal>
)}
</>
);
};
export const Search = memo(SearchComponent) as typeof SearchComponent;
export function useSearch(anchorRef?: React.RefObject<HTMLElement>): SearchProps {
const [searchAtom] = useState(atom(""));
const [indexAtom] = useState(atom(0));
const [numResultsAtom] = useState(atom(0));
const [isOpenAtom] = useState(atom(false));
anchorRef ??= useRef(null);
return { searchAtom, indexAtom, numResultsAtom, isOpenAtom, anchorRef };
}

View File

@ -1,32 +0,0 @@
// Copyright 2024, Command Line Inc.
// SPDX-License-Identifier: Apache-2.0
import type { Meta, StoryObj } from "@storybook/react";
import { SearchInput } from "./searchinput";
const meta: Meta<typeof SearchInput> = {
title: "Elements/SearchInput",
component: SearchInput,
argTypes: {
className: {
description: "Custom class for styling the input group",
control: { type: "text" },
},
},
};
export default meta;
type Story = StoryObj<typeof SearchInput>;
export const DefaultSearchInput: Story = {
render: (args) => {
const handleSearch = () => {
console.log("Search triggered");
};
return <SearchInput />;
},
args: {
className: "custom-search-input",
},
};

View File

@ -1,20 +0,0 @@
// Copyright 2024, Command Line Inc.
// SPDX-License-Identifier: Apache-2.0
import { Button } from "./button";
import { Input, InputGroup, InputRightElement } from "./input";
const SearchInput = () => {
return (
<InputGroup className="search-input-group">
<Input placeholder="Search..." />
<InputRightElement>
<Button className="search-button ghost grey">
<i className="fa-sharp fa-solid fa-magnifying-glass"></i>
</Button>
</InputRightElement>
</InputGroup>
);
};
export { SearchInput };

View File

@ -230,6 +230,7 @@
justify-content: center; justify-content: center;
button { button {
padding: 8px 20px;
font-size: 14px; font-size: 14px;
} }

View File

@ -153,6 +153,7 @@ const WorkspaceSwitcherItem = ({
const isCurrentWorkspace = activeWorkspace.oid === workspace.oid; const isCurrentWorkspace = activeWorkspace.oid === workspace.oid;
const setWorkspace = useCallback((newWorkspace: Workspace) => { const setWorkspace = useCallback((newWorkspace: Workspace) => {
setWorkspaceEntry({ ...workspaceEntry, workspace: newWorkspace });
if (newWorkspace.name != "") { if (newWorkspace.name != "") {
fireAndForget(() => fireAndForget(() =>
WorkspaceService.UpdateWorkspace( WorkspaceService.UpdateWorkspace(

View File

@ -83,8 +83,10 @@
--modal-bg-color: #232323; --modal-bg-color: #232323;
--modal-header-bottom-border-color: rgba(241, 246, 243, 0.15); --modal-header-bottom-border-color: rgba(241, 246, 243, 0.15);
--modal-border-color: rgba(255, 255, 255, 0.12); /* toggle colors */ --modal-border-color: rgba(255, 255, 255, 0.12); /* toggle colors */
--modal-border-radius: 6px;
--toggle-bg-color: var(--border-color); --toggle-bg-color: var(--border-color);
--modal-shadow-color: rgba(0, 0, 0, 0.8); --modal-shadow-color: rgba(0, 0, 0, 0.8);
--modal-box-shadow: box-shadow: 0px 8px 24px 0px var(--modal-shadow-color);
--toggle-thumb-color: var(--main-text-color); --toggle-thumb-color: var(--main-text-color);
--toggle-checked-bg-color: var(--accent-color); --toggle-checked-bg-color: var(--accent-color);

View File

@ -2,6 +2,7 @@
// SPDX-License-Identifier: Apache-2.0 // SPDX-License-Identifier: Apache-2.0
import { useOverrideConfigAtom } from "@/app/store/global"; import { useOverrideConfigAtom } from "@/app/store/global";
import { boundNumber } from "@/util/util";
import loader from "@monaco-editor/loader"; import loader from "@monaco-editor/loader";
import { Editor, Monaco } from "@monaco-editor/react"; import { Editor, Monaco } from "@monaco-editor/react";
import type * as MonacoTypes from "monaco-editor/esm/vs/editor/editor.api"; import type * as MonacoTypes from "monaco-editor/esm/vs/editor/editor.api";
@ -122,7 +123,7 @@ export function CodeEditor({ blockId, text, language, filename, meta, onChange,
const minimapEnabled = useOverrideConfigAtom(blockId, "editor:minimapenabled") ?? false; const minimapEnabled = useOverrideConfigAtom(blockId, "editor:minimapenabled") ?? false;
const stickyScrollEnabled = useOverrideConfigAtom(blockId, "editor:stickyscrollenabled") ?? false; const stickyScrollEnabled = useOverrideConfigAtom(blockId, "editor:stickyscrollenabled") ?? false;
const wordWrap = useOverrideConfigAtom(blockId, "editor:wordwrap") ?? false; const wordWrap = useOverrideConfigAtom(blockId, "editor:wordwrap") ?? false;
const fontSize = useOverrideConfigAtom(blockId, "editor:fontsize"); const fontSize = boundNumber(useOverrideConfigAtom(blockId, "editor:fontsize"), 6, 64);
const theme = "wave-theme-dark"; const theme = "wave-theme-dark";
React.useEffect(() => { React.useEffect(() => {

View File

@ -24,6 +24,7 @@ import {
} from "@/store/global"; } from "@/store/global";
import * as services from "@/store/services"; import * as services from "@/store/services";
import * as keyutil from "@/util/keyutil"; import * as keyutil from "@/util/keyutil";
import { boundNumber } from "@/util/util";
import clsx from "clsx"; import clsx from "clsx";
import debug from "debug"; import debug from "debug";
import * as jotai from "jotai"; import * as jotai from "jotai";
@ -62,6 +63,7 @@ class TermViewModel implements ViewModel {
vdomToolbarTarget: jotai.PrimitiveAtom<VDomTargetToolbar>; vdomToolbarTarget: jotai.PrimitiveAtom<VDomTargetToolbar>;
fontSizeAtom: jotai.Atom<number>; fontSizeAtom: jotai.Atom<number>;
termThemeNameAtom: jotai.Atom<string>; termThemeNameAtom: jotai.Atom<string>;
termTransparencyAtom: jotai.Atom<number>;
noPadding: jotai.PrimitiveAtom<boolean>; noPadding: jotai.PrimitiveAtom<boolean>;
endIconButtons: jotai.Atom<IconButtonDecl[]>; endIconButtons: jotai.Atom<IconButtonDecl[]>;
shellProcFullStatus: jotai.PrimitiveAtom<BlockControllerRuntimeStatus>; shellProcFullStatus: jotai.PrimitiveAtom<BlockControllerRuntimeStatus>;
@ -203,10 +205,17 @@ class TermViewModel implements ViewModel {
return get(getOverrideConfigAtom(this.blockId, "term:theme")) ?? DefaultTermTheme; return get(getOverrideConfigAtom(this.blockId, "term:theme")) ?? DefaultTermTheme;
}); });
}); });
this.termTransparencyAtom = useBlockAtom(blockId, "termtransparencyatom", () => {
return jotai.atom<number>((get) => {
let value = get(getOverrideConfigAtom(this.blockId, "term:transparency")) ?? 0.5;
return boundNumber(value, 0, 1);
});
});
this.blockBg = jotai.atom((get) => { this.blockBg = jotai.atom((get) => {
const fullConfig = get(atoms.fullConfigAtom); const fullConfig = get(atoms.fullConfigAtom);
const themeName = get(this.termThemeNameAtom); const themeName = get(this.termThemeNameAtom);
const [_, bgcolor] = computeTheme(fullConfig, themeName); const termTransparency = get(this.termTransparencyAtom);
const [_, bgcolor] = computeTheme(fullConfig, themeName, termTransparency);
if (bgcolor != null) { if (bgcolor != null) {
return { bg: bgcolor }; return { bg: bgcolor };
} }
@ -407,6 +416,11 @@ class TermViewModel implements ViewModel {
event.preventDefault(); event.preventDefault();
event.stopPropagation(); event.stopPropagation();
return false; return false;
} else if (keyutil.checkKeyPressed(waveEvent, "Cmd:k")) {
event.preventDefault();
event.stopPropagation();
this.termRef.current?.terminal?.clear();
return false;
} }
const shellProcStatus = globalStore.get(this.shellProcStatus); const shellProcStatus = globalStore.get(this.shellProcStatus);
if ((shellProcStatus == "done" || shellProcStatus == "init") && keyutil.checkKeyPressed(waveEvent, "Enter")) { if ((shellProcStatus == "done" || shellProcStatus == "init") && keyutil.checkKeyPressed(waveEvent, "Enter")) {
@ -453,6 +467,7 @@ class TermViewModel implements ViewModel {
const termThemeKeys = Object.keys(termThemes); const termThemeKeys = Object.keys(termThemes);
const curThemeName = globalStore.get(getBlockMetaKeyAtom(this.blockId, "term:theme")); const curThemeName = globalStore.get(getBlockMetaKeyAtom(this.blockId, "term:theme"));
const defaultFontSize = globalStore.get(getSettingsKeyAtom("term:fontsize")) ?? 12; const defaultFontSize = globalStore.get(getSettingsKeyAtom("term:fontsize")) ?? 12;
const transparencyMeta = globalStore.get(getBlockMetaKeyAtom(this.blockId, "term:transparency"));
const blockData = globalStore.get(this.blockAtom); const blockData = globalStore.get(this.blockAtom);
const overrideFontSize = blockData?.meta?.["term:fontsize"]; const overrideFontSize = blockData?.meta?.["term:fontsize"];
@ -474,6 +489,41 @@ class TermViewModel implements ViewModel {
checked: curThemeName == null, checked: curThemeName == null,
click: () => this.setTerminalTheme(null), click: () => this.setTerminalTheme(null),
}); });
const transparencySubMenu: ContextMenuItem[] = [];
transparencySubMenu.push({
label: "Default",
type: "checkbox",
checked: transparencyMeta == null,
click: () => {
RpcApi.SetMetaCommand(TabRpcClient, {
oref: WOS.makeORef("block", this.blockId),
meta: { "term:transparency": null },
});
},
});
transparencySubMenu.push({
label: "Transparent Background",
type: "checkbox",
checked: transparencyMeta == 0.5,
click: () => {
RpcApi.SetMetaCommand(TabRpcClient, {
oref: WOS.makeORef("block", this.blockId),
meta: { "term:transparency": 0.5 },
});
},
});
transparencySubMenu.push({
label: "No Transparency",
type: "checkbox",
checked: transparencyMeta == 0,
click: () => {
RpcApi.SetMetaCommand(TabRpcClient, {
oref: WOS.makeORef("block", this.blockId),
meta: { "term:transparency": 0 },
});
},
});
const fontSizeSubMenu: ContextMenuItem[] = [6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18].map( const fontSizeSubMenu: ContextMenuItem[] = [6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18].map(
(fontSize: number) => { (fontSize: number) => {
return { return {
@ -508,6 +558,10 @@ class TermViewModel implements ViewModel {
label: "Font Size", label: "Font Size",
submenu: fontSizeSubMenu, submenu: fontSizeSubMenu,
}); });
fullMenu.push({
label: "Transparency",
submenu: transparencySubMenu,
});
fullMenu.push({ type: "separator" }); fullMenu.push({ type: "separator" });
fullMenu.push({ fullMenu.push({
label: "Force Restart Controller", label: "Force Restart Controller",
@ -734,7 +788,8 @@ const TerminalView = ({ blockId, model }: TerminalViewProps) => {
React.useEffect(() => { React.useEffect(() => {
const fullConfig = globalStore.get(atoms.fullConfigAtom); const fullConfig = globalStore.get(atoms.fullConfigAtom);
const termThemeName = globalStore.get(model.termThemeNameAtom); const termThemeName = globalStore.get(model.termThemeNameAtom);
const [termTheme, _] = computeTheme(fullConfig, termThemeName); const termTransparency = globalStore.get(model.termTransparencyAtom);
const [termTheme, _] = computeTheme(fullConfig, termThemeName, termTransparency);
let termScrollback = 1000; let termScrollback = 1000;
if (termSettings?.["term:scrollback"]) { if (termSettings?.["term:scrollback"]) {
termScrollback = Math.floor(termSettings["term:scrollback"]); termScrollback = Math.floor(termSettings["term:scrollback"]);

View File

@ -17,7 +17,8 @@ interface TermThemeProps {
const TermThemeUpdater = ({ blockId, model, termRef }: TermThemeProps) => { const TermThemeUpdater = ({ blockId, model, termRef }: TermThemeProps) => {
const fullConfig = useAtomValue(atoms.fullConfigAtom); const fullConfig = useAtomValue(atoms.fullConfigAtom);
const blockTermTheme = useAtomValue(model.termThemeNameAtom); const blockTermTheme = useAtomValue(model.termThemeNameAtom);
const [theme, _] = computeTheme(fullConfig, blockTermTheme); const transparency = useAtomValue(model.termTransparencyAtom);
const [theme, _] = computeTheme(fullConfig, blockTermTheme, transparency);
useEffect(() => { useEffect(() => {
if (termRef.current?.terminal) { if (termRef.current?.terminal) {
termRef.current.terminal.options.theme = theme; termRef.current.terminal.options.theme = theme;

View File

@ -2,14 +2,32 @@
// SPDX-License-Identifier: Apache-2.0 // SPDX-License-Identifier: Apache-2.0
export const DefaultTermTheme = "default-dark"; export const DefaultTermTheme = "default-dark";
import { colord } from "colord";
// returns (theme, bgcolor) function applyTransparencyToColor(hexColor: string, transparency: number): string {
function computeTheme(fullConfig: FullConfigType, themeName: string): [TermThemeType, string] { const alpha = 1 - transparency; // transparency is already 0-1
return colord(hexColor).alpha(alpha).toHex();
}
// returns (theme, bgcolor, transparency (0 - 1.0))
function computeTheme(
fullConfig: FullConfigType,
themeName: string,
termTransparency: number
): [TermThemeType, string] {
let theme: TermThemeType = fullConfig?.termthemes?.[themeName]; let theme: TermThemeType = fullConfig?.termthemes?.[themeName];
if (theme == null) { if (theme == null) {
theme = fullConfig?.termthemes?.[DefaultTermTheme] || ({} as any); theme = fullConfig?.termthemes?.[DefaultTermTheme] || ({} as any);
} }
const themeCopy = { ...theme }; const themeCopy = { ...theme };
if (termTransparency != null && termTransparency > 0) {
if (themeCopy.background) {
themeCopy.background = applyTransparencyToColor(themeCopy.background, termTransparency);
}
if (themeCopy.selectionBackground) {
themeCopy.selectionBackground = applyTransparencyToColor(themeCopy.selectionBackground, termTransparency);
}
}
let bgcolor = themeCopy.background; let bgcolor = themeCopy.background;
themeCopy.background = "#00000000"; themeCopy.background = "#00000000";
return [themeCopy, bgcolor]; return [themeCopy, bgcolor];

View File

@ -4,28 +4,26 @@
.waveai { .waveai {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
overflow: hidden;
height: 100%; height: 100%;
width: 100%; width: 100%;
.waveai-chat { .waveai-chat {
flex-grow: 1; flex: 1 1 auto;
> .scrollable { overflow: hidden;
flex-flow: column nowrap; .chat-window-container {
margin-bottom: 0;
overflow-y: auto; overflow-y: auto;
min-height: 100%; margin-bottom: 0;
height: 100%;
.chat-window { .chat-window {
flex-flow: column nowrap;
display: flex; display: flex;
flex-direction: column;
gap: 8px; gap: 8px;
// This is the filler that will push the chat messages to the bottom until the chat window is full // This is the filler that will push the chat messages to the bottom until the chat window is full
.filler { .filler {
flex: 1 1 auto; flex: 1 1 auto;
} }
}
.chat-msg-container { .chat-msg-container {
display: flex; display: flex;
@ -103,8 +101,10 @@
} }
} }
} }
}
.waveai-controls { .waveai-controls {
flex: 0 0 auto;
display: flex; display: flex;
flex-direction: row; flex-direction: row;
align-items: center; align-items: center;

View File

@ -13,9 +13,11 @@ import { BlockService, ObjectService } from "@/store/services";
import { adaptFromReactOrNativeKeyEvent, checkKeyPressed } from "@/util/keyutil"; import { adaptFromReactOrNativeKeyEvent, checkKeyPressed } from "@/util/keyutil";
import { fireAndForget, isBlank, makeIconClass } from "@/util/util"; import { fireAndForget, isBlank, makeIconClass } from "@/util/util";
import { atom, Atom, PrimitiveAtom, useAtomValue, WritableAtom } from "jotai"; import { atom, Atom, PrimitiveAtom, useAtomValue, WritableAtom } from "jotai";
import { splitAtom } from "jotai/utils";
import type { OverlayScrollbars } from "overlayscrollbars"; import type { OverlayScrollbars } from "overlayscrollbars";
import { OverlayScrollbarsComponent, OverlayScrollbarsComponentRef } from "overlayscrollbars-react"; import { OverlayScrollbarsComponent, OverlayScrollbarsComponentRef } from "overlayscrollbars-react";
import { forwardRef, memo, useCallback, useEffect, useImperativeHandle, useMemo, useRef, useState } from "react"; import { forwardRef, memo, useCallback, useEffect, useImperativeHandle, useMemo, useRef, useState } from "react";
import { debounce, throttle } from "throttle-debounce";
import "./waveai.scss"; import "./waveai.scss";
interface ChatMessageType { interface ChatMessageType {
@ -29,7 +31,7 @@ const outline = "2px solid var(--accent-color)";
const slidingWindowSize = 30; const slidingWindowSize = 30;
interface ChatItemProps { interface ChatItemProps {
chatItem: ChatMessageType; chatItemAtom: Atom<ChatMessageType>;
model: WaveAiModel; model: WaveAiModel;
} }
@ -72,6 +74,8 @@ export class WaveAiModel implements ViewModel {
preIconButton?: Atom<IconButtonDecl>; preIconButton?: Atom<IconButtonDecl>;
endIconButtons?: Atom<IconButtonDecl[]>; endIconButtons?: Atom<IconButtonDecl[]>;
messagesAtom: PrimitiveAtom<Array<ChatMessageType>>; messagesAtom: PrimitiveAtom<Array<ChatMessageType>>;
messagesSplitAtom: SplitAtom<Array<ChatMessageType>>;
latestMessageAtom: Atom<ChatMessageType>;
addMessageAtom: WritableAtom<unknown, [message: ChatMessageType], void>; addMessageAtom: WritableAtom<unknown, [message: ChatMessageType], void>;
updateLastMessageAtom: WritableAtom<unknown, [text: string, isUpdating: boolean], void>; updateLastMessageAtom: WritableAtom<unknown, [text: string, isUpdating: boolean], void>;
removeLastMessageAtom: WritableAtom<unknown, [], void>; removeLastMessageAtom: WritableAtom<unknown, [], void>;
@ -92,6 +96,8 @@ export class WaveAiModel implements ViewModel {
this.viewIcon = atom("sparkles"); this.viewIcon = atom("sparkles");
this.viewName = atom("Wave AI"); this.viewName = atom("Wave AI");
this.messagesAtom = atom([]); this.messagesAtom = atom([]);
this.messagesSplitAtom = splitAtom(this.messagesAtom);
this.latestMessageAtom = atom((get) => get(this.messagesAtom).slice(-1)[0]);
this.presetKey = atom((get) => { this.presetKey = atom((get) => {
const metaPresetKey = get(this.blockAtom).meta["ai:preset"]; const metaPresetKey = get(this.blockAtom).meta["ai:preset"];
const globalPresetKey = get(atoms.settingsAtom)["ai:preset"]; const globalPresetKey = get(atoms.settingsAtom)["ai:preset"];
@ -405,10 +411,8 @@ export class WaveAiModel implements ViewModel {
} }
useWaveAi() { useWaveAi() {
const messages = useAtomValue(this.messagesAtom);
return { return {
messages, sendMessage: this.sendMessage.bind(this) as (text: string) => void,
sendMessage: this.sendMessage.bind(this),
}; };
} }
@ -431,10 +435,9 @@ function makeWaveAiViewModel(blockId: string): WaveAiModel {
return waveAiModel; return waveAiModel;
} }
const ChatItem = ({ chatItem, model }: ChatItemProps) => { const ChatItem = ({ chatItemAtom, model }: ChatItemProps) => {
const chatItem = useAtomValue(chatItemAtom);
const { user, text } = chatItem; const { user, text } = chatItem;
const cssVar = "--panel-bg-color";
const panelBgColor = getComputedStyle(document.documentElement).getPropertyValue(cssVar).trim();
const fontSize = useOverrideConfigAtom(model.blockId, "ai:fontsize"); const fontSize = useOverrideConfigAtom(model.blockId, "ai:fontsize");
const fixedFontSize = useOverrideConfigAtom(model.blockId, "ai:fixedfontsize"); const fixedFontSize = useOverrideConfigAtom(model.blockId, "ai:fixedfontsize");
const renderContent = useMemo(() => { const renderContent = useMemo(() => {
@ -503,43 +506,65 @@ const ChatItem = ({ chatItem, model }: ChatItemProps) => {
interface ChatWindowProps { interface ChatWindowProps {
chatWindowRef: React.RefObject<HTMLDivElement>; chatWindowRef: React.RefObject<HTMLDivElement>;
messages: ChatMessageType[];
msgWidths: Object; msgWidths: Object;
model: WaveAiModel; model: WaveAiModel;
} }
const ChatWindow = memo( const ChatWindow = memo(
forwardRef<OverlayScrollbarsComponentRef, ChatWindowProps>(({ chatWindowRef, messages, msgWidths, model }, ref) => { forwardRef<OverlayScrollbarsComponentRef, ChatWindowProps>(({ chatWindowRef, msgWidths, model }, ref) => {
const [isUserScrolling, setIsUserScrolling] = useState(false); const isUserScrolling = useRef(false);
const osRef = useRef<OverlayScrollbarsComponentRef>(null); const osRef = useRef<OverlayScrollbarsComponentRef>(null);
const prevMessagesLenRef = useRef(messages.length); const splitMessages = useAtomValue(model.messagesSplitAtom) as Atom<ChatMessageType>[];
const latestMessage = useAtomValue(model.latestMessageAtom);
const prevMessagesLenRef = useRef(splitMessages.length);
useImperativeHandle(ref, () => osRef.current as OverlayScrollbarsComponentRef); useImperativeHandle(ref, () => osRef.current as OverlayScrollbarsComponentRef);
useEffect(() => { const handleNewMessage = useCallback(
if (osRef.current && osRef.current.osInstance()) { throttle(100, (messagesLen: number) => {
if (osRef.current?.osInstance()) {
console.log("handleNewMessage", messagesLen, isUserScrolling.current);
const { viewport } = osRef.current.osInstance().elements(); const { viewport } = osRef.current.osInstance().elements();
const curMessagesLen = messages.length; if (prevMessagesLenRef.current !== messagesLen || !isUserScrolling.current) {
if (prevMessagesLenRef.current !== curMessagesLen || !isUserScrolling) {
setIsUserScrolling(false);
viewport.scrollTo({ viewport.scrollTo({
behavior: "auto", behavior: "auto",
top: chatWindowRef.current?.scrollHeight || 0, top: chatWindowRef.current?.scrollHeight || 0,
}); });
} }
prevMessagesLenRef.current = curMessagesLen; prevMessagesLenRef.current = messagesLen;
} }
}, [messages, isUserScrolling]); }),
[]
);
useEffect(() => { useEffect(() => {
if (osRef.current && osRef.current.osInstance()) { handleNewMessage(splitMessages.length);
const { viewport } = osRef.current.osInstance().elements(); }, [splitMessages, latestMessage]);
const handleUserScroll = () => { // Wait 300 ms after the user stops scrolling to determine if the user is within 300px of the bottom of the chat window.
setIsUserScrolling(true); // If so, unset the user scrolling flag.
}; const determineUnsetScroll = useCallback(
debounce(300, () => {
const { viewport } = osRef.current.osInstance().elements();
if (viewport.scrollTop > chatWindowRef.current?.clientHeight - viewport.clientHeight - 100) {
isUserScrolling.current = false;
}
}),
[]
);
const handleUserScroll = useCallback(
throttle(100, () => {
isUserScrolling.current = true;
determineUnsetScroll();
}),
[]
);
useEffect(() => {
if (osRef.current?.osInstance()) {
const { viewport } = osRef.current.osInstance().elements();
viewport.addEventListener("wheel", handleUserScroll, { passive: true }); viewport.addEventListener("wheel", handleUserScroll, { passive: true });
viewport.addEventListener("touchmove", handleUserScroll, { passive: true }); viewport.addEventListener("touchmove", handleUserScroll, { passive: true });
@ -571,14 +596,14 @@ const ChatWindow = memo(
return ( return (
<OverlayScrollbarsComponent <OverlayScrollbarsComponent
ref={osRef} ref={osRef}
className="scrollable" className="chat-window-container"
options={{ scrollbars: { autoHide: "leave" } }} options={{ scrollbars: { autoHide: "leave" } }}
events={{ initialized: handleScrollbarInitialized, updated: handleScrollbarUpdated }} events={{ initialized: handleScrollbarInitialized, updated: handleScrollbarUpdated }}
> >
<div ref={chatWindowRef} className="chat-window" style={msgWidths}> <div ref={chatWindowRef} className="chat-window" style={msgWidths}>
<div className="filler"></div> <div className="filler"></div>
{messages.map((chitem, idx) => ( {splitMessages.map((chitem, idx) => (
<ChatItem key={idx} chatItem={chitem} model={model} /> <ChatItem key={idx} chatItemAtom={chitem} model={model} />
))} ))}
</div> </div>
</OverlayScrollbarsComponent> </OverlayScrollbarsComponent>
@ -652,7 +677,7 @@ const ChatInput = forwardRef<HTMLTextAreaElement, ChatInputProps>(
); );
const WaveAi = ({ model }: { model: WaveAiModel; blockId: string }) => { const WaveAi = ({ model }: { model: WaveAiModel; blockId: string }) => {
const { messages, sendMessage } = model.useWaveAi(); const { sendMessage } = model.useWaveAi();
const waveaiRef = useRef<HTMLDivElement>(null); const waveaiRef = useRef<HTMLDivElement>(null);
const chatWindowRef = useRef<HTMLDivElement>(null); const chatWindowRef = useRef<HTMLDivElement>(null);
const osRef = useRef<OverlayScrollbarsComponentRef>(null); const osRef = useRef<OverlayScrollbarsComponentRef>(null);
@ -716,7 +741,7 @@ const WaveAi = ({ model }: { model: WaveAiModel; blockId: string }) => {
sendMessage(value); sendMessage(value);
setValue(""); setValue("");
setSelectedBlockIdx(null); setSelectedBlockIdx(null);
}, [messages, value]); }, [value]);
const updateScrollTop = () => { const updateScrollTop = () => {
const pres = chatWindowRef.current?.querySelectorAll("pre"); const pres = chatWindowRef.current?.querySelectorAll("pre");
@ -823,13 +848,7 @@ const WaveAi = ({ model }: { model: WaveAiModel; blockId: string }) => {
return ( return (
<div ref={waveaiRef} className="waveai"> <div ref={waveaiRef} className="waveai">
<div className="waveai-chat"> <div className="waveai-chat">
<ChatWindow <ChatWindow ref={osRef} chatWindowRef={chatWindowRef} msgWidths={msgWidths} model={model} />
ref={osRef}
chatWindowRef={chatWindowRef}
messages={messages}
msgWidths={msgWidths}
model={model}
/>
</div> </div>
<div className="waveai-controls"> <div className="waveai-controls">
<div className="waveai-input-wrapper"> <div className="waveai-input-wrapper">

View File

@ -491,6 +491,7 @@ declare global {
"term:scrollback"?: number; "term:scrollback"?: number;
"term:vdomblockid"?: string; "term:vdomblockid"?: string;
"term:vdomtoolbarblockid"?: string; "term:vdomtoolbarblockid"?: string;
"term:transparency"?: number;
"web:zoom"?: number; "web:zoom"?: number;
"markdown:fontsize"?: number; "markdown:fontsize"?: number;
"markdown:fixedfontsize"?: number; "markdown:fixedfontsize"?: number;
@ -641,6 +642,7 @@ declare global {
"term:localshellopts"?: string[]; "term:localshellopts"?: string[];
"term:scrollback"?: number; "term:scrollback"?: number;
"term:copyonselect"?: boolean; "term:copyonselect"?: boolean;
"term:transparency"?: number;
"editor:minimapenabled"?: boolean; "editor:minimapenabled"?: boolean;
"editor:stickyscrollenabled"?: boolean; "editor:stickyscrollenabled"?: boolean;
"editor:wordwrap"?: boolean; "editor:wordwrap"?: boolean;
@ -678,6 +680,8 @@ declare global {
"window:magnifiedblocksize"?: number; "window:magnifiedblocksize"?: number;
"window:magnifiedblockblurprimarypx"?: number; "window:magnifiedblockblurprimarypx"?: number;
"window:magnifiedblockblursecondarypx"?: number; "window:magnifiedblockblursecondarypx"?: number;
"window:confirmclose"?: boolean;
"window:savelastwindow"?: boolean;
"telemetry:*"?: boolean; "telemetry:*"?: boolean;
"telemetry:enabled"?: boolean; "telemetry:enabled"?: boolean;
"conn:*"?: boolean; "conn:*"?: boolean;

4
go.mod
View File

@ -45,9 +45,9 @@ require (
github.com/ubuntu/decorate v0.0.0-20230125165522-2d5b0a9bb117 // indirect github.com/ubuntu/decorate v0.0.0-20230125165522-2d5b0a9bb117 // indirect
github.com/yusufpapurcu/wmi v1.2.4 // indirect github.com/yusufpapurcu/wmi v1.2.4 // indirect
go.uber.org/atomic v1.7.0 // indirect go.uber.org/atomic v1.7.0 // indirect
golang.org/x/net v0.29.0 // indirect golang.org/x/net v0.33.0 // indirect
) )
replace github.com/kevinburke/ssh_config => github.com/wavetermdev/ssh_config v0.0.0-20241027232332-ed124367682d replace github.com/kevinburke/ssh_config => github.com/wavetermdev/ssh_config v0.0.0-20241219203747-6409e4292f34
replace github.com/creack/pty => github.com/photostorm/pty v1.1.19-0.20230903182454-31354506054b replace github.com/creack/pty => github.com/photostorm/pty v1.1.19-0.20230903182454-31354506054b

8
go.sum
View File

@ -90,16 +90,16 @@ github.com/ubuntu/gowsl v0.0.0-20240906163211-049fd49bd93b h1:wFBKF5k5xbJQU8bYgc
github.com/ubuntu/gowsl v0.0.0-20240906163211-049fd49bd93b/go.mod h1:N1CYNinssZru+ikvYTgVbVeSi21thHUTCoJ9xMvWe+s= github.com/ubuntu/gowsl v0.0.0-20240906163211-049fd49bd93b/go.mod h1:N1CYNinssZru+ikvYTgVbVeSi21thHUTCoJ9xMvWe+s=
github.com/wavetermdev/htmltoken v0.2.0 h1:sFVPPemlDv7/jg7n4Hx1AEF2m9MVAFjFpELWfhi/DlM= github.com/wavetermdev/htmltoken v0.2.0 h1:sFVPPemlDv7/jg7n4Hx1AEF2m9MVAFjFpELWfhi/DlM=
github.com/wavetermdev/htmltoken v0.2.0/go.mod h1:5FM0XV6zNYiNza2iaTcFGj+hnMtgqumFHO31Z8euquk= github.com/wavetermdev/htmltoken v0.2.0/go.mod h1:5FM0XV6zNYiNza2iaTcFGj+hnMtgqumFHO31Z8euquk=
github.com/wavetermdev/ssh_config v0.0.0-20241027232332-ed124367682d h1:ArHaUBaiQWUqBzM2G/oLlm3Be0kwUMDt9vTNOWIfOd0= github.com/wavetermdev/ssh_config v0.0.0-20241219203747-6409e4292f34 h1:I8VZVTZEXhnzfN7jB9a7TZYpzNO48sCUWMRXHM9XWSA=
github.com/wavetermdev/ssh_config v0.0.0-20241027232332-ed124367682d/go.mod h1:q2RIzfka+BXARoNexmF9gkxEX7DmvbW9P4hIVx2Kg4M= github.com/wavetermdev/ssh_config v0.0.0-20241219203747-6409e4292f34/go.mod h1:q2RIzfka+BXARoNexmF9gkxEX7DmvbW9P4hIVx2Kg4M=
github.com/yusufpapurcu/wmi v1.2.4 h1:zFUKzehAFReQwLys1b/iSMl+JQGSCSjtVqQn9bBrPo0= github.com/yusufpapurcu/wmi v1.2.4 h1:zFUKzehAFReQwLys1b/iSMl+JQGSCSjtVqQn9bBrPo0=
github.com/yusufpapurcu/wmi v1.2.4/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0= github.com/yusufpapurcu/wmi v1.2.4/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0=
go.uber.org/atomic v1.7.0 h1:ADUqmZGgLDDfbSL9ZmPxKTybcoEYHgpYfELNoN+7hsw= go.uber.org/atomic v1.7.0 h1:ADUqmZGgLDDfbSL9ZmPxKTybcoEYHgpYfELNoN+7hsw=
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
golang.org/x/crypto v0.31.0 h1:ihbySMvVjLAeSH1IbfcRTkD/iNscyz8rGzjF/E5hV6U= golang.org/x/crypto v0.31.0 h1:ihbySMvVjLAeSH1IbfcRTkD/iNscyz8rGzjF/E5hV6U=
golang.org/x/crypto v0.31.0/go.mod h1:kDsLvtWBEx7MV9tJOj9bnXsPbxwJQ6csT/x4KIN4Ssk= golang.org/x/crypto v0.31.0/go.mod h1:kDsLvtWBEx7MV9tJOj9bnXsPbxwJQ6csT/x4KIN4Ssk=
golang.org/x/net v0.29.0 h1:5ORfpBpCs4HzDYoodCDBbwHzdR5UrLBZ3sOnUJmFoHo= golang.org/x/net v0.33.0 h1:74SYHlV8BIgHIFC/LrYkOGIwL19eTYXQ5wc6TBuO36I=
golang.org/x/net v0.29.0/go.mod h1:gLkgy8jTGERgjzMic6DS9+SP0ajcu6Xu3Orq/SpETg0= golang.org/x/net v0.33.0/go.mod h1:HXLR5J+9DxmrqMwG9qjGCxZ+zKXxBru04zlTvWlWuN4=
golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20201204225414-ed752295db88/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201204225414-ed752295db88/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=

View File

@ -7,7 +7,7 @@
"productName": "Wave", "productName": "Wave",
"description": "Open-Source AI-Native Terminal Built for Seamless Workflows", "description": "Open-Source AI-Native Terminal Built for Seamless Workflows",
"license": "Apache-2.0", "license": "Apache-2.0",
"version": "0.10.2-beta.0", "version": "0.10.4",
"homepage": "https://waveterm.dev", "homepage": "https://waveterm.dev",
"build": { "build": {
"appId": "dev.commandline.waveterm" "appId": "dev.commandline.waveterm"

View File

@ -227,6 +227,9 @@ func (entry *CacheEntry) readAt(ctx context.Context, offset int64, size int64, r
offset += truncateAmt offset += truncateAmt
size -= truncateAmt size -= truncateAmt
} }
if size <= 0 {
return realDataOffset, nil, nil
}
} }
partMap := file.computePartMap(offset, size) partMap := file.computePartMap(offset, size)
dataEntryMap, err := entry.loadDataPartsForRead(ctx, getPartIdxsFromMap(partMap)) dataEntryMap, err := entry.loadDataPartsForRead(ctx, getPartIdxsFromMap(partMap))

View File

@ -23,6 +23,7 @@ import (
"github.com/kevinburke/ssh_config" "github.com/kevinburke/ssh_config"
"github.com/skeema/knownhosts" "github.com/skeema/knownhosts"
"github.com/wavetermdev/waveterm/pkg/panichandler"
"github.com/wavetermdev/waveterm/pkg/trimquotes" "github.com/wavetermdev/waveterm/pkg/trimquotes"
"github.com/wavetermdev/waveterm/pkg/userinput" "github.com/wavetermdev/waveterm/pkg/userinput"
"github.com/wavetermdev/waveterm/pkg/util/shellutil" "github.com/wavetermdev/waveterm/pkg/util/shellutil"
@ -750,7 +751,13 @@ func combineSshKeywords(userProvidedOpts *wshrpc.ConnKeywords, configKeywords *w
// note that a `var == "yes"` will default to false // note that a `var == "yes"` will default to false
// but `var != "no"` will default to true // but `var != "no"` will default to true
// when given unexpected strings // when given unexpected strings
func findSshConfigKeywords(hostPattern string) (*wshrpc.ConnKeywords, error) { func findSshConfigKeywords(hostPattern string) (connKeywords *wshrpc.ConnKeywords, outErr error) {
defer func() {
err := panichandler.PanicHandler("sshclient:find-ssh-config-keywords")
if err != nil {
outErr = err
}
}()
WaveSshConfigUserSettings().ReloadConfigs() WaveSshConfigUserSettings().ReloadConfigs()
sshKeywords := &wshrpc.ConnKeywords{} sshKeywords := &wshrpc.ConnKeywords{}
var err error var err error

View File

@ -45,10 +45,13 @@ func (svc *WorkspaceService) UpdateWorkspace_Meta() tsgenmeta.MethodMeta {
func (svc *WorkspaceService) UpdateWorkspace(ctx context.Context, workspaceId string, name string, icon string, color string, applyDefaults bool) (waveobj.UpdatesRtnType, error) { func (svc *WorkspaceService) UpdateWorkspace(ctx context.Context, workspaceId string, name string, icon string, color string, applyDefaults bool) (waveobj.UpdatesRtnType, error) {
ctx = waveobj.ContextWithUpdates(ctx) ctx = waveobj.ContextWithUpdates(ctx)
_, err := wcore.UpdateWorkspace(ctx, workspaceId, name, icon, color, applyDefaults) _, updated, err := wcore.UpdateWorkspace(ctx, workspaceId, name, icon, color, applyDefaults)
if err != nil { if err != nil {
return nil, fmt.Errorf("error updating workspace: %w", err) return nil, fmt.Errorf("error updating workspace: %w", err)
} }
if !updated {
return nil, nil
}
wps.Broker.Publish(wps.WaveEvent{ wps.Broker.Publish(wps.WaveEvent{
Event: wps.Event_WorkspaceUpdate}) Event: wps.Event_WorkspaceUpdate})

View File

@ -215,7 +215,7 @@ func StartWslShellProc(ctx context.Context, termSize waveobj.TermSize, cmdStr st
} }
if isZshShell(shellPath) { if isZshShell(shellPath) {
shellOpts = append(shellOpts, fmt.Sprintf(`ZDOTDIR="%s/.waveterm/%s"`, homeDir, shellutil.ZshIntegrationDir)) shellOpts = append(shellOpts, fmt.Sprintf(`ZDOTDIR=%s/.waveterm/%s`, homeDir, shellutil.ZshIntegrationDir))
} }
shellOpts = append(shellOpts, shellPath) shellOpts = append(shellOpts, shellPath)
shellOpts = append(shellOpts, subShellOpts...) shellOpts = append(shellOpts, subShellOpts...)

View File

@ -93,6 +93,7 @@ const (
MetaKey_TermScrollback = "term:scrollback" MetaKey_TermScrollback = "term:scrollback"
MetaKey_TermVDomSubBlockId = "term:vdomblockid" MetaKey_TermVDomSubBlockId = "term:vdomblockid"
MetaKey_TermVDomToolbarBlockId = "term:vdomtoolbarblockid" MetaKey_TermVDomToolbarBlockId = "term:vdomtoolbarblockid"
MetaKey_TermTransparency = "term:transparency"
MetaKey_WebZoom = "web:zoom" MetaKey_WebZoom = "web:zoom"

View File

@ -94,6 +94,7 @@ type MetaTSType struct {
TermScrollback *int `json:"term:scrollback,omitempty"` TermScrollback *int `json:"term:scrollback,omitempty"`
TermVDomSubBlockId string `json:"term:vdomblockid,omitempty"` TermVDomSubBlockId string `json:"term:vdomblockid,omitempty"`
TermVDomToolbarBlockId string `json:"term:vdomtoolbarblockid,omitempty"` TermVDomToolbarBlockId string `json:"term:vdomtoolbarblockid,omitempty"`
TermTransparency *float64 `json:"term:transparency,omitempty"` // default 0.5
WebZoom float64 `json:"web:zoom,omitempty"` WebZoom float64 `json:"web:zoom,omitempty"`

View File

@ -18,6 +18,8 @@
"window:magnifiedblocksize": 0.9, "window:magnifiedblocksize": 0.9,
"window:magnifiedblockblurprimarypx": 10, "window:magnifiedblockblurprimarypx": 10,
"window:magnifiedblockblursecondarypx": 2, "window:magnifiedblockblursecondarypx": 2,
"window:confirmclose": true,
"window:savelastwindow": true,
"telemetry:enabled": true, "telemetry:enabled": true,
"term:copyonselect": true "term:copyonselect": true
} }

View File

@ -22,12 +22,37 @@
"cmdtext": "#f0f0f0", "cmdtext": "#f0f0f0",
"foreground": "#c1c1c1", "foreground": "#c1c1c1",
"selectionBackground": "", "selectionBackground": "",
"background": "#00000077", "background": "#000000",
"cursor": "" "cursor": ""
}, },
"onedarkpro": {
"display:name": "One Dark Pro",
"display:order": 2,
"background": "#21252B",
"foreground": "#ABB2BF",
"cursor": "#D7DAE0",
"black": "#3F4451",
"red": "#E06C75",
"green": "#98C379",
"yellow": "#D18F52",
"blue": "#61AFEF",
"magenta": "#C678DD",
"cyan": "#42B3C2",
"white": "#D7DAE0",
"brightBlack": "#4F5666",
"brightRed": "#FF616E",
"brightGreen": "#A5E075",
"brightYellow": "#F0A45D",
"brightBlue": "#4DC4FF",
"brightMagenta": "#DE73FF",
"brightCyan": "#4CD1E0",
"brightWhite": "#E6E6E6",
"gray": "#495162",
"cmdtext": "#ABB2BF"
},
"dracula": { "dracula": {
"display:name": "Dracula", "display:name": "Dracula",
"display:order": 2, "display:order": 3,
"black": "#21222C", "black": "#21222C",
"red": "#FF5555", "red": "#FF5555",
"green": "#50FA7B", "green": "#50FA7B",
@ -47,13 +72,12 @@
"gray": "#6272A4", "gray": "#6272A4",
"cmdtext": "#F8F8F2", "cmdtext": "#F8F8F2",
"foreground": "#F8F8F2", "foreground": "#F8F8F2",
"selectionBackground": "#44475a",
"background": "#282a36", "background": "#282a36",
"cursor": "#f8f8f2" "cursor": "#f8f8f2"
}, },
"monokai": { "monokai": {
"display:name": "Monokai", "display:name": "Monokai",
"display:order": 3, "display:order": 4,
"black": "#1B1D1E", "black": "#1B1D1E",
"red": "#F92672", "red": "#F92672",
"green": "#A6E22E", "green": "#A6E22E",
@ -73,13 +97,12 @@
"gray": "#75715E", "gray": "#75715E",
"cmdtext": "#F8F8F2", "cmdtext": "#F8F8F2",
"foreground": "#F8F8F2", "foreground": "#F8F8F2",
"selectionBackground": "#49483E",
"background": "#272822", "background": "#272822",
"cursor": "#F8F8F2" "cursor": "#F8F8F2"
}, },
"campbell": { "campbell": {
"display:name": "Campbell", "display:name": "Campbell",
"display:order": 4, "display:order": 5,
"black": "#0C0C0C", "black": "#0C0C0C",
"red": "#C50F1F", "red": "#C50F1F",
"green": "#13A10E", "green": "#13A10E",
@ -99,13 +122,13 @@
"gray": "#767676", "gray": "#767676",
"cmdtext": "#CCCCCC", "cmdtext": "#CCCCCC",
"foreground": "#CCCCCC", "foreground": "#CCCCCC",
"selectionBackground": "#3A96DD", "selectionBackground": "#3A96DD77",
"background": "#0C0C0C", "background": "#0C0C0C",
"cursor": "#CCCCCC" "cursor": "#CCCCCC"
}, },
"warmyellow": { "warmyellow": {
"display:name": "Warm Yellow", "display:name": "Warm Yellow",
"display:order": 4, "display:order": 6,
"black": "#3C3228", "black": "#3C3228",
"red": "#E67E22", "red": "#E67E22",
"green": "#A5D6A7", "green": "#A5D6A7",
@ -124,33 +147,32 @@
"brightWhite": "#FFFFFF", "brightWhite": "#FFFFFF",
"background": "#2B2620", "background": "#2B2620",
"foreground": "#F2E6D4", "foreground": "#F2E6D4",
"selectionBackground": "#B7950B", "selectionBackground": "#B7950B77",
"cursor": "#F9D784" "cursor": "#F9D784"
}, },
"onedarkpro": { "rosepine": {
"display:name": "One Dark Pro", "display:name": "Rose Pine",
"display:order": 1.5, "display:order": 7,
"background": "#282C34", "black": "#26233a",
"foreground": "#ABB2BF", "red": "#eb6f92",
"cursor": "#D7DAE0", "green": "#3e8fb0",
"selectionBackground": "#528BFF", "yellow": "#f6c177",
"black": "#3F4451", "blue": "#9ccfd8",
"red": "#E05561", "magenta": "#c4a7e7",
"green": "#8CC265", "cyan": "#ebbcba",
"yellow": "#D18F52", "white": "#e0def4",
"blue": "#4AA5F0", "brightBlack": "#908caa",
"magenta": "#C162DE", "brightRed": "#ff8cab",
"cyan": "#42B3C2", "brightGreen": "#9ccfb0",
"white": "#D7DAE0", "brightYellow": "#ffd196",
"brightBlack": "#4F5666", "brightBlue": "#bee6e0",
"brightRed": "#FF616E", "brightMagenta": "#e2c4ff",
"brightGreen": "#A5E075", "brightCyan": "#ffd1d0",
"brightYellow": "#F0A45D", "brightWhite": "#fffaf3",
"brightBlue": "#4DC4FF", "gray": "#908caa",
"brightMagenta": "#DE73FF", "cmdtext": "#e0def4",
"brightCyan": "#4CD1E0", "foreground": "#e0def4",
"brightWhite": "#E6E6E6", "background": "#191724",
"gray": "#495162", "cursor": "#524f67"
"cmdtext": "#ABB2BF"
} }
} }

View File

@ -33,6 +33,7 @@ const (
ConfigKey_TermLocalShellOpts = "term:localshellopts" ConfigKey_TermLocalShellOpts = "term:localshellopts"
ConfigKey_TermScrollback = "term:scrollback" ConfigKey_TermScrollback = "term:scrollback"
ConfigKey_TermCopyOnSelect = "term:copyonselect" ConfigKey_TermCopyOnSelect = "term:copyonselect"
ConfigKey_TermTransparency = "term:transparency"
ConfigKey_EditorMinimapEnabled = "editor:minimapenabled" ConfigKey_EditorMinimapEnabled = "editor:minimapenabled"
ConfigKey_EditorStickyScrollEnabled = "editor:stickyscrollenabled" ConfigKey_EditorStickyScrollEnabled = "editor:stickyscrollenabled"
@ -79,6 +80,8 @@ const (
ConfigKey_WindowMagnifiedBlockSize = "window:magnifiedblocksize" ConfigKey_WindowMagnifiedBlockSize = "window:magnifiedblocksize"
ConfigKey_WindowMagnifiedBlockBlurPrimaryPx = "window:magnifiedblockblurprimarypx" ConfigKey_WindowMagnifiedBlockBlurPrimaryPx = "window:magnifiedblockblurprimarypx"
ConfigKey_WindowMagnifiedBlockBlurSecondaryPx = "window:magnifiedblockblursecondarypx" ConfigKey_WindowMagnifiedBlockBlurSecondaryPx = "window:magnifiedblockblursecondarypx"
ConfigKey_WindowConfirmClose = "window:confirmclose"
ConfigKey_WindowSaveLastWindow = "window:savelastwindow"
ConfigKey_TelemetryClear = "telemetry:*" ConfigKey_TelemetryClear = "telemetry:*"
ConfigKey_TelemetryEnabled = "telemetry:enabled" ConfigKey_TelemetryEnabled = "telemetry:enabled"

View File

@ -60,6 +60,7 @@ type SettingsType struct {
TermLocalShellOpts []string `json:"term:localshellopts,omitempty"` TermLocalShellOpts []string `json:"term:localshellopts,omitempty"`
TermScrollback *int64 `json:"term:scrollback,omitempty"` TermScrollback *int64 `json:"term:scrollback,omitempty"`
TermCopyOnSelect *bool `json:"term:copyonselect,omitempty"` TermCopyOnSelect *bool `json:"term:copyonselect,omitempty"`
TermTransparency *float64 `json:"term:transparency,omitempty"`
EditorMinimapEnabled bool `json:"editor:minimapenabled,omitempty"` EditorMinimapEnabled bool `json:"editor:minimapenabled,omitempty"`
EditorStickyScrollEnabled bool `json:"editor:stickyscrollenabled,omitempty"` EditorStickyScrollEnabled bool `json:"editor:stickyscrollenabled,omitempty"`
@ -106,6 +107,8 @@ type SettingsType struct {
WindowMagnifiedBlockSize *float64 `json:"window:magnifiedblocksize,omitempty"` WindowMagnifiedBlockSize *float64 `json:"window:magnifiedblocksize,omitempty"`
WindowMagnifiedBlockBlurPrimaryPx *int64 `json:"window:magnifiedblockblurprimarypx,omitempty"` WindowMagnifiedBlockBlurPrimaryPx *int64 `json:"window:magnifiedblockblurprimarypx,omitempty"`
WindowMagnifiedBlockBlurSecondaryPx *int64 `json:"window:magnifiedblockblursecondarypx,omitempty"` WindowMagnifiedBlockBlurSecondaryPx *int64 `json:"window:magnifiedblockblursecondarypx,omitempty"`
WindowConfirmClose bool `json:"window:confirmclose,omitempty"`
WindowSaveLastWindow bool `json:"window:savelastwindow,omitempty"`
TelemetryClear bool `json:"telemetry:*,omitempty"` TelemetryClear bool `json:"telemetry:*,omitempty"`
TelemetryEnabled bool `json:"telemetry:enabled,omitempty"` TelemetryEnabled bool `json:"telemetry:enabled,omitempty"`

View File

@ -25,6 +25,7 @@ func EnsureInitialData() error {
ctx, cancelFn := context.WithTimeout(context.Background(), 2*time.Second) ctx, cancelFn := context.WithTimeout(context.Background(), 2*time.Second)
defer cancelFn() defer cancelFn()
client, err := wstore.DBGetSingleton[*waveobj.Client](ctx) client, err := wstore.DBGetSingleton[*waveobj.Client](ctx)
firstLaunch := false
if err == wstore.ErrNotFound { if err == wstore.ErrNotFound {
client, err = CreateClient(ctx) client, err = CreateClient(ctx)
if err != nil { if err != nil {
@ -34,6 +35,7 @@ func EnsureInitialData() error {
if migrateErr != nil { if migrateErr != nil {
log.Printf("error migrating old history: %v\n", migrateErr) log.Printf("error migrating old history: %v\n", migrateErr)
} }
firstLaunch = true
} }
if client.TempOID == "" { if client.TempOID == "" {
log.Println("client.TempOID is empty") log.Println("client.TempOID is empty")
@ -53,12 +55,16 @@ func EnsureInitialData() error {
log.Println("client has windows") log.Println("client has windows")
return nil return nil
} }
log.Println("client has no windows, creating starter workspace") wsId := ""
if firstLaunch {
log.Println("client has no windows and first launch, creating starter workspace")
starterWs, err := CreateWorkspace(ctx, "Starter workspace", "custom@wave-logo-solid", "#58C142", false, true) starterWs, err := CreateWorkspace(ctx, "Starter workspace", "custom@wave-logo-solid", "#58C142", false, true)
if err != nil { if err != nil {
return fmt.Errorf("error creating starter workspace: %w", err) return fmt.Errorf("error creating starter workspace: %w", err)
} }
_, err = CreateWindow(ctx, nil, starterWs.OID) wsId = starterWs.OID
}
_, err = CreateWindow(ctx, nil, wsId)
if err != nil { if err != nil {
return fmt.Errorf("error creating window: %w", err) return fmt.Errorf("error creating window: %w", err)
} }

View File

@ -68,26 +68,34 @@ func CreateWorkspace(ctx context.Context, name string, icon string, color string
wps.Broker.Publish(wps.WaveEvent{ wps.Broker.Publish(wps.WaveEvent{
Event: wps.Event_WorkspaceUpdate}) Event: wps.Event_WorkspaceUpdate})
return UpdateWorkspace(ctx, ws.OID, name, icon, color, applyDefaults) ws, _, err = UpdateWorkspace(ctx, ws.OID, name, icon, color, applyDefaults)
return ws, err
} }
func UpdateWorkspace(ctx context.Context, workspaceId string, name string, icon string, color string, applyDefaults bool) (*waveobj.Workspace, error) { // Returns updated workspace, whether it was updated, error.
func UpdateWorkspace(ctx context.Context, workspaceId string, name string, icon string, color string, applyDefaults bool) (*waveobj.Workspace, bool, error) {
ws, err := GetWorkspace(ctx, workspaceId) ws, err := GetWorkspace(ctx, workspaceId)
updated := false
if err != nil { if err != nil {
return nil, fmt.Errorf("workspace %s not found: %w", workspaceId, err) return nil, updated, fmt.Errorf("workspace %s not found: %w", workspaceId, err)
} }
if name != "" { if name != "" {
ws.Name = name ws.Name = name
updated = true
} else if applyDefaults && ws.Name == "" { } else if applyDefaults && ws.Name == "" {
ws.Name = fmt.Sprintf("New Workspace (%s)", ws.OID[0:5]) ws.Name = fmt.Sprintf("New Workspace (%s)", ws.OID[0:5])
updated = true
} }
if icon != "" { if icon != "" {
ws.Icon = icon ws.Icon = icon
updated = true
} else if applyDefaults && ws.Icon == "" { } else if applyDefaults && ws.Icon == "" {
ws.Icon = WorkspaceIcons[0] ws.Icon = WorkspaceIcons[0]
updated = true
} }
if color != "" { if color != "" {
ws.Color = color ws.Color = color
updated = true
} else if applyDefaults && ws.Color == "" { } else if applyDefaults && ws.Color == "" {
wsList, err := ListWorkspaces(ctx) wsList, err := ListWorkspaces(ctx)
if err != nil { if err != nil {
@ -95,9 +103,12 @@ func UpdateWorkspace(ctx context.Context, workspaceId string, name string, icon
wsList = waveobj.WorkspaceList{} wsList = waveobj.WorkspaceList{}
} }
ws.Color = WorkspaceColors[len(wsList)%len(WorkspaceColors)] ws.Color = WorkspaceColors[len(wsList)%len(WorkspaceColors)]
updated = true
} }
if updated {
wstore.DBUpdate(ctx, ws) wstore.DBUpdate(ctx, ws)
return ws, nil }
return ws, updated, nil
} }
// If force is true, it will delete even if workspace is named. // If force is true, it will delete even if workspace is named.

View File

@ -342,7 +342,10 @@ func (w *WshRpc) runServer() {
continue continue
} }
if msg.IsRpcRequest() { if msg.IsRpcRequest() {
go w.handleRequest(&msg) go func() {
defer panichandler.PanicHandler("handleRequest:goroutine")
w.handleRequest(&msg)
}()
} else { } else {
respCh := w.getResponseCh(msg.ResId) respCh := w.getResponseCh(msg.ResId)
if respCh == nil { if respCh == nil {