mirror of
https://github.com/ghostty-org/ghostty.git
synced 2025-12-30 18:22:12 +00:00
macos: convert the transferable to a nsdraggingitem
This commit is contained in:
@@ -168,6 +168,7 @@
|
||||
"Helpers/Extensions/NSView+Extension.swift",
|
||||
"Helpers/Extensions/NSWindow+Extension.swift",
|
||||
"Helpers/Extensions/NSWorkspace+Extension.swift",
|
||||
"Helpers/Extensions/Transferable+Extension.swift",
|
||||
"Helpers/Extensions/UndoManager+Extension.swift",
|
||||
"Helpers/Extensions/View+Extension.swift",
|
||||
Helpers/Fullscreen.swift,
|
||||
|
||||
@@ -129,10 +129,8 @@ extension Ghostty {
|
||||
override func mouseDragged(with event: NSEvent) {
|
||||
guard !isTracking, let surfaceView = surfaceView else { return }
|
||||
|
||||
// Create the pasteboard item with the surface ID
|
||||
let data = withUnsafeBytes(of: surfaceView.id.uuid) { Data($0) }
|
||||
let pasteboardItem = NSPasteboardItem()
|
||||
pasteboardItem.setData(data, forType: .ghosttySurfaceId)
|
||||
// Create our dragging item from our transferable
|
||||
guard let pasteboardItem = surfaceView.pasteboardItem() else { return }
|
||||
let item = NSDraggingItem(pasteboardWriter: pasteboardItem)
|
||||
|
||||
// Create a scaled preview image from the surface snapshot
|
||||
|
||||
@@ -0,0 +1,58 @@
|
||||
import AppKit
|
||||
import CoreTransferable
|
||||
import UniformTypeIdentifiers
|
||||
|
||||
extension Transferable {
|
||||
/// Converts this Transferable to an NSPasteboardItem with lazy data loading.
|
||||
/// Data is only fetched when the pasteboard consumer requests it. This allows
|
||||
/// bridging a Transferable to NSDraggingSource.
|
||||
func pasteboardItem() -> NSPasteboardItem? {
|
||||
let itemProvider = NSItemProvider()
|
||||
itemProvider.register(self)
|
||||
|
||||
let types = itemProvider.registeredTypeIdentifiers.compactMap { UTType($0) }
|
||||
guard !types.isEmpty else { return nil }
|
||||
|
||||
let item = NSPasteboardItem()
|
||||
let dataProvider = TransferableDataProvider(itemProvider: itemProvider)
|
||||
let pasteboardTypes = types.map { NSPasteboard.PasteboardType($0.identifier) }
|
||||
item.setDataProvider(dataProvider, forTypes: pasteboardTypes)
|
||||
|
||||
return item
|
||||
}
|
||||
}
|
||||
|
||||
private final class TransferableDataProvider: NSObject, NSPasteboardItemDataProvider {
|
||||
private let itemProvider: NSItemProvider
|
||||
|
||||
init(itemProvider: NSItemProvider) {
|
||||
self.itemProvider = itemProvider
|
||||
super.init()
|
||||
}
|
||||
|
||||
func pasteboard(
|
||||
_ pasteboard: NSPasteboard?,
|
||||
item: NSPasteboardItem,
|
||||
provideDataForType type: NSPasteboard.PasteboardType
|
||||
) {
|
||||
// NSPasteboardItemDataProvider requires synchronous data return, but
|
||||
// NSItemProvider.loadDataRepresentation is async. We use a semaphore
|
||||
// to block until the async load completes. This is safe because AppKit
|
||||
// calls this method on a background thread during drag operations.
|
||||
let semaphore = DispatchSemaphore(value: 0)
|
||||
|
||||
var result: Data?
|
||||
itemProvider.loadDataRepresentation(forTypeIdentifier: type.rawValue) { data, _ in
|
||||
result = data
|
||||
semaphore.signal()
|
||||
}
|
||||
|
||||
// Wait for the data to load
|
||||
semaphore.wait()
|
||||
|
||||
// Set it. I honestly don't know what happens here if this fails.
|
||||
if let data = result {
|
||||
item.setData(data, forType: type)
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user