2022-10-31 13:41:19 +01:00
|
|
|
//
|
|
|
|
// AttachmentView.swift
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// Created by MainasuK on 2022-5-20.
|
|
|
|
//
|
|
|
|
|
|
|
|
import os.log
|
|
|
|
import UIKit
|
|
|
|
import SwiftUI
|
|
|
|
import Introspect
|
|
|
|
import AVKit
|
2022-11-08 12:40:58 +01:00
|
|
|
import MastodonAsset
|
2022-11-13 17:57:44 +01:00
|
|
|
import MastodonLocalization
|
|
|
|
import Introspect
|
2022-10-31 13:41:19 +01:00
|
|
|
|
|
|
|
public struct AttachmentView: View {
|
|
|
|
|
|
|
|
@ObservedObject var viewModel: AttachmentViewModel
|
2022-11-11 11:10:13 +01:00
|
|
|
|
2022-11-08 12:40:58 +01:00
|
|
|
var blurEffect: UIBlurEffect {
|
|
|
|
UIBlurEffect(style: .systemUltraThinMaterialDark)
|
|
|
|
}
|
2022-10-31 13:41:19 +01:00
|
|
|
|
|
|
|
public var body: some View {
|
2022-11-13 17:57:44 +01:00
|
|
|
Color.clear.aspectRatio(358.0/232.0, contentMode: .fill)
|
|
|
|
.overlay(
|
|
|
|
ZStack {
|
|
|
|
let image = viewModel.thumbnail ?? .placeholder(color: .secondarySystemFill)
|
|
|
|
Image(uiImage: image)
|
|
|
|
.resizable()
|
|
|
|
.aspectRatio(contentMode: .fill)
|
2022-11-08 12:40:58 +01:00
|
|
|
}
|
2022-11-13 17:57:44 +01:00
|
|
|
)
|
|
|
|
.overlay(
|
|
|
|
ZStack {
|
|
|
|
Color.clear
|
|
|
|
.overlay(
|
|
|
|
VStack(alignment: .leading) {
|
|
|
|
let placeholder: String = {
|
|
|
|
switch viewModel.output {
|
|
|
|
case .image: return L10n.Scene.Compose.Attachment.descriptionPhoto
|
|
|
|
case .video: return L10n.Scene.Compose.Attachment.descriptionVideo
|
|
|
|
case nil: return ""
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
Spacer()
|
|
|
|
TextField(placeholder, text: $viewModel.caption)
|
2022-11-13 18:19:39 +01:00
|
|
|
.lineLimit(1)
|
2022-11-13 17:57:44 +01:00
|
|
|
.textFieldStyle(.plain)
|
|
|
|
.foregroundColor(.white)
|
|
|
|
.placeholder(placeholder, when: viewModel.caption.isEmpty)
|
|
|
|
.padding(8)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
// loading…
|
|
|
|
if viewModel.output == nil, viewModel.error == nil {
|
|
|
|
ProgressView()
|
|
|
|
.progressViewStyle(.circular)
|
|
|
|
}
|
|
|
|
|
|
|
|
// load failed
|
|
|
|
// cannot re-entry
|
|
|
|
if viewModel.output == nil, let error = viewModel.error {
|
|
|
|
VisualEffectView(effect: blurEffect)
|
|
|
|
VStack {
|
2022-11-14 12:27:09 +01:00
|
|
|
Text(L10n.Scene.Compose.Attachment.loadFailed)
|
2022-11-13 17:57:44 +01:00
|
|
|
.font(.system(size: 13, weight: .semibold))
|
|
|
|
Text(error.localizedDescription)
|
|
|
|
.font(.system(size: 12, weight: .regular))
|
2022-11-08 12:40:58 +01:00
|
|
|
}
|
2022-11-13 17:57:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// loaded
|
|
|
|
// uploading… or upload failed
|
|
|
|
// could retry upload when error emit
|
|
|
|
if viewModel.output != nil, viewModel.uploadState != .finish {
|
|
|
|
VisualEffectView(effect: blurEffect)
|
|
|
|
VStack {
|
|
|
|
let action: AttachmentViewModel.Action = {
|
|
|
|
if let _ = viewModel.error {
|
|
|
|
return .retry
|
|
|
|
} else {
|
|
|
|
return .remove
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
Button {
|
|
|
|
viewModel.delegate?.attachmentViewModel(viewModel, actionButtonDidPressed: action)
|
|
|
|
} label: {
|
|
|
|
let image: UIImage = {
|
|
|
|
switch action {
|
|
|
|
case .remove:
|
|
|
|
return Asset.Scene.Compose.Attachment.stop.image.withRenderingMode(.alwaysTemplate)
|
|
|
|
case .retry:
|
|
|
|
return Asset.Scene.Compose.Attachment.retry.image.withRenderingMode(.alwaysTemplate)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
Image(uiImage: image)
|
|
|
|
.foregroundColor(.white)
|
|
|
|
.padding()
|
|
|
|
.background(Color(Asset.Scene.Compose.Attachment.indicatorButtonBackground.color))
|
|
|
|
.overlay(
|
|
|
|
Group {
|
|
|
|
switch viewModel.uploadState {
|
|
|
|
case .compressing:
|
|
|
|
CircleProgressView(progress: viewModel.videoCompressProgress)
|
|
|
|
.animation(.default, value: viewModel.videoCompressProgress)
|
|
|
|
case .uploading:
|
|
|
|
CircleProgressView(progress: viewModel.fractionCompleted)
|
|
|
|
.animation(.default, value: viewModel.fractionCompleted)
|
|
|
|
default:
|
|
|
|
EmptyView()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
.clipShape(Circle())
|
|
|
|
.padding()
|
2022-11-10 11:36:36 +01:00
|
|
|
}
|
2022-11-13 17:57:44 +01:00
|
|
|
|
|
|
|
let title: String = {
|
|
|
|
switch action {
|
|
|
|
case .remove:
|
2022-11-11 14:28:19 +01:00
|
|
|
switch viewModel.uploadState {
|
|
|
|
case .compressing:
|
2022-11-14 18:44:28 +01:00
|
|
|
return L10n.Scene.Compose.Attachment.compressingState
|
2022-11-11 14:28:19 +01:00
|
|
|
default:
|
2022-11-13 17:57:44 +01:00
|
|
|
if viewModel.fractionCompleted < 0.9 {
|
|
|
|
let totalSizeInByte = viewModel.outputSizeInByte
|
|
|
|
let uploadSizeInByte = Double(totalSizeInByte) * min(1.0, viewModel.fractionCompleted + 0.1) // 9:1
|
|
|
|
let total = viewModel.byteCountFormatter.string(fromByteCount: Int64(totalSizeInByte))
|
|
|
|
let upload = viewModel.byteCountFormatter.string(fromByteCount: Int64(uploadSizeInByte))
|
|
|
|
return "\(upload) / \(total)"
|
|
|
|
} else {
|
2022-11-14 18:44:28 +01:00
|
|
|
return L10n.Scene.Compose.Attachment.serverProcessingState
|
2022-11-13 17:57:44 +01:00
|
|
|
}
|
2022-11-11 14:28:19 +01:00
|
|
|
}
|
2022-11-13 17:57:44 +01:00
|
|
|
case .retry:
|
2022-11-14 12:27:09 +01:00
|
|
|
return L10n.Scene.Compose.Attachment.uploadFailed
|
2022-11-11 14:28:19 +01:00
|
|
|
}
|
2022-11-13 17:57:44 +01:00
|
|
|
}()
|
|
|
|
let subtitle: String = {
|
|
|
|
switch action {
|
|
|
|
case .remove:
|
|
|
|
if viewModel.progress.fractionCompleted < 1, viewModel.uploadState == .uploading {
|
|
|
|
if viewModel.progress.fractionCompleted < 0.9 {
|
|
|
|
return viewModel.remainTimeLocalizedString ?? ""
|
|
|
|
} else {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
} else if viewModel.videoCompressProgress < 1, viewModel.uploadState == .compressing {
|
|
|
|
return viewModel.percentageFormatter.string(from: NSNumber(floatLiteral: viewModel.videoCompressProgress)) ?? ""
|
|
|
|
} else {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
case .retry:
|
|
|
|
return viewModel.error?.localizedDescription ?? ""
|
2022-11-11 14:28:19 +01:00
|
|
|
}
|
2022-11-13 17:57:44 +01:00
|
|
|
}()
|
|
|
|
Text(title)
|
|
|
|
.font(.system(size: 13, weight: .semibold))
|
|
|
|
.foregroundColor(.white)
|
|
|
|
.padding(.horizontal)
|
|
|
|
Text(subtitle)
|
|
|
|
.font(.system(size: 12, weight: .regular))
|
|
|
|
.foregroundColor(.white)
|
|
|
|
.padding(.horizontal)
|
|
|
|
.lineLimit(nil)
|
|
|
|
.multilineTextAlignment(.center)
|
|
|
|
.frame(maxWidth: 240)
|
2022-11-08 12:40:58 +01:00
|
|
|
}
|
2022-11-13 17:57:44 +01:00
|
|
|
}
|
|
|
|
} // end ZStack
|
|
|
|
)
|
2022-10-31 13:41:19 +01:00
|
|
|
} // end body
|
|
|
|
|
|
|
|
}
|
2022-11-13 17:57:44 +01:00
|
|
|
|
|
|
|
// https://stackoverflow.com/a/57715771/3797903
|
|
|
|
extension View {
|
|
|
|
fileprivate func placeholder<Content: View>(
|
|
|
|
when shouldShow: Bool,
|
|
|
|
alignment: Alignment = .leading,
|
|
|
|
@ViewBuilder placeholder: () -> Content) -> some View {
|
|
|
|
|
|
|
|
ZStack(alignment: alignment) {
|
|
|
|
placeholder().opacity(shouldShow ? 1 : 0)
|
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fileprivate func placeholder(
|
|
|
|
_ text: String,
|
|
|
|
when shouldShow: Bool,
|
|
|
|
alignment: Alignment = .leading) -> some View {
|
|
|
|
|
|
|
|
placeholder(when: shouldShow, alignment: alignment) {
|
|
|
|
Text(text)
|
|
|
|
.foregroundColor(.white.opacity(0.7))
|
2022-11-13 18:19:39 +01:00
|
|
|
.lineLimit(1)
|
2022-11-13 17:57:44 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|