当前位置: 首页 > 面试题库 >

如何从NSApplication事件循环中启动Go的主要功能?

羊舌高爽
2023-03-14
问题内容

我正在尝试将Sparkle添加到我的Qt(绑定Go)应用程序中,以使其可以自动更新。

这是代码:https :
//github.com/sparkle-
project/Sparkle/blob/master/Sparkle/SUUIBasedUpdateDriver.m#L104

正如作者所指出的,原因是NSAlert需要运行循环才能起作用。

我找到了一些文档:

  • https://wiki.qt.io/Application_Start-up_Patterns
  • https://developer.apple.com/documentation/appkit/nsapplication

因此,据我了解,我们必须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的主要功能?


问题答案:

将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应用添加快速测试。

添加带有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()

生成并运行

为我们的日志记录消息打开loggin:

sudo log config --subsystem widget.example --mode level:debug

之后构建一个运行它:

$(go env GOPATH)/bin/qtdeploy test desktop examples/basic/widgets

测试

现在,我们可以在macOS控制台实用程序中看到计时器滴答声,证明运行循环正在运行

NSAlert

然后,您在问题中提到NSAlert需要运行循环才能工作。我们已经证明我们有一个,但是明确测试它是有道理的。

因此,我们可以修改timerrunlooptest.go来通知它,我们不仅要链接Foundation,还要链接Cocoa:

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应用程序按预期显示本机警报:

GO /
QT的本机警报

将Qt与本机代码混合

尽管我们似乎能够以上述方式显示本机警报,但是QT文档中有此提示可能有用也可能没有用:

Qt的事件分派器比Cocoa提供的灵活,并且让用户旋转事件分派器(并运行QEventLoop ::
exec),而无需考虑屏幕上是否显示模式对话框(与Cocoa相比有所不同)
。因此,我们需要在Qt中进行额外的管理才能正确处理此问题,不幸的是这使得混合本机面板变得困难。目前执行此操作的最佳方法是遵循以下模式,在该模式中,我们使用本地代码而不是直接调用将调用发布到函数中。然后我们知道Qt在显示本机面板之前已经干净地更新了所有未决的事件循环递归。

参见https://doc.qt.io/qt-5/macos-issues.html#using-native-cocoa-
panels

还有一个小代码示例。



 类似资料:
  • 现在我们已经初步了解了libevent的Reactor组件——event_base和事件管理框架,接下来就是libevent事件处理的中心部分——事件主循环,根据系统提供的事件多路分发机制执行事件循环,对已注册的就绪事件,调用注册事件的回调函数来处理事件。 1 阶段性的胜利 Libevent将I/O事件、定时器和信号事件处理很好的结合到了一起,本节也会介绍libevent是如何做到这一点的。   

  • 问题内容: 如何从单独的对象调用tkinter ? 我在寻找类似wxWidgets的东西。例如,如果我创建一个对象,并将其根实例传递给它,然后尝试从我的对象中调用该根窗口的方法,则我的应用程序将锁定。 我能想到的最好的方法是使用该方法并从单独的对象检查状态,但这似乎很浪费。 问题答案: 要回答“如何从单独的对象调用TkInter事件”这一特定问题,请使用命令。它允许您将事件注入到根窗口的事件队列中

  • 问题内容: 是否可以从Node.js代码执行本机函数,该函数输出包含有关事件循环中当前应包含的信息的数组或对象? 问题答案: 最近,有人要求查看io.js项目中事件循环的内容。在此注释中出现了两个函数名称, 让您保持活力 获取有关活动libuv请求的信息。 我想这是您可以从事件循环中收集到的最多信息。 注意: 它们都是未记录的功能,您不能在生产代码中依赖它们。

  • 我有一个具有两个实体的应用程序,它们具有双向一对多关系。车主和自行车。 因此,通过curl获得所有者将 这很好。 如果我创建一个在表中循环的模板, 然后我在浏览器中得到了预期的结果。我意识到上面的代码很糟糕,但我现在只对让thymeleaf工作感兴趣。 但是如果我执行以下代码 然后我得到以下控制台错误 嵌套的异常是org。百里香。例外情况。TemplateProcessingException:异

  • 我正在尝试创建一个脚本,该脚本将循环txt文件中列出的所有服务,检查服务启动类型是否正确(如果不正确,则更改它),并在需要时启动服务。我不太擅长Powershell,也不会真正从谷歌找到任何有用的东西。 我的文本文件: 我当前的脚本看起来是这样的,目前我能够从文本文件中打印每一个服务,但缺少下一步的信息。 困难的是每个服务都不具有相同的启动类型和状态,因此它更加复杂,例如 服务A需要手动并正在运行

  • 事件是应用程序与自身各个功能模块以及与操作系统进行通讯的手段,也是实现事件驱动编程模型的基础,应用程序如果要响应这些事件,通常是创建一个事件队列来集中存放它们,从事件队列取出事件并调用对应处理器就是一次事件响应,而往复执行这个操作的过程就是事件循环。 驱动接口 LCUI 对事件循环的操作有处理事件、绑定事件和解绑事件,驱动模块的职责就是基于操作系统接口向 LCUI 提供实现了这些操作的接口。首先我