将 QApplication 与 Runloop 一起使用
关于您如何将 QApplication 与 NSRunloop 一起使用的问题:您已经在这样做了。由于您使用的是 QApplication(而不是 QCoreApplication),因此您已经运行了一个 Runloop,
见http://code.qt.io/cgit/qt/qt.git/plain/src/gui/kernel/qeventdispatcher_mac.mm和 http://code.qt.io/cgit/qt/qt.git/plain /src/plugins/platforms/cocoa/qcocoaeventloopintegration.mm
证明
NSTimer 需要一个运行循环才能工作。因此,我们可以使用您在问题中引用的存储库中名为“widget”的现有示例 Qt 应用程序添加快速测试。
添加一个小的 Objective-C 测试类 TimerRunloopTest 和一个可以从 GO 调用的 C 函数包装器:
#import <Foundation/Foundation.h>
#include <os/log.h>
@interface TimerRunloopTest : NSObject
- (void)run;
@end
void runTimerRunloopTest() {
[[TimerRunloopTest new] run];
}
@implementation TimerRunloopTest
- (void)run {
os_log_t log = os_log_create("widget.example", "RunloopTest");
os_log(log, "setup happening at %f", NSDate.timeIntervalSinceReferenceDate);
[NSTimer scheduledTimerWithTimeInterval:1.0
target:self
selector:@selector(timerTick:)
userInfo:nil
repeats:YES];
}
- (void)timerTick:(NSTimer *)timer {
os_log_t log = os_log_create("widget.example", "RunloopTest");
os_log(log, "timer tick %f", NSDate.timeIntervalSinceReferenceDate);
}
@end
GO 对应 timerrunlooptest.go
package main
/*
#cgo LDFLAGS: -framework Foundation
void runTimerRunloopTest();
*/
import "C"
func runTimerRunloopTest() { C.runTimerRunloopTest() }
main.go 中的更改
在 app.Exec() 之前的最后添加这一行:
runTimerRunloopTest()
构建并运行它
为我们的日志消息打开登录:
sudo log config --subsystem widget.example --mode level:debug
然后构建运行它:
$(go env GOPATH)/bin/qtdeploy test desktop examples/basic/widgets
测试
在 macOS 控制台实用程序中,我们现在可以看到,显示了计时器滴答声,证明运行循环正在运行
安全警报
然后你在你的问题中引用了 NSAlert 需要一个运行循环才能工作。我们已经证明我们有一个,但是明确地测试它是有意义的。
所以我们可以修改 timerrunlooptest.go 来通知它,我们也想重新链接 Cocoa,而不仅仅是 Foundation:
package main
/*
#cgo LDFLAGS: -framework Foundation
#cgo LDFLAGS: -framework Cocoa
void runTimerRunloopTest();
*/
import "C"
func runTimerRunloopTest() { C.runTimerRunloopTest() }
然后我们可以在 TimerRunLoopTest 的 run 方法中添加如下代码:
#import <Cocoa/Cocoa.h>
...
NSAlert *alert = [[NSAlert alloc] init];
alert.messageText = @"Message";
alert.informativeText = @"Info";
[alert addButtonWithTitle:@"OK"];
[alert runModal];
结果
做了一个之后
$(go env GOPATH)/bin/qtdeploy test desktop examples/basic/widgets
本机警报按预期从 GO/QT 应用程序中显示:

将 Qt 与本机代码混合
尽管我们似乎能够以上述方式显示本机警报,但 QT 文档中有这个提示可能有用也可能没有用:
Qt 的事件分派器比 Cocoa 提供的更灵活,并且允许用户旋转事件分派器(并运行 QEventLoop::exec),而无需考虑屏幕上是否显示模式对话框(这与 Cocoa 相比有所不同) . 因此,我们需要在 Qt 中进行额外的管理以正确处理此问题,不幸的是,这使得混合原生面板变得困难。目前最好的方法是遵循下面的模式,我们使用本机代码发布对函数的调用,而不是直接调用它。然后我们知道 Qt 在显示本机面板之前已经干净地更新了任何未决的事件循环递归。
见https://doc.qt.io/qt-5/macos-issues.html#using-native-cocoa-panels
还有一个小代码示例。