summaryrefslogtreecommitdiff
path: root/Sora/Views/Settings/Section/SettingsSectionImportExportView.swift
blob: 381b6a4beff6a587023730a470b5e0a3414d08e8 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
import SwiftUI
import UniformTypeIdentifiers

struct SettingsSectionImportExportView: View {
  @EnvironmentObject private var settings: SettingsManager
  @State private var isFileExporterPresented = false
  @State private var isFileImporterPresented = false
  @State private var bookmarksExportDocument: JSONFileDocument?
  @State private var bookmarksExportFilename = "sora_bookmarks.json"
  @State private var exportError: Error?
  @State private var importError: Error?
  private let dateFormatter: DateFormatter = {
    let formatter = DateFormatter()
    formatter.dateFormat = "yyyy-MM-dd_HH-mm-ss"
    return formatter
  }()

  var body: some View {
    Group {
      Button("Import Bookmarks") {
        isFileImporterPresented = true
      }

      Button("Export Bookmarks") {
        prepareBookmarksExport()
      }
    }
    #if os(macOS)
      .trailingFrame()
    #endif
    .fileExporter(
      isPresented: $isFileExporterPresented,
      document: bookmarksExportDocument,
      contentType: .json,
      defaultFilename: bookmarksExportFilename
    ) { result in
      bookmarksExportDocument = nil

      switch result {
      case .success:
        break

      case .failure(let error):
        if !isUserCancelled(error) {
          exportError = error
        }
      }
    }
    .fileImporter(
      isPresented: $isFileImporterPresented,
      allowedContentTypes: [.json],
      allowsMultipleSelection: false
    ) { result in
      handleImportResult(result)
    }
    .alert(
      "Export Failed",
      isPresented: Binding(
        get: { exportError != nil },
        set: { if !$0 { exportError = nil } }
      )
    ) {
      Button("OK", role: .cancel) { () }
    } message: {
      Text(exportError?.localizedDescription ?? "An unknown error occurred while exporting.")
    }
    .alert(
      "Import Failed",
      isPresented: Binding(
        get: { importError != nil },
        set: { if !$0 { importError = nil } }
      )
    ) {
      Button("OK", role: .cancel) { () }
    } message: {
      Text(importError?.localizedDescription ?? "An unknown error occurred while importing.")
    }
  }

  private func prepareBookmarksExport() {
    do {
      let data = try settings.exportBookmarks()
      let timestamp = dateFormatter.string(from: Date())

      bookmarksExportDocument = JSONFileDocument(data)
      bookmarksExportFilename = "sora_bookmarks_\(timestamp).json"
      isFileExporterPresented = true
    } catch {
      exportError = error
    }
  }

  private func handleImportResult(_ result: Result<[URL], Error>) {
    do {
      guard let selectedFile = try result.get().first else { return }
      guard selectedFile.startAccessingSecurityScopedResource() else {
        throw ImportError.accessDenied
      }

      defer { selectedFile.stopAccessingSecurityScopedResource() }

      let data = try Data(contentsOf: selectedFile)

      try settings.importBookmarks(from: data)
    } catch {
      importError = error
    }
  }

  private func isUserCancelled(_ error: Error) -> Bool {
    let error = error as NSError

    return error.domain == NSCocoaErrorDomain
      && error.code == CocoaError.userCancelled.rawValue
  }

  private enum ImportError: Error {
    case accessDenied
  }
}