我有一个名为Foo_C_Func的C函数,我需要将其用作回调。应用程序的其余部分是用 Swift 编写的。据我了解,以下代码应该有效,但相反,我收到编译器错误。
// typealias makes our function signature easier to read
typealias Sig = ( EventHandlerCallRef, EventRef, UnsafeMutablePointer<Void>) -> OSStatus
// We need to make a CFunctionPointer to the C function
var ptr = UnsafeMutablePointer<Sig>.alloc(1)
ptr.initialize( Foo_C_Func )
let c_ptr = COpaquePointer( ptr )
let proc_ptr = CFunctionPointer<Sig>( c_ptr ) as EventHandlerProcPtr
// now we should be able to create the EventHandlerUPP
let handler_upp = NewEventHandlerUPP( proc_ptr )
尝试构建此版本失败,并显示以下错误:
建筑x86_64的未定义符号:
"_Foo_C_Func",引用自:
__TTOFSC10Foo_C_FuncFTVSs14COpaquePointerS_GVSs20UnsafeMutablePointerT___VSs5Int32 in Demo.o
"_NewEventHandlerUPP",引用自:
演示中的__TFC17DemocfMS0_FT_S0_
LD:找不到建筑x86_64符号
我还注意到,当我将鼠标悬停在最后一行上时,Xcode 工具提示显示 NewEventHandlerUpp 的返回类型为 (EventHandlerProcPtr) 而不是 EventHandlerProcPtr。
是我做错了,还是无法在 Swift 中创建 EventHandlerUPP?
我正在为用户 MAH 发布这个,尽管我已经放弃了原始项目。以下代码在 Swift 3 中编译。我还没有检查它是否有效。
import Carbon
func hotkey_callback( _:EventHandlerCallRef?, _ event:EventRef?, _ context:UnsafeMutablePointer<Void>? ) -> OSStatus {
// Do stuff
return noErr
}
let handler_ref_ptr = UnsafeMutablePointer<EventHandlerRef?>( allocatingCapacity: 1 )
let spec_ptr = UnsafeMutablePointer<EventTypeSpec>( allocatingCapacity: 1 )
spec_ptr.pointee = EventTypeSpec( eventClass: OSType( kEventClassKeyboard ), eventKind: UInt32( kEventHotKeyPressed ) )
let hotkey_callback_pointer = hotkey_callback as! EventHandlerUPP
let status = InstallEventHandler( GetEventDispatcherTarget(), hotkey_callback_pointer, 1, spec_ptr, nil, handler_ref_ptr )
使用的是 Swift 3,这里有一个替代答案。Swift 2 与 Swift 3 的不同之处在于前者很糟糕,而后者是可用的。
let hotkey_callback: EventHandlerUPP = { _, _, _ in
// Do stuff
return noErr
}
var handler: EventHandlerRef? = nil
var spec = EventTypeSpec(
eventClass: OSType(kEventClassKeyboard), eventKind: UInt32(kEventHotKeyPressed)
)
let status = InstallEventHandler(
GetEventDispatcherTarget(), hotkey_callback, 1, &spec, nil, &handler
)
请注意,我们现在可以使用"&"来避免手动分配指针,并注意遵守非正式的"EventHandlerUPP"协议现在是显式的,不需要我们手动声明每个参数的类型。如此容易阅读:)