297 lines
10 KiB
Swift
297 lines
10 KiB
Swift
//
|
|
// ComposeView.swift
|
|
// Tusker
|
|
//
|
|
// Created by Shadowfacts on 8/18/20.
|
|
// Copyright © 2020 Shadowfacts. All rights reserved.
|
|
//
|
|
|
|
import SwiftUI
|
|
import Pachyderm
|
|
import Combine
|
|
|
|
@propertyWrapper struct OptionalStateObject<T: ObservableObject>: DynamicProperty {
|
|
private class Republisher: ObservableObject {
|
|
var cancellable: AnyCancellable?
|
|
var wrapped: T? {
|
|
didSet {
|
|
cancellable?.cancel()
|
|
cancellable = wrapped?.objectWillChange
|
|
.receive(on: RunLoop.main)
|
|
.sink { [unowned self] _ in
|
|
self.objectWillChange.send()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
@StateObject private var republisher = Republisher()
|
|
@State private var object: T?
|
|
var wrappedValue: T? {
|
|
get {
|
|
object
|
|
}
|
|
nonmutating set {
|
|
object = newValue
|
|
}
|
|
}
|
|
|
|
func update() {
|
|
republisher.wrapped = wrappedValue
|
|
}
|
|
}
|
|
|
|
struct ComposeView: View {
|
|
@ObservedObject var draft: Draft
|
|
@EnvironmentObject var mastodonController: MastodonController
|
|
@EnvironmentObject var uiState: ComposeUIState
|
|
|
|
@State private var globalFrameOutsideList: CGRect = .zero
|
|
|
|
@OptionalStateObject private var poster: PostService?
|
|
@State private var isShowingPostErrorAlert = false
|
|
@State private var postError: PostService.Error?
|
|
|
|
private var isPosting: Bool {
|
|
poster != nil
|
|
}
|
|
|
|
private let stackPadding: CGFloat = 8
|
|
|
|
init(draft: Draft) {
|
|
self.draft = draft
|
|
}
|
|
|
|
var charactersRemaining: Int {
|
|
let limit = mastodonController.instanceFeatures.maxStatusChars
|
|
let cwCount = draft.contentWarningEnabled ? draft.contentWarning.count : 0
|
|
return limit - (cwCount + CharacterCounter.count(text: draft.text, for: mastodonController.instance))
|
|
}
|
|
|
|
var requiresAttachmentDescriptions: Bool {
|
|
guard Preferences.shared.requireAttachmentDescriptions else { return false }
|
|
let attachmentIds = draft.attachments.map(\.id)
|
|
return attachmentIds.contains { uiState.attachmentsMissingDescriptions.contains($0) }
|
|
}
|
|
|
|
var postButtonEnabled: Bool {
|
|
draft.hasContent && charactersRemaining >= 0 && !isPosting && !requiresAttachmentDescriptions && (draft.poll == nil || draft.poll!.options.allSatisfy { !$0.text.isEmpty })
|
|
}
|
|
|
|
var body: some View {
|
|
ZStack(alignment: .top) {
|
|
mainList
|
|
.scrollDismissesKeyboardInteractivelyIfAvailable()
|
|
|
|
if let poster = poster {
|
|
// can't use SwiftUI.ProgressView because there's no UIProgressView.Style.bar equivalent, see FB8587149
|
|
WrappedProgressView(value: poster.currentStep, total: poster.totalSteps)
|
|
}
|
|
|
|
autocompleteSuggestions
|
|
}
|
|
.background(GeometryReader { proxy in
|
|
Color.clear
|
|
.preference(key: GlobalFrameOutsideListPrefKey.self, value: proxy.frame(in: .global))
|
|
.onPreferenceChange(GlobalFrameOutsideListPrefKey.self) { frame in
|
|
globalFrameOutsideList = frame
|
|
}
|
|
})
|
|
.navigationTitle(navTitle)
|
|
.sheet(isPresented: $uiState.isShowingDraftsList) {
|
|
DraftsView(currentDraft: draft)
|
|
}
|
|
.actionSheet(isPresented: $uiState.isShowingSaveDraftSheet, content: self.saveAndCloseSheet)
|
|
.alert(isPresented: $isShowingPostErrorAlert) {
|
|
Alert(
|
|
title: Text("Error Posting Status"),
|
|
message: Text(postError?.localizedDescription ?? ""),
|
|
dismissButton: .default(Text("OK"))
|
|
)
|
|
}
|
|
.toolbar {
|
|
ToolbarItem(placement: .cancellationAction) { cancelButton }
|
|
ToolbarItem(placement: .confirmationAction) { postButton }
|
|
}
|
|
}
|
|
|
|
@ViewBuilder
|
|
var autocompleteSuggestions: some View {
|
|
VStack(spacing: 0) {
|
|
Spacer()
|
|
if let state = uiState.autocompleteState {
|
|
ComposeAutocompleteView(autocompleteState: state)
|
|
}
|
|
}
|
|
.transition(.move(edge: .bottom))
|
|
.animation(.default, value: uiState.autocompleteState)
|
|
}
|
|
|
|
var mainList: some View {
|
|
List {
|
|
if let id = draft.inReplyToID,
|
|
let status = mastodonController.persistentContainer.status(for: id) {
|
|
ComposeReplyView(
|
|
status: status,
|
|
rowTopInset: 8,
|
|
globalFrameOutsideList: globalFrameOutsideList
|
|
)
|
|
.listRowInsets(EdgeInsets(top: 8, leading: 8, bottom: 4, trailing: 8))
|
|
.listRowSeparator(.hidden)
|
|
}
|
|
|
|
header
|
|
.listRowInsets(EdgeInsets(top: draft.inReplyToID == nil ? 8 : 4, leading: 8, bottom: 4, trailing: 8))
|
|
.listRowSeparator(.hidden)
|
|
|
|
if draft.contentWarningEnabled {
|
|
ComposeEmojiTextField(text: $draft.contentWarning, placeholder: "Write your warning here")
|
|
.listRowInsets(EdgeInsets(top: 4, leading: 8, bottom: 4, trailing: 8))
|
|
.listRowSeparator(.hidden)
|
|
}
|
|
|
|
MainComposeTextView(draft: draft)
|
|
.listRowInsets(EdgeInsets(top: 4, leading: 8, bottom: 4, trailing: 8))
|
|
.listRowSeparator(.hidden)
|
|
|
|
if let poll = draft.poll {
|
|
ComposePollView(draft: draft, poll: poll)
|
|
.listRowInsets(EdgeInsets(top: 4, leading: 8, bottom: 4, trailing: 8))
|
|
.listRowSeparator(.hidden)
|
|
}
|
|
|
|
ComposeAttachmentsList(
|
|
draft: draft
|
|
)
|
|
.listRowInsets(EdgeInsets(top: 4, leading: 8, bottom: 8, trailing: 8))
|
|
}
|
|
.animation(.default, value: draft.poll?.options.count)
|
|
.scrollDismissesKeyboardInteractivelyIfAvailable()
|
|
.listStyle(.plain)
|
|
.disabled(isPosting)
|
|
.padding(.bottom, uiState.autocompleteState != nil ? 46 : 0)
|
|
}
|
|
|
|
private var header: some View {
|
|
HStack(alignment: .top) {
|
|
ComposeCurrentAccount()
|
|
Spacer()
|
|
Text(verbatim: charactersRemaining.description)
|
|
.foregroundColor(charactersRemaining < 0 ? .red : .secondary)
|
|
.font(Font.body.monospacedDigit())
|
|
.accessibility(label: Text(charactersRemaining < 0 ? "\(-charactersRemaining) characters too many" : "\(charactersRemaining) characters remaining"))
|
|
}.frame(height: 50)
|
|
}
|
|
|
|
private var navTitle: Text {
|
|
if let id = draft.inReplyToID,
|
|
let status = mastodonController.persistentContainer.status(for: id) {
|
|
return Text("Reply to @\(status.account.acct)")
|
|
} else {
|
|
return Text("New Post")
|
|
}
|
|
}
|
|
|
|
private var cancelButton: some View {
|
|
Button(action: self.cancel) {
|
|
Text("Cancel")
|
|
// otherwise all Buttons in the nav bar are made semibold
|
|
.font(.system(size: 17, weight: .regular))
|
|
}
|
|
}
|
|
|
|
private var postButton: some View {
|
|
Button {
|
|
Task {
|
|
await self.postStatus()
|
|
}
|
|
} label: {
|
|
Text("Post")
|
|
}
|
|
.disabled(!postButtonEnabled)
|
|
}
|
|
|
|
private func cancel() {
|
|
if Preferences.shared.automaticallySaveDrafts {
|
|
// draft is already stored in drafts manager, drafts manager is saved by ComposeHostingController.viewWillDisappear
|
|
uiState.delegate?.dismissCompose(mode: .cancel)
|
|
} else {
|
|
// if the draft doesn't have content, it doesn't need to be saved
|
|
if draft.hasContent {
|
|
uiState.isShowingSaveDraftSheet = true
|
|
} else {
|
|
DraftsManager.shared.remove(draft)
|
|
uiState.delegate?.dismissCompose(mode: .cancel)
|
|
}
|
|
}
|
|
}
|
|
|
|
private func saveAndCloseSheet() -> ActionSheet {
|
|
ActionSheet(title: Text("Do you want to save the current post as a draft?"), buttons: [
|
|
.default(Text("Save Draft"), action: {
|
|
// draft is already stored in drafts manager, drafts manager is saved by ComposeHostingController.viewWillDisappear
|
|
uiState.isShowingSaveDraftSheet = false
|
|
uiState.delegate?.dismissCompose(mode: .cancel)
|
|
}),
|
|
.destructive(Text("Delete Draft"), action: {
|
|
DraftsManager.shared.remove(draft)
|
|
uiState.isShowingSaveDraftSheet = false
|
|
uiState.delegate?.dismissCompose(mode: .cancel)
|
|
}),
|
|
.cancel(),
|
|
])
|
|
}
|
|
|
|
private func postStatus() async {
|
|
guard !isPosting,
|
|
draft.hasContent else {
|
|
return
|
|
}
|
|
|
|
let poster = PostService(mastodonController: mastodonController, draft: draft)
|
|
self.poster = poster
|
|
|
|
do {
|
|
try await poster.post()
|
|
|
|
// wait .25 seconds so the user can see the progress bar has completed
|
|
try? await Task.sleep(nanoseconds: 250_000_000)
|
|
|
|
uiState.delegate?.dismissCompose(mode: .post)
|
|
|
|
} catch let error as PostService.Error {
|
|
self.isShowingPostErrorAlert = true
|
|
self.postError = error
|
|
} catch {
|
|
fatalError("Unreachable")
|
|
}
|
|
|
|
self.poster = nil
|
|
}
|
|
}
|
|
|
|
private extension View {
|
|
@available(iOS, obsoleted: 16.0)
|
|
@ViewBuilder
|
|
func scrollDismissesKeyboardInteractivelyIfAvailable() -> some View {
|
|
if #available(iOS 16.0, *) {
|
|
self.scrollDismissesKeyboard(.interactively)
|
|
} else {
|
|
self
|
|
}
|
|
}
|
|
}
|
|
|
|
private struct GlobalFrameOutsideListPrefKey: PreferenceKey {
|
|
static var defaultValue: CGRect = .zero
|
|
static func reduce(value: inout CGRect, nextValue: () -> CGRect) {
|
|
value = nextValue()
|
|
}
|
|
}
|
|
|
|
//struct ComposeView_Previews: PreviewProvider {
|
|
// static var previews: some View {
|
|
// ComposeView()
|
|
// }
|
|
//}
|