mirror of
https://github.com/MODSetter/SurfSense.git
synced 2026-06-30 21:59:46 +02:00
Remove desktop suggestion UI and refresh settings surfaces
This commit is contained in:
parent
3f97b77ab6
commit
8034f372e7
8 changed files with 49 additions and 849 deletions
|
|
@ -20,7 +20,6 @@ import { searchSpacesApiService } from "@/lib/apis/search-spaces-api.service";
|
||||||
export function DesktopContent() {
|
export function DesktopContent() {
|
||||||
const api = useElectronAPI();
|
const api = useElectronAPI();
|
||||||
const [loading, setLoading] = useState(true);
|
const [loading, setLoading] = useState(true);
|
||||||
const [enabled, setEnabled] = useState(true);
|
|
||||||
|
|
||||||
const [searchSpaces, setSearchSpaces] = useState<SearchSpace[]>([]);
|
const [searchSpaces, setSearchSpaces] = useState<SearchSpace[]>([]);
|
||||||
const [activeSpaceId, setActiveSpaceId] = useState<string | null>(null);
|
const [activeSpaceId, setActiveSpaceId] = useState<string | null>(null);
|
||||||
|
|
@ -41,14 +40,12 @@ export function DesktopContent() {
|
||||||
setAutoLaunchSupported(hasAutoLaunchApi);
|
setAutoLaunchSupported(hasAutoLaunchApi);
|
||||||
|
|
||||||
Promise.all([
|
Promise.all([
|
||||||
api.getAutocompleteEnabled(),
|
|
||||||
api.getActiveSearchSpace?.() ?? Promise.resolve(null),
|
api.getActiveSearchSpace?.() ?? Promise.resolve(null),
|
||||||
searchSpacesApiService.getSearchSpaces(),
|
searchSpacesApiService.getSearchSpaces(),
|
||||||
hasAutoLaunchApi ? api.getAutoLaunch() : Promise.resolve(null),
|
hasAutoLaunchApi ? api.getAutoLaunch() : Promise.resolve(null),
|
||||||
])
|
])
|
||||||
.then(([autoEnabled, spaceId, spaces, autoLaunch]) => {
|
.then(([spaceId, spaces, autoLaunch]) => {
|
||||||
if (!mounted) return;
|
if (!mounted) return;
|
||||||
setEnabled(autoEnabled);
|
|
||||||
setActiveSpaceId(spaceId);
|
setActiveSpaceId(spaceId);
|
||||||
if (spaces) setSearchSpaces(spaces);
|
if (spaces) setSearchSpaces(spaces);
|
||||||
if (autoLaunch) {
|
if (autoLaunch) {
|
||||||
|
|
@ -86,11 +83,6 @@ export function DesktopContent() {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleToggle = async (checked: boolean) => {
|
|
||||||
setEnabled(checked);
|
|
||||||
await api.setAutocompleteEnabled(checked);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleAutoLaunchToggle = async (checked: boolean) => {
|
const handleAutoLaunchToggle = async (checked: boolean) => {
|
||||||
if (!autoLaunchSupported || !api.setAutoLaunch) {
|
if (!autoLaunchSupported || !api.setAutoLaunch) {
|
||||||
toast.error("Please update the desktop app to configure launch on startup");
|
toast.error("Please update the desktop app to configure launch on startup");
|
||||||
|
|
@ -133,13 +125,11 @@ export function DesktopContent() {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="space-y-4 md:space-y-6">
|
<div className="space-y-4 md:space-y-6">
|
||||||
{/* Default Search Space */}
|
|
||||||
<Card>
|
<Card>
|
||||||
<CardHeader className="px-3 md:px-6 pt-3 md:pt-6 pb-2 md:pb-3">
|
<CardHeader className="px-3 md:px-6 pt-3 md:pt-6 pb-2 md:pb-3">
|
||||||
<CardTitle className="text-base md:text-lg">Default Search Space</CardTitle>
|
<CardTitle className="text-base md:text-lg">Default Search Space</CardTitle>
|
||||||
<CardDescription className="text-xs md:text-sm">
|
<CardDescription className="text-xs md:text-sm">
|
||||||
Choose which search space General Assist, Quick Assist, and Extreme Assist operate
|
Choose which search space General Assist and Quick Assist use by default.
|
||||||
against.
|
|
||||||
</CardDescription>
|
</CardDescription>
|
||||||
</CardHeader>
|
</CardHeader>
|
||||||
<CardContent className="px-3 md:px-6 pb-3 md:pb-6">
|
<CardContent className="px-3 md:px-6 pb-3 md:pb-6">
|
||||||
|
|
@ -164,7 +154,6 @@ export function DesktopContent() {
|
||||||
</CardContent>
|
</CardContent>
|
||||||
</Card>
|
</Card>
|
||||||
|
|
||||||
{/* Launch on Startup */}
|
|
||||||
<Card>
|
<Card>
|
||||||
<CardHeader className="px-3 md:px-6 pt-3 md:pt-6 pb-2 md:pb-3">
|
<CardHeader className="px-3 md:px-6 pt-3 md:pt-6 pb-2 md:pb-3">
|
||||||
<CardTitle className="text-base md:text-lg flex items-center gap-2">
|
<CardTitle className="text-base md:text-lg flex items-center gap-2">
|
||||||
|
|
@ -215,29 +204,6 @@ export function DesktopContent() {
|
||||||
</div>
|
</div>
|
||||||
</CardContent>
|
</CardContent>
|
||||||
</Card>
|
</Card>
|
||||||
|
|
||||||
{/* Extreme Assist Toggle */}
|
|
||||||
<Card>
|
|
||||||
<CardHeader className="px-3 md:px-6 pt-3 md:pt-6 pb-2 md:pb-3">
|
|
||||||
<CardTitle className="text-base md:text-lg">Extreme Assist</CardTitle>
|
|
||||||
<CardDescription className="text-xs md:text-sm">
|
|
||||||
Get inline writing suggestions powered by your knowledge base as you type in any app.
|
|
||||||
</CardDescription>
|
|
||||||
</CardHeader>
|
|
||||||
<CardContent className="px-3 md:px-6 pb-3 md:pb-6">
|
|
||||||
<div className="flex items-center justify-between rounded-lg border p-4">
|
|
||||||
<div className="space-y-0.5">
|
|
||||||
<Label htmlFor="autocomplete-toggle" className="text-sm font-medium cursor-pointer">
|
|
||||||
Enable Extreme Assist
|
|
||||||
</Label>
|
|
||||||
<p className="text-xs text-muted-foreground">
|
|
||||||
Show suggestions while typing in other applications.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<Switch id="autocomplete-toggle" checked={enabled} onCheckedChange={handleToggle} />
|
|
||||||
</div>
|
|
||||||
</CardContent>
|
|
||||||
</Card>
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,6 +1,6 @@
|
||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { BrainCog, Rocket, RotateCcw, Zap } from "lucide-react";
|
import { Rocket, RotateCcw, Zap } from "lucide-react";
|
||||||
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
||||||
import { toast } from "sonner";
|
import { toast } from "sonner";
|
||||||
import { DEFAULT_SHORTCUTS, keyEventToAccelerator } from "@/components/desktop/shortcut-recorder";
|
import { DEFAULT_SHORTCUTS, keyEventToAccelerator } from "@/components/desktop/shortcut-recorder";
|
||||||
|
|
@ -9,13 +9,12 @@ import { ShortcutKbd } from "@/components/ui/shortcut-kbd";
|
||||||
import { Spinner } from "@/components/ui/spinner";
|
import { Spinner } from "@/components/ui/spinner";
|
||||||
import { useElectronAPI } from "@/hooks/use-platform";
|
import { useElectronAPI } from "@/hooks/use-platform";
|
||||||
|
|
||||||
type ShortcutKey = "generalAssist" | "quickAsk" | "autocomplete";
|
type ShortcutKey = "generalAssist" | "quickAsk";
|
||||||
type ShortcutMap = typeof DEFAULT_SHORTCUTS;
|
type ShortcutMap = typeof DEFAULT_SHORTCUTS;
|
||||||
|
|
||||||
const HOTKEY_ROWS: Array<{ key: ShortcutKey; label: string; icon: React.ElementType }> = [
|
const HOTKEY_ROWS: Array<{ key: ShortcutKey; label: string; icon: React.ElementType }> = [
|
||||||
{ key: "generalAssist", label: "General Assist", icon: Rocket },
|
{ key: "generalAssist", label: "General Assist", icon: Rocket },
|
||||||
{ key: "quickAsk", label: "Quick Assist", icon: Zap },
|
{ key: "quickAsk", label: "Quick Assist", icon: Zap },
|
||||||
{ key: "autocomplete", label: "Extreme Assist", icon: BrainCog },
|
|
||||||
];
|
];
|
||||||
|
|
||||||
function acceleratorToKeys(accel: string, isMac: boolean): string[] {
|
function acceleratorToKeys(accel: string, isMac: boolean): string[] {
|
||||||
|
|
@ -111,9 +110,7 @@ function HotkeyRow({
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
{recording ? (
|
{recording ? (
|
||||||
<span className="px-2 text-[9px] text-primary whitespace-nowrap">
|
<span className="px-2 text-[9px] text-primary whitespace-nowrap">Press hotkeys...</span>
|
||||||
Press hotkeys...
|
|
||||||
</span>
|
|
||||||
) : (
|
) : (
|
||||||
<ShortcutKbd keys={displayKeys} className="ml-0 px-1.5 text-foreground/85" />
|
<ShortcutKbd keys={displayKeys} className="ml-0 px-1.5 text-foreground/85" />
|
||||||
)}
|
)}
|
||||||
|
|
@ -155,15 +152,14 @@ export function DesktopShortcutsContent() {
|
||||||
if (!api) {
|
if (!api) {
|
||||||
return (
|
return (
|
||||||
<div className="flex flex-col items-center justify-center py-12 text-center">
|
<div className="flex flex-col items-center justify-center py-12 text-center">
|
||||||
<p className="text-sm text-muted-foreground">Hotkeys are only available in the SurfSense desktop app.</p>
|
<p className="text-sm text-muted-foreground">
|
||||||
|
Hotkeys are only available in the SurfSense desktop app.
|
||||||
|
</p>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const updateShortcut = (
|
const updateShortcut = (key: ShortcutKey, accelerator: string) => {
|
||||||
key: "generalAssist" | "quickAsk" | "autocomplete",
|
|
||||||
accelerator: string
|
|
||||||
) => {
|
|
||||||
setShortcuts((prev) => {
|
setShortcuts((prev) => {
|
||||||
const updated = { ...prev, [key]: accelerator };
|
const updated = { ...prev, [key]: accelerator };
|
||||||
api.setShortcuts?.({ [key]: accelerator }).catch(() => {
|
api.setShortcuts?.({ [key]: accelerator }).catch(() => {
|
||||||
|
|
@ -178,28 +174,26 @@ export function DesktopShortcutsContent() {
|
||||||
updateShortcut(key, DEFAULT_SHORTCUTS[key]);
|
updateShortcut(key, DEFAULT_SHORTCUTS[key]);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return shortcutsLoaded ? (
|
||||||
shortcutsLoaded ? (
|
<div className="flex flex-col gap-3">
|
||||||
<div className="flex flex-col gap-3">
|
<div>
|
||||||
<div>
|
{HOTKEY_ROWS.map((row) => (
|
||||||
{HOTKEY_ROWS.map((row) => (
|
<HotkeyRow
|
||||||
<HotkeyRow
|
key={row.key}
|
||||||
key={row.key}
|
label={row.label}
|
||||||
label={row.label}
|
value={shortcuts[row.key]}
|
||||||
value={shortcuts[row.key]}
|
defaultValue={DEFAULT_SHORTCUTS[row.key]}
|
||||||
defaultValue={DEFAULT_SHORTCUTS[row.key]}
|
icon={row.icon}
|
||||||
icon={row.icon}
|
isMac={isMac}
|
||||||
isMac={isMac}
|
onChange={(accel) => updateShortcut(row.key, accel)}
|
||||||
onChange={(accel) => updateShortcut(row.key, accel)}
|
onReset={() => resetShortcut(row.key)}
|
||||||
onReset={() => resetShortcut(row.key)}
|
/>
|
||||||
/>
|
))}
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
) : (
|
</div>
|
||||||
<div className="flex justify-center py-4">
|
) : (
|
||||||
<Spinner size="sm" />
|
<div className="flex justify-center py-4">
|
||||||
</div>
|
<Spinner size="sm" />
|
||||||
)
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -2,7 +2,7 @@
|
||||||
|
|
||||||
import { IconBrandGoogleFilled } from "@tabler/icons-react";
|
import { IconBrandGoogleFilled } from "@tabler/icons-react";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { BrainCog, Eye, EyeOff, Rocket, RotateCcw, Zap } from "lucide-react";
|
import { Eye, EyeOff, Rocket, RotateCcw, Zap } from "lucide-react";
|
||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
import { useRouter } from "next/navigation";
|
import { useRouter } from "next/navigation";
|
||||||
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
||||||
|
|
@ -21,10 +21,15 @@ import { setBearerToken } from "@/lib/auth-utils";
|
||||||
import { AUTH_TYPE, BACKEND_URL } from "@/lib/env-config";
|
import { AUTH_TYPE, BACKEND_URL } from "@/lib/env-config";
|
||||||
|
|
||||||
const isGoogleAuth = AUTH_TYPE === "GOOGLE";
|
const isGoogleAuth = AUTH_TYPE === "GOOGLE";
|
||||||
type ShortcutKey = "generalAssist" | "quickAsk" | "autocomplete";
|
type ShortcutKey = "generalAssist" | "quickAsk";
|
||||||
type ShortcutMap = typeof DEFAULT_SHORTCUTS;
|
type ShortcutMap = typeof DEFAULT_SHORTCUTS;
|
||||||
|
|
||||||
const HOTKEY_ROWS: Array<{ key: ShortcutKey; label: string; description: string; icon: React.ElementType }> = [
|
const HOTKEY_ROWS: Array<{
|
||||||
|
key: ShortcutKey;
|
||||||
|
label: string;
|
||||||
|
description: string;
|
||||||
|
icon: React.ElementType;
|
||||||
|
}> = [
|
||||||
{
|
{
|
||||||
key: "generalAssist",
|
key: "generalAssist",
|
||||||
label: "General Assist",
|
label: "General Assist",
|
||||||
|
|
@ -37,12 +42,6 @@ const HOTKEY_ROWS: Array<{ key: ShortcutKey; label: string; description: string;
|
||||||
description: "Select text anywhere, then ask AI to explain, rewrite, or act on it",
|
description: "Select text anywhere, then ask AI to explain, rewrite, or act on it",
|
||||||
icon: Zap,
|
icon: Zap,
|
||||||
},
|
},
|
||||||
{
|
|
||||||
key: "autocomplete",
|
|
||||||
label: "Extreme Assist",
|
|
||||||
description: "AI drafts text using your screen context and knowledge base",
|
|
||||||
icon: BrainCog,
|
|
||||||
},
|
|
||||||
];
|
];
|
||||||
|
|
||||||
function acceleratorToKeys(accel: string, isMac: boolean): string[] {
|
function acceleratorToKeys(accel: string, isMac: boolean): string[] {
|
||||||
|
|
@ -182,7 +181,7 @@ export default function DesktopLoginPage() {
|
||||||
}, [api]);
|
}, [api]);
|
||||||
|
|
||||||
const updateShortcut = useCallback(
|
const updateShortcut = useCallback(
|
||||||
(key: "generalAssist" | "quickAsk" | "autocomplete", accelerator: string) => {
|
(key: ShortcutKey, accelerator: string) => {
|
||||||
setShortcuts((prev) => {
|
setShortcuts((prev) => {
|
||||||
const updated = { ...prev, [key]: accelerator };
|
const updated = { ...prev, [key]: accelerator };
|
||||||
api?.setShortcuts?.({ [key]: accelerator }).catch(() => {
|
api?.setShortcuts?.({ [key]: accelerator }).catch(() => {
|
||||||
|
|
@ -196,7 +195,7 @@ export default function DesktopLoginPage() {
|
||||||
);
|
);
|
||||||
|
|
||||||
const resetShortcut = useCallback(
|
const resetShortcut = useCallback(
|
||||||
(key: "generalAssist" | "quickAsk" | "autocomplete") => {
|
(key: ShortcutKey) => {
|
||||||
updateShortcut(key, DEFAULT_SHORTCUTS[key]);
|
updateShortcut(key, DEFAULT_SHORTCUTS[key]);
|
||||||
},
|
},
|
||||||
[updateShortcut]
|
[updateShortcut]
|
||||||
|
|
@ -369,7 +368,9 @@ export default function DesktopLoginPage() {
|
||||||
|
|
||||||
<Button type="submit" disabled={isLoggingIn} className="relative h-9 mt-1">
|
<Button type="submit" disabled={isLoggingIn} className="relative h-9 mt-1">
|
||||||
<span className={isLoggingIn ? "opacity-0" : ""}>Sign in</span>
|
<span className={isLoggingIn ? "opacity-0" : ""}>Sign in</span>
|
||||||
{isLoggingIn && <Spinner size="sm" className="absolute text-primary-foreground" />}
|
{isLoggingIn && (
|
||||||
|
<Spinner size="sm" className="absolute text-primary-foreground" />
|
||||||
|
)}
|
||||||
</Button>
|
</Button>
|
||||||
</form>
|
</form>
|
||||||
)}
|
)}
|
||||||
|
|
|
||||||
|
|
@ -1,9 +0,0 @@
|
||||||
import "./suggestion.css";
|
|
||||||
|
|
||||||
export const metadata = {
|
|
||||||
title: "SurfSense Suggestion",
|
|
||||||
};
|
|
||||||
|
|
||||||
export default function SuggestionLayout({ children }: { children: React.ReactNode }) {
|
|
||||||
return <div className="suggestion-body">{children}</div>;
|
|
||||||
}
|
|
||||||
|
|
@ -1,384 +0,0 @@
|
||||||
"use client";
|
|
||||||
|
|
||||||
import { useCallback, useEffect, useRef, useState } from "react";
|
|
||||||
import { useElectronAPI } from "@/hooks/use-platform";
|
|
||||||
import { ensureTokensFromElectron, getBearerToken } from "@/lib/auth-utils";
|
|
||||||
|
|
||||||
type SSEEvent =
|
|
||||||
| { type: "text-delta"; id: string; delta: string }
|
|
||||||
| { type: "text-start"; id: string }
|
|
||||||
| { type: "text-end"; id: string }
|
|
||||||
| { type: "start"; messageId: string }
|
|
||||||
| { type: "finish" }
|
|
||||||
| { type: "error"; errorText: string }
|
|
||||||
| {
|
|
||||||
type: "data-thinking-step";
|
|
||||||
data: { id: string; title: string; status: string; items: string[] };
|
|
||||||
}
|
|
||||||
| {
|
|
||||||
type: "data-suggestions";
|
|
||||||
data: { options: string[] };
|
|
||||||
};
|
|
||||||
|
|
||||||
interface AgentStep {
|
|
||||||
id: string;
|
|
||||||
title: string;
|
|
||||||
status: string;
|
|
||||||
items: string[];
|
|
||||||
}
|
|
||||||
|
|
||||||
type FriendlyError = { message: string; isSetup?: boolean };
|
|
||||||
|
|
||||||
function friendlyError(raw: string | number): FriendlyError {
|
|
||||||
if (typeof raw === "number") {
|
|
||||||
if (raw === 401) return { message: "Please sign in to use suggestions." };
|
|
||||||
if (raw === 403) return { message: "You don\u2019t have permission for this." };
|
|
||||||
if (raw === 404) return { message: "Suggestion service not found. Is the backend running?" };
|
|
||||||
if (raw >= 500) return { message: "Something went wrong on the server. Try again." };
|
|
||||||
return { message: "Something went wrong. Try again." };
|
|
||||||
}
|
|
||||||
const lower = raw.toLowerCase();
|
|
||||||
if (lower.includes("not authenticated") || lower.includes("unauthorized"))
|
|
||||||
return { message: "Please sign in to use suggestions." };
|
|
||||||
if (lower.includes("no vision llm configured") || lower.includes("no llm configured"))
|
|
||||||
return {
|
|
||||||
message: "Configure a vision-capable model (e.g. GPT-4o, Gemini) to enable autocomplete.",
|
|
||||||
isSetup: true,
|
|
||||||
};
|
|
||||||
if (lower.includes("does not support vision"))
|
|
||||||
return {
|
|
||||||
message: "The selected model doesn\u2019t support vision. Choose a vision-capable model.",
|
|
||||||
isSetup: true,
|
|
||||||
};
|
|
||||||
if (lower.includes("fetch") || lower.includes("network") || lower.includes("econnrefused"))
|
|
||||||
return { message: "Can\u2019t reach the server. Check your connection." };
|
|
||||||
return { message: "Something went wrong. Try again." };
|
|
||||||
}
|
|
||||||
|
|
||||||
const AUTO_DISMISS_MS = 3000;
|
|
||||||
|
|
||||||
function StepIcon({ status }: { status: string }) {
|
|
||||||
if (status === "complete") {
|
|
||||||
return (
|
|
||||||
<svg
|
|
||||||
className="step-icon step-icon-done"
|
|
||||||
viewBox="0 0 16 16"
|
|
||||||
fill="none"
|
|
||||||
aria-label="Step complete"
|
|
||||||
>
|
|
||||||
<circle cx="8" cy="8" r="7" stroke="#4ade80" strokeWidth="1.5" />
|
|
||||||
<path
|
|
||||||
d="M5 8.5l2 2 4-4.5"
|
|
||||||
stroke="#4ade80"
|
|
||||||
strokeWidth="1.5"
|
|
||||||
strokeLinecap="round"
|
|
||||||
strokeLinejoin="round"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return <span className="step-spinner" />;
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function SuggestionPage() {
|
|
||||||
const api = useElectronAPI();
|
|
||||||
const [options, setOptions] = useState<string[]>([]);
|
|
||||||
const [isLoading, setIsLoading] = useState(true);
|
|
||||||
const [error, setError] = useState<FriendlyError | null>(null);
|
|
||||||
const [steps, setSteps] = useState<AgentStep[]>([]);
|
|
||||||
const [expandedOption, setExpandedOption] = useState<number | null>(null);
|
|
||||||
const abortRef = useRef<AbortController | null>(null);
|
|
||||||
|
|
||||||
const isDesktop = !!api?.onAutocompleteContext;
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!api?.onAutocompleteContext) {
|
|
||||||
setIsLoading(false);
|
|
||||||
}
|
|
||||||
}, [api]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!error || error.isSetup) return;
|
|
||||||
const timer = setTimeout(() => {
|
|
||||||
api?.dismissSuggestion?.();
|
|
||||||
}, AUTO_DISMISS_MS);
|
|
||||||
return () => clearTimeout(timer);
|
|
||||||
}, [error, api]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (isLoading || error || options.length > 0) return;
|
|
||||||
const timer = setTimeout(() => {
|
|
||||||
api?.dismissSuggestion?.();
|
|
||||||
}, AUTO_DISMISS_MS);
|
|
||||||
return () => clearTimeout(timer);
|
|
||||||
}, [isLoading, error, options, api]);
|
|
||||||
|
|
||||||
const fetchSuggestion = useCallback(
|
|
||||||
async (screenshot: string, searchSpaceId: string, appName?: string, windowTitle?: string) => {
|
|
||||||
abortRef.current?.abort();
|
|
||||||
const controller = new AbortController();
|
|
||||||
abortRef.current = controller;
|
|
||||||
|
|
||||||
setIsLoading(true);
|
|
||||||
setOptions([]);
|
|
||||||
setError(null);
|
|
||||||
setSteps([]);
|
|
||||||
setExpandedOption(null);
|
|
||||||
|
|
||||||
let token = getBearerToken();
|
|
||||||
if (!token) {
|
|
||||||
await ensureTokensFromElectron();
|
|
||||||
token = getBearerToken();
|
|
||||||
}
|
|
||||||
if (!token) {
|
|
||||||
setError(friendlyError("not authenticated"));
|
|
||||||
setIsLoading(false);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const backendUrl = process.env.NEXT_PUBLIC_FASTAPI_BACKEND_URL || "http://localhost:8000";
|
|
||||||
|
|
||||||
try {
|
|
||||||
const response = await fetch(`${backendUrl}/api/v1/autocomplete/vision/stream`, {
|
|
||||||
method: "POST",
|
|
||||||
headers: {
|
|
||||||
Authorization: `Bearer ${token}`,
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
body: JSON.stringify({
|
|
||||||
screenshot,
|
|
||||||
search_space_id: parseInt(searchSpaceId, 10),
|
|
||||||
app_name: appName || "",
|
|
||||||
window_title: windowTitle || "",
|
|
||||||
}),
|
|
||||||
signal: controller.signal,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!response.ok) {
|
|
||||||
setError(friendlyError(response.status));
|
|
||||||
setIsLoading(false);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!response.body) {
|
|
||||||
setError(friendlyError("network error"));
|
|
||||||
setIsLoading(false);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const reader = response.body.getReader();
|
|
||||||
const decoder = new TextDecoder();
|
|
||||||
let buffer = "";
|
|
||||||
|
|
||||||
while (true) {
|
|
||||||
const { done, value } = await reader.read();
|
|
||||||
if (done) break;
|
|
||||||
|
|
||||||
buffer += decoder.decode(value, { stream: true });
|
|
||||||
const events = buffer.split(/\r?\n\r?\n/);
|
|
||||||
buffer = events.pop() || "";
|
|
||||||
|
|
||||||
for (const event of events) {
|
|
||||||
const lines = event.split(/\r?\n/);
|
|
||||||
for (const line of lines) {
|
|
||||||
if (!line.startsWith("data: ")) continue;
|
|
||||||
const data = line.slice(6).trim();
|
|
||||||
if (!data || data === "[DONE]") continue;
|
|
||||||
|
|
||||||
try {
|
|
||||||
const parsed: SSEEvent = JSON.parse(data);
|
|
||||||
if (parsed.type === "data-suggestions") {
|
|
||||||
setOptions(parsed.data.options);
|
|
||||||
} else if (parsed.type === "error") {
|
|
||||||
setError(friendlyError(parsed.errorText));
|
|
||||||
} else if (parsed.type === "data-thinking-step") {
|
|
||||||
const { id, title, status, items } = parsed.data;
|
|
||||||
setSteps((prev) => {
|
|
||||||
const existing = prev.findIndex((s) => s.id === id);
|
|
||||||
if (existing >= 0) {
|
|
||||||
const updated = [...prev];
|
|
||||||
updated[existing] = { id, title, status, items };
|
|
||||||
return updated;
|
|
||||||
}
|
|
||||||
return [...prev, { id, title, status, items }];
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} catch {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (err) {
|
|
||||||
if (err instanceof DOMException && err.name === "AbortError") return;
|
|
||||||
setError(friendlyError("network error"));
|
|
||||||
} finally {
|
|
||||||
setIsLoading(false);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!api?.onAutocompleteContext) return;
|
|
||||||
|
|
||||||
const cleanup = api.onAutocompleteContext((data) => {
|
|
||||||
const searchSpaceId = data.searchSpaceId || "1";
|
|
||||||
if (data.screenshot) {
|
|
||||||
fetchSuggestion(data.screenshot, searchSpaceId, data.appName, data.windowTitle);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return cleanup;
|
|
||||||
}, [fetchSuggestion, api]);
|
|
||||||
|
|
||||||
if (!isDesktop) {
|
|
||||||
return (
|
|
||||||
<div className="suggestion-tooltip">
|
|
||||||
<span className="suggestion-error-text">
|
|
||||||
This page is only available in the SurfSense desktop app.
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (error) {
|
|
||||||
if (error.isSetup) {
|
|
||||||
return (
|
|
||||||
<div className="suggestion-tooltip suggestion-setup">
|
|
||||||
<div className="setup-icon">
|
|
||||||
<svg viewBox="0 0 24 24" fill="none" width="28" height="28" aria-hidden="true">
|
|
||||||
<path
|
|
||||||
d="M1 12C1 12 5 4 12 4C19 4 23 12 23 12C23 12 19 20 12 20C5 20 1 12 1 12Z"
|
|
||||||
stroke="#a78bfa"
|
|
||||||
strokeWidth="1.5"
|
|
||||||
strokeLinecap="round"
|
|
||||||
strokeLinejoin="round"
|
|
||||||
/>
|
|
||||||
<circle
|
|
||||||
cx="12"
|
|
||||||
cy="12"
|
|
||||||
r="3"
|
|
||||||
stroke="#a78bfa"
|
|
||||||
strokeWidth="1.5"
|
|
||||||
strokeLinecap="round"
|
|
||||||
strokeLinejoin="round"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
<div className="setup-content">
|
|
||||||
<span className="setup-title">Vision Model Required</span>
|
|
||||||
<span className="setup-message">{error.message}</span>
|
|
||||||
<span className="setup-hint">Settings → Vision Models</span>
|
|
||||||
</div>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
className="setup-dismiss"
|
|
||||||
onClick={() => api?.dismissSuggestion?.()}
|
|
||||||
>
|
|
||||||
✕
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
<div className="suggestion-tooltip suggestion-error">
|
|
||||||
<span className="suggestion-error-text">{error.message}</span>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const showLoading = isLoading && options.length === 0;
|
|
||||||
|
|
||||||
if (showLoading) {
|
|
||||||
return (
|
|
||||||
<div className="suggestion-tooltip">
|
|
||||||
<div className="agent-activity">
|
|
||||||
{steps.length === 0 && (
|
|
||||||
<div className="activity-initial">
|
|
||||||
<span className="step-spinner" />
|
|
||||||
<span className="activity-label">Preparing…</span>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
{steps.length > 0 && (
|
|
||||||
<div className="activity-steps">
|
|
||||||
{steps.map((step) => (
|
|
||||||
<div key={step.id} className="activity-step">
|
|
||||||
<StepIcon status={step.status} />
|
|
||||||
<span className="step-label">
|
|
||||||
{step.title}
|
|
||||||
{step.items.length > 0 && (
|
|
||||||
<span className="step-detail"> · {step.items[0]}</span>
|
|
||||||
)}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleSelect = (text: string) => {
|
|
||||||
api?.acceptSuggestion?.(text);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleDismiss = () => {
|
|
||||||
api?.dismissSuggestion?.();
|
|
||||||
};
|
|
||||||
|
|
||||||
const TRUNCATE_LENGTH = 120;
|
|
||||||
|
|
||||||
if (options.length === 0) {
|
|
||||||
return (
|
|
||||||
<div className="suggestion-tooltip suggestion-error">
|
|
||||||
<span className="suggestion-error-text">No suggestions available.</span>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className="suggestion-tooltip">
|
|
||||||
<div className="suggestion-options">
|
|
||||||
{options.map((option, index) => {
|
|
||||||
const isExpanded = expandedOption === index;
|
|
||||||
const needsTruncation = option.length > TRUNCATE_LENGTH;
|
|
||||||
const displayText =
|
|
||||||
needsTruncation && !isExpanded ? option.slice(0, TRUNCATE_LENGTH) + "…" : option;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
key={index}
|
|
||||||
className="suggestion-option"
|
|
||||||
onClick={() => handleSelect(option)}
|
|
||||||
>
|
|
||||||
<span className="option-number">{index + 1}</span>
|
|
||||||
<span className="option-text">{displayText}</span>
|
|
||||||
{needsTruncation && (
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
className="option-expand"
|
|
||||||
onClick={(e) => {
|
|
||||||
e.stopPropagation();
|
|
||||||
setExpandedOption(isExpanded ? null : index);
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{isExpanded ? "less" : "more"}
|
|
||||||
</button>
|
|
||||||
)}
|
|
||||||
</button>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</div>
|
|
||||||
<div className="suggestion-actions">
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
className="suggestion-btn suggestion-btn-dismiss"
|
|
||||||
onClick={handleDismiss}
|
|
||||||
>
|
|
||||||
Dismiss
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
@ -1,352 +0,0 @@
|
||||||
html:has(.suggestion-body),
|
|
||||||
body:has(.suggestion-body) {
|
|
||||||
margin: 0 !important;
|
|
||||||
padding: 0 !important;
|
|
||||||
background: transparent !important;
|
|
||||||
overflow: hidden !important;
|
|
||||||
height: auto !important;
|
|
||||||
width: 100% !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-body {
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
background: transparent;
|
|
||||||
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
|
|
||||||
-webkit-font-smoothing: antialiased;
|
|
||||||
user-select: none;
|
|
||||||
-webkit-app-region: no-drag;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-tooltip {
|
|
||||||
box-sizing: border-box;
|
|
||||||
background: #1e1e1e;
|
|
||||||
border: 1px solid #3c3c3c;
|
|
||||||
border-radius: 8px;
|
|
||||||
padding: 8px 12px;
|
|
||||||
margin: 4px;
|
|
||||||
max-width: 400px;
|
|
||||||
/* MAX_HEIGHT in suggestion-window.ts is 400px. Subtract 8px for margin
|
|
||||||
(4px * 2) so the tooltip + margin fits within the Electron window.
|
|
||||||
box-sizing: border-box ensures padding + border are included. */
|
|
||||||
max-height: 392px;
|
|
||||||
box-shadow: 0 4px 16px rgba(0, 0, 0, 0.5);
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-text {
|
|
||||||
color: #d4d4d4;
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 1.45;
|
|
||||||
margin: 0 0 6px 0;
|
|
||||||
word-wrap: break-word;
|
|
||||||
white-space: pre-wrap;
|
|
||||||
overflow-y: auto;
|
|
||||||
flex: 1 1 auto;
|
|
||||||
min-height: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-text::-webkit-scrollbar {
|
|
||||||
width: 5px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-text::-webkit-scrollbar-track {
|
|
||||||
background: transparent;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-text::-webkit-scrollbar-thumb {
|
|
||||||
background: #555;
|
|
||||||
border-radius: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-text::-webkit-scrollbar-thumb:hover {
|
|
||||||
background: #777;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-actions {
|
|
||||||
display: flex;
|
|
||||||
justify-content: flex-end;
|
|
||||||
gap: 4px;
|
|
||||||
border-top: 1px solid #2a2a2a;
|
|
||||||
padding-top: 6px;
|
|
||||||
flex-shrink: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-btn {
|
|
||||||
padding: 2px 8px;
|
|
||||||
border-radius: 3px;
|
|
||||||
border: 1px solid #3c3c3c;
|
|
||||||
font-family: inherit;
|
|
||||||
font-size: 10px;
|
|
||||||
font-weight: 500;
|
|
||||||
cursor: pointer;
|
|
||||||
line-height: 16px;
|
|
||||||
transition:
|
|
||||||
background 0.15s,
|
|
||||||
border-color 0.15s;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-btn-accept {
|
|
||||||
background: #2563eb;
|
|
||||||
border-color: #3b82f6;
|
|
||||||
color: #fff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-btn-accept:hover {
|
|
||||||
background: #1d4ed8;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-btn-dismiss {
|
|
||||||
background: #2a2a2a;
|
|
||||||
color: #999;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-btn-dismiss:hover {
|
|
||||||
background: #333;
|
|
||||||
color: #ccc;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-error {
|
|
||||||
border-color: #5c2626;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-error-text {
|
|
||||||
color: #f48771;
|
|
||||||
font-size: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* --- Setup prompt (vision model not configured) --- */
|
|
||||||
|
|
||||||
.suggestion-setup {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: row;
|
|
||||||
align-items: flex-start;
|
|
||||||
gap: 10px;
|
|
||||||
border-color: #3b2d6b;
|
|
||||||
padding: 10px 14px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setup-icon {
|
|
||||||
flex-shrink: 0;
|
|
||||||
margin-top: 1px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setup-content {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 3px;
|
|
||||||
min-width: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setup-title {
|
|
||||||
font-size: 13px;
|
|
||||||
font-weight: 600;
|
|
||||||
color: #c4b5fd;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setup-message {
|
|
||||||
font-size: 11.5px;
|
|
||||||
color: #a1a1aa;
|
|
||||||
line-height: 1.4;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setup-hint {
|
|
||||||
font-size: 10.5px;
|
|
||||||
color: #7c6dac;
|
|
||||||
margin-top: 2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setup-dismiss {
|
|
||||||
flex-shrink: 0;
|
|
||||||
align-self: flex-start;
|
|
||||||
background: none;
|
|
||||||
border: none;
|
|
||||||
color: #6b6b7b;
|
|
||||||
font-size: 14px;
|
|
||||||
cursor: pointer;
|
|
||||||
padding: 2px 4px;
|
|
||||||
line-height: 1;
|
|
||||||
border-radius: 4px;
|
|
||||||
transition:
|
|
||||||
color 0.15s,
|
|
||||||
background 0.15s;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setup-dismiss:hover {
|
|
||||||
color: #c4b5fd;
|
|
||||||
background: rgba(124, 109, 172, 0.15);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* --- Agent activity indicator --- */
|
|
||||||
|
|
||||||
.agent-activity {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 4px;
|
|
||||||
overflow-y: auto;
|
|
||||||
max-height: 340px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.agent-activity::-webkit-scrollbar {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.activity-initial {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 8px;
|
|
||||||
padding: 2px 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.activity-label {
|
|
||||||
color: #a1a1aa;
|
|
||||||
font-size: 12px;
|
|
||||||
white-space: nowrap;
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
}
|
|
||||||
|
|
||||||
.activity-steps {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.activity-step {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
min-height: 18px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.step-label {
|
|
||||||
color: #d4d4d4;
|
|
||||||
font-size: 12px;
|
|
||||||
white-space: nowrap;
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
}
|
|
||||||
|
|
||||||
.step-detail {
|
|
||||||
color: #71717a;
|
|
||||||
font-size: 11px;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Spinner (in_progress) */
|
|
||||||
.step-spinner {
|
|
||||||
width: 14px;
|
|
||||||
height: 14px;
|
|
||||||
flex-shrink: 0;
|
|
||||||
border: 1.5px solid #3f3f46;
|
|
||||||
border-top-color: #a78bfa;
|
|
||||||
border-radius: 50%;
|
|
||||||
animation: step-spin 0.7s linear infinite;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Checkmark icon (complete) */
|
|
||||||
.step-icon {
|
|
||||||
width: 14px;
|
|
||||||
height: 14px;
|
|
||||||
flex-shrink: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@keyframes step-spin {
|
|
||||||
to {
|
|
||||||
transform: rotate(360deg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* --- Suggestion option cards --- */
|
|
||||||
|
|
||||||
.suggestion-options {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 4px;
|
|
||||||
overflow-y: auto;
|
|
||||||
flex: 1 1 auto;
|
|
||||||
min-height: 0;
|
|
||||||
margin-bottom: 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-options::-webkit-scrollbar {
|
|
||||||
width: 5px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-options::-webkit-scrollbar-track {
|
|
||||||
background: transparent;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-options::-webkit-scrollbar-thumb {
|
|
||||||
background: #555;
|
|
||||||
border-radius: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-option {
|
|
||||||
display: flex;
|
|
||||||
align-items: flex-start;
|
|
||||||
gap: 8px;
|
|
||||||
padding: 6px 8px;
|
|
||||||
border-radius: 5px;
|
|
||||||
border: 1px solid #333;
|
|
||||||
background: #262626;
|
|
||||||
cursor: pointer;
|
|
||||||
text-align: left;
|
|
||||||
font-family: inherit;
|
|
||||||
transition:
|
|
||||||
background 0.15s,
|
|
||||||
border-color 0.15s;
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-option:hover {
|
|
||||||
background: #2a2d3a;
|
|
||||||
border-color: #3b82f6;
|
|
||||||
}
|
|
||||||
|
|
||||||
.option-number {
|
|
||||||
flex-shrink: 0;
|
|
||||||
width: 18px;
|
|
||||||
height: 18px;
|
|
||||||
border-radius: 50%;
|
|
||||||
background: #3f3f46;
|
|
||||||
color: #d4d4d4;
|
|
||||||
font-size: 10px;
|
|
||||||
font-weight: 600;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
margin-top: 1px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.suggestion-option:hover .option-number {
|
|
||||||
background: #2563eb;
|
|
||||||
color: #fff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.option-text {
|
|
||||||
color: #d4d4d4;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 1.45;
|
|
||||||
word-wrap: break-word;
|
|
||||||
white-space: pre-wrap;
|
|
||||||
flex: 1 1 auto;
|
|
||||||
min-width: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.option-expand {
|
|
||||||
flex-shrink: 0;
|
|
||||||
background: none;
|
|
||||||
border: none;
|
|
||||||
color: #71717a;
|
|
||||||
font-size: 10px;
|
|
||||||
cursor: pointer;
|
|
||||||
padding: 0 2px;
|
|
||||||
font-family: inherit;
|
|
||||||
margin-top: 1px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.option-expand:hover {
|
|
||||||
color: #a1a1aa;
|
|
||||||
}
|
|
||||||
|
|
@ -36,9 +36,8 @@ export function acceleratorToDisplay(accel: string): string[] {
|
||||||
}
|
}
|
||||||
|
|
||||||
export const DEFAULT_SHORTCUTS = {
|
export const DEFAULT_SHORTCUTS = {
|
||||||
generalAssist: "CommandOrControl+Shift+S",
|
generalAssist: "Alt+Shift+G",
|
||||||
quickAsk: "CommandOrControl+Alt+S",
|
quickAsk: "Alt+Shift+Q",
|
||||||
autocomplete: "CommandOrControl+Shift+Space",
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
|
|
|
||||||
25
surfsense_web/types/window.d.ts
vendored
25
surfsense_web/types/window.d.ts
vendored
|
|
@ -71,6 +71,7 @@ interface ElectronAPI {
|
||||||
openExternal: (url: string) => void;
|
openExternal: (url: string) => void;
|
||||||
getAppVersion: () => Promise<string>;
|
getAppVersion: () => Promise<string>;
|
||||||
onDeepLink: (callback: (url: string) => void) => () => void;
|
onDeepLink: (callback: (url: string) => void) => () => void;
|
||||||
|
onChatScreenCapture: (callback: (dataUrl: string) => void) => () => void;
|
||||||
getQuickAskText: () => Promise<string>;
|
getQuickAskText: () => Promise<string>;
|
||||||
setQuickAskMode: (mode: string) => Promise<void>;
|
setQuickAskMode: (mode: string) => Promise<void>;
|
||||||
getQuickAskMode: () => Promise<string>;
|
getQuickAskMode: () => Promise<string>;
|
||||||
|
|
@ -83,19 +84,6 @@ interface ElectronAPI {
|
||||||
requestAccessibility: () => Promise<void>;
|
requestAccessibility: () => Promise<void>;
|
||||||
requestScreenRecording: () => Promise<void>;
|
requestScreenRecording: () => Promise<void>;
|
||||||
restartApp: () => Promise<void>;
|
restartApp: () => Promise<void>;
|
||||||
// Autocomplete
|
|
||||||
onAutocompleteContext: (
|
|
||||||
callback: (data: {
|
|
||||||
screenshot: string;
|
|
||||||
searchSpaceId?: string;
|
|
||||||
appName?: string;
|
|
||||||
windowTitle?: string;
|
|
||||||
}) => void
|
|
||||||
) => () => void;
|
|
||||||
acceptSuggestion: (text: string) => Promise<void>;
|
|
||||||
dismissSuggestion: () => Promise<void>;
|
|
||||||
setAutocompleteEnabled: (enabled: boolean) => Promise<void>;
|
|
||||||
getAutocompleteEnabled: () => Promise<boolean>;
|
|
||||||
// Folder sync
|
// Folder sync
|
||||||
selectFolder: () => Promise<string | null>;
|
selectFolder: () => Promise<string | null>;
|
||||||
addWatchedFolder: (config: WatchedFolderConfig) => Promise<WatchedFolderConfig[]>;
|
addWatchedFolder: (config: WatchedFolderConfig) => Promise<WatchedFolderConfig[]>;
|
||||||
|
|
@ -115,18 +103,15 @@ interface ElectronAPI {
|
||||||
browseFiles: () => Promise<string[] | null>;
|
browseFiles: () => Promise<string[] | null>;
|
||||||
readLocalFiles: (paths: string[]) => Promise<LocalFileData[]>;
|
readLocalFiles: (paths: string[]) => Promise<LocalFileData[]>;
|
||||||
readAgentLocalFileText: (virtualPath: string) => Promise<LocalTextFileResult>;
|
readAgentLocalFileText: (virtualPath: string) => Promise<LocalTextFileResult>;
|
||||||
writeAgentLocalFileText: (
|
writeAgentLocalFileText: (virtualPath: string, content: string) => Promise<LocalTextFileResult>;
|
||||||
virtualPath: string,
|
|
||||||
content: string
|
|
||||||
) => Promise<LocalTextFileResult>;
|
|
||||||
// Auth token sync across windows
|
// Auth token sync across windows
|
||||||
getAuthTokens: () => Promise<{ bearer: string; refresh: string } | null>;
|
getAuthTokens: () => Promise<{ bearer: string; refresh: string } | null>;
|
||||||
setAuthTokens: (bearer: string, refresh: string) => Promise<void>;
|
setAuthTokens: (bearer: string, refresh: string) => Promise<void>;
|
||||||
// Keyboard shortcut configuration
|
// Keyboard shortcut configuration
|
||||||
getShortcuts: () => Promise<{ generalAssist: string; quickAsk: string; autocomplete: string }>;
|
getShortcuts: () => Promise<{ generalAssist: string; quickAsk: string }>;
|
||||||
setShortcuts: (
|
setShortcuts: (
|
||||||
config: Partial<{ generalAssist: string; quickAsk: string; autocomplete: string }>
|
config: Partial<{ generalAssist: string; quickAsk: string }>
|
||||||
) => Promise<{ generalAssist: string; quickAsk: string; autocomplete: string }>;
|
) => Promise<{ generalAssist: string; quickAsk: string }>;
|
||||||
// Launch on system startup
|
// Launch on system startup
|
||||||
getAutoLaunch: () => Promise<{
|
getAutoLaunch: () => Promise<{
|
||||||
enabled: boolean;
|
enabled: boolean;
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue