Bark/Controller/SoundsViewModel.swift
2024-08-07 16:31:23 +08:00

231 lines
8.0 KiB
Swift
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

//
// SoundsViewModel.swift
// Bark
//
// Created by huangfeng on 2020/11/17.
// Copyright © 2020 Fin. All rights reserved.
//
import AVKit
import Foundation
import RxCocoa
import RxDataSources
import RxSwift
enum SoundItem {
case sound(model: SoundCellViewModel)
case addSound
}
class SoundsViewModel: ViewModel, ViewModelType {
///
struct Dependencies {
///
let soundFileStorage: SoundFileStorageProtocol
}
private let dependencies: Dependencies
init(dependencies: Dependencies = Dependencies(soundFileStorage: SoundFileStorage())) {
self.dependencies = dependencies
}
struct Input {
///
var soundSelected: Driver<SoundItem>
///
var importSound: Driver<URL>
///
var soundDeleted: Driver<SoundItem>
}
struct Output {
///
var audios: Observable<[SectionModel<String, SoundItem>]>
///
var copyNameAction: Driver<String>
///
var playAction: Driver<CFURL>
///
var pickerFile: Driver<Void>
}
/// URL SoundItem
func getSounds(urls: [URL]) -> [SoundItem] {
let urls = urls.sorted { u1, u2 -> Bool in
u1.lastPathComponent.localizedStandardCompare(u2.lastPathComponent) == ComparisonResult.orderedAscending
}
return urls
.map { AVURLAsset(url: $0) }
.map { SoundCellViewModel(model: $0) }
.map { SoundItem.sound(model: $0) }
}
///
func getFilesInDirectory(directory: String, suffix: String) -> [URL] {
let fileManager = FileManager.default
do {
let files = try fileManager.contentsOfDirectory(atPath: directory)
return files.compactMap { file -> URL? in
if file.hasSuffix(suffix) {
return URL(fileURLWithPath: directory).appendingPathComponent(file)
}
return nil
}
} catch {
return []
}
}
func transform(input: Input) -> Output {
//
input
.importSound
.drive { [unowned self] url in
self.dependencies.soundFileStorage.saveSound(url: url)
}
.disposed(by: rx.disposeBag)
//
input.soundDeleted.drive(onNext: { item in
guard case SoundItem.sound(let model) = item else {
return
}
self.dependencies.soundFileStorage.deleteSound(url: model.model.url)
}).disposed(by: rx.disposeBag)
//
let soundsListUpdated = Observable.merge(
//
Observable.just(()),
//
input.importSound.map { _ in () }.asObservable(),
//
input.soundDeleted.map { _ in () }.asObservable()
).share(replay: 1)
//
let sounds: Observable<([SoundItem], [SoundItem])> = soundsListUpdated.map { _ in
let defaultSounds = self.getSounds(
urls: Bundle.main.urls(forResourcesWithExtension: "caf", subdirectory: nil) ?? []
)
let customSounds: [SoundItem] = {
guard let soundsDirectoryUrl = NSSearchPathForDirectoriesInDomains(.libraryDirectory, .userDomainMask, true).first?.appending("/Sounds") else {
return [.addSound]
}
return self.getSounds(
urls: self.getFilesInDirectory(directory: soundsDirectoryUrl, suffix: "caf")
) + [.addSound]
}()
return (customSounds, defaultSounds)
}.share(replay: 1)
// RxDataSource
let dataSource = sounds.map { sounds in
return [
SectionModel(model: "customSounds", items: sounds.0),
SectionModel(model: "defaultSounds", items: sounds.1)
]
}
//
let copyAction = sounds.flatMapLatest { sounds in
let observables = (sounds.0 + sounds.1).compactMap { item in
if case SoundItem.sound(let model) = item {
return model
}
return nil
}.map { model in
return model.copyNameAction.asObservable()
}
return Observable.merge(observables)
}.asDriver(onErrorDriveWith: .empty())
return Output(
audios: dataSource,
copyNameAction: copyAction,
playAction: input.soundSelected
.compactMap { item in
if case SoundItem.sound(let model) = item {
return model
}
return nil
}
.map { $0.model.url as CFURL },
pickerFile: input.soundSelected
.compactMap { item in
if case SoundItem.addSound = item {
return ()
}
return nil
}
)
}
}
///
protocol SoundFileStorageProtocol {
func saveSound(url: URL)
func deleteSound(url: URL)
}
/// /Library/Sounds
class SoundFileStorage: SoundFileStorageProtocol {
let fileManager: FileManager
init() {
fileManager = FileManager()
}
/// Library/Sound
func saveSound(url: URL) {
// Sounds
let soundsDirectoryUrl = getSoundsDirectory()
let soundUrl = soundsDirectoryUrl.appendingPathComponent(url.lastPathComponent)
try? fileManager.copyItem(at: url, to: soundUrl)
//
saveSoundToGroupDirectory(url: url)
}
func deleteSound(url: URL) {
// sounds
try? fileManager.removeItem(at: url)
//
if let groupSoundUrl = getSoundsGroupDirectory()?.appendingPathComponent(url.lastPathComponent) {
try? fileManager.removeItem(at: groupSoundUrl)
}
}
/// Library Sounds
///
private func getSoundsDirectory() -> URL {
let soundsDirectoryUrl = NSSearchPathForDirectoriesInDomains(.libraryDirectory, .userDomainMask, true).first!.appending("/Sounds")
if !fileManager.fileExists(atPath: soundsDirectoryUrl) {
try? fileManager.createDirectory(atPath: soundsDirectoryUrl, withIntermediateDirectories: true, attributes: nil)
}
return URL(fileURLWithPath: soundsDirectoryUrl)
}
/// NotificationServiceExtension 使
private func saveSoundToGroupDirectory(url: URL) {
guard let groupUrl = getSoundsGroupDirectory() else {
return
}
let soundUrl = groupUrl.appendingPathComponent(url.lastPathComponent)
try? fileManager.copyItem(at: url, to: soundUrl)
}
/// Sounds
///
private func getSoundsGroupDirectory() -> URL? {
if let directoryUrl = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: "group.bark")?.appendingPathComponent("Sounds") {
if !fileManager.fileExists(atPath: directoryUrl.absoluteString) {
try? fileManager.createDirectory(atPath: directoryUrl.absoluteString, withIntermediateDirectories: true, attributes: nil)
}
return directoryUrl
}
return nil
}
}