mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-23 08:58:09 +00:00
314 lines
12 KiB
Swift
314 lines
12 KiB
Swift
import OpenClawChatUI
|
|
import OpenClawProtocol
|
|
import SwiftUI
|
|
|
|
struct ChatProTab: View {
|
|
@Environment(NodeAppModel.self) private var appModel
|
|
@Environment(\.colorScheme) private var colorScheme
|
|
@State private var viewModel: OpenClawChatViewModel?
|
|
@State private var viewModelTransportModeID = ""
|
|
let headerLeadingAction: OpenClawSidebarHeaderAction?
|
|
let headerTitle: String?
|
|
let headerSubtitle: String?
|
|
let showsAgentBadge: Bool
|
|
let openSettings: (() -> Void)?
|
|
|
|
init(
|
|
headerLeadingAction: OpenClawSidebarHeaderAction? = nil,
|
|
headerTitle: String? = nil,
|
|
headerSubtitle: String? = nil,
|
|
showsAgentBadge: Bool = true,
|
|
openSettings: (() -> Void)? = nil)
|
|
{
|
|
self.headerLeadingAction = headerLeadingAction
|
|
self.headerTitle = headerTitle
|
|
self.headerSubtitle = headerSubtitle
|
|
self.showsAgentBadge = showsAgentBadge
|
|
self.openSettings = openSettings
|
|
}
|
|
|
|
var body: some View {
|
|
NavigationStack {
|
|
ZStack {
|
|
OpenClawProBackground()
|
|
VStack(spacing: 0) {
|
|
self.header
|
|
if let viewModel {
|
|
OpenClawChatView(
|
|
viewModel: viewModel,
|
|
drawsBackground: false,
|
|
showsSessionSwitcher: false,
|
|
userAccent: self.chatUserAccent,
|
|
assistantName: self.agentDisplayName,
|
|
assistantAvatarText: self.agentBadge,
|
|
assistantAvatarTint: OpenClawBrand.accent,
|
|
showsAssistantAvatars: false,
|
|
composerChrome: .clean,
|
|
isComposerEnabled: self.gatewayConnected,
|
|
messagePlaceholder: self.messagePlaceholder,
|
|
talkControl: self.talkControl)
|
|
.id(ObjectIdentifier(viewModel))
|
|
.frame(maxWidth: .infinity, maxHeight: .infinity, alignment: .top)
|
|
} else {
|
|
ProCard {
|
|
VStack(alignment: .leading, spacing: 8) {
|
|
Text("Chat is preparing")
|
|
.font(.headline)
|
|
Text("The operator session will attach when the gateway is ready.")
|
|
.font(.subheadline)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
}
|
|
.padding()
|
|
Spacer()
|
|
}
|
|
}
|
|
.frame(maxWidth: .infinity, maxHeight: .infinity, alignment: .top)
|
|
.safeAreaPadding(.top, 8)
|
|
}
|
|
.frame(maxWidth: .infinity, maxHeight: .infinity, alignment: .top)
|
|
.navigationBarHidden(true)
|
|
}
|
|
.task {
|
|
self.syncChatViewModel()
|
|
}
|
|
.onChange(of: self.appModel.chatSessionKey) { _, _ in
|
|
self.syncChatViewModel()
|
|
}
|
|
.onChange(of: self.appModel.isAppleReviewDemoModeEnabled) { _, _ in
|
|
self.syncChatViewModel()
|
|
self.viewModel?.refresh()
|
|
}
|
|
.onChange(of: self.appModel.isScreenshotFixtureModeEnabled) { _, _ in
|
|
self.syncChatViewModel()
|
|
self.viewModel?.refresh()
|
|
}
|
|
.onChange(of: self.appModel.isOperatorGatewayConnected) { _, connected in
|
|
guard connected else { return }
|
|
self.syncChatViewModel()
|
|
self.viewModel?.refresh()
|
|
}
|
|
}
|
|
|
|
private var header: some View {
|
|
OpenClawAdaptiveHeaderRow(
|
|
title: self.headerDisplayTitle,
|
|
subtitle: self.headerDisplaySubtitle,
|
|
titleFont: .headline.weight(.semibold),
|
|
subtitleFont: .caption,
|
|
subtitleLineLimit: 1)
|
|
{
|
|
HStack(spacing: 11) {
|
|
if let headerLeadingAction {
|
|
OpenClawSidebarHeaderLeadingSlot(action: headerLeadingAction)
|
|
}
|
|
self.headerIdentityBadge
|
|
}
|
|
} accessory: {
|
|
self.connectionPillButton
|
|
}
|
|
.padding(.horizontal, OpenClawProMetric.pagePadding)
|
|
.padding(.bottom, 4)
|
|
}
|
|
|
|
@ViewBuilder
|
|
private var headerIdentityBadge: some View {
|
|
if self.showsAgentBadge {
|
|
Text(self.agentBadge)
|
|
.font(.system(size: self.agentBadge.count > 2 ? 13 : 16, weight: .bold, design: .rounded))
|
|
.foregroundStyle(.white)
|
|
.minimumScaleFactor(0.6)
|
|
.lineLimit(1)
|
|
.frame(width: 38, height: 38)
|
|
.background(
|
|
Circle()
|
|
.fill(
|
|
LinearGradient(
|
|
colors: [
|
|
OpenClawBrand.accent,
|
|
OpenClawBrand.accentHot,
|
|
],
|
|
startPoint: .topLeading,
|
|
endPoint: .bottomTrailing)))
|
|
.overlay(Circle().strokeBorder(.white.opacity(0.18), lineWidth: 1))
|
|
.shadow(color: OpenClawBrand.accent.opacity(0.18), radius: 10, y: 5)
|
|
} else {
|
|
ProIconBadge(systemName: "bubble.left", color: OpenClawBrand.accent)
|
|
}
|
|
}
|
|
|
|
private func syncChatViewModel() {
|
|
let sessionKey = self.appModel.chatSessionKey
|
|
let transportModeID = self.appModel.chatTransportModeID
|
|
guard let viewModel else {
|
|
self.viewModelTransportModeID = transportModeID
|
|
self.viewModel = OpenClawChatViewModel(
|
|
sessionKey: sessionKey,
|
|
transport: self.appModel.makeChatTransport(),
|
|
onSessionChanged: { sessionKey in
|
|
self.appModel.focusChatSession(sessionKey)
|
|
},
|
|
diagnosticsLog: { message in
|
|
GatewayDiagnostics.log(message)
|
|
})
|
|
return
|
|
}
|
|
if self.viewModelTransportModeID != transportModeID {
|
|
self.viewModelTransportModeID = transportModeID
|
|
self.viewModel = OpenClawChatViewModel(
|
|
sessionKey: sessionKey,
|
|
transport: self.appModel.makeChatTransport(),
|
|
onSessionChanged: { sessionKey in
|
|
self.appModel.focusChatSession(sessionKey)
|
|
},
|
|
diagnosticsLog: { message in
|
|
GatewayDiagnostics.log(message)
|
|
})
|
|
return
|
|
}
|
|
guard viewModel.sessionKey != sessionKey else { return }
|
|
viewModel.syncSession(to: sessionKey)
|
|
}
|
|
|
|
private var talkControl: OpenClawChatTalkControl {
|
|
OpenClawChatTalkControl(
|
|
isEnabled: self.appModel.talkMode.isEnabled,
|
|
isListening: self.appModel.talkMode.isListening,
|
|
isSpeaking: self.appModel.talkMode.isSpeaking,
|
|
isGatewayConnected: self.appModel.talkMode.isGatewayConnected,
|
|
statusText: self.appModel.talkMode.statusText,
|
|
providerLabel: self.appModel.talkMode.gatewayTalkProviderLabel,
|
|
toggle: { sessionKey in
|
|
self.appModel.focusChatSession(sessionKey)
|
|
self.appModel.setTalkEnabled(!self.appModel.talkMode.isEnabled)
|
|
})
|
|
}
|
|
|
|
private var activeAgentID: String {
|
|
self.normalized(self.appModel.chatAgentId)
|
|
?? "main"
|
|
}
|
|
|
|
@ViewBuilder
|
|
private var connectionPillButton: some View {
|
|
if let openSettings {
|
|
Button(action: openSettings) {
|
|
self.connectionPill
|
|
}
|
|
.buttonStyle(.plain)
|
|
.accessibilityHint("Opens Settings / Gateway")
|
|
} else {
|
|
self.connectionPill
|
|
}
|
|
}
|
|
|
|
private var connectionPill: some View {
|
|
HStack(spacing: 6) {
|
|
ProStatusDot(color: self.gatewayPillColor)
|
|
Text(Self.gatewayPillTitle(state: self.gatewayDisplayState, isGatewayUsable: self.gatewayConnected))
|
|
.font(.caption.weight(.semibold))
|
|
.lineLimit(1)
|
|
}
|
|
.foregroundStyle(self.gatewayPillColor)
|
|
.padding(.horizontal, 10)
|
|
.frame(height: 30)
|
|
.background {
|
|
Capsule()
|
|
.fill(self.gatewayPillColor.opacity(0.11))
|
|
}
|
|
.overlay {
|
|
Capsule()
|
|
.strokeBorder(self.gatewayPillColor.opacity(0.16), lineWidth: 1)
|
|
}
|
|
}
|
|
|
|
private var gatewayConnected: Bool {
|
|
guard self.gatewayDisplayState == .connected else {
|
|
return false
|
|
}
|
|
return self.appModel.isLocalChatFixtureEnabled || self.appModel.isOperatorGatewayConnected
|
|
}
|
|
|
|
private var gatewayDisplayState: GatewayDisplayState {
|
|
GatewayStatusBuilder.build(appModel: self.appModel)
|
|
}
|
|
|
|
private var gatewayPillColor: Color {
|
|
switch self.gatewayDisplayState {
|
|
case .connected:
|
|
self.gatewayConnected ? OpenClawBrand.ok : .secondary
|
|
case .connecting:
|
|
OpenClawBrand.accent
|
|
case .error:
|
|
OpenClawBrand.warn
|
|
case .disconnected:
|
|
.secondary
|
|
}
|
|
}
|
|
|
|
nonisolated static func gatewayPillTitle(state: GatewayDisplayState, isGatewayUsable: Bool) -> String {
|
|
switch state {
|
|
case .connected:
|
|
isGatewayUsable ? "Connected" : "Unavailable"
|
|
case .connecting:
|
|
"Connecting"
|
|
case .error:
|
|
"Attention"
|
|
case .disconnected:
|
|
"Offline"
|
|
}
|
|
}
|
|
|
|
private var messagePlaceholder: String {
|
|
self.gatewayConnected ? "Message \(self.agentDisplayName)..." : "Connect to a gateway"
|
|
}
|
|
|
|
private var headerDisplayTitle: String {
|
|
self.normalized(self.headerTitle)
|
|
?? Self.defaultHeaderTitle(showsAgentBadge: self.showsAgentBadge, agentDisplayName: self.agentDisplayName)
|
|
}
|
|
|
|
private var headerDisplaySubtitle: String {
|
|
self.normalized(self.headerSubtitle) ?? "AI Assistant"
|
|
}
|
|
|
|
nonisolated static func defaultHeaderTitle(showsAgentBadge: Bool, agentDisplayName: String) -> String {
|
|
showsAgentBadge ? agentDisplayName : "Chat"
|
|
}
|
|
|
|
private var chatUserAccent: Color {
|
|
self.colorScheme == .light ? Color(red: 0 / 255.0, green: 122 / 255.0, blue: 255 / 255.0) : OpenClawBrand.accent
|
|
}
|
|
|
|
private var activeAgent: AgentSummary? {
|
|
self.appModel.gatewayAgents.first { $0.id == self.activeAgentID }
|
|
}
|
|
|
|
private var agentDisplayName: String {
|
|
self.normalized(self.activeAgent?.name) ?? self.appModel.chatAgentName
|
|
}
|
|
|
|
private var agentBadge: String {
|
|
if let identity = self.activeAgent?.identity,
|
|
let emoji = identity["emoji"]?.value as? String,
|
|
let normalizedEmoji = self.normalized(emoji)
|
|
{
|
|
return normalizedEmoji
|
|
}
|
|
let words = self.agentDisplayName
|
|
.split(whereSeparator: { $0.isWhitespace || $0 == "-" || $0 == "_" })
|
|
.prefix(2)
|
|
let initials = words.compactMap(\.first).map(String.init).joined()
|
|
if !initials.isEmpty {
|
|
return initials.uppercased()
|
|
}
|
|
return "OC"
|
|
}
|
|
|
|
private func normalized(_ value: String?) -> String? {
|
|
guard let value else { return nil }
|
|
let trimmed = value.trimmingCharacters(in: .whitespacesAndNewlines)
|
|
return trimmed.isEmpty ? nil : trimmed
|
|
}
|
|
}
|