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