Merge remote-tracking branch 'origin/main'

main
Peter Steinberger 2025-12-12 16:10:02 +00:00
commit 3f1bcac077
2 changed files with 34 additions and 34 deletions

View File

@ -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

View File

@ -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
} }
} }