You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
session-ios/Session/Settings/HelpViewModel.swift

290 lines
11 KiB
Swift

// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
import Foundation
import CryptoKit
import GRDB
import DifferenceKit
import SessionUIKit
import SessionMessagingKit
import SessionUtilitiesKit
class HelpViewModel: SessionTableViewModel, NavigatableStateHolder, ObservableTableSource {
typealias TableItem = Section
public let dependencies: Dependencies
public let navigatableState: NavigatableState = NavigatableState()
public let state: TableDataState<Section, TableItem> = TableDataState()
public let observableState: ObservableTableSourceState<Section, TableItem> = ObservableTableSourceState()
private static var documentPickerResult: DocumentPickerResult?
// MARK: - Initialization
init(using dependencies: Dependencies = Dependencies()) {
self.dependencies = dependencies
}
// MARK: - Section
public enum Section: SessionTableSection {
case report
case translate
case feedback
case faq
case support
var style: SessionTableSectionStyle { .padding }
}
// MARK: - Content
let title: String = "sessionHelp".localized()
lazy var observation: TargetObservation = [
SectionModel(
model: .report,
elements: [
SessionCell.Info(
id: .report,
title: "helpReportABug".localized(),
1 year ago
subtitle: "helpReportABugExportLogsDescription"
.put(key: "app_name", value: Constants.app_name)
1 year ago
.localized(),
rightAccessory: .highlightingBackgroundLabel(
title: "helpReportABugExportLogs".localized()
),
onTapView: { HelpViewModel.shareLogs(targetView: $0) }
)
]
),
SectionModel(
model: .translate,
elements: [
SessionCell.Info(
id: .translate,
1 year ago
title: "helpHelpUsTranslateSession"
.put(key: "app_name", value: Constants.app_name)
1 year ago
.localized(),
rightAccessory: .icon(
UIImage(systemName: "arrow.up.forward.app")?
.withRenderingMode(.alwaysTemplate),
size: .small
),
onTap: {
guard let url: URL = URL(string: "https://getsession.org/translate") else {
return
}
UIApplication.shared.open(url)
}
)
]
),
SectionModel(
model: .feedback,
elements: [
SessionCell.Info(
id: .feedback,
title: "helpWedLoveYourFeedback".localized(),
rightAccessory: .icon(
UIImage(systemName: "arrow.up.forward.app")?
.withRenderingMode(.alwaysTemplate),
size: .small
),
onTap: {
guard let url: URL = URL(string: "https://getsession.org/survey") else {
return
}
UIApplication.shared.open(url)
}
)
]
),
SectionModel(
model: .faq,
elements: [
SessionCell.Info(
id: .faq,
title: "helpFAQ".localized(),
rightAccessory: .icon(
UIImage(systemName: "arrow.up.forward.app")?
.withRenderingMode(.alwaysTemplate),
size: .small
),
onTap: {
guard let url: URL = URL(string: "https://getsession.org/faq") else {
return
}
UIApplication.shared.open(url)
}
)
]
),
SectionModel(
model: .support,
elements: [
SessionCell.Info(
id: .support,
title: "helpSupport".localized(),
rightAccessory: .icon(
UIImage(systemName: "arrow.up.forward.app")?
.withRenderingMode(.alwaysTemplate),
size: .small
),
onTap: {
guard let url: URL = URL(string: "https://sessionapp.zendesk.com/hc/en-us") else {
return
}
UIApplication.shared.open(url)
}
)
]
)
]
// MARK: - Functions
public static func shareLogs(
viewControllerToDismiss: UIViewController? = nil,
targetView: UIView? = nil,
animated: Bool = true,
onShareComplete: (() -> ())? = nil
) {
guard
let latestLogFilePath: String = Log.logFilePath(),
Singleton.hasAppContext,
let viewController: UIViewController = Singleton.appContext.frontmostViewController
else { return }
let modal: ConfirmationModal = ConfirmationModal(
info: ConfirmationModal.Info(
title: "helpReportABugExportLogs".localized(),
body: .text("helpReportABugExportLogsDescription"
.put(key: "app_name", value: Constants.app_name)
.localized()),
confirmTitle: "save".localized(),
cancelTitle: "share".localized(),
cancelStyle: .alert_text,
hasCloseButton: true,
dismissOnConfirm: false,
onConfirm: { modal in
#if targetEnvironment(simulator)
UIPasteboard.general.string = latestLogFilePath
#endif
modal.dismiss(animated: true) {
HelpViewModel.shareLogsInternal(
viaShareSheet: false,
viewControllerToDismiss: viewControllerToDismiss,
targetView: targetView,
animated: animated,
onShareComplete: onShareComplete
)
}
},
onCancel: { modal in
#if targetEnvironment(simulator)
UIPasteboard.general.string = latestLogFilePath
#endif
modal.dismiss(animated: true) {
HelpViewModel.shareLogsInternal(
viaShareSheet: true,
viewControllerToDismiss: viewControllerToDismiss,
targetView: targetView,
animated: animated,
onShareComplete: onShareComplete
)
}
}
)
)
viewController.present(modal, animated: animated, completion: nil)
}
private static func shareLogsInternal(
viaShareSheet: Bool,
viewControllerToDismiss: UIViewController? = nil,
targetView: UIView? = nil,
animated: Bool = true,
onShareComplete: (() -> ())? = nil
) {
Log.info("[Version] \(SessionApp.versionInfo)")
Log.flush()
guard
let latestLogFilePath: String = Log.logFilePath(),
Singleton.hasAppContext,
let viewController: UIViewController = Singleton.appContext.frontmostViewController
else { return }
let showExportOption: () -> () = {
switch viaShareSheet {
case true:
let shareVC = UIActivityViewController(
activityItems: [ URL(fileURLWithPath: latestLogFilePath) ],
applicationActivities: nil
)
shareVC.completionWithItemsHandler = { _, _, _, _ in onShareComplete?() }
if UIDevice.current.isIPad {
shareVC.excludedActivityTypes = []
shareVC.popoverPresentationController?.permittedArrowDirections = (targetView != nil ? [.up] : [])
shareVC.popoverPresentationController?.sourceView = (targetView ?? viewController.view)
shareVC.popoverPresentationController?.sourceRect = (targetView ?? viewController.view).bounds
}
viewController.present(shareVC, animated: animated, completion: nil)
case false:
// Create and present the document picker
let documentPickerResult: DocumentPickerResult = DocumentPickerResult { _ in
HelpViewModel.documentPickerResult = nil
onShareComplete?()
}
HelpViewModel.documentPickerResult = documentPickerResult
let documentPicker: UIDocumentPickerViewController = UIDocumentPickerViewController(
forExporting: [URL(fileURLWithPath: latestLogFilePath)]
)
documentPicker.delegate = documentPickerResult
documentPicker.modalPresentationStyle = .formSheet
viewController.present(documentPicker, animated: animated, completion: nil)
}
}
guard let viewControllerToDismiss: UIViewController = viewControllerToDismiss else {
showExportOption()
return
}
viewControllerToDismiss.dismiss(animated: animated) {
showExportOption()
}
}
}
private class DocumentPickerResult: NSObject, UIDocumentPickerDelegate {
private let onResult: (URL?) -> Void
init(onResult: @escaping (URL?) -> Void) {
self.onResult = onResult
}
// MARK: - UIDocumentPickerDelegate
func documentPicker(_ controller: UIDocumentPickerViewController, didPickDocumentsAt urls: [URL]) {
guard let url: URL = urls.first else {
self.onResult(nil)
return
}
self.onResult(url)
}
func documentPickerWasCancelled(_ controller: UIDocumentPickerViewController) {
self.onResult(nil)
}
}