Nir*_*rMH 6 python treeview ttk python-3.x
我有一个tkinter包含ttk.treeview小部件的 python应用程序。
小部件显示在具有特定扩展名的特定目录树上找到的文件列表 - 使用tt.treeview小部件构建这很简单。
有一个请求启用树的“即时”过滤 - 例如,用户输入Entry某个字符串,当他/她输入时,树删除到目前为止与输入的字符串不匹配的元素。
我正在浏览Treeview文档,尝试了detach和reattach方法,但没有运气。
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)
期待阅读您的建议。
为了让我的答案可以被任何人重复使用,我必须说的不仅仅是直接回答你的问题。如果你想直接看我如何获取分离项(从而不使用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)