From 5d487bf5fe31cefcb32e70bc038755dd9d96e525 Mon Sep 17 00:00:00 2001 From: MikeTheWatchGuy Date: Tue, 9 Oct 2018 21:44:29 -0400 Subject: [PATCH] NEW Tree Element! --- Demo_Tree_Element.py | 29 ++++++++++++++++++++ PySimpleGUI.py | 63 +++++++++++++++++++++++++++++++++++++++----- docs/index.md | 46 ++++++++++++++++++++++++++++++++ readme.md | 46 ++++++++++++++++++++++++++++++++ 4 files changed, 177 insertions(+), 7 deletions(-) create mode 100644 Demo_Tree_Element.py diff --git a/Demo_Tree_Element.py b/Demo_Tree_Element.py new file mode 100644 index 00000000..a3faf6ea --- /dev/null +++ b/Demo_Tree_Element.py @@ -0,0 +1,29 @@ +#!/usr/bin/env python +import sys +if sys.version_info[0] >= 3: + import PySimpleGUI as sg +else: + import PySimpleGUI27 as sg + +treedata = sg.TreeData() + +treedata.Insert("", '_A_', 'A', [1,2,3]) +treedata.Insert("", '_B_', 'B', [4,5,6]) +treedata.Insert("_A_", '_A1_', 'A1', ['can','be','anything']) +treedata.Insert("", '_C_', 'C', []) +treedata.Insert("_C_", '_C1_', 'C1', ['or']) +treedata.Insert("_A_", '_A2_', 'A2', [None, None]) +treedata.Insert("_A1_", '_A3_', 'A30', ['getting deep']) +treedata.Insert("_C_", '_C2_', 'C2', ['nothing', 'at', 'all']) + +layout = [[ sg.Text('Tree Test') ], + [ sg.Tree(data=treedata, headings=['col1', 'col2', 'col3'], auto_size_columns=True, num_rows=10, col0_width=10)], + [ sg.RButton('Read')]] + +window = sg.Window('Tree Element Test').Layout(layout) + +while True: # Event Loop + button, value = window.Read() + if button is None: + break + print(button, value) diff --git a/PySimpleGUI.py b/PySimpleGUI.py index dc94abd5..da68e2a3 100644 --- a/PySimpleGUI.py +++ b/PySimpleGUI.py @@ -2249,7 +2249,7 @@ class Table(Element): # Tree # # ---------------------------------------------------------------------- # class Tree(Element): - def __init__(self, headings=None, visible_column_map=None, col_widths=None, def_col_width=10, auto_size_columns=True, max_col_width=20, select_mode=None, font=None, justification='right', text_color=None, background_color=None, num_rows=None, pad=None, key=None, tooltip=None): + def __init__(self, data=None, headings=None, visible_column_map=None, col_widths=None, col0_width=10, def_col_width=10, auto_size_columns=True, max_col_width=20, select_mode=None, font=None, justification='right', text_color=None, background_color=None, num_rows=None, pad=None, key=None, tooltip=None): ''' Tree Element :param headings: @@ -2268,6 +2268,7 @@ class Tree(Element): :param key: :param tooltip: ''' + self.TreeData = data self.ColumnHeadings = headings self.ColumnsToDisplay = visible_column_map self.ColumnWidths = col_widths @@ -2280,16 +2281,52 @@ class Tree(Element): self.InitialState = None self.SelectMode = select_mode self.NumRows = num_rows + self.Col0Width = col0_width self.TKTreeview = None super().__init__(ELEM_TYPE_TREE, text_color=text_color, background_color=background_color, font=font, pad=pad, key=key, tooltip=tooltip) return + def __del__(self): super().__del__() +class TreeData(object): + class Node(object): + def __init__(self, parent, key, text, values): + self.parent = parent + self.children = [] + self.key = key + self.text = text + self.values = values + + def _Add(self, node): + self.children.append(node) + + def __init__(self): + self.tree_dict = {} + self.root_node = self.Node("", "", 'root', []) + self.tree_dict [""] = self.root_node + + def _AddNode(self, key, node): + self.tree_dict[key] = node + + def Insert(self, parent, key, text, values): + node = self.Node(parent, key, text,values) + self.tree_dict[key] = node + parent_node = self.tree_dict[parent] + parent_node._Add(node) + + def Print(self): + self._print_node(self.root_node) + + def _print_node(self, node): + print(f'Node: {node.text}') + print(f'Children = {[c.text for c in node.children]}') + for node in node.children: + self._print_node(node) @@ -2662,6 +2699,7 @@ class Window: CloseNonBlockingForm = CloseNonBlocking def OnClosingCallback(self): + # print('Got closing callback') return @@ -2677,7 +2715,7 @@ class Window: def UnHide(self): self.TKroot.deiconify() - def Disapper(self): + def Disappear(self): self.TKroot.attributes('-alpha', 0) def Reappear(self): @@ -3801,7 +3839,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): element.TooltipObject = ToolTip(element.TKTreeview, text=element.Tooltip, timeout=DEFAULT_TOOLTIP_TIME) # ------------------------- Tree element ------------------------- # elif element_type == ELEM_TYPE_TREE: - width, height = element_size + height = element.NumRows if element.Justification == 'left': # justification anchor = tk.W elif element.Justification == 'right': @@ -3819,18 +3857,28 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): column_headings= element.ColumnHeadings # ------------- GET THE TREEVIEW WIDGET ------------- element.TKTreeview = ttk.Treeview(tk_row_frame, columns=column_headings, - displaycolumns=displaycolumns, show='headings', height=height, selectmode=element.SelectMode) + displaycolumns=displaycolumns, show='tree headings', height=height, selectmode=element.SelectMode, ) treeview = element.TKTreeview for i, heading in enumerate(element.ColumnHeadings): # Configure cols + headings treeview.heading(heading, text=heading) if element.AutoSizeColumns: - width = min(element.MaxColumnWidth, len(heading)) + width = min(element.MaxColumnWidth, len(heading)+1) else: try: width = element.ColumnWidths[i] except: width = element.DefaultColumnWidth treeview.column(heading, width=width*CharWidthInPixels(), anchor=anchor) + + def add_treeview_data(node): + # print(f'Inserting {node.key} under parent {node.parent}') + if node.key != '': + treeview.insert(node.parent, 'end', node.key, text=node.text, values=node.values) + for node in node.children: + add_treeview_data(node) + + add_treeview_data(element.TreeData.root_node) + treeview.column('#0', width=element.Col0Width*CharWidthInPixels(), anchor=anchor) # ----- configure colors ----- if element.BackgroundColor is not None and element.BackgroundColor != COLOR_SYSTEM_DEFAULT: ttk.Style().configure("Treeview", background=element.BackgroundColor, fieldbackground=element.BackgroundColor) @@ -3941,7 +3989,8 @@ def StartupTK(my_flex_form): duration = DEFAULT_AUTOCLOSE_TIME if my_flex_form.AutoCloseDuration is None else my_flex_form.AutoCloseDuration my_flex_form.TKAfterID = root.after(duration * 1000, my_flex_form._AutoCloseAlarmCallback) if my_flex_form.NonBlocking: - my_flex_form.TKroot.protocol("WM_WINDOW_DESTROYED", my_flex_form.OnClosingCallback()) + pass + # my_flex_form.TKroot.protocol("WM_DELETE_WINDOW", my_flex_form.OnClosingCallback()) else: # it's a blocking form # print('..... CALLING MainLoop') my_flex_form.TKroot.mainloop() @@ -4885,7 +4934,7 @@ def Popup(*args, button_color=None, background_color=None, text_color=None, butt if non_blocking: button, values = window.ReadNonBlocking() else: - button, values = window.Show() + button, values = window.Read() return button diff --git a/docs/index.md b/docs/index.md index 2e9e386b..06f87692 100644 --- a/docs/index.md +++ b/docs/index.md @@ -2067,6 +2067,52 @@ Let me say up front that the Table Element has Beta status. The reason is that s tooltip - tooltip text +## Tree Element + +The Tree Element and Table Element are close cousins. Many of the parameters found in the Table Element apply to Tree Elements. In particular the heading information, column widths, etc. + +``` +class Tree(data=None - data in TreeData format + headings=None - list of strings representing your headings + visible_column_map=None - list of bools indicating which columns to display + col_widths=None - list of column widths + col0_width=10 - width of the first column which has the text data + def_col_width=10 - default column width + auto_size_columns=True - if true will autosize columns (currenly only sizes to col heading width) + max_col_width=20 - max width for columns in characters + select_mode=None - not yet used + font=None - the display font + justification='right' - justification for data display + text_color=None- color of text to display + background_color=None - background color + num_rows=None - number of rows to display + pad=None - element padding + key=None - key for element + tooltip=None - tooltip +``` + +Unlike Tables there is no standard format for trees. Thus the data structure passed to the Tree Element must be constructed. This is done using the TreeData class. The process is as follows: +* Get a TreeData Object +* "Insert" data into the tree +* Pass the filled in TreeData object to Tree Element + +To "insert" data into the tree the TreeData method Insert is called. + +`Insert(parent_key, key, display_text, values)` + +To indicate insertion at the head of the tree, use a parent key of "". So, every top-level node in the tree will have a parent node = "" + +This code creates a TreeData object and populates with 3 values +```python +treedata = sg.TreeData() + +treedata.Insert("", '_A_', 'A', [1,2,3]) +treedata.Insert("", '_B_', 'B', [4,5,6]) +treedata.Insert("_A_", '_A1_', 'A1', ['can','be','anything']) +``` + +Note that you can use the same values for display_text and keys. The only thing you have to watch for is that you cannot repeat keys. + ## Tab and Tab Group Elements diff --git a/readme.md b/readme.md index 2e9e386b..06f87692 100644 --- a/readme.md +++ b/readme.md @@ -2067,6 +2067,52 @@ Let me say up front that the Table Element has Beta status. The reason is that s tooltip - tooltip text +## Tree Element + +The Tree Element and Table Element are close cousins. Many of the parameters found in the Table Element apply to Tree Elements. In particular the heading information, column widths, etc. + +``` +class Tree(data=None - data in TreeData format + headings=None - list of strings representing your headings + visible_column_map=None - list of bools indicating which columns to display + col_widths=None - list of column widths + col0_width=10 - width of the first column which has the text data + def_col_width=10 - default column width + auto_size_columns=True - if true will autosize columns (currenly only sizes to col heading width) + max_col_width=20 - max width for columns in characters + select_mode=None - not yet used + font=None - the display font + justification='right' - justification for data display + text_color=None- color of text to display + background_color=None - background color + num_rows=None - number of rows to display + pad=None - element padding + key=None - key for element + tooltip=None - tooltip +``` + +Unlike Tables there is no standard format for trees. Thus the data structure passed to the Tree Element must be constructed. This is done using the TreeData class. The process is as follows: +* Get a TreeData Object +* "Insert" data into the tree +* Pass the filled in TreeData object to Tree Element + +To "insert" data into the tree the TreeData method Insert is called. + +`Insert(parent_key, key, display_text, values)` + +To indicate insertion at the head of the tree, use a parent key of "". So, every top-level node in the tree will have a parent node = "" + +This code creates a TreeData object and populates with 3 values +```python +treedata = sg.TreeData() + +treedata.Insert("", '_A_', 'A', [1,2,3]) +treedata.Insert("", '_B_', 'B', [4,5,6]) +treedata.Insert("_A_", '_A1_', 'A1', ['can','be','anything']) +``` + +Note that you can use the same values for display_text and keys. The only thing you have to watch for is that you cannot repeat keys. + ## Tab and Tab Group Elements