mirror of
https://github.com/simplex-chat/simplex-chat.git
synced 2026-05-31 13:15:42 +00:00
9d1329498b
* ios: open chat on first unread, "scroll" to quoted items that were not loaded * more changes * changes * unused * fix reveal logic * debug * changes * test * Revert "test" This reverts commit553be124d5. * change * change * changes * changes * changes * commented deceleration logic * changes * fixes * optimized item identifiers to use merged item directly * fixed counters * encreased initial and preload counters * fix initial loading and trimming items * optimize * allow marking read * 10 instead of 5 * performance * one more parameter in hash * disable trimming * performance * performance - in background * optimization * next/prev * changes * markread * finally * less logs * read * change after merge * trimming, edge cases * wait until items loaded * Revert "wait until items loaded" This reverts commit895218b978. * progress indicator * optimization * disable scroll helper * experiment * Revert "experiment" This reverts commitc952c9e623. * jump * no read * layoutIfNeeded * changes * EndlessScrollView * read * changes * changes * changes * reduce time to open a chat (by ~300ms) * open from the first unread when clicking member chat * refactored and removed unused code * handling search emptiness to scroll to correct position * changes * read state maintain * remove protocol * avoid parsing chatId * pass chat * changes * remove reveal * refactor spaghetti * remove ItemsScrollModel * rename * remove setUpdateListener * unused * optimization * scrollToTop * fix * scrollbar working again * scrollToBottom * fix * scrollBar hiding when not many items on screen * small safer change --------- Co-authored-by: Evgeny Poberezkin <evgeny@poberezkin.com>
311 lines
14 KiB
Swift
311 lines
14 KiB
Swift
//
|
|
// CIFileView.swift
|
|
// SimpleX
|
|
//
|
|
// Created by JRoberts on 28/04/2022.
|
|
// Copyright © 2022 SimpleX Chat. All rights reserved.
|
|
//
|
|
|
|
import SwiftUI
|
|
import SimpleXChat
|
|
|
|
struct CIFileView: View {
|
|
@EnvironmentObject var m: ChatModel
|
|
@EnvironmentObject var theme: AppTheme
|
|
let file: CIFile?
|
|
let edited: Bool
|
|
var smallViewSize: CGFloat?
|
|
|
|
var body: some View {
|
|
if smallViewSize != nil {
|
|
fileIndicator()
|
|
.onTapGesture(perform: fileAction)
|
|
} else {
|
|
let metaReserve = edited
|
|
? " "
|
|
: " "
|
|
Button(action: fileAction) {
|
|
HStack(alignment: .bottom, spacing: 6) {
|
|
fileIndicator()
|
|
.padding(.top, 5)
|
|
.padding(.bottom, 3)
|
|
if let file = file {
|
|
let prettyFileSize = ByteCountFormatter.string(fromByteCount: file.fileSize, countStyle: .binary)
|
|
VStack(alignment: .leading, spacing: 2) {
|
|
Text(file.fileName)
|
|
.lineLimit(1)
|
|
.multilineTextAlignment(.leading)
|
|
.foregroundColor(theme.colors.onBackground)
|
|
Text(prettyFileSize + metaReserve)
|
|
.font(.caption)
|
|
.lineLimit(1)
|
|
.multilineTextAlignment(.leading)
|
|
.foregroundColor(theme.colors.secondary)
|
|
}
|
|
} else {
|
|
Text(metaReserve)
|
|
}
|
|
}
|
|
.padding(.top, 4)
|
|
.padding(.bottom, 6)
|
|
.padding(.leading, 10)
|
|
.padding(.trailing, 12)
|
|
}
|
|
.disabled(!itemInteractive)
|
|
}
|
|
}
|
|
|
|
private var itemInteractive: Bool {
|
|
if let file = file {
|
|
switch (file.fileStatus) {
|
|
case .sndStored: return file.fileProtocol == .local
|
|
case .sndTransfer: return false
|
|
case .sndComplete: return true
|
|
case .sndCancelled: return false
|
|
case .sndError: return true
|
|
case .sndWarning: return true
|
|
case .rcvInvitation: return true
|
|
case .rcvAccepted: return true
|
|
case .rcvTransfer: return false
|
|
case .rcvAborted: return true
|
|
case .rcvComplete: return true
|
|
case .rcvCancelled: return false
|
|
case .rcvError: return true
|
|
case .rcvWarning: return true
|
|
case .invalid: return false
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
private func fileAction() {
|
|
logger.debug("CIFileView fileAction")
|
|
if let file = file {
|
|
switch (file.fileStatus) {
|
|
case .rcvInvitation, .rcvAborted:
|
|
if fileSizeValid(file) {
|
|
Task {
|
|
logger.debug("CIFileView fileAction - in .rcvInvitation, .rcvAborted, in Task")
|
|
if let user = m.currentUser {
|
|
await receiveFile(user: user, fileId: file.fileId)
|
|
}
|
|
}
|
|
} else {
|
|
let prettyMaxFileSize = ByteCountFormatter.string(fromByteCount: getMaxFileSize(file.fileProtocol), countStyle: .binary)
|
|
AlertManager.shared.showAlertMsg(
|
|
title: "Large file!",
|
|
message: "Your contact sent a file that is larger than currently supported maximum size (\(prettyMaxFileSize))."
|
|
)
|
|
}
|
|
case .rcvAccepted:
|
|
switch file.fileProtocol {
|
|
case .xftp:
|
|
AlertManager.shared.showAlertMsg(
|
|
title: "Waiting for file",
|
|
message: "File will be received when your contact completes uploading it."
|
|
)
|
|
case .smp:
|
|
AlertManager.shared.showAlertMsg(
|
|
title: "Waiting for file",
|
|
message: "File will be received when your contact is online, please wait or check later!"
|
|
)
|
|
case .local: ()
|
|
}
|
|
case .rcvComplete:
|
|
logger.debug("CIFileView fileAction - in .rcvComplete")
|
|
if let fileSource = getLoadedFileSource(file) {
|
|
saveCryptoFile(fileSource)
|
|
}
|
|
case let .rcvError(rcvFileError):
|
|
logger.debug("CIFileView fileAction - in .rcvError")
|
|
showFileErrorAlert(rcvFileError)
|
|
case let .rcvWarning(rcvFileError):
|
|
logger.debug("CIFileView fileAction - in .rcvWarning")
|
|
showFileErrorAlert(rcvFileError, temporary: true)
|
|
case .sndStored:
|
|
logger.debug("CIFileView fileAction - in .sndStored")
|
|
if file.fileProtocol == .local, let fileSource = getLoadedFileSource(file) {
|
|
saveCryptoFile(fileSource)
|
|
}
|
|
case .sndComplete:
|
|
logger.debug("CIFileView fileAction - in .sndComplete")
|
|
if let fileSource = getLoadedFileSource(file) {
|
|
saveCryptoFile(fileSource)
|
|
}
|
|
case let .sndError(sndFileError):
|
|
logger.debug("CIFileView fileAction - in .sndError")
|
|
showFileErrorAlert(sndFileError)
|
|
case let .sndWarning(sndFileError):
|
|
logger.debug("CIFileView fileAction - in .sndWarning")
|
|
showFileErrorAlert(sndFileError, temporary: true)
|
|
default: break
|
|
}
|
|
}
|
|
}
|
|
|
|
@ViewBuilder private func fileIndicator() -> some View {
|
|
if let file = file {
|
|
switch file.fileStatus {
|
|
case .sndStored:
|
|
switch file.fileProtocol {
|
|
case .xftp: progressView()
|
|
case .smp: fileIcon("doc.fill")
|
|
case .local: fileIcon("doc.fill")
|
|
}
|
|
case let .sndTransfer(sndProgress, sndTotal):
|
|
switch file.fileProtocol {
|
|
case .xftp: progressCircle(sndProgress, sndTotal)
|
|
case .smp: progressView()
|
|
case .local: EmptyView()
|
|
}
|
|
case .sndComplete: fileIcon("doc.fill", innerIcon: "checkmark", innerIconSize: 10)
|
|
case .sndCancelled: fileIcon("doc.fill", innerIcon: "xmark", innerIconSize: 10)
|
|
case .sndError: fileIcon("doc.fill", innerIcon: "xmark", innerIconSize: 10)
|
|
case .sndWarning: fileIcon("doc.fill", innerIcon: "exclamationmark.triangle.fill", innerIconSize: 10)
|
|
case .rcvInvitation:
|
|
if fileSizeValid(file) {
|
|
fileIcon("arrow.down.doc.fill", color: theme.colors.primary)
|
|
} else {
|
|
fileIcon("doc.fill", color: .orange, innerIcon: "exclamationmark", innerIconSize: 12)
|
|
}
|
|
case .rcvAccepted: fileIcon("doc.fill", innerIcon: "ellipsis", innerIconSize: 12)
|
|
case let .rcvTransfer(rcvProgress, rcvTotal):
|
|
if file.fileProtocol == .xftp && rcvProgress < rcvTotal {
|
|
progressCircle(rcvProgress, rcvTotal)
|
|
} else {
|
|
progressView()
|
|
}
|
|
case .rcvAborted:
|
|
fileIcon("doc.fill", color: theme.colors.primary, innerIcon: "exclamationmark.arrow.circlepath", innerIconSize: 12)
|
|
case .rcvComplete: fileIcon("doc.fill")
|
|
case .rcvCancelled: fileIcon("doc.fill", innerIcon: "xmark", innerIconSize: 10)
|
|
case .rcvError: fileIcon("doc.fill", innerIcon: "xmark", innerIconSize: 10)
|
|
case .rcvWarning: fileIcon("doc.fill", innerIcon: "exclamationmark.triangle.fill", innerIconSize: 10)
|
|
case .invalid: fileIcon("doc.fill", innerIcon: "questionmark", innerIconSize: 10)
|
|
}
|
|
} else {
|
|
fileIcon("doc.fill")
|
|
}
|
|
}
|
|
|
|
private func fileIcon(_ icon: String, color: Color = Color(uiColor: .tertiaryLabel), innerIcon: String? = nil, innerIconSize: CGFloat? = nil) -> some View {
|
|
let size = smallViewSize ?? 30
|
|
return ZStack(alignment: .center) {
|
|
Image(systemName: icon)
|
|
.resizable()
|
|
.aspectRatio(contentMode: .fit)
|
|
.frame(width: size, height: size)
|
|
.foregroundColor(color)
|
|
if let innerIcon = innerIcon,
|
|
let innerIconSize = innerIconSize, (smallViewSize == nil || file?.showStatusIconInSmallView == true) {
|
|
Image(systemName: innerIcon)
|
|
.resizable()
|
|
.aspectRatio(contentMode: .fit)
|
|
.frame(maxHeight: 16)
|
|
.frame(width: innerIconSize, height: innerIconSize)
|
|
.foregroundColor(.white)
|
|
.padding(.top, size / 2.5)
|
|
}
|
|
}
|
|
}
|
|
|
|
private func progressView() -> some View {
|
|
ProgressView().frame(width: 30, height: 30)
|
|
}
|
|
|
|
private func progressCircle(_ progress: Int64, _ total: Int64) -> some View {
|
|
Circle()
|
|
.trim(from: 0, to: Double(progress) / Double(total))
|
|
.stroke(
|
|
Color(uiColor: .tertiaryLabel),
|
|
style: StrokeStyle(lineWidth: 3)
|
|
)
|
|
.rotationEffect(.degrees(-90))
|
|
.frame(width: 30, height: 30)
|
|
}
|
|
}
|
|
|
|
func fileSizeValid(_ file: CIFile?) -> Bool {
|
|
if let file = file {
|
|
return file.fileSize <= getMaxFileSize(file.fileProtocol)
|
|
}
|
|
return false
|
|
}
|
|
|
|
func saveCryptoFile(_ fileSource: CryptoFile) {
|
|
if let cfArgs = fileSource.cryptoArgs {
|
|
let url = getAppFilePath(fileSource.filePath)
|
|
let tempUrl = getTempFilesDirectory().appendingPathComponent(fileSource.filePath)
|
|
Task {
|
|
do {
|
|
try decryptCryptoFile(fromPath: url.path, cryptoArgs: cfArgs, toPath: tempUrl.path)
|
|
await MainActor.run {
|
|
showShareSheet(items: [tempUrl]) {
|
|
removeFile(tempUrl)
|
|
}
|
|
}
|
|
} catch {
|
|
await MainActor.run {
|
|
AlertManager.shared.showAlertMsg(title: "Error decrypting file", message: "Error: \(error.localizedDescription)")
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
let url = getAppFilePath(fileSource.filePath)
|
|
showShareSheet(items: [url])
|
|
}
|
|
}
|
|
|
|
func showFileErrorAlert(_ err: FileError, temporary: Bool = false) {
|
|
let title: String = if temporary {
|
|
NSLocalizedString("Temporary file error", comment: "file error alert title")
|
|
} else {
|
|
NSLocalizedString("File error", comment: "file error alert title")
|
|
}
|
|
if let btn = err.moreInfoButton {
|
|
showAlert(title, message: err.errorInfo) {
|
|
[
|
|
okAlertAction,
|
|
UIAlertAction(title: NSLocalizedString("How it works", comment: "alert button"), style: .default, handler: { _ in
|
|
UIApplication.shared.open(contentModerationPostLink)
|
|
})
|
|
]
|
|
}
|
|
} else {
|
|
showAlert(title, message: err.errorInfo)
|
|
}
|
|
}
|
|
|
|
struct CIFileView_Previews: PreviewProvider {
|
|
static var previews: some View {
|
|
let sentFile: ChatItem = ChatItem(
|
|
chatDir: .directSnd,
|
|
meta: CIMeta.getSample(1, .now, "", .sndSent(sndProgress: .complete), itemEdited: true),
|
|
content: .sndMsgContent(msgContent: .file("")),
|
|
quotedItem: nil,
|
|
file: CIFile.getSample(fileStatus: .sndComplete)
|
|
)
|
|
let fileChatItemWtFile = ChatItem(
|
|
chatDir: .directRcv,
|
|
meta: CIMeta.getSample(1, .now, "", .rcvRead),
|
|
content: .rcvMsgContent(msgContent: .file("")),
|
|
quotedItem: nil,
|
|
file: nil
|
|
)
|
|
Group {
|
|
ChatItemView(chat: Chat.sampleData, chatItem: sentFile, scrollToItemId: { _ in })
|
|
ChatItemView(chat: Chat.sampleData, chatItem: ChatItem.getFileMsgContentSample(), scrollToItemId: { _ in })
|
|
ChatItemView(chat: Chat.sampleData, chatItem: ChatItem.getFileMsgContentSample(fileName: "some_long_file_name_here", fileStatus: .rcvInvitation), scrollToItemId: { _ in })
|
|
ChatItemView(chat: Chat.sampleData, chatItem: ChatItem.getFileMsgContentSample(fileStatus: .rcvAccepted), scrollToItemId: { _ in })
|
|
ChatItemView(chat: Chat.sampleData, chatItem: ChatItem.getFileMsgContentSample(fileStatus: .rcvTransfer(rcvProgress: 7, rcvTotal: 10)), scrollToItemId: { _ in })
|
|
ChatItemView(chat: Chat.sampleData, chatItem: ChatItem.getFileMsgContentSample(fileStatus: .rcvCancelled), scrollToItemId: { _ in })
|
|
ChatItemView(chat: Chat.sampleData, chatItem: ChatItem.getFileMsgContentSample(fileSize: 1_000_000_000, fileStatus: .rcvInvitation), scrollToItemId: { _ in })
|
|
ChatItemView(chat: Chat.sampleData, chatItem: ChatItem.getFileMsgContentSample(text: "Hello there", fileStatus: .rcvInvitation), scrollToItemId: { _ in })
|
|
ChatItemView(chat: Chat.sampleData, chatItem: ChatItem.getFileMsgContentSample(text: "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.", fileStatus: .rcvInvitation), scrollToItemId: { _ in })
|
|
ChatItemView(chat: Chat.sampleData, chatItem: fileChatItemWtFile, scrollToItemId: { _ in })
|
|
}
|
|
.environment(\.revealed, false)
|
|
.previewLayout(.fixed(width: 360, height: 360))
|
|
}
|
|
}
|