NEW Tree Element!

This commit is contained in:
MikeTheWatchGuy 2018-10-09 21:44:29 -04:00
parent 6d123a44eb
commit 5d487bf5fe
4 changed files with 177 additions and 7 deletions

29
Demo_Tree_Element.py Normal file
View File

@ -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)

View File

@ -2249,7 +2249,7 @@ class Table(Element):
# Tree # # Tree #
# ---------------------------------------------------------------------- # # ---------------------------------------------------------------------- #
class Tree(Element): 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 Tree Element
:param headings: :param headings:
@ -2268,6 +2268,7 @@ class Tree(Element):
:param key: :param key:
:param tooltip: :param tooltip:
''' '''
self.TreeData = data
self.ColumnHeadings = headings self.ColumnHeadings = headings
self.ColumnsToDisplay = visible_column_map self.ColumnsToDisplay = visible_column_map
self.ColumnWidths = col_widths self.ColumnWidths = col_widths
@ -2280,16 +2281,52 @@ class Tree(Element):
self.InitialState = None self.InitialState = None
self.SelectMode = select_mode self.SelectMode = select_mode
self.NumRows = num_rows self.NumRows = num_rows
self.Col0Width = col0_width
self.TKTreeview = None self.TKTreeview = None
super().__init__(ELEM_TYPE_TREE, text_color=text_color, background_color=background_color, font=font, pad=pad, key=key, tooltip=tooltip) super().__init__(ELEM_TYPE_TREE, text_color=text_color, background_color=background_color, font=font, pad=pad, key=key, tooltip=tooltip)
return return
def __del__(self): def __del__(self):
super().__del__() 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 CloseNonBlockingForm = CloseNonBlocking
def OnClosingCallback(self): def OnClosingCallback(self):
# print('Got closing callback')
return return
@ -2677,7 +2715,7 @@ class Window:
def UnHide(self): def UnHide(self):
self.TKroot.deiconify() self.TKroot.deiconify()
def Disapper(self): def Disappear(self):
self.TKroot.attributes('-alpha', 0) self.TKroot.attributes('-alpha', 0)
def Reappear(self): 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) element.TooltipObject = ToolTip(element.TKTreeview, text=element.Tooltip, timeout=DEFAULT_TOOLTIP_TIME)
# ------------------------- Tree element ------------------------- # # ------------------------- Tree element ------------------------- #
elif element_type == ELEM_TYPE_TREE: elif element_type == ELEM_TYPE_TREE:
width, height = element_size height = element.NumRows
if element.Justification == 'left': # justification if element.Justification == 'left': # justification
anchor = tk.W anchor = tk.W
elif element.Justification == 'right': elif element.Justification == 'right':
@ -3819,18 +3857,28 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
column_headings= element.ColumnHeadings column_headings= element.ColumnHeadings
# ------------- GET THE TREEVIEW WIDGET ------------- # ------------- GET THE TREEVIEW WIDGET -------------
element.TKTreeview = ttk.Treeview(tk_row_frame, columns=column_headings, 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 treeview = element.TKTreeview
for i, heading in enumerate(element.ColumnHeadings): # Configure cols + headings for i, heading in enumerate(element.ColumnHeadings): # Configure cols + headings
treeview.heading(heading, text=heading) treeview.heading(heading, text=heading)
if element.AutoSizeColumns: if element.AutoSizeColumns:
width = min(element.MaxColumnWidth, len(heading)) width = min(element.MaxColumnWidth, len(heading)+1)
else: else:
try: try:
width = element.ColumnWidths[i] width = element.ColumnWidths[i]
except: except:
width = element.DefaultColumnWidth width = element.DefaultColumnWidth
treeview.column(heading, width=width*CharWidthInPixels(), anchor=anchor) 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 ----- # ----- configure colors -----
if element.BackgroundColor is not None and element.BackgroundColor != COLOR_SYSTEM_DEFAULT: if element.BackgroundColor is not None and element.BackgroundColor != COLOR_SYSTEM_DEFAULT:
ttk.Style().configure("Treeview", background=element.BackgroundColor, fieldbackground=element.BackgroundColor) 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 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) my_flex_form.TKAfterID = root.after(duration * 1000, my_flex_form._AutoCloseAlarmCallback)
if my_flex_form.NonBlocking: 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 else: # it's a blocking form
# print('..... CALLING MainLoop') # print('..... CALLING MainLoop')
my_flex_form.TKroot.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: if non_blocking:
button, values = window.ReadNonBlocking() button, values = window.ReadNonBlocking()
else: else:
button, values = window.Show() button, values = window.Read()
return button return button

View File

@ -2067,6 +2067,52 @@ Let me say up front that the Table Element has Beta status. The reason is that s
tooltip - tooltip text 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 ## Tab and Tab Group Elements

View File

@ -2067,6 +2067,52 @@ Let me say up front that the Table Element has Beta status. The reason is that s
tooltip - tooltip text 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 ## Tab and Tab Group Elements