Merge remote-tracking branch 'origin/main'
commit
3f1bcac077
|
|
@ -147,11 +147,11 @@ struct ToolsSettings: View {
|
||||||
method: .brew(formula: "openhue/cli/openhue-cli", binary: "openhue"),
|
method: .brew(formula: "openhue/cli/openhue-cli", binary: "openhue"),
|
||||||
kind: .tool),
|
kind: .tool),
|
||||||
ToolEntry(
|
ToolEntry(
|
||||||
id: "openai-whisper",
|
id: "gog",
|
||||||
name: "OpenAI Whisper",
|
name: "gog",
|
||||||
url: URL(string: "https://github.com/openai/whisper")!,
|
url: URL(string: "https://github.com/steipete/gog")!,
|
||||||
description: "On-device speech-to-text for quick note taking or voicemail transcription.",
|
description: "Unified Google CLI for Gmail, Calendar, Drive, and Contacts. Replaces MCP servers.",
|
||||||
method: .brew(formula: "openai-whisper", binary: "whisper"),
|
method: .brew(formula: "steipete/tap/gog", binary: "gog"),
|
||||||
kind: .tool),
|
kind: .tool),
|
||||||
ToolEntry(
|
ToolEntry(
|
||||||
id: "gemini-cli",
|
id: "gemini-cli",
|
||||||
|
|
@ -179,26 +179,6 @@ struct ToolsSettings: View {
|
||||||
url: "https://github.com/badlogic/agent-tools.git",
|
url: "https://github.com/badlogic/agent-tools.git",
|
||||||
destination: "\(NSHomeDirectory())/agent-tools"),
|
destination: "\(NSHomeDirectory())/agent-tools"),
|
||||||
kind: .tool),
|
kind: .tool),
|
||||||
ToolEntry(
|
|
||||||
id: "gmail-mcp",
|
|
||||||
name: "Gmail MCP",
|
|
||||||
url: URL(string: "https://www.npmjs.com/package/@gongrzhe/server-gmail-autoauth-mcp")!,
|
|
||||||
description: "Model Context Protocol server that exposes Gmail search, read, and send tools.",
|
|
||||||
method: .mcporter(
|
|
||||||
name: "gmail",
|
|
||||||
command: "npx -y @gongrzhe/server-gmail-autoauth-mcp",
|
|
||||||
summary: "Adds Gmail MCP via mcporter (stdio transport, auto-auth)."),
|
|
||||||
kind: .mcp),
|
|
||||||
ToolEntry(
|
|
||||||
id: "google-calendar-mcp",
|
|
||||||
name: "Google Calendar MCP",
|
|
||||||
url: URL(string: "https://www.npmjs.com/package/@cocal/google-calendar-mcp")!,
|
|
||||||
description: "MCP server to list, create, and update calendar events for scheduling automations.",
|
|
||||||
method: .mcporter(
|
|
||||||
name: "google-calendar",
|
|
||||||
command: "npx -y @cocal/google-calendar-mcp",
|
|
||||||
summary: "Adds Google Calendar MCP via mcporter (stdio transport)."),
|
|
||||||
kind: .mcp),
|
|
||||||
]
|
]
|
||||||
|
|
||||||
@AppStorage("tools.packageManager") private var packageManagerRaw = NodePackageManager.npm.rawValue
|
@AppStorage("tools.packageManager") private var packageManagerRaw = NodePackageManager.npm.rawValue
|
||||||
|
|
|
||||||
|
|
@ -538,8 +538,10 @@ final class WebChatManager {
|
||||||
static let shared = WebChatManager()
|
static let shared = WebChatManager()
|
||||||
private var windowController: WebChatWindowController?
|
private var windowController: WebChatWindowController?
|
||||||
private var panelController: WebChatWindowController?
|
private var panelController: WebChatWindowController?
|
||||||
|
private var panelSessionKey: String?
|
||||||
private var swiftWindowController: WebChatSwiftUIWindowController?
|
private var swiftWindowController: WebChatSwiftUIWindowController?
|
||||||
private var swiftPanelController: WebChatSwiftUIWindowController?
|
private var swiftPanelController: WebChatSwiftUIWindowController?
|
||||||
|
private var swiftPanelSessionKey: String?
|
||||||
private var browserTunnel: WebChatTunnel?
|
private var browserTunnel: WebChatTunnel?
|
||||||
var onPanelVisibilityChanged: ((Bool) -> Void)?
|
var onPanelVisibilityChanged: ((Bool) -> Void)?
|
||||||
|
|
||||||
|
|
@ -575,12 +577,18 @@ final class WebChatManager {
|
||||||
func togglePanel(sessionKey: String, anchorProvider: @escaping () -> NSRect?) {
|
func togglePanel(sessionKey: String, anchorProvider: @escaping () -> NSRect?) {
|
||||||
if AppStateStore.webChatSwiftUIEnabled {
|
if AppStateStore.webChatSwiftUIEnabled {
|
||||||
if let controller = self.swiftPanelController {
|
if let controller = self.swiftPanelController {
|
||||||
if controller.isVisible {
|
if self.swiftPanelSessionKey != sessionKey {
|
||||||
controller.close()
|
controller.close()
|
||||||
|
self.swiftPanelController = nil
|
||||||
|
self.swiftPanelSessionKey = nil
|
||||||
} else {
|
} else {
|
||||||
controller.presentAnchored(anchorProvider: anchorProvider)
|
if controller.isVisible {
|
||||||
|
controller.close()
|
||||||
|
} else {
|
||||||
|
controller.presentAnchored(anchorProvider: anchorProvider)
|
||||||
|
}
|
||||||
|
return
|
||||||
}
|
}
|
||||||
return
|
|
||||||
}
|
}
|
||||||
let controller = WebChatSwiftUIWindowController(
|
let controller = WebChatSwiftUIWindowController(
|
||||||
sessionKey: sessionKey,
|
sessionKey: sessionKey,
|
||||||
|
|
@ -592,21 +600,30 @@ final class WebChatManager {
|
||||||
self?.onPanelVisibilityChanged?(visible)
|
self?.onPanelVisibilityChanged?(visible)
|
||||||
}
|
}
|
||||||
self.swiftPanelController = controller
|
self.swiftPanelController = controller
|
||||||
|
self.swiftPanelSessionKey = sessionKey
|
||||||
controller.presentAnchored(anchorProvider: anchorProvider)
|
controller.presentAnchored(anchorProvider: anchorProvider)
|
||||||
} else {
|
} else {
|
||||||
if let controller = self.panelController {
|
if let controller = self.panelController {
|
||||||
if controller.window?.isVisible == true {
|
if self.panelSessionKey != sessionKey {
|
||||||
controller.closePanel()
|
controller.shutdown()
|
||||||
|
controller.close()
|
||||||
|
self.panelController = nil
|
||||||
|
self.panelSessionKey = nil
|
||||||
} else {
|
} else {
|
||||||
controller.presentAnchoredPanel(anchorProvider: anchorProvider)
|
if controller.window?.isVisible == true {
|
||||||
|
controller.closePanel()
|
||||||
|
} else {
|
||||||
|
controller.presentAnchoredPanel(anchorProvider: anchorProvider)
|
||||||
|
}
|
||||||
|
return
|
||||||
}
|
}
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let controller = WebChatWindowController(
|
let controller = WebChatWindowController(
|
||||||
sessionKey: sessionKey,
|
sessionKey: sessionKey,
|
||||||
presentation: .panel(anchorProvider: anchorProvider))
|
presentation: .panel(anchorProvider: anchorProvider))
|
||||||
self.panelController = controller
|
self.panelController = controller
|
||||||
|
self.panelSessionKey = sessionKey
|
||||||
controller.onPanelClosed = { [weak self] in
|
controller.onPanelClosed = { [weak self] in
|
||||||
self?.panelHidden()
|
self?.panelHidden()
|
||||||
}
|
}
|
||||||
|
|
@ -656,10 +673,12 @@ final class WebChatManager {
|
||||||
self.panelController?.shutdown()
|
self.panelController?.shutdown()
|
||||||
self.panelController?.close()
|
self.panelController?.close()
|
||||||
self.panelController = nil
|
self.panelController = nil
|
||||||
|
self.panelSessionKey = nil
|
||||||
self.swiftWindowController?.close()
|
self.swiftWindowController?.close()
|
||||||
self.swiftWindowController = nil
|
self.swiftWindowController = nil
|
||||||
self.swiftPanelController?.close()
|
self.swiftPanelController?.close()
|
||||||
self.swiftPanelController = nil
|
self.swiftPanelController = nil
|
||||||
|
self.swiftPanelSessionKey = nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@MainActor
|
@MainActor
|
||||||
|
|
@ -712,17 +731,18 @@ final class WebChatManager {
|
||||||
self.panelController?.shutdown()
|
self.panelController?.shutdown()
|
||||||
self.panelController?.close()
|
self.panelController?.close()
|
||||||
self.panelController = nil
|
self.panelController = nil
|
||||||
|
self.panelSessionKey = nil
|
||||||
|
|
||||||
self.swiftWindowController?.close()
|
self.swiftWindowController?.close()
|
||||||
self.swiftWindowController = nil
|
self.swiftWindowController = nil
|
||||||
self.swiftPanelController?.close()
|
self.swiftPanelController?.close()
|
||||||
self.swiftPanelController = nil
|
self.swiftPanelController = nil
|
||||||
|
self.swiftPanelSessionKey = nil
|
||||||
}
|
}
|
||||||
|
|
||||||
private func panelHidden() {
|
private func panelHidden() {
|
||||||
self.onPanelVisibilityChanged?(false)
|
self.onPanelVisibilityChanged?(false)
|
||||||
self.panelController = nil
|
// Keep panel controllers cached so reopening doesn't re-bootstrap.
|
||||||
self.swiftPanelController = nil
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue