Adds ability to import ControllerSkin’s
This commit is contained in:
parent
a7037ee45a
commit
023fe13c6a
@ -32,15 +32,124 @@ final class DatabaseManager: NSPersistentContainer
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extension DatabaseManager
|
||||||
|
{
|
||||||
|
override func newBackgroundContext() -> NSManagedObjectContext
|
||||||
|
{
|
||||||
|
let context = super.newBackgroundContext()
|
||||||
|
context.mergePolicy = NSMergeByPropertyObjectTrumpMergePolicy
|
||||||
|
return context
|
||||||
|
}
|
||||||
|
|
||||||
|
override func loadPersistentStores(completionHandler block: @escaping (NSPersistentStoreDescription, Error?) -> Void)
|
||||||
|
{
|
||||||
|
super.loadPersistentStores { (description, error) in
|
||||||
|
self.prepareDatabase {
|
||||||
|
block(description, error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//MARK: - Preparation -
|
||||||
|
private extension DatabaseManager
|
||||||
|
{
|
||||||
|
func prepareDatabase(completion: @escaping (Void) -> Void)
|
||||||
|
{
|
||||||
|
self.performBackgroundTask { (context) in
|
||||||
|
|
||||||
|
for gameType in Game.supportedTypes
|
||||||
|
{
|
||||||
|
guard let deltaControllerSkin = DeltaCore.ControllerSkin.standardControllerSkin(for: gameType) else { continue }
|
||||||
|
|
||||||
|
let controllerSkin = ControllerSkin(context: context)
|
||||||
|
controllerSkin.isStandard = true
|
||||||
|
controllerSkin.filename = deltaControllerSkin.fileURL.lastPathComponent
|
||||||
|
controllerSkin.name = deltaControllerSkin.name
|
||||||
|
controllerSkin.identifier = deltaControllerSkin.identifier
|
||||||
|
controllerSkin.gameType = deltaControllerSkin.gameType
|
||||||
|
}
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
try context.save()
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
print("Failed to import standard controller skins:", error)
|
||||||
|
}
|
||||||
|
|
||||||
|
completion()
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
//MARK: - Importing -
|
//MARK: - Importing -
|
||||||
/// Importing
|
/// Importing
|
||||||
extension DatabaseManager
|
extension DatabaseManager
|
||||||
{
|
{
|
||||||
func importGames(at urls: [URL], completion: (([String]) -> Void)?)
|
func importControllerSkins(at urls: [URL], completion: ((Set<String>) -> Void)?)
|
||||||
{
|
{
|
||||||
self.performBackgroundTask { (context) in
|
self.performBackgroundTask { (context) in
|
||||||
|
|
||||||
var identifiers: [String] = []
|
var identifiers = Set<String>()
|
||||||
|
|
||||||
|
for url in urls
|
||||||
|
{
|
||||||
|
guard let deltaControllerSkin = DeltaCore.ControllerSkin(fileURL: url) else { continue }
|
||||||
|
|
||||||
|
let controllerSkin = ControllerSkin(context: context)
|
||||||
|
|
||||||
|
// Manually copy values to be stored in database.
|
||||||
|
// Remaining ControllerSkinProtocol requirements will be provided by the ControllerSkin's private DeltaCore.ControllerSkin instance.
|
||||||
|
controllerSkin.filename = deltaControllerSkin.identifier + ".deltaskin"
|
||||||
|
controllerSkin.name = deltaControllerSkin.name
|
||||||
|
controllerSkin.identifier = deltaControllerSkin.identifier
|
||||||
|
controllerSkin.gameType = deltaControllerSkin.gameType
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if FileManager.default.fileExists(atPath: controllerSkin.fileURL.path)
|
||||||
|
{
|
||||||
|
// Normally we'd replace item instead of delete + move, but it's crashing as of iOS 10
|
||||||
|
// FileManager.default.replaceItemAt(controllerSkin.fileURL, withItemAt: url)
|
||||||
|
|
||||||
|
// Controller skin exists, but we replace it with the new skin
|
||||||
|
try FileManager.default.removeItem(at: controllerSkin.fileURL)
|
||||||
|
}
|
||||||
|
|
||||||
|
try FileManager.default.moveItem(at: url, to: controllerSkin.fileURL)
|
||||||
|
|
||||||
|
identifiers.insert(controllerSkin.identifier)
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
print("Import Controller Skins error:", error)
|
||||||
|
controllerSkin.managedObjectContext?.delete(controllerSkin)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
try context.save()
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
print("Failed to save controller skin import context:", error)
|
||||||
|
|
||||||
|
identifiers.removeAll()
|
||||||
|
}
|
||||||
|
|
||||||
|
completion?(identifiers)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func importGames(at urls: [URL], completion: ((Set<String>) -> Void)?)
|
||||||
|
{
|
||||||
|
self.performBackgroundTask { (context) in
|
||||||
|
|
||||||
|
var identifiers = Set<String>()
|
||||||
|
|
||||||
for url in urls
|
for url in urls
|
||||||
{
|
{
|
||||||
@ -63,6 +172,7 @@ extension DatabaseManager
|
|||||||
|
|
||||||
if FileManager.default.fileExists(atPath: destinationURL.path)
|
if FileManager.default.fileExists(atPath: destinationURL.path)
|
||||||
{
|
{
|
||||||
|
// Game already exists, so we choose not to override it and just delete the new game instead
|
||||||
try FileManager.default.removeItem(at: url)
|
try FileManager.default.removeItem(at: url)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -70,10 +180,11 @@ extension DatabaseManager
|
|||||||
try FileManager.default.moveItem(at: url, to: destinationURL)
|
try FileManager.default.moveItem(at: url, to: destinationURL)
|
||||||
}
|
}
|
||||||
|
|
||||||
identifiers.append(game.identifier)
|
identifiers.insert(game.identifier)
|
||||||
}
|
}
|
||||||
catch
|
catch
|
||||||
{
|
{
|
||||||
|
print("Import Games error:", error)
|
||||||
game.managedObjectContext?.delete(game)
|
game.managedObjectContext?.delete(game)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -90,11 +201,7 @@ extension DatabaseManager
|
|||||||
identifiers.removeAll()
|
identifiers.removeAll()
|
||||||
}
|
}
|
||||||
|
|
||||||
if let completion = completion
|
completion?(identifiers)
|
||||||
{
|
|
||||||
completion(identifiers)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -14,7 +14,7 @@ import DeltaCore
|
|||||||
public class ControllerSkin: _ControllerSkin
|
public class ControllerSkin: _ControllerSkin
|
||||||
{
|
{
|
||||||
public var fileURL: URL {
|
public var fileURL: URL {
|
||||||
let fileURL = DatabaseManager.controllerSkinsDirectoryURL(for: self.gameType).appendingPathComponent(self.filename)
|
let fileURL = self.isStandard ? self.controllerSkin!.fileURL : DatabaseManager.controllerSkinsDirectoryURL(for: self.gameType).appendingPathComponent(self.filename)
|
||||||
return fileURL
|
return fileURL
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -22,7 +22,10 @@ public class ControllerSkin: _ControllerSkin
|
|||||||
return self.controllerSkin?.isDebugModeEnabled ?? false
|
return self.controllerSkin?.isDebugModeEnabled ?? false
|
||||||
}
|
}
|
||||||
|
|
||||||
fileprivate lazy var controllerSkin: DeltaCore.ControllerSkin? = DeltaCore.ControllerSkin(fileURL: self.fileURL)
|
fileprivate lazy var controllerSkin: DeltaCore.ControllerSkin? = {
|
||||||
|
let controllerSkin = self.isStandard ? DeltaCore.ControllerSkin.standardControllerSkin(for: self.gameType) : DeltaCore.ControllerSkin(fileURL: self.fileURL)
|
||||||
|
return controllerSkin
|
||||||
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
extension ControllerSkin: ControllerSkinProtocol
|
extension ControllerSkin: ControllerSkinProtocol
|
||||||
|
|||||||
@ -27,6 +27,7 @@
|
|||||||
</userInfo>
|
</userInfo>
|
||||||
</attribute>
|
</attribute>
|
||||||
<attribute name="identifier" attributeType="String" syncable="YES"/>
|
<attribute name="identifier" attributeType="String" syncable="YES"/>
|
||||||
|
<attribute name="isStandard" attributeType="Boolean" defaultValueString="NO" usesScalarValueType="YES" syncable="YES"/>
|
||||||
<attribute name="name" attributeType="String" syncable="YES"/>
|
<attribute name="name" attributeType="String" syncable="YES"/>
|
||||||
<uniquenessConstraints>
|
<uniquenessConstraints>
|
||||||
<uniquenessConstraint>
|
<uniquenessConstraint>
|
||||||
@ -98,7 +99,7 @@
|
|||||||
</entity>
|
</entity>
|
||||||
<elements>
|
<elements>
|
||||||
<element name="Cheat" positionX="-198" positionY="-63" width="128" height="165"/>
|
<element name="Cheat" positionX="-198" positionY="-63" width="128" height="165"/>
|
||||||
<element name="ControllerSkin" positionX="-387" positionY="90" width="128" height="105"/>
|
<element name="ControllerSkin" positionX="-387" positionY="90" width="128" height="120"/>
|
||||||
<element name="Game" positionX="-378" positionY="-54" width="128" height="180"/>
|
<element name="Game" positionX="-378" positionY="-54" width="128" height="180"/>
|
||||||
<element name="GameCollection" positionX="-585" positionY="-27" width="128" height="90"/>
|
<element name="GameCollection" positionX="-585" positionY="-27" width="128" height="90"/>
|
||||||
<element name="SaveState" positionX="-198" positionY="113" width="128" height="165"/>
|
<element name="SaveState" positionX="-198" positionY="113" width="128" height="165"/>
|
||||||
|
|||||||
@ -75,8 +75,8 @@ extension Game
|
|||||||
|
|
||||||
extension Game
|
extension Game
|
||||||
{
|
{
|
||||||
class func supportedTypeIdentifiers() -> Set<String>
|
class var supportedTypes: Set<GameType>
|
||||||
{
|
{
|
||||||
return [GameType.snes.rawValue, GameType.gba.rawValue]
|
return [GameType.snes, GameType.gba]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -20,6 +20,8 @@ public class _ControllerSkin: NSManagedObject
|
|||||||
|
|
||||||
@NSManaged public var identifier: String
|
@NSManaged public var identifier: String
|
||||||
|
|
||||||
|
@NSManaged public var isStandard: Bool
|
||||||
|
|
||||||
@NSManaged public var name: String
|
@NSManaged public var name: String
|
||||||
|
|
||||||
// MARK: - Relationships
|
// MARK: - Relationships
|
||||||
|
|||||||
@ -1,165 +0,0 @@
|
|||||||
//
|
|
||||||
// GamePickerController.swift
|
|
||||||
// Delta
|
|
||||||
//
|
|
||||||
// Created by Riley Testut on 10/10/15.
|
|
||||||
// Copyright © 2015 Riley Testut. All rights reserved.
|
|
||||||
//
|
|
||||||
|
|
||||||
import UIKit
|
|
||||||
import ObjectiveC
|
|
||||||
|
|
||||||
import DeltaCore
|
|
||||||
|
|
||||||
protocol GamePickerControllerDelegate
|
|
||||||
{
|
|
||||||
func gamePickerController(_ gamePickerController: GamePickerController, didImportGames games: [Game])
|
|
||||||
|
|
||||||
/** Optional **/
|
|
||||||
func gamePickerControllerDidCancel(_ gamePickerController: GamePickerController)
|
|
||||||
}
|
|
||||||
|
|
||||||
extension GamePickerControllerDelegate
|
|
||||||
{
|
|
||||||
func gamePickerControllerDidCancel(_ gamePickerController: GamePickerController)
|
|
||||||
{
|
|
||||||
// Empty Implementation
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class GamePickerController: NSObject
|
|
||||||
{
|
|
||||||
var delegate: GamePickerControllerDelegate?
|
|
||||||
|
|
||||||
fileprivate weak var presentingViewController: UIViewController?
|
|
||||||
|
|
||||||
fileprivate func presentGamePickerControllerFromPresentingViewController(_ presentingViewController: UIViewController, animated: Bool, completion: ((Void) -> Void)?)
|
|
||||||
{
|
|
||||||
self.presentingViewController = presentingViewController
|
|
||||||
|
|
||||||
#if os(iOS)
|
|
||||||
let documentMenuController = UIDocumentMenuViewController(documentTypes: Array(Game.supportedTypeIdentifiers()), in: .import)
|
|
||||||
documentMenuController.delegate = self
|
|
||||||
documentMenuController.addOption(withTitle: NSLocalizedString("iTunes", comment: ""), image: nil, order: .first) { self.importFromiTunes(nil) }
|
|
||||||
self.presentingViewController?.present(documentMenuController, animated: true, completion: nil)
|
|
||||||
#else
|
|
||||||
self.importFromiTunes(completion)
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
private func importFromiTunes(_ completion: ((Void) -> Void)?)
|
|
||||||
{
|
|
||||||
let alertController = UIAlertController(title: NSLocalizedString("Import from iTunes?", comment: ""), message: NSLocalizedString("Delta will import the games copied over via iTunes.", comment: ""), preferredStyle: .alert)
|
|
||||||
|
|
||||||
let importAction = UIAlertAction(title: NSLocalizedString("Import", comment: ""), style: .default) { action in
|
|
||||||
|
|
||||||
let documentsDirectoryURL = DatabaseManager.defaultDirectoryURL().deletingLastPathComponent()
|
|
||||||
|
|
||||||
do
|
|
||||||
{
|
|
||||||
let contents = try FileManager.default.contentsOfDirectory(at: documentsDirectoryURL, includingPropertiesForKeys: nil, options: .skipsHiddenFiles)
|
|
||||||
|
|
||||||
DatabaseManager.shared.performBackgroundTask { (context) in
|
|
||||||
let gameURLs = contents.filter({ GameCollection.gameSystemCollectionForPathExtension($0.pathExtension, inManagedObjectContext: context).identifier != GameType.delta.rawValue })
|
|
||||||
self.importGamesAtURLs(gameURLs)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
catch let error as NSError
|
|
||||||
{
|
|
||||||
print(error)
|
|
||||||
}
|
|
||||||
|
|
||||||
self.presentingViewController?.gamePickerController = nil
|
|
||||||
|
|
||||||
}
|
|
||||||
alertController.addAction(importAction)
|
|
||||||
|
|
||||||
let cancelAction = UIAlertAction(title: NSLocalizedString("Cancel", comment: ""), style: .cancel) { action in
|
|
||||||
self.delegate?.gamePickerControllerDidCancel(self)
|
|
||||||
self.presentingViewController?.gamePickerController = nil
|
|
||||||
}
|
|
||||||
alertController.addAction(cancelAction)
|
|
||||||
|
|
||||||
self.presentingViewController?.present(alertController, animated: true, completion: completion)
|
|
||||||
}
|
|
||||||
|
|
||||||
fileprivate func importGamesAtURLs(_ URLs: [URL])
|
|
||||||
{
|
|
||||||
DatabaseManager.shared.importGames(at: URLs) { identifiers in
|
|
||||||
|
|
||||||
DatabaseManager.shared.viewContext.perform() {
|
|
||||||
|
|
||||||
let predicate = NSPredicate(format: "%K IN (%@)", #keyPath(Game.identifier), identifiers)
|
|
||||||
let games = Game.instancesWithPredicate(predicate, inManagedObjectContext: DatabaseManager.shared.viewContext, type: Game.self)
|
|
||||||
|
|
||||||
self.delegate?.gamePickerController(self, didImportGames: games)
|
|
||||||
|
|
||||||
self.presentingViewController?.gamePickerController = nil
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#if os(iOS)
|
|
||||||
|
|
||||||
extension GamePickerController: UIDocumentMenuDelegate
|
|
||||||
{
|
|
||||||
func documentMenu(_ documentMenu: UIDocumentMenuViewController, didPickDocumentPicker documentPicker: UIDocumentPickerViewController)
|
|
||||||
{
|
|
||||||
documentPicker.delegate = self
|
|
||||||
self.presentingViewController?.present(documentPicker, animated: true, completion: nil)
|
|
||||||
}
|
|
||||||
|
|
||||||
func documentMenuWasCancelled(_ documentMenu: UIDocumentMenuViewController)
|
|
||||||
{
|
|
||||||
self.delegate?.gamePickerControllerDidCancel(self)
|
|
||||||
|
|
||||||
self.presentingViewController?.gamePickerController = nil
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
extension GamePickerController: UIDocumentPickerDelegate
|
|
||||||
{
|
|
||||||
func documentPicker(_ controller: UIDocumentPickerViewController, didPickDocumentAt url: URL)
|
|
||||||
{
|
|
||||||
self.importGamesAtURLs([url])
|
|
||||||
|
|
||||||
self.presentingViewController?.gamePickerController = nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func documentPickerWasCancelled(_ controller: UIDocumentPickerViewController)
|
|
||||||
{
|
|
||||||
self.delegate?.gamePickerControllerDidCancel(self)
|
|
||||||
|
|
||||||
self.presentingViewController?.gamePickerController = nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
private var GamePickerControllerKey: UInt8 = 0
|
|
||||||
|
|
||||||
extension UIViewController
|
|
||||||
{
|
|
||||||
fileprivate(set) var gamePickerController: GamePickerController?
|
|
||||||
{
|
|
||||||
set
|
|
||||||
{
|
|
||||||
objc_setAssociatedObject(self, &GamePickerControllerKey, newValue, .OBJC_ASSOCIATION_RETAIN_NONATOMIC)
|
|
||||||
}
|
|
||||||
get
|
|
||||||
{
|
|
||||||
return objc_getAssociatedObject(self, &GamePickerControllerKey) as? GamePickerController
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func presentGamePickerController(_ gamePickerController: GamePickerController, animated: Bool, completion: ((Void) -> Void)?)
|
|
||||||
{
|
|
||||||
self.gamePickerController = gamePickerController
|
|
||||||
|
|
||||||
gamePickerController.presentGamePickerControllerFromPresentingViewController(self, animated: animated, completion: completion)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
196
Common/Importing/ImportController.swift
Normal file
196
Common/Importing/ImportController.swift
Normal file
@ -0,0 +1,196 @@
|
|||||||
|
//
|
||||||
|
// ImportController.swift
|
||||||
|
// Delta
|
||||||
|
//
|
||||||
|
// Created by Riley Testut on 10/10/15.
|
||||||
|
// Copyright © 2015 Riley Testut. All rights reserved.
|
||||||
|
//
|
||||||
|
|
||||||
|
import UIKit
|
||||||
|
import ObjectiveC
|
||||||
|
|
||||||
|
import DeltaCore
|
||||||
|
|
||||||
|
protocol ImportControllerDelegate
|
||||||
|
{
|
||||||
|
func importController(_ importController: ImportController, didImport games: Set<Game>)
|
||||||
|
func importController(_ importController: ImportController, didImport controllerSkins: Set<ControllerSkin>)
|
||||||
|
|
||||||
|
/** Optional **/
|
||||||
|
func importControllerDidCancel(_ importController: ImportController)
|
||||||
|
}
|
||||||
|
|
||||||
|
extension ImportControllerDelegate
|
||||||
|
{
|
||||||
|
func importControllerDidCancel(_ importController: ImportController)
|
||||||
|
{
|
||||||
|
// Empty Implementation
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class ImportController: NSObject
|
||||||
|
{
|
||||||
|
var delegate: ImportControllerDelegate?
|
||||||
|
|
||||||
|
fileprivate weak var presentingViewController: UIViewController?
|
||||||
|
|
||||||
|
fileprivate func presentImportController(from presentingViewController: UIViewController, animated: Bool, completion: ((Void) -> Void)?)
|
||||||
|
{
|
||||||
|
self.presentingViewController = presentingViewController
|
||||||
|
|
||||||
|
var documentTypes = Game.supportedTypes.map { $0.rawValue }
|
||||||
|
documentTypes.append(kUTTypeDeltaControllerSkin as String)
|
||||||
|
|
||||||
|
#if os(iOS)
|
||||||
|
let documentMenuController = UIDocumentMenuViewController(documentTypes: documentTypes, in: .import)
|
||||||
|
documentMenuController.delegate = self
|
||||||
|
documentMenuController.addOption(withTitle: NSLocalizedString("iTunes", comment: ""), image: nil, order: .first) { self.importFromiTunes(nil) }
|
||||||
|
self.presentingViewController?.present(documentMenuController, animated: true, completion: nil)
|
||||||
|
#else
|
||||||
|
self.importFromiTunes(completion)
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
private func importFromiTunes(_ completion: ((Void) -> Void)?)
|
||||||
|
{
|
||||||
|
let alertController = UIAlertController(title: NSLocalizedString("Import from iTunes?", comment: ""), message: NSLocalizedString("Delta will import the games and controller skins copied over via iTunes.", comment: ""), preferredStyle: .alert)
|
||||||
|
|
||||||
|
let importAction = UIAlertAction(title: NSLocalizedString("Import", comment: ""), style: .default) { action in
|
||||||
|
|
||||||
|
let documentsDirectoryURL = DatabaseManager.defaultDirectoryURL().deletingLastPathComponent()
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
let contents = try FileManager.default.contentsOfDirectory(at: documentsDirectoryURL, includingPropertiesForKeys: nil, options: .skipsHiddenFiles)
|
||||||
|
|
||||||
|
DatabaseManager.shared.performBackgroundTask { (context) in
|
||||||
|
let controllerSkinURLs = contents.filter { $0.pathExtension == "deltaskin" }
|
||||||
|
self.importControllerSkins(at: controllerSkinURLs)
|
||||||
|
|
||||||
|
let gameURLs = contents.filter { GameCollection.gameSystemCollectionForPathExtension($0.pathExtension, inManagedObjectContext: context).identifier != GameType.delta.rawValue }
|
||||||
|
self.importGames(at: gameURLs)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
catch let error as NSError
|
||||||
|
{
|
||||||
|
print(error)
|
||||||
|
}
|
||||||
|
|
||||||
|
self.presentingViewController?.importController = nil
|
||||||
|
|
||||||
|
}
|
||||||
|
alertController.addAction(importAction)
|
||||||
|
|
||||||
|
let cancelAction = UIAlertAction(title: NSLocalizedString("Cancel", comment: ""), style: .cancel) { action in
|
||||||
|
self.delegate?.importControllerDidCancel(self)
|
||||||
|
self.presentingViewController?.importController = nil
|
||||||
|
}
|
||||||
|
alertController.addAction(cancelAction)
|
||||||
|
|
||||||
|
self.presentingViewController?.present(alertController, animated: true, completion: completion)
|
||||||
|
}
|
||||||
|
|
||||||
|
fileprivate func importGames(at urls: [URL])
|
||||||
|
{
|
||||||
|
DatabaseManager.shared.importGames(at: urls) { identifiers in
|
||||||
|
|
||||||
|
DatabaseManager.shared.viewContext.perform() {
|
||||||
|
|
||||||
|
let predicate = NSPredicate(format: "%K IN (%@)", #keyPath(Game.identifier), identifiers)
|
||||||
|
let games = Game.instancesWithPredicate(predicate, inManagedObjectContext: DatabaseManager.shared.viewContext, type: Game.self)
|
||||||
|
|
||||||
|
self.delegate?.importController(self, didImport: Set(games))
|
||||||
|
|
||||||
|
self.presentingViewController?.importController = nil
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fileprivate func importControllerSkins(at urls: [URL])
|
||||||
|
{
|
||||||
|
DatabaseManager.shared.importControllerSkins(at: urls) { identifiers in
|
||||||
|
|
||||||
|
DatabaseManager.shared.viewContext.perform() {
|
||||||
|
|
||||||
|
let predicate = NSPredicate(format: "%K IN (%@)", #keyPath(ControllerSkin.identifier), identifiers)
|
||||||
|
let controllerSkins = ControllerSkin.instancesWithPredicate(predicate, inManagedObjectContext: DatabaseManager.shared.viewContext, type: ControllerSkin.self)
|
||||||
|
|
||||||
|
self.delegate?.importController(self, didImport: Set(controllerSkins))
|
||||||
|
|
||||||
|
self.presentingViewController?.importController = nil
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#if os(iOS)
|
||||||
|
|
||||||
|
extension ImportController: UIDocumentMenuDelegate
|
||||||
|
{
|
||||||
|
func documentMenu(_ documentMenu: UIDocumentMenuViewController, didPickDocumentPicker documentPicker: UIDocumentPickerViewController)
|
||||||
|
{
|
||||||
|
documentPicker.delegate = self
|
||||||
|
self.presentingViewController?.present(documentPicker, animated: true, completion: nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func documentMenuWasCancelled(_ documentMenu: UIDocumentMenuViewController)
|
||||||
|
{
|
||||||
|
self.delegate?.importControllerDidCancel(self)
|
||||||
|
|
||||||
|
self.presentingViewController?.importController = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
extension ImportController: UIDocumentPickerDelegate
|
||||||
|
{
|
||||||
|
func documentPicker(_ controller: UIDocumentPickerViewController, didPickDocumentAt url: URL)
|
||||||
|
{
|
||||||
|
if url.pathExtension == "deltaskin"
|
||||||
|
{
|
||||||
|
self.importControllerSkins(at: [url])
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
self.importGames(at: [url])
|
||||||
|
}
|
||||||
|
|
||||||
|
self.presentingViewController?.importController = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func documentPickerWasCancelled(_ controller: UIDocumentPickerViewController)
|
||||||
|
{
|
||||||
|
self.delegate?.importControllerDidCancel(self)
|
||||||
|
|
||||||
|
self.presentingViewController?.importController = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
private var ImportControllerKey: UInt8 = 0
|
||||||
|
|
||||||
|
extension UIViewController
|
||||||
|
{
|
||||||
|
fileprivate(set) var importController: ImportController?
|
||||||
|
{
|
||||||
|
set
|
||||||
|
{
|
||||||
|
objc_setAssociatedObject(self, &ImportControllerKey, newValue, .OBJC_ASSOCIATION_RETAIN_NONATOMIC)
|
||||||
|
}
|
||||||
|
get
|
||||||
|
{
|
||||||
|
return objc_getAssociatedObject(self, &ImportControllerKey) as? ImportController
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func present(_ importController: ImportController, animated: Bool, completion: ((Void) -> Void)?)
|
||||||
|
{
|
||||||
|
self.importController = importController
|
||||||
|
|
||||||
|
importController.presentImportController(from: self, animated: animated, completion: completion)
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1 +1 @@
|
|||||||
Subproject commit 6558b6e5198ef9ae20e5ba9e68e4389fb210b66f
|
Subproject commit 43208853591dd1e24bd6aedaa81a6271a2016e28
|
||||||
@ -63,7 +63,7 @@
|
|||||||
BFC9B7391CEFCD34008629BB /* CheatsViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFC9B7381CEFCD34008629BB /* CheatsViewController.swift */; };
|
BFC9B7391CEFCD34008629BB /* CheatsViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFC9B7381CEFCD34008629BB /* CheatsViewController.swift */; };
|
||||||
BFCEA67E1D56FF640061A534 /* UIViewControllerContextTransitioning+Conveniences.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFCEA67D1D56FF640061A534 /* UIViewControllerContextTransitioning+Conveniences.swift */; };
|
BFCEA67E1D56FF640061A534 /* UIViewControllerContextTransitioning+Conveniences.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFCEA67D1D56FF640061A534 /* UIViewControllerContextTransitioning+Conveniences.swift */; };
|
||||||
BFD097211D3A01B8005A44C2 /* SaveStatesViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF3540041C5DA70400C1184C /* SaveStatesViewController.swift */; };
|
BFD097211D3A01B8005A44C2 /* SaveStatesViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF3540041C5DA70400C1184C /* SaveStatesViewController.swift */; };
|
||||||
BFDB28451BC9DA7B001D0C83 /* GamePickerController.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFDB28441BC9DA7B001D0C83 /* GamePickerController.swift */; };
|
BFDB28451BC9DA7B001D0C83 /* ImportController.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFDB28441BC9DA7B001D0C83 /* ImportController.swift */; };
|
||||||
BFDD04EF1D5E27DB002D450E /* NSFetchedResultsController+Conveniences.m in Sources */ = {isa = PBXBuildFile; fileRef = BF02BCFF1D361BD1000892F2 /* NSFetchedResultsController+Conveniences.m */; };
|
BFDD04EF1D5E27DB002D450E /* NSFetchedResultsController+Conveniences.m in Sources */ = {isa = PBXBuildFile; fileRef = BF02BCFF1D361BD1000892F2 /* NSFetchedResultsController+Conveniences.m */; };
|
||||||
BFDD04F11D5E2C27002D450E /* GameCollectionViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFDD04F01D5E2C27002D450E /* GameCollectionViewController.swift */; };
|
BFDD04F11D5E2C27002D450E /* GameCollectionViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFDD04F01D5E2C27002D450E /* GameCollectionViewController.swift */; };
|
||||||
BFE4269E1D9C68E600DC913F /* SaveStatesStoryboardSegue.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFE4269D1D9C68E600DC913F /* SaveStatesStoryboardSegue.swift */; };
|
BFE4269E1D9C68E600DC913F /* SaveStatesStoryboardSegue.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFE4269D1D9C68E600DC913F /* SaveStatesStoryboardSegue.swift */; };
|
||||||
@ -161,7 +161,7 @@
|
|||||||
BFC853361DB039B500E8C372 /* ControllerSkin.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ControllerSkin.swift; sourceTree = "<group>"; };
|
BFC853361DB039B500E8C372 /* ControllerSkin.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ControllerSkin.swift; sourceTree = "<group>"; };
|
||||||
BFC9B7381CEFCD34008629BB /* CheatsViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = CheatsViewController.swift; path = "Pause Menu/Cheats/CheatsViewController.swift"; sourceTree = "<group>"; };
|
BFC9B7381CEFCD34008629BB /* CheatsViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = CheatsViewController.swift; path = "Pause Menu/Cheats/CheatsViewController.swift"; sourceTree = "<group>"; };
|
||||||
BFCEA67D1D56FF640061A534 /* UIViewControllerContextTransitioning+Conveniences.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "UIViewControllerContextTransitioning+Conveniences.swift"; sourceTree = "<group>"; };
|
BFCEA67D1D56FF640061A534 /* UIViewControllerContextTransitioning+Conveniences.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "UIViewControllerContextTransitioning+Conveniences.swift"; sourceTree = "<group>"; };
|
||||||
BFDB28441BC9DA7B001D0C83 /* GamePickerController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = GamePickerController.swift; sourceTree = "<group>"; };
|
BFDB28441BC9DA7B001D0C83 /* ImportController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ImportController.swift; sourceTree = "<group>"; };
|
||||||
BFDD04F01D5E2C27002D450E /* GameCollectionViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = GameCollectionViewController.swift; sourceTree = "<group>"; };
|
BFDD04F01D5E2C27002D450E /* GameCollectionViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = GameCollectionViewController.swift; sourceTree = "<group>"; };
|
||||||
BFE4269D1D9C68E600DC913F /* SaveStatesStoryboardSegue.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = SaveStatesStoryboardSegue.swift; path = Segues/SaveStatesStoryboardSegue.swift; sourceTree = "<group>"; };
|
BFE4269D1D9C68E600DC913F /* SaveStatesStoryboardSegue.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = SaveStatesStoryboardSegue.swift; path = Segues/SaveStatesStoryboardSegue.swift; sourceTree = "<group>"; };
|
||||||
BFEC732C1AAECC4A00650035 /* Roxas.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; path = Roxas.framework; sourceTree = BUILT_PRODUCTS_DIR; };
|
BFEC732C1AAECC4A00650035 /* Roxas.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; path = Roxas.framework; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||||
@ -410,7 +410,7 @@
|
|||||||
BFDB28431BC9D9D1001D0C83 /* Importing */ = {
|
BFDB28431BC9D9D1001D0C83 /* Importing */ = {
|
||||||
isa = PBXGroup;
|
isa = PBXGroup;
|
||||||
children = (
|
children = (
|
||||||
BFDB28441BC9DA7B001D0C83 /* GamePickerController.swift */,
|
BFDB28441BC9DA7B001D0C83 /* ImportController.swift */,
|
||||||
);
|
);
|
||||||
path = Importing;
|
path = Importing;
|
||||||
sourceTree = "<group>";
|
sourceTree = "<group>";
|
||||||
@ -706,7 +706,7 @@
|
|||||||
BF5E7F441B9A650B00AE44F8 /* SettingsViewController.swift in Sources */,
|
BF5E7F441B9A650B00AE44F8 /* SettingsViewController.swift in Sources */,
|
||||||
BF2898231DAAFD720023D8E9 /* _GameCollection.swift in Sources */,
|
BF2898231DAAFD720023D8E9 /* _GameCollection.swift in Sources */,
|
||||||
BF353FF21C5D7FB000C1184C /* PauseViewController.swift in Sources */,
|
BF353FF21C5D7FB000C1184C /* PauseViewController.swift in Sources */,
|
||||||
BFDB28451BC9DA7B001D0C83 /* GamePickerController.swift in Sources */,
|
BFDB28451BC9DA7B001D0C83 /* ImportController.swift in Sources */,
|
||||||
BF2898151DAAFC2A0023D8E9 /* Cheat.swift in Sources */,
|
BF2898151DAAFC2A0023D8E9 /* Cheat.swift in Sources */,
|
||||||
BF696B801D9B2B02009639E0 /* Theme.swift in Sources */,
|
BF696B801D9B2B02009639E0 /* Theme.swift in Sources */,
|
||||||
BF7AE8081C2E858400B1B5BC /* PauseMenuViewController.swift in Sources */,
|
BF7AE8081C2E858400B1B5BC /* PauseMenuViewController.swift in Sources */,
|
||||||
|
|||||||
@ -246,20 +246,25 @@ private extension GamesViewController
|
|||||||
|
|
||||||
//MARK: - Importing -
|
//MARK: - Importing -
|
||||||
/// Importing
|
/// Importing
|
||||||
extension GamesViewController: GamePickerControllerDelegate
|
extension GamesViewController: ImportControllerDelegate
|
||||||
{
|
{
|
||||||
@IBAction fileprivate func importFiles()
|
@IBAction fileprivate func importFiles()
|
||||||
{
|
{
|
||||||
let gamePickerController = GamePickerController()
|
let importController = ImportController()
|
||||||
gamePickerController.delegate = self
|
importController.delegate = self
|
||||||
self.presentGamePickerController(gamePickerController, animated: true, completion: nil)
|
self.present(importController, animated: true, completion: nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
//MARK: - GamePickerControllerDelegate
|
//MARK: - ImportControllerDelegate
|
||||||
func gamePickerController(_ gamePickerController: GamePickerController, didImportGames games: [Game])
|
@nonobjc func importController(_ importController: ImportController, didImport games: Set<Game>)
|
||||||
{
|
{
|
||||||
print(games)
|
print(games)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@nonobjc func importController(_ importController: ImportController, didImport controllerSkins: Set<ControllerSkin>)
|
||||||
|
{
|
||||||
|
print(controllerSkins)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//MARK: - UIPageViewController -
|
//MARK: - UIPageViewController -
|
||||||
|
|||||||
@ -32,6 +32,20 @@
|
|||||||
<string>com.rileytestut.delta.game.gba</string>
|
<string>com.rileytestut.delta.game.gba</string>
|
||||||
</array>
|
</array>
|
||||||
</dict>
|
</dict>
|
||||||
|
<dict>
|
||||||
|
<key>CFBundleTypeIconFiles</key>
|
||||||
|
<array/>
|
||||||
|
<key>CFBundleTypeName</key>
|
||||||
|
<string>Delta Controller Skin</string>
|
||||||
|
<key>CFBundleTypeRole</key>
|
||||||
|
<string>Viewer</string>
|
||||||
|
<key>LSHandlerRank</key>
|
||||||
|
<string>Owner</string>
|
||||||
|
<key>LSItemContentTypes</key>
|
||||||
|
<array>
|
||||||
|
<string>com.rileytestut.delta.skin</string>
|
||||||
|
</array>
|
||||||
|
</dict>
|
||||||
</array>
|
</array>
|
||||||
<key>CFBundleExecutable</key>
|
<key>CFBundleExecutable</key>
|
||||||
<string>$(EXECUTABLE_NAME)</string>
|
<string>$(EXECUTABLE_NAME)</string>
|
||||||
@ -118,6 +132,22 @@
|
|||||||
<string>gba</string>
|
<string>gba</string>
|
||||||
</dict>
|
</dict>
|
||||||
</dict>
|
</dict>
|
||||||
|
<dict>
|
||||||
|
<key>UTTypeConformsTo</key>
|
||||||
|
<array>
|
||||||
|
<string>public.data</string>
|
||||||
|
<string>public.content</string>
|
||||||
|
</array>
|
||||||
|
<key>UTTypeDescription</key>
|
||||||
|
<string>Delta Controller Skin</string>
|
||||||
|
<key>UTTypeIdentifier</key>
|
||||||
|
<string>com.rileytestut.delta.skin</string>
|
||||||
|
<key>UTTypeTagSpecification</key>
|
||||||
|
<dict>
|
||||||
|
<key>public.filename-extension</key>
|
||||||
|
<string>deltaskin</string>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
</array>
|
</array>
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user