NEW Tree Element!
This commit is contained in:
parent
6d123a44eb
commit
5d487bf5fe
|
@ -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)
|
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
46
readme.md
46
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
|
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
|
||||||
|
|
Loading…
Reference in New Issue