我正在尝试将Sparkle添加到我的Qt(binding for Go)应用程序中,以使其可以自动更新。
问题:there is no popup dialog when running the latest version
代码如下:https://github.com/sparkle-project/Sparkle/blob/master/Sparkle/SUUIBasedUpdateDriver.m#L104
作者pointed out是NSAlert
的原因需要运行循环。
我找到了一些文档:
据我所知,我们必须在创建NSApplication
之前实例化QApplication
。
void NSApplicationMain(int argc, char *argv[]) {
[NSApplication sharedApplication];
[NSBundle loadNibNamed:@"myMain" owner:NSApp];
[NSApp run];
}
My Go的主要功能如下:
func main() {
widgets.NewQApplication(len(os.Args), os.Args)
...
action := widgets.NewQMenuBar(nil).AddMenu2("").AddAction("Check for Updates...")
// http://doc.qt.io/qt-5/qaction.html#MenuRole-enum
action.SetMenuRole(widgets.QAction__ApplicationSpecificRole)
action.ConnectTriggered(func(bool) { sparkle_checkUpdates() })
...
widgets.QApplication_Exec()
}
问题:如何在NSApplicationMain
事件循环中启动Go的主要功能?
答案 0 :(得分:2)
将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需要运行循环才能工作。因此,我们可以使用问题中引用的存储库中现有的示例Qt应用程序“ widget”添加快速测试。
添加带有C函数包装器的小型Objective-C测试类TimerRunloopTest,可以从GO中调用它:
#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
转到对应的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
然后您在问题中提到,NSAlert需要运行循环才能工作。我们已经证明我们有一个,但是明确测试它是有道理的。
因此,我们可以修改timerrunlooptest.go以通知它,我们不仅要链接基金会,还要链接可可。
package main
/*
#cgo LDFLAGS: -framework Foundation
#cgo LDFLAGS: -framework Cocoa
void runTimerRunloopTest();
*/
import "C"
func runTimerRunloopTest() { C.runTimerRunloopTest() }
然后,我们可以将以下代码添加到TimerRunLoopTest的运行方法中:
#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),而无需考虑屏幕上是否显示模式对话框(相比之下,有所不同)到可可)。因此,我们需要在Qt中进行额外的管理才能正确处理此问题,不幸的是,这使得混合本机面板变得困难。目前执行此操作的最佳方法是遵循以下模式,在该模式中,我们使用本机代码而不是直接调用函数来发布对函数的调用。然后我们知道Qt在显示本机面板之前已经干净地更新了所有未决的事件循环递归。
请参阅https://doc.qt.io/qt-5/macos-issues.html#using-native-cocoa-panels
还有一个小代码示例。