NSPasteboard Copy Перетаскиваемый файл
Хотя у меня есть рабочий код, я не понимаю, правильно ли я отношусь к NSPasteboard
и я пытаюсь выяснить, может ли кто-нибудь помочь мне прояснить лучший подход.
Похоже, что не было полной документации, написанной Apple, о перетаскивании с 2012 года, и в это время, похоже, произошел износ и изменения (особенно в приложениях с песочницей), в которых вы должны его реализовывать. После долгих раздумий я обнаружил, что URL-адрес файла, предоставляемый перетаскиваемой панелью, показывает свой истинный путь (в отличие от анонимной строки чисел, составляющих URL), когда вы получаете доступ к свойству пути, и это позволяет делать обычную копию с использованием FileManager
,
У меня вопрос, следую ли я правильной процедуре здесь? Кастинг NSURL
полученный из картона в URL
введите и используйте его в copyItem(at:URL, to:URL)
не работает (хотя копирование и вставка одного и того же URL в Safari позволяет мне просматривать файл). Поэтому я использую вместо copyItem(atPath:String, toPath:String)
, Для дальнейшей работы над этим пунктом есть метод для URL
тип, который заменяет NSURL(from:NSPasteboard)
? Кажется, должно быть.
В любом случае, вот код, который у меня работает (пожалуйста, не обращайте внимания на код внутри draggingEntered: сейчас он просто работает для тестирования и тот факт, что я зарегистрировался для NSFilenamesPboardType
но затем игнорировать его также можно не заметить).
import Cocoa
class DraggerView: NSView {
let types = [NSURLPboardType, NSFilenamesPboardType]
var directory:URL!
override init(frame: NSRect) {
super.init(frame: frame)
register(forDraggedTypes: types)
if let dir = URL(string:NSTemporaryDirectory())
{
directory = dir
}
}
required init?(coder: NSCoder) {
super.init(coder: coder)
register(forDraggedTypes: types)
if let dir = URL(string:NSTemporaryDirectory())
{
directory = dir
}
}
override func draw(_ dirtyRect: NSRect) {
super.draw(dirtyRect)
NSColor.white.set()
NSRectFill(dirtyRect)
}
override func draggingEntered(_ sender: NSDraggingInfo) -> NSDragOperation {
return NSDragOperation.copy
}
override func performDragOperation(_ sender: NSDraggingInfo) -> Bool {
let pboard = sender.draggingPasteboard()
if let types = pboard.types, types.contains(NSURLPboardType) {
// add file name to save to loaction
if let fileURL = NSURL(from:pboard), let filename = fileURL.lastPathComponent {
directory.appendPathComponent(filename)
do {
try FileManager.default.copyItem(atPath:fileURL.path!, toPath:directory.path)
}
catch {
// something went wrong
}
}
return true
}
else { return false }
}
}
И последнее замечание: документы продолжают ссылаться на использование UTI при регистрации для перетаскиваемых типов, но мне не ясно, насколько это далеко. Должен ли я заменить NSURLPboardType
а также NSFilenamesPboardType
с ИМП?
1 ответ
Следуя советам на форумах разработчиков Apple, я смог отказаться от большого количества кода и реализовать readObjects:
метод. Это значительно упрощает вещи.
Теперь у меня работает следующий код (с использованием предпочтительного метода URL-адресов, а не путей).
import Cocoa
class DraggerView: NSView {
override init(frame: NSRect) {
super.init(frame: frame)
self.register(forDraggedTypes: [NSURLPboardType])
}
required init?(coder: NSCoder) {
super.init(coder: coder)
self.register(forDraggedTypes: [NSURLPboardType])
}
override func draw(_ dirtyRect: NSRect) {
super.draw(dirtyRect)
NSColor.white.set()
NSRectFill(dirtyRect)
}
override func draggingEntered(_ sender: NSDraggingInfo) -> NSDragOperation {
return NSDragOperation.copy
}
override func performDragOperation(_ sender: NSDraggingInfo) -> Bool {
let pboard = sender.draggingPasteboard()
if let urls = pboard.readObjects(forClasses: [NSURL.self], options: [:]) as? [URL] {
for url in urls {
do {
var directory = URL(fileURLWithPath:NSTemporaryDirectory())
directory.appendPathComponent(url.lastPathComponent)
try FileManager.default.copyItem(at:url, to:directory)
}
catch {
// something went wrong
}
}
}
return true
}
}
Примечание. Приведенный выше код копирует любые перетаскиваемые файлы или папки (включая файлы и подпапки) во временный каталог вашего приложения.