我正在为OS X 10.9开发一个带有快速,沙盒的应用程序。
该应用需要访问SQLite数据库文件。我让用户使用NSOpenPanel选择/打开文件。然后,我使用NSUserDefaults保存文件路径,以备后用。
我希望每次启动应用程序时自动打开此文件。我从NSUserDefault获取存储的路径,但是当我使用该路径打开文件时,出现错误,提示我无权访问该文件。
(它可以在没有沙箱的情况下工作)
看来书签是我的问题的解决方案。
是否有一个很好的教程,如何在OSX应用程序中快速使用书签?
还有其他建议吗?
最佳答案
这是我的答案,我刚刚在http://swiftrien.blogspot.com/2015/07/persisting-file-access-rights-between.html的帮助下在Swift 3中工作
import Foundation
import Cocoa
var bookmarks = [URL: Data]()
func bookmarkPath() -> String
{
var url = app.applicationDocumentsDirectory
url = url.appendingPathComponent("Bookmarks.dict")
return url.path
}
func loadBookmarks()
{
let path = bookmarkPath()
bookmarks = NSKeyedUnarchiver.unarchiveObject(withFile: path) as! [URL: Data]
for bookmark in bookmarks
{
restoreBookmark(bookmark)
}
}
func saveBookmarks()
{
let path = bookmarkPath()
NSKeyedArchiver.archiveRootObject(bookmarks, toFile: path)
}
func storeBookmark(url: URL)
{
do
{
let data = try url.bookmarkData(options: NSURL.BookmarkCreationOptions.withSecurityScope, includingResourceValuesForKeys: nil, relativeTo: nil)
bookmarks[url] = data
}
catch
{
Swift.print ("Error storing bookmarks")
}
}
func restoreBookmark(_ bookmark: (key: URL, value: Data))
{
let restoredUrl: URL?
var isStale = false
Swift.print ("Restoring \(bookmark.key)")
do
{
restoredUrl = try URL.init(resolvingBookmarkData: bookmark.value, options: NSURL.BookmarkResolutionOptions.withSecurityScope, relativeTo: nil, bookmarkDataIsStale: &isStale)
}
catch
{
Swift.print ("Error restoring bookmarks")
restoredUrl = nil
}
if let url = restoredUrl
{
if isStale
{
Swift.print ("URL is stale")
}
else
{
if !url.startAccessingSecurityScopedResource()
{
Swift.print ("Couldn't access: \(url.path)")
}
}
}
}
func allowFolder() -> URL?
{
let openPanel = NSOpenPanel()
openPanel.allowsMultipleSelection = false
openPanel.canChooseDirectories = true
openPanel.canCreateDirectories = true
openPanel.canChooseFiles = false
openPanel.begin
{ (result) -> Void in
if result == NSFileHandlingPanelOKButton
{
let url = openPanel.url
storeBookmark(url: url!)
}
}
return openPanel.url
}
Swift 4(更新):
import Foundation
import Cocoa
var bookmarks = [URL: Data]()
func fileExists(_ url: URL) -> Bool
{
var isDir = ObjCBool(false)
let exists = FileManager.default.fileExists(atPath: url.path, isDirectory: &isDir)
return exists
}
func bookmarkURL() -> URL
{
let urls = FileManager.default.urls(for: .applicationSupportDirectory, in: .userDomainMask)
let appSupportURL = urls[urls.count - 1]
let url = appSupportURL.appendingPathComponent("Bookmarks.dict")
return url
}
func loadBookmarks()
{
let url = bookmarkURL()
if fileExists(url)
{
do
{
let fileData = try Data(contentsOf: url)
if let fileBookmarks = try NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(fileData) as! [URL: Data]?
{
bookmarks = fileBookmarks
for bookmark in bookmarks
{
restoreBookmark(bookmark)
}
}
}
catch
{
print ("Couldn't load bookmarks")
}
}
}
func saveBookmarks()
{
let url = bookmarkURL()
do
{
let data = try NSKeyedArchiver.archivedData(withRootObject: bookmarks, requiringSecureCoding: false)
try data.write(to: url)
}
catch
{
print("Couldn't save bookmarks")
}
}
func storeBookmark(url: URL)
{
do
{
let data = try url.bookmarkData(options: NSURL.BookmarkCreationOptions.withSecurityScope, includingResourceValuesForKeys: nil, relativeTo: nil)
bookmarks[url] = data
}
catch
{
Swift.print ("Error storing bookmarks")
}
}
func restoreBookmark(_ bookmark: (key: URL, value: Data))
{
let restoredUrl: URL?
var isStale = false
Swift.print ("Restoring \(bookmark.key)")
do
{
restoredUrl = try URL.init(resolvingBookmarkData: bookmark.value, options: NSURL.BookmarkResolutionOptions.withSecurityScope, relativeTo: nil, bookmarkDataIsStale: &isStale)
}
catch
{
Swift.print ("Error restoring bookmarks")
restoredUrl = nil
}
if let url = restoredUrl
{
if isStale
{
Swift.print ("URL is stale")
}
else
{
if !url.startAccessingSecurityScopedResource()
{
Swift.print ("Couldn't access: \(url.path)")
}
}
}
}
func allowFolder() -> URL?
{
let openPanel = NSOpenPanel()
openPanel.allowsMultipleSelection = false
openPanel.canChooseDirectories = true
openPanel.canCreateDirectories = true
openPanel.canChooseFiles = false
openPanel.begin
{ (result) -> Void in
if result == NSFileHandlingPanelOKButton
{
let url = openPanel.url
storeBookmark(url: url!)
}
}
return openPanel.url
}
要使用此代码,您必须首先调用NSOpenPanel,以便用户可以选择要授予您访问权限的文件夹。 NSOpenPanel必须作为书签存储并保存到磁盘。
let url = allowFolder()
saveBookmarks()
重新启动应用程序时,您必须调用
loadBookmarks()
那么您的应用将具有与用户选择文件夹相同的访问级别。希望这对某人有帮助。