如何在 Python 中过滤 ttk.treeview?

Nir*_*rMH 6 python treeview ttk python-3.x

我有一个tkinter包含ttk.treeview小部件的 python应用程序。

小部件显示在具有特定扩展名的特定目录树上找到的文件列表 - 使用tt.treeview小部件构建这很简单。

有一个请求启用树的“即时”过滤 - 例如,用户输入Entry某个字符串,当他/她输入时,树删除到目前为止与输入的字符串不匹配的元素。

我正在浏览Treeview文档,尝试了detachreattach方法,但没有运气。

detach确实从树中删除了不匹配的元素,但是如果用户点击Backspace,我将无法再在树上正确迭代以恢复那些分离的元素,因为get_children方法不会返回它们。

def filter_tree(self):
    search_by = self.search_entry.get()
    self.tree_detach_leaf_by_regex(self.current_loaded_folder, search_by, "")

def tree_detach_leaf_by_regex(self, root, regex, parent):
    if self.treeview.get_children(root):
        for child in self.treeview.get_children(root):
            self.tree_detach_leaf_by_regex(child, regex, root)
    else:
        if not re.match(regex, self.treeview.item(root)["text"]):
            self.elements_index_within_parent[root] = self.treeview.index(root)
            self.elements_parents[parent] = 1
            self.treeview.detach(root)
        else:
            self.treeview.reattach(root, parent, self.elements_index_within_parent[root])
Run Code Online (Sandbox Code Playgroud)

期待阅读您的建议。

kee*_*ive 5

为了让我的答案可以被任何人重复使用,我必须说的不仅仅是直接回答你的问题。如果你想直接看我如何获取分离项(从而不使用get_children无法获取分离项的 id 的方法),请跳转到名称为 的方法的定义_columns_searcher


介绍

让我们首先定义一些属性。

@property
def _to_search(self):
    key = 'to_search'
    if key not in self._cache:
        self._cache[key] = tk.StringVar()
    return self._cache[key] 

def _set_search_entry(self):  
    ent = ttk.Entry(
        self.root, # or canvas, or frame ...
        #...
        textvariable=self._to_search
    )
    ent.grid(
        #...
    )
    ent.bind(
        '<Return>',
        self._columns_searcher
    )
    return ent

@property
def search_entry(self):
    key = 'search_entry'
    if key not in self._cache:
        self._cache[key] = self._set_search_entry()
    return self._cache[key]
Run Code Online (Sandbox Code Playgroud)


核心答案

接下来是直接展示如何重新附加用户分离项目的部分。首先请注意,正如 OP 所提到的,get_children只返回附加项目的 id。其次要注意的是,您唯一需要重新附加分离的项目的是它们的 id。这意味着在它们分离时跟踪/保存它们以便能够重新连接它们。

_detached = set()
def _columns_searcher(self, event):
    #              originally a set            returns a tuple
    children = list(self._detached) + list(self.tree.get_children())
    self._detached = set()
    query = self._to_search.get()

    self._brut_searcher(children, query.lower())
Run Code Online (Sandbox Code Playgroud)

请注意,children上面包含所有项目,将它们分开。

def _brut_searcher(self, children, query):
    i_r = -1
    for item_id in children:
        text = self.tree.item(item_id)['text'] # already contains the strin-concatenation (over columns) of the row's values
        if query in text:
            i_r += 1
            self.tree.reattach(item_id, '', i_r)
        else:
            self._detached.add(item_id)
            self.tree.detach(item_id)
Run Code Online (Sandbox Code Playgroud)