Files
Sybil-2/ios/Packages/Sybil/Sources/Sybil/SybilWorkspaceView.swift

412 lines
15 KiB
Swift
Raw Normal View History

2026-02-20 00:09:02 -08:00
import Observation
2026-05-02 19:47:38 -07:00
import PhotosUI
2026-02-20 00:09:02 -08:00
import SwiftUI
2026-05-02 19:47:38 -07:00
import UniformTypeIdentifiers
import UIKit
2026-02-20 00:09:02 -08:00
struct SybilWorkspaceView: View {
@Bindable var viewModel: SybilViewModel
@FocusState private var composerFocused: Bool
2026-05-02 19:47:38 -07:00
@State private var isShowingAttachmentOptions = false
@State private var isShowingFileImporter = false
@State private var isShowingPhotoPicker = false
@State private var photoPickerItems: [PhotosPickerItem] = []
@State private var isComposerDropTargeted = false
2026-02-20 00:09:02 -08:00
private var isSettingsSelected: Bool {
if case .settings = viewModel.selectedItem {
return true
}
return false
}
2026-05-02 16:23:00 -07:00
private var showsHeader: Bool {
2026-05-02 17:03:02 -07:00
viewModel.errorMessage != nil
2026-05-02 16:23:00 -07:00
}
2026-05-02 22:25:24 -07:00
private var transcriptScrollContextID: String {
if viewModel.draftKind == .chat {
return "draft-chat"
}
if case let .chat(chatID) = viewModel.selectedItem {
return "chat:\(chatID)"
}
return "chat:none"
}
2026-02-20 00:09:02 -08:00
var body: some View {
VStack(spacing: 0) {
2026-05-02 16:23:00 -07:00
if showsHeader {
header
2026-02-20 00:09:02 -08:00
2026-05-02 16:23:00 -07:00
Divider()
.overlay(SybilTheme.border)
}
2026-02-20 00:09:02 -08:00
Group {
if isSettingsSelected {
SybilSettingsView(viewModel: viewModel)
} else if viewModel.isSearchMode {
SybilSearchResultsView(
search: viewModel.selectedSearch,
isLoading: viewModel.isLoadingSelection,
2026-05-02 16:48:01 -07:00
isRunning: viewModel.isSending,
isStartingChat: viewModel.isCreatingSearchChat
) {
Task {
await viewModel.startChatFromSelectedSearch()
}
}
2026-02-20 00:09:02 -08:00
} else {
SybilChatTranscriptView(
messages: viewModel.displayedMessages,
isLoading: viewModel.isLoadingSelection,
isSending: viewModel.isSending
)
2026-05-02 22:25:24 -07:00
.id(transcriptScrollContextID)
2026-02-20 00:09:02 -08:00
}
}
.frame(maxWidth: .infinity, maxHeight: .infinity)
if viewModel.showsComposer {
Divider()
.overlay(SybilTheme.border)
composerBar
}
}
2026-05-02 17:19:34 -07:00
.navigationTitle(viewModel.selectedTitle)
2026-05-02 17:03:02 -07:00
.navigationBarTitleDisplayMode(.inline)
2026-05-02 17:19:34 -07:00
.toolbarRole(.editor)
2026-05-02 16:23:00 -07:00
.toolbar {
2026-05-02 17:03:02 -07:00
if !isSettingsSelected {
2026-05-02 16:23:00 -07:00
ToolbarItem(placement: .topBarTrailing) {
2026-05-02 17:03:02 -07:00
if viewModel.isSearchMode {
searchModeChip
} else {
providerModelMenu
}
2026-05-02 16:23:00 -07:00
}
}
}
2026-02-20 00:09:02 -08:00
.background(SybilTheme.background)
.frame(maxWidth: .infinity, maxHeight: .infinity, alignment: .topLeading)
}
private var header: some View {
VStack(alignment: .leading, spacing: 12) {
if let error = viewModel.errorMessage {
Text(error)
2026-05-02 16:23:00 -07:00
.font(.sybil(.footnote))
2026-02-20 00:09:02 -08:00
.foregroundStyle(SybilTheme.danger)
.frame(maxWidth: .infinity, alignment: .leading)
}
}
.padding(.horizontal, 16)
.padding(.vertical, 12)
2026-05-02 16:23:00 -07:00
.background(SybilTheme.panelGradient.opacity(0.58))
}
2026-05-02 17:03:02 -07:00
private var providerModelMenu: some View {
2026-05-02 16:23:00 -07:00
Menu {
Text("\(viewModel.provider.displayName)\(viewModel.model)")
.font(.sybil(.caption))
Divider()
ForEach(Provider.allCases, id: \.self) { candidate in
Menu(candidate.displayName) {
let models = viewModel.modelOptions(for: candidate)
if models.isEmpty {
Text("No models")
} else {
ForEach(models, id: \.self) { candidateModel in
Button {
viewModel.setProvider(candidate, model: candidateModel)
} label: {
if viewModel.provider == candidate && viewModel.model == candidateModel {
Label(candidateModel, systemImage: "checkmark")
} else {
Text(candidateModel)
}
}
}
}
}
}
} label: {
Image(systemName: "ellipsis")
.font(.system(size: 18, weight: .semibold))
.foregroundStyle(SybilTheme.text)
.frame(width: 34, height: 34)
.background(
Circle()
.fill(SybilTheme.surface.opacity(0.78))
)
.overlay(
Circle()
.stroke(SybilTheme.border.opacity(0.82), lineWidth: 1)
)
}
.accessibilityLabel("Provider and model")
2026-02-20 00:09:02 -08:00
}
2026-05-02 17:03:02 -07:00
private var searchModeChip: some View {
Label("Search", systemImage: "globe")
.font(.sybil(.caption, weight: .medium))
.foregroundStyle(SybilTheme.accent)
.padding(.horizontal, 10)
.padding(.vertical, 7)
.background(
Capsule()
.fill(SybilTheme.accent.opacity(0.10))
.overlay(
Capsule()
.stroke(SybilTheme.accent.opacity(0.24), lineWidth: 1)
2026-02-20 00:09:02 -08:00
)
2026-05-02 17:03:02 -07:00
)
2026-02-20 00:09:02 -08:00
}
private var composerBar: some View {
2026-05-02 19:47:38 -07:00
VStack(alignment: .leading, spacing: 10) {
if !viewModel.isSearchMode && !viewModel.composerAttachments.isEmpty {
SybilAttachmentListView(
attachments: viewModel.composerAttachments,
tone: .composer
) { attachmentID in
viewModel.removeComposerAttachment(id: attachmentID)
2026-02-20 00:09:02 -08:00
}
}
2026-05-02 19:47:38 -07:00
HStack(alignment: .bottom, spacing: 10) {
if !viewModel.isSearchMode {
Button {
isShowingAttachmentOptions = true
} label: {
Image(systemName: "paperclip")
.font(.system(size: 17, weight: .semibold))
.frame(width: 40, height: 40)
.background(
Circle()
.fill(SybilTheme.surface)
2026-05-02 16:23:00 -07:00
)
2026-05-02 19:47:38 -07:00
.overlay(
Circle()
.stroke(SybilTheme.border.opacity(0.82), lineWidth: 1)
)
.foregroundStyle(viewModel.isSending ? SybilTheme.textMuted : SybilTheme.text)
}
.buttonStyle(.plain)
.disabled(viewModel.isSending)
.accessibilityLabel("Attach file")
}
TextField(
viewModel.isSearchMode ? "Search the web" : "Message Sybil",
text: $viewModel.composer,
axis: .vertical
)
.focused($composerFocused)
.textInputAutocapitalization(.sentences)
.autocorrectionDisabled(false)
.lineLimit(1 ... 6)
.submitLabel(.send)
.onSubmit {
Task {
await viewModel.sendComposer()
}
}
.padding(.horizontal, 12)
.padding(.vertical, 10)
.background(
RoundedRectangle(cornerRadius: 12)
.fill(SybilTheme.composerGradient)
.overlay(
RoundedRectangle(cornerRadius: 12)
.stroke(SybilTheme.primary.opacity(0.34), lineWidth: 1)
)
)
.foregroundStyle(SybilTheme.text)
Button {
Task {
await viewModel.sendComposer()
}
} label: {
Image(systemName: viewModel.isSearchMode ? "magnifyingglass" : "arrow.up")
.font(.system(size: 17, weight: .semibold))
.frame(width: 40, height: 40)
.background(
Circle()
.fill(
viewModel.canSendComposer
? AnyShapeStyle(SybilTheme.primaryGradient)
: AnyShapeStyle(SybilTheme.surface)
)
)
.foregroundStyle(viewModel.canSendComposer ? SybilTheme.text : SybilTheme.textMuted)
}
.buttonStyle(.plain)
.disabled(!viewModel.canSendComposer)
2026-02-20 00:09:02 -08:00
}
}
.padding(.horizontal, 14)
.padding(.vertical, 12)
2026-05-02 16:23:00 -07:00
.background(
LinearGradient(
colors: [
SybilTheme.background.opacity(0.18),
SybilTheme.background.opacity(0.96)
],
startPoint: .top,
endPoint: .bottom
)
)
2026-05-02 19:47:38 -07:00
.overlay {
if isComposerDropTargeted && !viewModel.isSearchMode {
RoundedRectangle(cornerRadius: 18)
.stroke(SybilTheme.accent.opacity(0.78), style: StrokeStyle(lineWidth: 1.5, dash: [7, 5]))
.padding(.horizontal, 14)
.padding(.vertical, 10)
}
}
.onDrop(of: [UTType.fileURL.identifier, UTType.image.identifier], isTargeted: $isComposerDropTargeted) { providers in
if viewModel.isSearchMode || viewModel.isSending {
return false
}
Task {
await importAttachmentsFromItemProviders(providers)
}
return true
}
.confirmationDialog("Add attachment", isPresented: $isShowingAttachmentOptions, titleVisibility: .visible) {
Button("Photo Library") {
isShowingPhotoPicker = true
}
Button("Files") {
isShowingFileImporter = true
}
if canPasteFromClipboard {
Button("Paste from Clipboard") {
Task {
await pasteAttachmentsFromClipboard()
}
}
}
Button("Cancel", role: .cancel) {}
}
.photosPicker(
isPresented: $isShowingPhotoPicker,
selection: $photoPickerItems,
maxSelectionCount: max(1, SybilChatAttachmentSupport.maxAttachmentsPerMessage - viewModel.composerAttachments.count),
matching: .images
)
.fileImporter(
isPresented: $isShowingFileImporter,
allowedContentTypes: [.item],
allowsMultipleSelection: true
) { result in
Task {
do {
let urls = try result.get()
let attachments = try SybilChatAttachmentSupport.buildAttachments(from: urls)
try await MainActor.run {
try viewModel.appendComposerAttachments(attachments)
}
composerFocused = true
} catch {
await MainActor.run {
viewModel.errorMessage = error.localizedDescription
}
SybilLog.error(SybilLog.ui, "File import failed", error: error)
}
}
}
.onChange(of: photoPickerItems) { _, items in
guard !items.isEmpty else { return }
Task {
do {
let attachments = try await loadAttachmentsFromPhotoPickerItems(items)
try await MainActor.run {
try viewModel.appendComposerAttachments(attachments)
photoPickerItems = []
}
composerFocused = true
} catch {
await MainActor.run {
viewModel.errorMessage = error.localizedDescription
photoPickerItems = []
}
SybilLog.error(SybilLog.ui, "Photo import failed", error: error)
}
}
}
}
private var canPasteFromClipboard: Bool {
let pasteboard = UIPasteboard.general
if pasteboard.hasImages {
return true
}
if let url = pasteboard.url, url.isFileURL {
return true
}
if let string = pasteboard.string, !string.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty {
return true
}
return false
}
@MainActor
private func importAttachmentsFromItemProviders(_ providers: [NSItemProvider]) async {
do {
let attachments = try await SybilChatAttachmentSupport.buildAttachments(from: providers)
try viewModel.appendComposerAttachments(attachments)
composerFocused = true
} catch {
viewModel.errorMessage = error.localizedDescription
SybilLog.error(SybilLog.ui, "Clipboard/drop attachment import failed", error: error)
}
}
private func loadAttachmentsFromPhotoPickerItems(_ items: [PhotosPickerItem]) async throws -> [ChatAttachment] {
var attachments: [ChatAttachment] = []
for item in items {
guard let data = try await item.loadTransferable(type: Data.self) else {
continue
}
let contentType = item.supportedContentTypes.first(where: { $0.conforms(to: .image) })
let filename = contentType?.preferredFilenameExtension.map { "photo.\($0)" } ?? "photo.jpg"
attachments.append(try SybilChatAttachmentSupport.buildImageAttachment(data: data, filename: filename, contentType: contentType))
}
return attachments
}
@MainActor
private func pasteAttachmentsFromClipboard() async {
do {
let pasteboard = UIPasteboard.general
var attachments: [ChatAttachment] = []
if let image = pasteboard.image {
attachments.append(try SybilChatAttachmentSupport.buildImageAttachment(image: image))
}
if let url = pasteboard.url, url.isFileURL {
attachments.append(contentsOf: try SybilChatAttachmentSupport.buildAttachments(from: [url]))
} else if let text = pasteboard.string?.trimmingCharacters(in: .whitespacesAndNewlines), !text.isEmpty {
attachments.append(try SybilChatAttachmentSupport.buildTextAttachment(text: text))
}
try viewModel.appendComposerAttachments(attachments)
composerFocused = true
} catch {
viewModel.errorMessage = error.localizedDescription
SybilLog.error(SybilLog.ui, "Clipboard attachment import failed", error: error)
}
2026-02-20 00:09:02 -08:00
}
}