首页 > 后端开发 > Golang > 您可以将 GTK4 applicationWindow 从 ui 文件添加到 GTK4 应用程序吗?

您可以将 GTK4 applicationWindow 从 ui 文件添加到 GTK4 应用程序吗?

WBOY
发布: 2024-02-09 19:24:10
转载
727 人浏览过

您可以将 GTK4 applicationWindow 从 ui 文件添加到 GTK4 应用程序吗?

php小编子墨很高兴为大家解答一个常见问题:“您可以将GTK4 applicationWindow从ui文件添加到GTK4应用程序吗?”答案是肯定的!在GTK4中,您可以使用GtkBuilder来加载UI文件,并将其中的applicationWindow部件添加到您的GTK应用程序中。这样,您可以更方便地设计和管理应用程序的界面,提高开发效率。不过,请注意,要成功加载UI文件,您需要确保在您的系统中安装了GTK4的相关库和工具。希望这个回答对您有所帮助!

问题内容

我才使用 GTK 几天,因为我的 go 程序需要一个 GUI,所以这是一个新手问题。作为新手,我采用了我能找到的最简单的方法来开始使用 Cambalache 生成一个 .UI 文件,然后在我的 go 应用程序中调用它。

我将所有内容放在 .UI 文件中的 ApplicationWindow 下,并让它加载并与 gkt4-builder-tool 一起正常工作,但在我的 go 应用程序中我没有GUI 中的事件。查看示例代码,我发现我需要将 ApplicationWindow 附加到在我的 go 程序中实例化的 GTK 应用程序,并且我通过更改 >ApplicationWindow 只是一个窗口,它工作得很好,因为我可以向 GTK 应用程序添加一个普通窗口。

问题是我无法找到从 UI 文件中提取 ApplicationWindow 后将其附加到应用程序的方法。将窗口附加到应用程序的唯一方法是 app.AddWindow() 但它仅添加 GtkWindow,而不是 GtkApplicationWindow 并且采用强类型语言,例如go 是“no go”(没有双关语)。

您可以将 GTK 应用程序与 ApplicationWindow 一起直接导出到 .UI 文件中,理论上,如果您能成功完成此操作,您可以将它们都提取出来,也许可以通过这种方式将它们连接起来,但是,我尚未成功导出 GTK 应用程序、ApplicationWindow 和菜单,因为 gkt4-builder-tool 总是出现验证错误(在尝试多种配置后),因此 Cambalache 不知道如何导出该组合;或者我不知道如何让它这样做;或者它实际上并不是 .UI 文件中的合法组合。

所以我的问题是,我是否应该放弃尝试在 .UI 文件中存储 ApplicationWindow 并仅在代码中构建 ApplicationWindow 小部件,或者还有其他选择吗?还没学会?

如果不是真的有必要,我可以放弃 ApplicationWindow 并使用 Windows 作为另一条路径。

感谢您的知识和经验。

谢谢!

更新:

根据 Kripto 的评论更加清楚。

这是代码。这不会直接运行,因为我从一个更大的程序中提取了 main() 中的相关片段,并且 env, log := boot.Initialize() 在此片段中不存在,但理解问题并不重要.

我知道 GTK GUI 中只有一个应用程序窗口。这个概念即使不完全相同,也类似于我以前的 Visual Basic 时代的应用程序窗口

下面是go代码以及包含.UI文件中相关元素的精简版本:

如果我更改 .UI 文件中的第 8 行:

<object class="GtkApplicationWindow" id="appWin">

对此:

<object class="GtkWindow" id="appWin">

Go 代码 将打开 .UI 文件并且它可以工作,但现在我没有应用程序窗口。

如果我按原样保留 .UI 文件中的第 8 行,那么它仍然会打开该文件,但生成的 GUI 没有交互,并且仅当我停止我正在运行的 Go 程序时它才会关闭在 GoLand 中以调试模式运行。这是因为要运行它,我必须对 go 程序进行两处更改。:

  1. 更改此行:

appWindow := builder.GetObject(appWin).Cast().(*gtk.Window)

appWindow := builder.GetObject(appWin).Cast().(*gtk.ApplicationWindow)

  1. 注释掉这一行:

app.AddWindow(appWindow)

第二个是问题,因为现在窗口没有附加到应用程序,我确信这就是它非交互式的原因。

package main

import (
    "os"

    "github.com/diamondburned/gotk4/pkg/gio/v2"
    "github.com/diamondburned/gotk4/pkg/gtk/v4"

    "gitlab.com/trading5124936/core.git/loggers"
)

func main() {
    env, log := boot.Initialize()
    var app *gtk.Application
    app = gtk.NewApplication(`site.TradingAnalyzer`, gio.ApplicationFlagsNone)
    app.ConnectActivate(func() { activate(app, log, env.Paths.GUIFile) })

    if code := app.Run(os.Args); code > 0 {
        os.Exit(code)
    }

}

func activate(app *gtk.Application, log *loggers.Logger, guiPath string) {
    // You can build UIs using Cambalache (https://flathub.org/apps/details/ar.xjuan.Cambalache)

    b, err := os.ReadFile(guiPath)
    if err != nil {
        log.Critical(err)
        return
    }
    uiXML := string(b)

    builder := gtk.NewBuilderFromString(uiXML, len(uiXML))

    // MainWindow and Button are object IDs from the UI file
    appWindow := builder.GetObject(`appWin`).Cast().(*gtk.Window)

    entry := builder.GetObject(`GeneralSetup.Timezone`).Cast().(*gtk.Entry)
    entry.Connect("changed", func() {
        println(`Changed`)
    })

    app.AddWindow(appWindow)
    appWindow.Show()

}
登录后复制

这是 .UI 文件:

<?xml version='1.0' encoding='UTF-8'?>
<!-- Created with Cambalache 0.16.0 -->
<interface domain="ta.site">
  <!-- interface-name TradingAnalyzer.ui -->
  <!-- interface-authors Reg Proctor -->
  <requires lib="gtk" version="4.6"/>
  <object class="GtkApplication" id="app"/>
  <object class="GtkApplicationWindow" id="appWin">
    <property name="default-height">925</property>
    <property name="default-width">1200</property>
    <child>
      <object class="GtkPaned">
        <child>
          <object class="GtkFrame">
            <property name="label">Settings</property>
            <child>
              <object class="GtkStackSidebar">
                <property name="halign">start</property>
                <property name="height-request">900</property>
                <property name="stack">pages</property>
                <property name="valign">start</property>
              </object>
            </child>
          </object>
        </child>
        <child>
          <object class="GtkFrame">
            <child>
              <object class="GtkStack" id="pages">
                <property name="name">Timezone</property>
                <child>
                  <object class="GtkStackPage" id="GeneralSetup">
                    <property name="child">
                      <object class="GtkFlowBox">
                        <property name="margin-bottom">20</property>
                        <property name="margin-end">50</property>
                        <property name="margin-start">50</property>
                        <property name="margin-top">20</property>
                        <property name="name">Timezone</property>
                        <child>
                          <object class="GtkEntry" id="GeneralSetup.Timezone">
                            <property name="activates-default">True</property>
                            <property name="halign">start</property>
                            <property name="height-request">10</property>
                            <property name="input-purpose">alpha</property>
                            <property name="placeholder-text">Timezone</property>
                            <property name="text">America/Phoenix</property>
                            <property name="tooltip-text">Enter your timezone</property>
                            <property name="valign">center</property>
                            <property name="width-request">50</property>
                          </object>
                        </child>
                      </object>
                    </property>
                    <property name="name">General Setup</property>
                    <property name="title">General Setup</property>
                  </object>
                </child>
              </object>
            </child>
          </object>
        </child>
      </object>
    </child>
  </object>
</interface>
登录后复制

更新2

我找到了部分答案。您可以做到这一点,但您需要将 App 添加到 ApplicationWindow 中,而不是相反,因此这是一个不同的功能。这是它的工作方式(它也有点短,因为我了解到有一个从文件加载的函数):

func activate(app *gtk.Application, log *loggers.Logger, guiPath string, appName string) {

    const ext = `.ui`

    guiPath += `/`
    builder := gtk.NewBuilderFromFile(guiPath + appName + ext)
    // builder := gtk.NewBuilderFromFile(guiPath + `Another template File.ui`)

    appWindow := builder.GetObject(`appWin`).Cast().(*gtk.ApplicationWindow)

    entry := builder.GetObject(`GeneralSetup.Timezone`).Cast().(*gtk.Entry)
    entry.Connect("changed", func() {
        println(`Changed`)
    })

    // ***** THE LINE THAT MAKES THE DIFFERENCE ***
    appWindow.SetApplication(app)
    appWindow.Show()

}

登录后复制

我仍然不确定是否存在将应用程序对象导出到这些 .UI 文件之一的情况。

我还没有找到任何人这样做的例子,我倾向于相信这不是你应该做的,但我仍在学习,所以很容易出错。

解决方法

更新 2 提供了此问题的大部分答案。

此外,到目前为止,我认为没有任何理由将应用程序添加到 UI 文件中。无论如何,它往往会抛出错误。

以上是您可以将 GTK4 applicationWindow 从 ui 文件添加到 GTK4 应用程序吗?的详细内容。更多信息请关注PHP中文网其他相关文章!

来源:stackoverflow.com
本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板