首頁 >後端開發 >Golang >您可以將 GTK4 applicationWindow 從 ui 檔案新增至 GTK4 應用程式嗎?

您可以將 GTK4 applicationWindow 從 ui 檔案新增至 GTK4 應用程式嗎?

WBOY
WBOY轉載
2024-02-09 19:24:10721瀏覽

您可以将 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 行:

97a58757273a657561b448c8783884df

#對此:

2474e20c2716df4ddfdb5cde8a14f2e0

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刪除