如何在gtk.ScrolledWindow中的GtkTreeView中显示列头?

11 投票
1 回答
3089 浏览
提问于 2025-04-17 00:06

我在使用 Gtk.TreeviewGtk.ScrolledWindow 时遇到了问题。我有很多数据,放不下在我的普通窗口里,所以我把 Treeview 放进了一个 ScrolledWindow 里,这似乎是个正常的做法。不过,这样做有个问题,就是当我滚动的时候,列标题会消失。我觉得这种情况有点奇怪,因为 ScrolledWindow 应该负责处理里面的控件。

那么,我该怎么做才能让我的 Treeview 可以滚动,同时又能让列标题始终显示在第一行呢?

我修改了 basictreeview.py 示例来展示我的问题:

#!/usr/bin/env python

# example basictreeview.py

import pygtk
pygtk.require('2.0')
import gtk

class BasicTreeViewExample:

    # close the window and quit
    def delete_event(self, widget, event, data=None):
        gtk.main_quit()
        return False

    def __init__(self):
        # Create a new window
        self.window = gtk.Window(gtk.WINDOW_TOPLEVEL)
        self.window.set_title("Basic TreeView Example")
        self.window.set_size_request(200, 200)
        self.window.connect("delete_event", self.delete_event)

        # create a TreeStore with one string column to use as the model
        self.liststore = gtk.ListStore(str)

        # we'll add some data now - 4 rows with 3 child rows each
        for i in range(10):
            self.liststore.append([str(i)])

        # create the TreeView using treestore
        self.treeview = gtk.TreeView(self.liststore)

        # create the TreeViewColumn to display the data
        self.tvcolumn = gtk.TreeViewColumn("Column-Name", gtk.CellRendererText(), text=0)

        # add tvcolumn to treeview
        self.treeview.append_column(self.tvcolumn)

        # make it searchable
        self.treeview.set_search_column(0)

        # Allow sorting on the column
        self.tvcolumn.set_sort_column_id(0)

        # Allow drag and drop reordering of rows
        self.treeview.set_reorderable(True)

        # Add Treeview to a ScrolledWindow
        self.scrolledwindow = gtk.ScrolledWindow(hadjustment=None, vadjustment=None)
        self.scrolledwindow.set_policy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC)
        self.scrolledwindow.add_with_viewport(self.treeview)

        self.window.add(self.scrolledwindow)

        self.window.show_all()

def main():
    gtk.main()

if __name__ == "__main__":
    tvexample = BasicTreeViewExample()
    main()

1 个回答

14

self.scrolledwindow.add_with_viewport 改成 self.scrolledwindow.add 就可以正常工作了。根据文档的说明:

add_with_viewport() 这个方法是用来把一个没有自带滚动功能的控件(也就是你指定的 child)添加到滚动窗口里的。这个方法其实就是把 child 先放到一个 gtk.Viewport 里,然后再把这个 viewport 放到滚动窗口里。如果你添加的控件本身就有滚动功能(比如 gtk.TextView、gtk.TreeView、gtk.Layout),那就应该用 gtk.Container.add() 这个方法。

gtk.TreeView 的“自带滚动”功能可以让列标题保持在顶部,但如果你先把整个 treeview 控件放到一个 gtk.Viewport 里,那么整个控件就会一起滚动。

撰写回答