有没有人能够限制在 SwiftUI 3 中 mac 上放置的对象的类型?
is anyone able to restrict the type of the objects dropped on the mac in SwiftUI 3?
根据文档,它应该非常简单。 List
的示例:https://developer.apple.com/documentation/swiftui/list/ondrop(of:istargeted:perform:)-75hvy#
UTType
应该是限制 SwiftUI 对象可以接收的参数。就我而言,我只想接受 Apps
。 UTType
是 .applicationBundle
:https://developer.apple.com/documentation/uniformtypeidentifiers/uttype/3551459-applicationbundle
但它不起作用。 SwiftUI 对象永远不会改变状态,也永远不会接受放置。闭包永远不会 运行。无论是 Lists
、H/VStacks
、Buttons
还是什么。 pdf
类型似乎也不起作用,许多其他类型也是如此。如果 fileURL
,我唯一可以使用的类型,主要是没有限制。
我不确定我是否做错了什么,或者 SwiftUI 是否为 mac.
工作了一半
代码如下:
List(appsToIgnore, id: \.self, selection: $selection) {
Text([=11=])
}
.onDrop(of: [.applicationBundle, .application], isTargeted: isTargeted) { providers in
print("hehe")
return true
}
在 UTType
数组中替换或仅添加 .fileURL
可以使放置工作但没有任何类型限制。
我也曾尝试在 ForEach
上使用 .onInsert
而不是 (https://developer.apple.com/documentation/swiftui/foreach/oninsert(of:perform:)-2whxl#), and to go through a proper DropDelegate (https://developer.apple.com/documentation/swiftui/dropdelegate#),但一直得到相同的结果。 macOS 的 SwiftUI drop 似乎还没有工作,但我找不到关于此的任何官方信息。在文档中它被写成 macOS 11.0+
所以我希望它能工作?
感谢任何信息!谢谢。
需要手动验证,使用DropDelegate
拖过来的是什么类型的文件。
这是可能方法的简化演示。使用 Xcode 13 / macOS 11.6
进行测试
let delegate = MyDelegate()
...
List(appsToIgnore, id: \.self, selection: $selection) {
Text([=10=])
}
.onDrop(of: [.fileURL], delegate: delegate) // << accept file URLs
和验证部分如
class MyDelegate: DropDelegate {
func validateDrop(info: DropInfo) -> Bool {
// find provider with file URL
guard info.hasItemsConforming(to: [.fileURL]) else { return false }
guard let provider = info.itemProviders(for: [.fileURL]).first else { return false }
var result = false
if provider.canLoadObject(ofClass: String.self) {
let group = DispatchGroup()
group.enter() // << make decoding sync
// decode URL from item provider
_ = provider.loadObject(ofClass: String.self) { value, _ in
defer { group.leave() }
guard let fileURL = value, let url = URL(string: fileURL) else { return }
// verify type of content by URL
let flag = try? url.resourceValues(forKeys: [.contentTypeKey]).contentType == .applicationBundle
result = flag ?? false
}
// wait a bit for verification result
_ = group.wait(timeout: .now() + 0.5)
}
return result
}
func performDrop(info: DropInfo) -> Bool {
// handling code is here
return true
}
}
根据文档,它应该非常简单。 List
的示例:https://developer.apple.com/documentation/swiftui/list/ondrop(of:istargeted:perform:)-75hvy#
UTType
应该是限制 SwiftUI 对象可以接收的参数。就我而言,我只想接受 Apps
。 UTType
是 .applicationBundle
:https://developer.apple.com/documentation/uniformtypeidentifiers/uttype/3551459-applicationbundle
但它不起作用。 SwiftUI 对象永远不会改变状态,也永远不会接受放置。闭包永远不会 运行。无论是 Lists
、H/VStacks
、Buttons
还是什么。 pdf
类型似乎也不起作用,许多其他类型也是如此。如果 fileURL
,我唯一可以使用的类型,主要是没有限制。
我不确定我是否做错了什么,或者 SwiftUI 是否为 mac.
工作了一半代码如下:
List(appsToIgnore, id: \.self, selection: $selection) {
Text([=11=])
}
.onDrop(of: [.applicationBundle, .application], isTargeted: isTargeted) { providers in
print("hehe")
return true
}
在 UTType
数组中替换或仅添加 .fileURL
可以使放置工作但没有任何类型限制。
我也曾尝试在 ForEach
上使用 .onInsert
而不是 (https://developer.apple.com/documentation/swiftui/foreach/oninsert(of:perform:)-2whxl#), and to go through a proper DropDelegate (https://developer.apple.com/documentation/swiftui/dropdelegate#),但一直得到相同的结果。 macOS 的 SwiftUI drop 似乎还没有工作,但我找不到关于此的任何官方信息。在文档中它被写成 macOS 11.0+
所以我希望它能工作?
感谢任何信息!谢谢。
需要手动验证,使用DropDelegate
拖过来的是什么类型的文件。
这是可能方法的简化演示。使用 Xcode 13 / macOS 11.6
进行测试let delegate = MyDelegate()
...
List(appsToIgnore, id: \.self, selection: $selection) {
Text([=10=])
}
.onDrop(of: [.fileURL], delegate: delegate) // << accept file URLs
和验证部分如
class MyDelegate: DropDelegate {
func validateDrop(info: DropInfo) -> Bool {
// find provider with file URL
guard info.hasItemsConforming(to: [.fileURL]) else { return false }
guard let provider = info.itemProviders(for: [.fileURL]).first else { return false }
var result = false
if provider.canLoadObject(ofClass: String.self) {
let group = DispatchGroup()
group.enter() // << make decoding sync
// decode URL from item provider
_ = provider.loadObject(ofClass: String.self) { value, _ in
defer { group.leave() }
guard let fileURL = value, let url = URL(string: fileURL) else { return }
// verify type of content by URL
let flag = try? url.resourceValues(forKeys: [.contentTypeKey]).contentType == .applicationBundle
result = flag ?? false
}
// wait a bit for verification result
_ = group.wait(timeout: .now() + 0.5)
}
return result
}
func performDrop(info: DropInfo) -> Bool {
// handling code is here
return true
}
}