将可编辑的Gtk.TreeView与其ListStore同步

时间:2018-06-09 16:00:24

标签: python gtk glade

我正在使用Glade在Gtk3 / Python3中构建一个接口,在TreeView中显示来自ListStore的数据,并允许用户在其中一列中编辑值。

如上所述,我正在使用林间空地,并且在林间空地文件中写了ListStore列为每个GtkCellRenderer提供哪些数据。第一个有趣的事情是我不知道如何获取这些信息,我正在引入冗余以保持工作。

接下来更有趣的问题:当编辑的单元格失去焦点但焦点停留在TreeView中时,我会调用edited信号的回调,并且可以将编辑后的值存储在ListStore中。

当编辑的单元格失去焦点,并且焦点转移到另一个小部件时(如示例中的更无用的按钮),不会调用已注册的回调。此外,如果窗口失去焦点,则恢复先前的值。最后一个在“焦点跟随鼠标”的情况下特别不实用。

这笔交易是什么?

import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
from gi.repository import Gdk


class TestWindow:

    def __init__(self, builder):
        import os.path
        path, name = os.path.split(__file__)
        builder.add_from_file(os.path.join(path, "treeedit.glade"))
        builder.connect_signals(self)
        self.window = builder.get_object("window1")
        self.window.connect("destroy", Gtk.main_quit)
        self.notes_list = builder.get_object("notes_list")
        self.notes_list.append(('a',))
        self.notes_list.append(('b',))
        self.notes_list.append(('c',))
        self.notes_list.append(('d',))

    def on_cell_edited(self, widget, path, text):
        self.notes_list[path][0] = text

    def show_all(self):
        self.window.show_all()


builder = Gtk.Builder()
win = TestWindow(builder)
win.show_all()
Gtk.main()

这是林间空地文件:

<?xml version="1.0" encoding="UTF-8"?>
<!-- Generated with glade 3.20.0 -->
<interface>
  <requires lib="gtk+" version="3.20"/>
  <object class="GtkListStore" id="notes_list">
    <columns>
      <!-- column-name content -->
      <column type="gchararray"/>
    </columns>
  </object>
  <object class="GtkDialog" id="window1">
    <property name="can_focus">False</property>
    <property name="type_hint">dialog</property>
    <child internal-child="vbox">
      <object class="GtkBox">
        <property name="orientation">vertical</property>
        <property name="spacing">2</property>
        <child>
          <object class="GtkButton">
            <property name="label" translatable="yes">button</property>
          </object>
        </child>
        <child>
          <object class="GtkFrame">
            <property name="visible">True</property>
            <child>
              <object class="GtkAlignment">
                <property name="visible">True</property>
                <property name="can_focus">False</property>
                <property name="left_padding">12</property>
                <child>
                  <object class="GtkTreeView" id="notes_tree">
                    <property name="visible">True</property>
                    <property name="can_focus">True</property>
                    <property name="hexpand">True</property>
                    <property name="vexpand">True</property>
                    <property name="model">notes_list</property>
                    <property name="headers_clickable">False</property>
                    <child internal-child="selection">
                      <object class="GtkTreeSelection"/>
                    </child>
                    <child>
                      <object class="GtkTreeViewColumn">
                        <property name="resizable">True</property>
                        <property name="title" translatable="yes">value</property>
                        <child>
                          <object class="GtkCellRendererText" id="prefs_cellrenderertext1">
                            <property name="editable">True</property>
                            <signal name="edited" handler="on_cell_edited" swapped="no"/>
                          </object>
                          <attributes>
                            <attribute name="text">0</attribute>
                          </attributes>
                        </child>
                      </object>
                    </child>
                  </object>
                </child>
              </object>
            </child>
            <child type="label">
              <object class="GtkLabel">
                <property name="visible">True</property>
                <property name="can_focus">False</property>
                <property name="label" translatable="yes">Notes</property>
              </object>
            </child>
          </object>
        </child>
      </object>
    </child>
  </object>
</interface>

我自己的答案只是一个部分答案,因为如果我需要编辑多个列怎么办?截至目前,我已经使用editing-started信号写了多个回调,如下所示:

def on_cell_editing_started_col0(self, *args):
    self.column = 0

def on_cell_editing_started_col1(self, *args):
    self.column = 1

然后我在两个回调中使用self.column值,Gtk.CellRendererText edited信号和Gtk.TreeView set-focus-child信号。

1 个答案:

答案 0 :(得分:0)

看起来我可以为容器set-focus-child的信号TreeView注册回调。当可编辑的子项开始编辑时,以及编辑完成时,将调用此方法。在这两种情况下,回调都会收到两个参数,第一个参数是TreeView本身。

编辑开始时,第二个参数是临时Gtk.Entry,用户可以与之交互。编辑结束时,第二个参数是None

我可能会使用这段代码,我确信我不知道这可能有多强大:

def on_focus_child(self, tree, entry):
    if entry is not None:
        self.last_entry = entry
    else:
        tv, path = tree.get_selection().get_selected()
        self.notes_list[path][0] = self.last_entry.get_text()