mirror of
https://github.com/simplex-chat/simplex-chat.git
synced 2026-04-28 01:56:25 +00:00
d90c4261b8
* refactor image utils and initial link metadata tools * remove LPMetadata conversion as we will build our own view to avoid network calls * initial very basic preview outline, remove icon loading * connect preview view to compose view * v0.1 barely working * minor refactor * refactor * collect images effectively * link up to api for send/receive * rework async get metadata logic * show previews in chat * refactor resizing logic * checkpoint before view editing * ui changes * housekeeping * ui tweaks * typo * improve link preview design/logic * resize image to target data size * fix link preview state machine * tidy up * fix typo Co-authored-by: Evgeny Poberezkin <2769109+epoberezkin@users.noreply.github.com>
159 lines
4.8 KiB
Swift
159 lines
4.8 KiB
Swift
//
|
|
// ComposeView.swift
|
|
// SimpleX
|
|
//
|
|
// Created by Evgeny on 13/03/2022.
|
|
// Copyright © 2022 SimpleX Chat. All rights reserved.
|
|
//
|
|
|
|
import SwiftUI
|
|
|
|
// TODO
|
|
//enum ComposeState {
|
|
// case plain
|
|
// case quoted(quotedItem: ChatItem)
|
|
// case editing(editingItem: ChatItem)
|
|
//}
|
|
|
|
struct ComposeView: View {
|
|
@Binding var message: String
|
|
@Binding var quotedItem: ChatItem?
|
|
@Binding var editingItem: ChatItem?
|
|
@Binding var linkPreview: LinkPreview?
|
|
|
|
var sendMessage: (String) -> Void
|
|
var resetMessage: () -> Void
|
|
var inProgress: Bool = false
|
|
@FocusState.Binding var keyboardVisible: Bool
|
|
@State var editing: Bool = false
|
|
@State var linkUrl: URL? = nil
|
|
@State var prevLinkUrl: URL? = nil
|
|
@State var pendingLinkUrl: URL? = nil
|
|
@State var cancelledLinks: Set<String> = []
|
|
|
|
|
|
private func isValidLink(link: String) -> Bool {
|
|
return !(link.starts(with: "https://simplex.chat") || link.starts(with: "http://simplex.chat"))
|
|
}
|
|
|
|
func cancelPreview() {
|
|
if let uri = linkPreview?.uri.absoluteString {
|
|
cancelledLinks.insert(uri)
|
|
}
|
|
linkPreview = nil
|
|
}
|
|
|
|
func parseMessage(_ msg: String) -> URL? {
|
|
do {
|
|
if let parsedMsg = try apiParseMarkdown(text: msg),
|
|
let link = parsedMsg.first(where: {
|
|
$0.format == .uri && !cancelledLinks.contains($0.text)
|
|
}),
|
|
isValidLink(link: link.text) {
|
|
return URL(string: link.text)
|
|
} else {
|
|
return nil
|
|
}
|
|
} catch {
|
|
logger.error("apiParseMarkdown error: \(error.localizedDescription)")
|
|
return nil
|
|
}
|
|
}
|
|
|
|
var body: some View {
|
|
VStack(spacing: 0) {
|
|
if let metadata = linkPreview {
|
|
ComposeLinkView(linkPreview: metadata, cancelPreview: cancelPreview)
|
|
}
|
|
if (quotedItem != nil) {
|
|
ContextItemView(contextItem: $quotedItem, editing: $editing)
|
|
} else if (editingItem != nil) {
|
|
ContextItemView(contextItem: $editingItem, editing: $editing, resetMessage: resetMessage)
|
|
}
|
|
SendMessageView(
|
|
sendMessage: { text in
|
|
sendMessage(text)
|
|
resetLinkPreview()
|
|
},
|
|
inProgress: inProgress,
|
|
message: $message,
|
|
keyboardVisible: $keyboardVisible,
|
|
editing: $editing
|
|
)
|
|
.background(.background)
|
|
}
|
|
.onChange(of: message) { _ in
|
|
if message.count > 0 {
|
|
prevLinkUrl = linkUrl
|
|
linkUrl = parseMessage(message)
|
|
if let url = linkUrl {
|
|
if prevLinkUrl == linkUrl {
|
|
loadLinkPreview(url)
|
|
} else {
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + 1.5) {
|
|
loadLinkPreview(url)
|
|
}
|
|
}
|
|
} else {
|
|
linkPreview = nil
|
|
}
|
|
} else {
|
|
resetLinkPreview()
|
|
}
|
|
}
|
|
.onChange(of: editingItem == nil) { _ in
|
|
editing = (editingItem != nil)
|
|
}
|
|
}
|
|
|
|
func loadLinkPreview(_ url: URL) {
|
|
if url != linkPreview?.uri && url != pendingLinkUrl {
|
|
pendingLinkUrl = url
|
|
getLinkPreview(url: url) { lp in
|
|
if pendingLinkUrl == url {
|
|
linkPreview = lp
|
|
pendingLinkUrl = nil
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func resetLinkPreview() {
|
|
linkUrl = nil
|
|
prevLinkUrl = nil
|
|
pendingLinkUrl = nil
|
|
cancelledLinks = []
|
|
}
|
|
}
|
|
|
|
struct ComposeView_Previews: PreviewProvider {
|
|
static var previews: some View {
|
|
@State var message: String = ""
|
|
@FocusState var keyboardVisible: Bool
|
|
@State var item: ChatItem? = ChatItem.getSample(1, .directSnd, .now, "hello")
|
|
@State var nilItem: ChatItem? = nil
|
|
@State var linkPreview: LinkPreview? = nil
|
|
|
|
return Group {
|
|
ComposeView(
|
|
message: $message,
|
|
quotedItem: $item,
|
|
editingItem: $nilItem,
|
|
linkPreview: $linkPreview,
|
|
sendMessage: { print ($0) },
|
|
resetMessage: {},
|
|
keyboardVisible: $keyboardVisible
|
|
)
|
|
ComposeView(
|
|
message: $message,
|
|
quotedItem: $nilItem,
|
|
editingItem: $item,
|
|
linkPreview: $linkPreview,
|
|
sendMessage: { print ($0) },
|
|
resetMessage: {},
|
|
keyboardVisible: $keyboardVisible
|
|
)
|
|
}
|
|
}
|
|
}
|