Merge pull request #1390 from PySimpleGUI/Dev-latest

Tabs and Menu support being worked on.
This commit is contained in:
MikeTheWatchGuy 2019-05-10 15:18:23 -04:00 committed by GitHub
commit c947e506f8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
1 changed files with 164 additions and 133 deletions

View File

@ -69,7 +69,9 @@ def TimerStop():
""" """
# Because looks matter... # Because looks matter...
DEFAULT_BASE64_ICON = b'iVBORw0KGgoAAAANSUhEUgAAACEAAAAgCAMAAACrZuH4AAAABGdBTUEAALGPC/xhBQAAAwBQTFRFAAAAMGmYMGqZMWqaMmubMmycM22dNGuZNm2bNm6bNG2dN26cNG6dNG6eNW+fN3CfOHCeOXGfNXCgNnGhN3KiOHOjOXSjOHSkOnWmOnamOnanPHSiPXakPnalO3eoPnimO3ioPHioPHmpPHmqPXqqPnurPnusPnytP3yuQHimQnurQn2sQH2uQX6uQH6vR32qRn+sSXujSHynTH2mTn+nSX6pQH6wTIGsTYKuTYSvQoCxQoCyRIK0R4S1RYS2Roa4SIe4SIe6SIi7Soq7SYm8SYq8Sou+TY2/UYStUYWvVIWtUYeyVoewUIi0VIizUI6+Vo+8WImxXJG5YI2xZI+xZ5CzZJC0ZpG1b5a3apW4aZm/cZi4dJ2/eJ69fJ+9XZfEZZnCZJzHaZ/Jdp/AeKTI/tM8/9Q7/9Q8/9Q9/9Q+/tQ//9VA/9ZA/9ZB/9ZC/9dD/9ZE/tdJ/9dK/9hF/9hG/9hH/9hI/9hJ/9hK/9lL/9pK/9pL/thO/9pM/9pN/9tO/9tP/9xP/tpR/9xQ/9xR/9xS/9xT/91U/91V/t1W/95W/95X/95Y/95Z/99a/99b/txf/txh/txk/t5l/t1q/t5v/+Bb/+Bc/+Bd/+Be/+Bf/+Bg/+Fh/+Fi/+Jh/+Ji/uJk/uJl/+Jm/+Rm/uJo/+Ro/+Rr/+Zr/+Vs/+Vu/+Zs/+Zu/uF0/uVw/+dw/+dz/+d2/uB5/uB6/uJ9/uR7/uR+/uV//+hx/+hy/+h0/+h2/+l4/+l7/+h8gKXDg6vLgazOhKzMiqrEj6/KhK/Qka/Hk7HJlLHJlLPMmLTLmbbOkLXSmLvXn77XoLrPpr/Tn8DaocLdpcHYrcjdssfZus/g/uOC/uOH/uaB/uWE/uaF/uWK/+qA/uqH/uqI/uuN/uyM/ueS/ueW/ueY/umQ/uqQ/uuS/uuW/uyU/uyX/uqa/uue/uye/uyf/u6f/uyq/u+r/u+t/vCm/vCp/vCu/vCy/vC2/vK2/vO8/vO/wtTjwtXlzdrl/vTA/vPQAAAAiNpY5gAAAQB0Uk5T////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////AFP3ByUAAAAJcEhZcwAAFw8AABcPASe7rwsAAAAYdEVYdFNvZnR3YXJlAHBhaW50Lm5ldCA0LjEuMWMqnEsAAAKUSURBVDhPhdB3WE1xHMdxt5JV0dANoUiyd8kqkey996xclUuTlEKidO3qVnTbhIyMW/bee5NskjJLmR/f3++cK/94vP76Ps/n/Zx7z6mE/6koJowcK154vvHOL/GsKCZXkUgkWlf4vWGWq5tsDz+JWIzSokAiqXGe7nWu3HxhEYof7fhOqp1GtptQuMruVhQdxZ05U5G47tYUHbQ4oah6Fg9Z4ubm7i57JhQjdHS0RSzUPoG17u6zZTKZh8c8XlytqW9YWUOH1LqFOZ6enl5ec+XybFb0rweM1tPTM6yuq6vLs0lYJJfLvb19fHwDWGF0jh5lYNAe4/QFemOwxtfXz8/fPyBgwVMqzAcCF4ybAZ2MRCexJGBhYGBQUHDw4u1UHDG1G2ZqB/Q1MTHmzAE+kpCwL1RghlTaBt/6SaXS2kx9YH1IaOjSZST8vfA9JtoDnSngGgL7wkg4WVkofA9mcF1Sx8zMzBK4v3wFiYiMVLxlEy9u21syFhYNmgN7IyJXEYViNZvEYoCVVWOmUVvgQVSUQqGIjolRFvOAFd8HWVs34VoA+6OjY2JjY5Vxm4BC1UuhGG5jY9OUaQXci1MqlfHx8YmqjyhOViW9ZsUN29akJRmPFwkJCZsTSXIpilJffXiTzorLXYgtcxRJKpUqKTklJQ0oSt9FP/EonxVdNY4jla1kK4q2ZB6mIr+AipvduzFUzMSOtLT09IyMzMxtJKug/F0u/6dTexAWDcXXLGEjapKjfsILOLKEuYiSnTQeYCt3UHhbwEHjGMrETfBJU5zq5dSTcXC8hLJccSWP2cgLXHPu7cQNAcpyxF1dyjehAKb0cSYUAOXCUw6V8OFPgevTXFymC+fPPLU677Nw/1X8A/AbfAKGulaqFlIAAAAASUVORK5CYII=' DEFAULT_BASE64_ICON_OLD = b'iVBORw0KGgoAAAANSUhEUgAAACEAAAAgCAMAAACrZuH4AAAABGdBTUEAALGPC/xhBQAAAwBQTFRFAAAAMGmYMGqZMWqaMmubMmycM22dNGuZNm2bNm6bNG2dN26cNG6dNG6eNW+fN3CfOHCeOXGfNXCgNnGhN3KiOHOjOXSjOHSkOnWmOnamOnanPHSiPXakPnalO3eoPnimO3ioPHioPHmpPHmqPXqqPnurPnusPnytP3yuQHimQnurQn2sQH2uQX6uQH6vR32qRn+sSXujSHynTH2mTn+nSX6pQH6wTIGsTYKuTYSvQoCxQoCyRIK0R4S1RYS2Roa4SIe4SIe6SIi7Soq7SYm8SYq8Sou+TY2/UYStUYWvVIWtUYeyVoewUIi0VIizUI6+Vo+8WImxXJG5YI2xZI+xZ5CzZJC0ZpG1b5a3apW4aZm/cZi4dJ2/eJ69fJ+9XZfEZZnCZJzHaZ/Jdp/AeKTI/tM8/9Q7/9Q8/9Q9/9Q+/tQ//9VA/9ZA/9ZB/9ZC/9dD/9ZE/tdJ/9dK/9hF/9hG/9hH/9hI/9hJ/9hK/9lL/9pK/9pL/thO/9pM/9pN/9tO/9tP/9xP/tpR/9xQ/9xR/9xS/9xT/91U/91V/t1W/95W/95X/95Y/95Z/99a/99b/txf/txh/txk/t5l/t1q/t5v/+Bb/+Bc/+Bd/+Be/+Bf/+Bg/+Fh/+Fi/+Jh/+Ji/uJk/uJl/+Jm/+Rm/uJo/+Ro/+Rr/+Zr/+Vs/+Vu/+Zs/+Zu/uF0/uVw/+dw/+dz/+d2/uB5/uB6/uJ9/uR7/uR+/uV//+hx/+hy/+h0/+h2/+l4/+l7/+h8gKXDg6vLgazOhKzMiqrEj6/KhK/Qka/Hk7HJlLHJlLPMmLTLmbbOkLXSmLvXn77XoLrPpr/Tn8DaocLdpcHYrcjdssfZus/g/uOC/uOH/uaB/uWE/uaF/uWK/+qA/uqH/uqI/uuN/uyM/ueS/ueW/ueY/umQ/uqQ/uuS/uuW/uyU/uyX/uqa/uue/uye/uyf/u6f/uyq/u+r/u+t/vCm/vCp/vCu/vCy/vC2/vK2/vO8/vO/wtTjwtXlzdrl/vTA/vPQAAAAiNpY5gAAAQB0Uk5T////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////AFP3ByUAAAAJcEhZcwAAFw8AABcPASe7rwsAAAAYdEVYdFNvZnR3YXJlAHBhaW50Lm5ldCA0LjEuMWMqnEsAAAKUSURBVDhPhdB3WE1xHMdxt5JV0dANoUiyd8kqkey996xclUuTlEKidO3qVnTbhIyMW/bee5NskjJLmR/f3++cK/94vP76Ps/n/Zx7z6mE/6koJowcK154vvHOL/GsKCZXkUgkWlf4vWGWq5tsDz+JWIzSokAiqXGe7nWu3HxhEYof7fhOqp1GtptQuMruVhQdxZ05U5G47tYUHbQ4oah6Fg9Z4ubm7i57JhQjdHS0RSzUPoG17u6zZTKZh8c8XlytqW9YWUOH1LqFOZ6enl5ec+XybFb0rweM1tPTM6yuq6vLs0lYJJfLvb19fHwDWGF0jh5lYNAe4/QFemOwxtfXz8/fPyBgwVMqzAcCF4ybAZ2MRCexJGBhYGBQUHDw4u1UHDG1G2ZqB/Q1MTHmzAE+kpCwL1RghlTaBt/6SaXS2kx9YH1IaOjSZST8vfA9JtoDnSngGgL7wkg4WVkofA9mcF1Sx8zMzBK4v3wFiYiMVLxlEy9u21syFhYNmgN7IyJXEYViNZvEYoCVVWOmUVvgQVSUQqGIjolRFvOAFd8HWVs34VoA+6OjY2JjY5Vxm4BC1UuhGG5jY9OUaQXci1MqlfHx8YmqjyhOViW9ZsUN29akJRmPFwkJCZsTSXIpilJffXiTzorLXYgtcxRJKpUqKTklJQ0oSt9FP/EonxVdNY4jla1kK4q2ZB6mIr+AipvduzFUzMSOtLT09IyMzMxtJKug/F0u/6dTexAWDcXXLGEjapKjfsILOLKEuYiSnTQeYCt3UHhbwEHjGMrETfBJU5zq5dSTcXC8hLJccSWP2cgLXHPu7cQNAcpyxF1dyjehAKb0cSYUAOXCUw6V8OFPgevTXFymC+fPPLU677Nw/1X8A/AbfAKGulaqFlIAAAAASUVORK5CYII='
DEFAULT_BASE64_ICON = b'iVBORw0KGgoAAAANSUhEUgAAACEAAAAgCAMAAACrZuH4AAAABGdBTUEAALGPC/xhBQAAAwBQTFRFAAAAMGmYMGqZMWqaMmubMmycM22dNGuZNm2bNm6bNG2dN26cNG6dNG6eNW+fN3CfOHCeOXGfNXCgNnGhN3KiOHOjOXSjOHSkOnWmOnamOnanPHSiPXakPnalO3eoPnimO3ioPHioPHmpPHmqPXqqPnurPnusPnytP3yuQHimQnurQn2sQH2uQX6uQH6vR32qRn+sSXujSHynTH2mTn+nSX6pQH6wTIGsTYKuTYSvQoCxQoCyRIK0R4S1RYS2Roa4SIe4SIe6SIi7Soq7SYm8SYq8Sou+TY2/UYStUYWvVIWtUYeyVoewUIi0VIizUI6+Vo+8WImxXJG5YI2xZI+xZ5CzZJC0ZpG1b5a3apW4aZm/cZi4dJ2/eJ69fJ+9XZfEZZnCZJzHaZ/Jdp/AeKTI/tM8/9Q7/9Q8/9Q9/9Q+/tQ//9VA/9ZA/9ZB/9ZC/9dD/9ZE/tdJ/9dK/9hF/9hG/9hH/9hI/9hJ/9hK/9lL/9pK/9pL/thO/9pM/9pN/9tO/9tP/9xP/tpR/9xQ/9xR/9xS/9xT/91U/91V/t1W/95W/95X/95Y/95Z/99a/99b/txf/txh/txk/t5l/t1q/t5v/+Bb/+Bc/+Bd/+Be/+Bf/+Bg/+Fh/+Fi/+Jh/+Ji/uJk/uJl/+Jm/+Rm/uJo/+Ro/+Rr/+Zr/+Vs/+Vu/+Zs/+Zu/uF0/uVw/+dw/+dz/+d2/uB5/uB6/uJ9/uR7/uR+/uV//+hx/+hy/+h0/+h2/+l4/+l7/+h8gKXDg6vLgazOhKzMiqrEj6/KhK/Qka/Hk7HJlLHJlLPMmLTLmbbOkLXSmLvXn77XoLrPpr/Tn8DaocLdpcHYrcjdssfZus/g/uOC/uOH/uaB/uWE/uaF/uWK/+qA/uqH/uqI/uuN/uyM/ueS/ueW/ueY/umQ/uqQ/uuS/uuW/uyU/uyX/uqa/uue/uye/uyf/u6f/uyq/u+r/u+t/vCm/vCp/vCu/vCy/vC2/vK2/vO8/vO/wtTjwtXlzdrl/vTA/vPQAAAAiNpY5gAAAQB0Uk5T////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////AFP3ByUAAAAJcEhZcwAAFw4AABcOAfX8Te8AAAAYdEVYdFNvZnR3YXJlAHBhaW50Lm5ldCA0LjEuNWRHWFIAAAKUSURBVDhPhdB3WE1xHMdxt5JV0dANoUiyd8kqkey996xclUuTlEKidO3qVnTbhIyMW/bee5NskjJLmR/f3++cK/94vP76Ps/n/Zx7z6mE/6koJowcK154vvHOL/GsKCZXkUgkWlf4vWGWq5tsDz+JWIzSokAiqXGe7nWu3HxhEYof7fhOqp1GtptQuMruVhQdxZ05U5G47tYUHbQ4oah6Fg9Z4ubm7i57JhQjdHS0RSzUPoG17u6zZTKZh8c8XlytqW9YWUOH1LqFOZ6enl5ec+XybFb0rweM1tPTM6yuq6vLs0lYJJfLvb19fHwDWGF0jh5lYNAe4/QFemOwxtfXz8/fPyBgwVMqzAcCF4ybAZ2MRCexJGBhYGBQUHDw4u1UHDG1G2ZqB/Q1MTHmzAE+kpCwL1RghlTaBt/6SaXS2kx9YH1IaOjSZST8vfA9JtoDnSngGgL7wkg4WVkofA9mcF1Sx8zMzBK4v3wFiYiMVLxlEy9u21syFhYNmgN7IyJXEYViNZvEYoCVVWOmUVvgQVSUQqGIjolRFvOAFd8HWVs34VoA+6OjY2JjY5Vxm4BC1UuhGG5jY9OUaQXci1MqlfHx8YmqjyhOViW9ZsUN29akJRmPFwkJCZsTSXIpilJffXiTzorLXYgtcxRJKpUqKTklJQ0oSt9FP/EonxVdNY4jla1kK4q2ZB6mIr+AipvduzFUzMSOtLT09IyMzMxtJKug/F0u/6dTexAWDcXXLGEjapKjfsILOLKEuYiSnTQeYCt3UHhbwEHjGMrETfBJU5zq5dSTcXC8hLJccSWP2cgLXHPu7cQNAcpyxF1dyjehAKb0cSYUAOXCUw6V8OFPgevTXFymC+fPPLU677Nw/1X8A/AbfAKGulaqFlIAAAAASUVORK5CYII='
# ----====----====----==== Constants the user CAN safely change ====----====----====----# # ----====----====----==== Constants the user CAN safely change ====----====----====----#
@ -204,6 +206,12 @@ TIMEOUT_KEY = '__TIMEOUT__'
# Key indicating should not create any return values for element # Key indicating should not create any return values for element
WRITE_ONLY_KEY = '__WRITE ONLY__' WRITE_ONLY_KEY = '__WRITE ONLY__'
# MENU Constants, can be changed by user if desired
MENU_DISABLED_CHARACTER = '!'
MENU_KEY_SEPARATOR = '::'
# a shameful global variable. This represents the top-level window information. Needed because opening a second window is different than opening the first. # a shameful global variable. This represents the top-level window information. Needed because opening a second window is different than opening the first.
class MyWindows(): class MyWindows():
@ -757,17 +765,18 @@ class Listbox(Element):
background_color=bg, text_color=fg, key=key, pad=pad, tooltip=tooltip, visible=visible, size_px=size_px) background_color=bg, text_color=fg, key=key, pad=pad, tooltip=tooltip, visible=visible, size_px=size_px)
def Update(self, values=None, disabled=None, set_to_index=None,background_color=None, text_color=None, font=None, visible=None): def Update(self, values=None, disabled=None, set_to_index=None,background_color=None, text_color=None, font=None, visible=None):
return
if values is not None: if values is not None:
self.Values = values self.Values = values
self.Widget.set_value(values) self.Widget.empty()
for item in values:
self.Widget.append(remi.gui.ListItem(item))
# if disabled == True: # if disabled == True:
# self.QT_ListWidget.setDisabled(True) # self.QT_ListWidget.setDisabled(True)
# elif disabled == False: # elif disabled == False:
# self.QT_ListWidget.setDisabled(False) # self.QT_ListWidget.setDisabled(False)
# if set_to_index is not None: # if set_to_index is not None:
# self.QT_ListWidget.setCurrentRow(set_to_index) # self.QT_ListWidget.setCurrentRow(set_to_index)
super().Update(self.Widget, background_color=background_color, text_color=text_color, font=font, visible=visible) super().Update(self.Widget, background_color=background_color, text_color=text_color, font=font, visible=visible, disabled=disabled)
return return
@ -2231,7 +2240,7 @@ class Tab(Element):
# ---------------------------------------------------------------------- # # ---------------------------------------------------------------------- #
class TabGroup(Element): class TabGroup(Element):
def __init__(self, layout, tab_location=None, title_color=None, selected_title_color=None, background_color=None, def __init__(self, layout, tab_location=None, title_color=None, selected_title_color=None, background_color=None,
font=None, change_submits=False, pad=None, border_width=None, theme=None, key=None, tooltip=None): font=None, change_submits=False, enable_events=False,pad=None, border_width=None, theme=None, key=None, tooltip=None, visible=True):
''' '''
TabGroup Element TabGroup Element
:param layout: :param layout:
@ -2262,9 +2271,10 @@ class TabGroup(Element):
self.BorderWidth = border_width self.BorderWidth = border_width
self.Theme = theme self.Theme = theme
self.BackgroundColor = background_color if background_color is not None else DEFAULT_BACKGROUND_COLOR self.BackgroundColor = background_color if background_color is not None else DEFAULT_BACKGROUND_COLOR
self.ChangeSubmits = change_submits self.ChangeSubmits = enable_events or change_submits
self.TabLocation = tab_location self.TabLocation = tab_location
self.Visible = visible
self.Disabled = False
self.Layout(layout) self.Layout(layout)
super().__init__(ELEM_TYPE_TAB_GROUP, background_color=background_color, text_color=title_color, font=font, super().__init__(ELEM_TYPE_TAB_GROUP, background_color=background_color, text_color=title_color, font=font,
@ -2462,6 +2472,7 @@ class Menu(Element):
self.MenuDefinition = menu_definition self.MenuDefinition = menu_definition
self.TKMenu = None self.TKMenu = None
self.Tearoff = tearoff self.Tearoff = tearoff
self.Widget = None # type: remi.gui.MenuBar
super().__init__(ELEM_TYPE_MENUBAR, background_color=background_color, size=size, pad=pad, key=key) super().__init__(ELEM_TYPE_MENUBAR, background_color=background_color, size=size, pad=pad, key=key)
return return
@ -2788,6 +2799,7 @@ class Window:
stdout_string_io = None stdout_string_io = None
stdout_location = None stdout_location = None
port_number = 6900 port_number = 6900
menu_element = None
active_windows = [ ] # type: [Window] active_windows = [ ] # type: [Window]
App = None # type: remi.App App = None # type: remi.App
@ -3440,8 +3452,6 @@ class Window:
# remi.start(self.MyApp, standalone=True, debug=True, userdata=(self,) ) # Can't do this because of a threading problem # remi.start(self.MyApp, standalone=True, debug=True, userdata=(self,) ) # Can't do this because of a threading problem
print('Returned from Remi Start command... now sending None event') print('Returned from Remi Start command... now sending None event')
# self.App.server_starter_instance._alive = False
# self.App.server_starter_instance._sserver.shutdown()
self.MessageQueue.put(None) # if returned from start call, then the window has been destroyed and a None event should be generated self.MessageQueue.put(None) # if returned from start call, then the window has been destroyed and a None event should be generated
@ -3456,6 +3466,7 @@ class Window:
self.window = userdata2 # type: Window self.window = userdata2 # type: Window
self.master_widget = None self.master_widget = None
self.window.App = self self.window.App = self
if userdata2 is None: if userdata2 is None:
# res_path = os.path.dirname(os.path.abspath(__file__)) # res_path = os.path.dirname(os.path.abspath(__file__))
# print('res path', res_path) # print('res path', res_path)
@ -3497,8 +3508,13 @@ class Window:
# self.master_widget.add_child("onunloadevent", tag) # self.master_widget.add_child("onunloadevent", tag)
self.master_widget = setup_remi_window(self, self.window) self.master_widget = setup_remi_window(self, self.window)
self.window.MessageQueue.put('Layout complete') # signal the main code that the layout is all done
self.window.master_widget = self.master_widget self.window.master_widget = self.master_widget
# if self.window.WindowIcon:
# print('placing icon')
# self.page.children['head'].set_icon_file("/res:logo.png")
# self.page.children['head'].set_icon_data( base64_data=self.window.WindowIcon, mimetype="image/png" )
self.window.MessageQueue.put('Layout complete') # signal the main code that the layout is all done
return self.master_widget # returning the root widget return self.master_widget # returning the root widget
@ -4176,8 +4192,12 @@ def _FindElementWithFocusInSubForm(form):
return element return element
if sys.version_info[0] >= 3: def AddMenuItem(top_menu, sub_menu_info, element, is_sub_menu=False, skip=False):
def AddMenuItem(top_menu, sub_menu_info, element, is_sub_menu=False, skip=False): # m3 = gui.MenuItem('Dialog', width=100, height=30)
# m3.onclick.connect(self.menu_dialog_clicked)
# menu.append([m1, m2, m3])
return_val = None
if type(sub_menu_info) is str: if type(sub_menu_info) is str:
if not is_sub_menu and not skip: if not is_sub_menu and not skip:
# print(f'Adding command {sub_menu_info}') # print(f'Adding command {sub_menu_info}')
@ -4186,55 +4206,44 @@ if sys.version_info[0] >= 3:
if pos == 0 or sub_menu_info[pos - 1] != "\\": if pos == 0 or sub_menu_info[pos - 1] != "\\":
sub_menu_info = sub_menu_info[:pos] + sub_menu_info[pos + 1:] sub_menu_info = sub_menu_info[:pos] + sub_menu_info[pos + 1:]
if sub_menu_info == '---': if sub_menu_info == '---':
top_menu.add('separator') # top_menu.add('separator')
pass
else: else:
top_menu.add_command(label=sub_menu_info, underline=pos, try:
command=lambda: Menu.MenuItemChosenCallback(element, sub_menu_info)) item_without_key = sub_menu_info[:sub_menu_info.index(MENU_KEY_SEPARATOR)]
except:
item_without_key = sub_menu_info
if item_without_key[0] == MENU_DISABLED_CHARACTER:
menu_item = remi.gui.MenuItem(item_without_key[1:], width=100, height=30)
menu_item.set_enabled(False)
top_menu.append([menu_item,])
# TODO add callback here!
# TODO disable entry
else:
menu_item = remi.gui.MenuItem(item_without_key, width=100, height=30)
top_menu.append([menu_item,])
else: else:
i = 0 i = 0
while i < (len(sub_menu_info)): while i < (len(sub_menu_info)):
item = sub_menu_info[i] item = sub_menu_info[i]
if i != len(sub_menu_info) - 1: if i != len(sub_menu_info) - 1:
if type(sub_menu_info[i + 1]) == list: if type(sub_menu_info[i + 1]) == list:
new_menu = tk.Menu(top_menu, tearoff=element.Tearoff)
pos = sub_menu_info[i].find('&') pos = sub_menu_info[i].find('&')
if pos != -1: if pos != -1:
if pos == 0 or sub_menu_info[i][pos - 1] != "\\": if pos == 0 or sub_menu_info[i][pos - 1] != "\\":
sub_menu_info[i] = sub_menu_info[i][:pos] + sub_menu_info[i][pos + 1:] sub_menu_info[i] = sub_menu_info[i][:pos] + sub_menu_info[i][pos + 1:]
top_menu.add_cascade(label=sub_menu_info[i], menu=new_menu, underline=pos) if sub_menu_info[i][0] == MENU_DISABLED_CHARACTER:
AddMenuItem(new_menu, sub_menu_info[i + 1], element, is_sub_menu=True) new_menu = remi.gui.MenuItem(sub_menu_info[i][len(MENU_DISABLED_CHARACTER):], width=100, height=30)
i += 1 # skip the next one new_menu.set_enabled(False)
else:
AddMenuItem(top_menu, item, element) # TODO Disable Entry
else: else:
AddMenuItem(top_menu, item, element) new_menu = remi.gui.MenuItem(sub_menu_info[i], width=100, height=30)
i += 1
else: top_menu.append([new_menu,])
def AddMenuItem(top_menu, sub_menu_info, element, is_sub_menu=False, skip=False): return_val = new_menu
if isinstance(sub_menu_info, types.StringType):
if not is_sub_menu and not skip:
# print(f'Adding command {sub_menu_info}')
pos = sub_menu_info.find('&')
if pos != -1:
if pos == 0 or sub_menu_info[pos - 1] != "\\":
sub_menu_info = sub_menu_info[:pos] + sub_menu_info[pos + 1:]
if sub_menu_info == '---':
top_menu.add('separator')
else:
top_menu.add_command(label=sub_menu_info, underline=pos,
command=lambda: Menu.MenuItemChosenCallback(element, sub_menu_info))
else:
i = 0
while i < (len(sub_menu_info)):
item = sub_menu_info[i]
if i != len(sub_menu_info) - 1:
if not isinstance(sub_menu_info[i + 1], types.StringType):
new_menu = tk.Menu(top_menu, tearoff=element.Tearoff)
pos = sub_menu_info[i].find('&')
if pos != -1:
if pos == 0 or sub_menu_info[i][pos - 1] != "\\":
sub_menu_info[i] = sub_menu_info[i][:pos] + sub_menu_info[i][pos + 1:]
top_menu.add_cascade(label=sub_menu_info[i], menu=new_menu, underline=pos)
AddMenuItem(new_menu, sub_menu_info[i + 1], element, is_sub_menu=True) AddMenuItem(new_menu, sub_menu_info[i + 1], element, is_sub_menu=True)
i += 1 # skip the next one i += 1 # skip the next one
else: else:
@ -4242,6 +4251,7 @@ else:
else: else:
AddMenuItem(top_menu, item, element) AddMenuItem(top_menu, item, element)
i += 1 i += 1
return return_val
""" """
::::::::: :::::::::: ::: ::: ::::::::::: ::::::::: :::::::::: ::: ::: :::::::::::
@ -4832,25 +4842,48 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
# element._TKCanvas2.bind('<Motion>', element.MotionCallBack) # element._TKCanvas2.bind('<Motion>', element.MotionCallBack)
# ------------------------- MENUBAR element ------------------------- # # ------------------------- MENUBAR element ------------------------- #
elif element_type == ELEM_TYPE_MENUBAR: elif element_type == ELEM_TYPE_MENUBAR:
pass element = element # type: Menu
# menu_def = element.MenuDefinition menu = remi.gui.Menu(width='100%', height='30px')
# element.TKMenu = tk.Menu(toplevel_form.TKroot, tearoff=element.Tearoff) # create the menubar menu_def = element.MenuDefinition
# menubar = element.TKMenu for menu_entry in menu_def:
# for menu_entry in menu_def: # print(f'Adding a Menubar ENTRY {menu_entry}')
# # print(f'Adding a Menubar ENTRY {menu_entry}') pos = menu_entry[0].find('&')
# baritem = tk.Menu(menubar, tearoff=element.Tearoff) # print(pos)
# pos = menu_entry[0].find('&') if pos != -1:
# # print(pos) if pos == 0 or menu_entry[0][pos - 1] != "\\":
# if pos != -1: menu_entry[0] = menu_entry[0][:pos] + menu_entry[0][pos + 1:]
# if pos == 0 or menu_entry[0][pos - 1] != "\\": if menu_entry[0][0] == MENU_DISABLED_CHARACTER:
# menu_entry[0] = menu_entry[0][:pos] + menu_entry[0][pos + 1:] item = remi.gui.MenuItem(menu_entry[0][1:], width=100, height=30)
# menubar.add_cascade(label=menu_entry[0], menu=baritem, underline=pos) item.set_enabled(False)
# if len(menu_entry) > 1: else:
# AddMenuItem(baritem, menu_entry[1], element) item = remi.gui.MenuItem(menu_entry[0], width=100, height=30)
# toplevel_form.TKroot.configure(menu=element.TKMenu)
menu.append([item,])
if len(menu_entry) > 1:
AddMenuItem(item, menu_entry[1], element)
element.Widget = menubar = remi.gui.MenuBar(width='100%', height='30px')
element.Widget.style['z-index'] = '11'
element.Widget.style['order'] = '11'
menubar.append(menu)
tk_row_frame.append(element.Widget)
# ------------------------- Frame element ------------------------- # # ------------------------- Frame element ------------------------- #
elif element_type == ELEM_TYPE_FRAME: elif element_type == ELEM_TYPE_FRAME:
pass element = element # type: Frame
element.Widget = column_widget = remi.gui.VBox()
if element.Justification.startswith('c'):
# print('CENTERING')
column_widget.style['align-items'] = 'center'
column_widget.style['justify-content'] = 'center'
else:
column_widget.style['justify-content'] = 'flex-start'
column_widget.style['align-items'] = 'baseline'
if element.BackgroundColor not in (None, COLOR_SYSTEM_DEFAULT):
column_widget.style['background-color'] = element.BackgroundColor
PackFormIntoFrame(element, column_widget, toplevel_form)
tk_row_frame.append(element.Widget)
# labeled_frame = tk.LabelFrame(tk_row_frame, text=element.Title, relief=element.Relief) # labeled_frame = tk.LabelFrame(tk_row_frame, text=element.Title, relief=element.Relief)
# PackFormIntoFrame(element, labeled_frame, toplevel_form) # PackFormIntoFrame(element, labeled_frame, toplevel_form)
# labeled_frame.pack(side=tk.LEFT, padx=element.Pad[0], pady=element.Pad[1]) # labeled_frame.pack(side=tk.LEFT, padx=element.Pad[0], pady=element.Pad[1])
@ -4918,6 +4951,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
elif element_type == ELEM_TYPE_TAB_GROUP: elif element_type == ELEM_TYPE_TAB_GROUP:
element = element # type: TabGroup element = element # type: TabGroup
element.Widget = remi.gui.TabBox() element.Widget = remi.gui.TabBox()
# do_font_and_color(element.Widget)
PackFormIntoFrame(element ,element.Widget, toplevel_form) PackFormIntoFrame(element ,element.Widget, toplevel_form)
tk_row_frame.append(element.Widget) tk_row_frame.append(element.Widget)
@ -4975,41 +5009,11 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
do_font_and_color(element.Widget) do_font_and_color(element.Widget)
if element.ChangeSubmits: if element.ChangeSubmits:
element.Widget.onchange.connect(element.SliderCallback) element.Widget.onchange.connect(element.SliderCallback)
element.Widget.style['orientation'] = 'vertical'
element.Widget.attributes['orientation'] = 'vertical'
# print(f'slider = {element.Widget.style, element.Widget.attributes}')
tk_row_frame.append(element.Widget) # slider_length = element_size[0] * CharWidthInPixels() tk_row_frame.append(element.Widget) # slider_length = element_size[0] * CharWidthInPixels()
# slider_width = element_size[1]
# element.TKIntVar = tk.IntVar()
# element.TKIntVar.set(element.DefaultValue)
# if element.Orientation[0] == 'v':
# range_from = element.Range[1]
# range_to = element.Range[0]
# slider_length += DEFAULT_MARGINS[1] * (element_size[0] * 2) # add in the padding
# else:
# range_from = element.Range[0]
# range_to = element.Range[1]
# if element.ChangeSubmits:
# tkscale = tk.Scale(tk_row_frame, orient=element.Orientation, variable=element.TKIntVar,
# from_=range_from, to_=range_to, resolution=element.Resolution,
# length=slider_length, width=slider_width, bd=element.BorderWidth,
# relief=element.Relief, font=font, tickinterval=element.TickInterval,
# command=element.SliderChangedHandler)
# else:
# tkscale = tk.Scale(tk_row_frame, orient=element.Orientation, variable=element.TKIntVar,
# from_=range_from, to_=range_to, resolution=element.Resolution,
# length=slider_length, width=slider_width, bd=element.BorderWidth,
# relief=element.Relief, font=font, tickinterval=element.TickInterval)
# tkscale.config(highlightthickness=0)
# if element.BackgroundColor is not None and element.BackgroundColor != COLOR_SYSTEM_DEFAULT:
# tkscale.configure(background=element.BackgroundColor)
# if DEFAULT_SCROLLBAR_COLOR != COLOR_SYSTEM_DEFAULT:
# tkscale.config(troughcolor=DEFAULT_SCROLLBAR_COLOR)
# if text_color is not None and text_color != COLOR_SYSTEM_DEFAULT:
# tkscale.configure(fg=text_color)
# tkscale.pack(side=tk.LEFT, padx=element.Pad[0], pady=element.Pad[1])
# element.TKScale = tkscale
# if element.Disabled == True:
# element.TKScale['state'] = 'disabled'
# if element.Tooltip is not None:
# element.TooltipObject = ToolTip(element.TKScale, text=element.Tooltip, timeout=DEFAULT_TOOLTIP_TIME)
# ------------------------- TABLE element ------------------------- # # ------------------------- TABLE element ------------------------- #
elif element_type == ELEM_TYPE_TABLE: elif element_type == ELEM_TYPE_TABLE:
element = element # type: Table element = element # type: Table
@ -5294,6 +5298,7 @@ def setup_remi_window(app:Window.MyApp, window:Window):
print('* ERROR PACKING FORM *') print('* ERROR PACKING FORM *')
print(traceback.format_exc()) print(traceback.format_exc())
if window.BackgroundImage: if window.BackgroundImage:
master_widget.style['background-image'] = "url('{}')".format('/' + window.BackgroundImage) master_widget.style['background-image'] = "url('{}')".format('/' + window.BackgroundImage)
# print(f'background info',self.master_widget.attributes['background-image'] ) # print(f'background info',self.master_widget.attributes['background-image'] )
@ -5310,6 +5315,7 @@ def setup_remi_window(app:Window.MyApp, window:Window):
if window.ReturnKeyDownEvents: if window.ReturnKeyDownEvents:
app.page.children['body'].onkeydown.connect(window.on_key_down) app.page.children['body'].onkeydown.connect(window.on_key_down)
# if window.WindowIcon: # if window.WindowIcon:
# if type(window.WindowIcon) is bytes or len(window.WindowIcon) > 200: # if type(window.WindowIcon) is bytes or len(window.WindowIcon) > 200:
# app.page.children['head'].set_icon_data( base64_data=str(window.WindowIcon), mimetype="image/gif" ) # app.page.children['head'].set_icon_data( base64_data=str(window.WindowIcon), mimetype="image/gif" )
@ -5359,6 +5365,7 @@ def StartupTK(window:Window):
Window.active_windows.append(window) Window.active_windows.append(window)
Window.App = window.App Window.App = window.App
else: else:
# print('Starting second page')
# margin 0px auto allows to center the app to the screen # margin 0px auto allows to center the app to the screen
# master_widget = remi.gui.VBox() # master_widget = remi.gui.VBox()
# master_widget.style['justify-content'] = 'flex-start' # master_widget.style['justify-content'] = 'flex-start'
@ -6806,7 +6813,7 @@ def PopupGetFolder(message, default_path='', no_window=False, size=(None, None),
return folder_name return folder_name
layout = [[Text(message, auto_size_text=True, text_color=text_color, background_color=background_color)], layout = [[Text(message, auto_size_text=True, text_color=text_color, background_color=background_color)],
[InputText(default_text=default_path, size=size), FolderBrowse(initial_folder=initial_folder)], [InputText(default_text=default_path, size=size, key='_INPUT_'), FolderBrowse(initial_folder=initial_folder)],
[Button('Ok', size=(5, 1), bind_return_key=True), Button('Cancel', size=(5, 1))]] [Button('Ok', size=(5, 1), bind_return_key=True), Button('Cancel', size=(5, 1))]]
window = Window(title=message, icon=icon, auto_size_text=True, button_color=button_color, window = Window(title=message, icon=icon, auto_size_text=True, button_color=button_color,
@ -6874,22 +6881,23 @@ def PopupGetFile(message, default_path='', default_extension='', save_as=False,
file_types=file_types, initial_folder=initial_folder) file_types=file_types, initial_folder=initial_folder)
layout = [[Text(message, auto_size_text=True, text_color=text_color, background_color=background_color)], layout = [[Text(message, auto_size_text=True, text_color=text_color, background_color=background_color)],
[InputText(default_text=default_path, size=size), browse_button], [InputText(default_text=default_path, size=size, key='_INPUT_'), browse_button],
[Button('Ok', size=(6, 1), bind_return_key=True), Button('Cancel', size=(6, 1))]] [Button('Ok', size=(6, 1), bind_return_key=True), Button('Cancel', size=(6, 1))]]
window = Window(title=message, icon=icon, auto_size_text=True, button_color=button_color, font=font, window = Window(title=message, layout = layout, icon=icon, auto_size_text=True, button_color=button_color, font=font,
background_color=background_color, background_color=background_color,
no_titlebar=no_titlebar, grab_anywhere=grab_anywhere, keep_on_top=keep_on_top, location=location) no_titlebar=no_titlebar, grab_anywhere=grab_anywhere, keep_on_top=keep_on_top, location=location)
(button, input_values) = window.Layout(layout).Read() button, values = window.Read()
window.Close() window.Close()
if button != 'Ok': if button != 'Ok':
return None return None
else: else:
path = input_values[0] path = values['_INPUT_']
return path return path
# --------------------------- PopupGetText --------------------------- # --------------------------- PopupGetText ---------------------------
def PopupGetText(message, default_text='', password_char='', size=(None, None), button_color=None, def PopupGetText(message, default_text='', password_char='', size=(None, None), button_color=None,
@ -6914,19 +6922,22 @@ def PopupGetText(message, default_text='', password_char='', size=(None, None),
""" """
layout = [[Text(message, auto_size_text=True, text_color=text_color, background_color=background_color, font=font)], layout = [[Text(message, auto_size_text=True, text_color=text_color, background_color=background_color, font=font)],
[InputText(default_text=default_text, size=size, password_char=password_char)], [InputText(default_text=default_text, size=size, key='_INPUT_', password_char=password_char)],
[Button('Ok', size=(5, 1), bind_return_key=True), Button('Cancel', size=(5, 1))]] [Button('Ok', size=(5, 1), bind_return_key=True), Button('Cancel', size=(5, 1))]]
window = Window(title=message, icon=icon, auto_size_text=True, button_color=button_color, no_titlebar=no_titlebar, window = Window(title=message, layout=layout, icon=icon, auto_size_text=True, button_color=button_color, no_titlebar=no_titlebar,
background_color=background_color, grab_anywhere=grab_anywhere, keep_on_top=keep_on_top, background_color=background_color, grab_anywhere=grab_anywhere, keep_on_top=keep_on_top,
location=location) location=location)
(button, input_values) = window.Layout(layout).Read() button, values = window.Read()
window.Close() window.Close()
if button != 'Ok': if button != 'Ok':
return None return None
else: else:
return input_values[0] path = values['_INPUT_']
return path
def main(): def main():
@ -6943,9 +6954,26 @@ def main():
# [Text('Destination Folder', justification='right', size=(40,1)), InputText('Dest'), FolderBrowse()], # [Text('Destination Folder', justification='right', size=(40,1)), InputText('Dest'), FolderBrowse()],
# [Ok(), Cancel(disabled=True), Exit(tooltip='Exit button'), Button('Hidden Button', visible=False)]] # [Ok(), Cancel(disabled=True), Exit(tooltip='Exit button'), Button('Hidden Button', visible=False)]]
menu_def = [['&File', ['&Open', '&Save', 'E&xit', 'Properties']],
['&Edit', ['Paste', ['Special', 'Normal', ], '!Undo'], ],
['!&Disabled', ['Paste', ['Special', 'Normal', ], '!Undo'], ],
['&Help', '&About...'], ]
menu_def = [['File', ['&Open', '&Save', 'E&xit', 'Properties']],
['Edit', ['!Paste', ['Special', 'Normal', ], '!Undo'], ],
['!Disabled', ['Has Sub', ['Item1', 'Item2', ], 'No Sub'], ],
['Help', 'About...'], ]
col1 = [[Text('Column 1 line 1', background_color='red')], [Text('Column 1 line 2')]] col1 = [[Text('Column 1 line 1', background_color='red')], [Text('Column 1 line 2')]]
layout = [ layout = [
[Menu(menu_def)],
# [T('123435', size=(1,8))],
[Image(data=DEFAULT_BASE64_ICON)],
[Image(data=DEFAULT_BASE64_ICON)],
[Image(data=DEFAULT_BASE64_ICON)],
[Image(data=DEFAULT_BASE64_ICON)],
[Image(data=DEFAULT_BASE64_ICON)], [Image(data=DEFAULT_BASE64_ICON)],
[Text('PySimpleGUIWeb Welcomes You...', tooltip='text', font=('Comic sans ms', 20),size=(40,1), text_color='red', enable_events=True, key='_PySimpleGUIWeb_')], [Text('PySimpleGUIWeb Welcomes You...', tooltip='text', font=('Comic sans ms', 20),size=(40,1), text_color='red', enable_events=True, key='_PySimpleGUIWeb_')],
[T('Current Time '), Text('Text', key='_TEXT_', font='Arial 18', text_color='black', size=(30,1)), Column(col1, background_color='red')], [T('Current Time '), Text('Text', key='_TEXT_', font='Arial 18', text_color='black', size=(30,1)), Column(col1, background_color='red')],
@ -6964,7 +6992,10 @@ def main():
[OK(), Button('Hidden', visible=False, key='_HIDDEN_'), Button('Values'), Button('Exit', button_color=('white', 'red')), Button('UnHide'), B('Popup')] [OK(), Button('Hidden', visible=False, key='_HIDDEN_'), Button('Values'), Button('Exit', button_color=('white', 'red')), Button('UnHide'), B('Popup')]
] ]
window = Window('PySimpleGUIWeb Test Harness Window', layout, font='Arial 18',default_element_size=(12,1), window = Window('PySimpleGUIWeb Test Harness Window', layout,
font='Arial 18',
icon=DEFAULT_BASE64_ICON_OLD,
default_element_size=(12,1),
auto_size_buttons=False) auto_size_buttons=False)
start_time = datetime.datetime.now() start_time = datetime.datetime.now()