diff --git a/PySimpleGUI.py b/PySimpleGUI.py index 5fb80fd8..6697b16b 100644 --- a/PySimpleGUI.py +++ b/PySimpleGUI.py @@ -1,5 +1,5 @@ #!/usr/bin/python3 -__version__ = "4.0.0 Unreleased" +__version__ = "4.0.0" import sys @@ -1122,7 +1122,10 @@ class Radio(Element): self.TKIntVar.set(0) def Get(self): - """ """ + """ + A snapshot of the value of Radio Button (True/False) + :return: Bool - True is radio button is chosen + """ return self.TKIntVar.get() == self.EncodedRadioValue diff --git a/PySimpleGUI27.py b/PySimpleGUI27.py index 86bee6c3..a5f92c33 100644 --- a/PySimpleGUI27.py +++ b/PySimpleGUI27.py @@ -12,6 +12,8 @@ from builtins import str from builtins import object from future import standard_library standard_library.install_aliases() +__version__ = "2.0.0" + import sys if sys.version_info[0] >= 3: @@ -55,12 +57,14 @@ g_time_delta = 0 def TimerStart(): + """ """ global g_time_start g_time_start = time.time() def TimerStop(): + """ """ global g_time_delta, g_time_end g_time_end = time.time() @@ -256,7 +260,7 @@ BUTTON_TYPE_COLOR_CHOOSER = 40 BUTTON_TYPE_SHOW_DEBUGGER = 50 # ------------------------- Element types ------------------------- # -# class ElementType(Enum): + ELEM_TYPE_TEXT = 'text' ELEM_TYPE_INPUT_TEXT = 'input' ELEM_TYPE_INPUT_COMBO = 'combo' @@ -303,12 +307,17 @@ POPUP_BUTTONS_NO_BUTTONS = 5 # ------------------------------------------------------------------------- # class ToolTip(object): - """ Create a tooltip for a given widget - + """Create a tooltip for a given widget (inspired by https://stackoverflow.com/a/36221216) """ - def __init__(self, widget, text, timeout=DEFAULT_TOOLTIP_TIME): + """ + + :param widget: + :param text: + :param timeout: (Default value = DEFAULT_TOOLTIP_TIME) + + """ self.widget = widget self.text = text self.timeout = timeout @@ -321,24 +330,37 @@ class ToolTip(object): self.widget.bind("", self.leave) def enter(self, event=None): + """ + + :param event: (Default value = None) + + """ self.x = event.x self.y = event.y self.schedule() def leave(self, event=None): + """ + + :param event: (Default value = None) + + """ self.unschedule() self.hidetip() def schedule(self): + """ """ self.unschedule() self.id = self.widget.after(self.timeout, self.showtip) def unschedule(self): + """ """ if self.id: self.widget.after_cancel(self.id) self.id = None def showtip(self): + """ """ if self.tipwindow: return x = self.widget.winfo_rootx() + self.x + DEFAULT_TOOLTIP_OFFSET[0] @@ -353,6 +375,7 @@ class ToolTip(object): label.pack() def hidetip(self): + """ """ if self.tipwindow: self.tipwindow.destroy() self.tipwindow = None @@ -369,21 +392,26 @@ class ToolTip(object): # Element CLASS # # ------------------------------------------------------------------------- # class Element(object): - def __init__(self, type, size=(None, None), auto_size_text=None, font=None, background_color=None, text_color=None, - key=None, pad=None, tooltip=None, visible=True): - ''' - Element - :param type: - :param size: - :param auto_size_text: - :param font: - :param background_color: - :param text_color: - :param key: - :param pad: - :param tooltip: - :param visible: - ''' + """The base class for all Elements. + Holds the basic description of an Element like size and colors + + + """ + def __init__(self, type, size=(None, None), auto_size_text=None, font=None, background_color=None, text_color=None, key=None, pad=None, tooltip=None, visible=True): + """ + + :param type: + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: element's text color (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ self.Size = size self.Type = type self.AutoSizeText = auto_size_text @@ -410,10 +438,20 @@ class Element(object): self.Widget = None # Set when creating window. Has the main tkinter widget for element def _RightClickMenuCallback(self, event): + """ + + :param event: + + """ self.TKRightClickMenu.tk_popup(event.x_root, event.y_root, 0) self.TKRightClickMenu.grab_release() - def MenuItemChosenCallback(self, item_chosen): # TEXT Menu item callback + def _MenuItemChosenCallback(self, item_chosen): # TEXT Menu item callback + """ + + :param item_chosen: + + """ # print('IN MENU ITEM CALLBACK', item_chosen) self.MenuItemChosen = item_chosen.replace('&', '') self.ParentForm.LastButtonClicked = self.MenuItemChosen @@ -421,35 +459,45 @@ class Element(object): if self.ParentForm.CurrentlyRunningMainloop: self.ParentForm.TKroot.quit() # kick the users out of the mainloop - def FindReturnKeyBoundButton(self, form): + def _FindReturnKeyBoundButton(self, form): + """ + + :param form: + + """ for row in form.Rows: for element in row: if element.Type == ELEM_TYPE_BUTTON: if element.BindReturnKey: return element if element.Type == ELEM_TYPE_COLUMN: - rc = self.FindReturnKeyBoundButton(element) + rc = self._FindReturnKeyBoundButton(element) if rc is not None: return rc if element.Type == ELEM_TYPE_FRAME: - rc = self.FindReturnKeyBoundButton(element) + rc = self._FindReturnKeyBoundButton(element) if rc is not None: return rc if element.Type == ELEM_TYPE_TAB_GROUP: - rc = self.FindReturnKeyBoundButton(element) + rc = self._FindReturnKeyBoundButton(element) if rc is not None: return rc if element.Type == ELEM_TYPE_TAB: - rc = self.FindReturnKeyBoundButton(element) + rc = self._FindReturnKeyBoundButton(element) if rc is not None: return rc if element.Type == ELEM_TYPE_PANE: - rc = self.FindReturnKeyBoundButton(element) + rc = self._FindReturnKeyBoundButton(element) if rc is not None: return rc return None - def TextClickedHandler(self, event): + def _TextClickedHandler(self, event): + """ + + :param event: + + """ if self.Key is not None: self.ParentForm.LastButtonClicked = self.Key else: @@ -458,13 +506,23 @@ class Element(object): if self.ParentForm.CurrentlyRunningMainloop: self.ParentForm.TKroot.quit() # kick the users out of the mainloop - def ReturnKeyHandler(self, event): + def _ReturnKeyHandler(self, event): + """ + + :param event: + + """ MyForm = self.ParentForm - button_element = self.FindReturnKeyBoundButton(MyForm) + button_element = self._FindReturnKeyBoundButton(MyForm) if button_element is not None: button_element.ButtonCallBack() - def ListboxSelectHandler(self, event): + def _ListboxSelectHandler(self, event): + """ + + :param event: + + """ # first, get the results table built # modify the Results table in the parent FlexForm object if self.Key is not None: @@ -475,7 +533,12 @@ class Element(object): if self.ParentForm.CurrentlyRunningMainloop: self.ParentForm.TKroot.quit() # kick the users out of the mainloop - def ComboboxSelectHandler(self, event): + def _ComboboxSelectHandler(self, event): + """ + + :param event: + + """ # first, get the results table built # modify the Results table in the parent FlexForm object if self.Key is not None: @@ -486,7 +549,8 @@ class Element(object): if self.ParentForm.CurrentlyRunningMainloop: self.ParentForm.TKroot.quit() # kick the users out of the mainloop - def RadioHandler(self): + def _RadioHandler(self): + """ """ if self.Key is not None: self.ParentForm.LastButtonClicked = self.Key else: @@ -495,7 +559,8 @@ class Element(object): if self.ParentForm.CurrentlyRunningMainloop: self.ParentForm.TKroot.quit() - def CheckboxHandler(self): + def _CheckboxHandler(self): + """ """ if self.Key is not None: self.ParentForm.LastButtonClicked = self.Key else: @@ -504,7 +569,12 @@ class Element(object): if self.ParentForm.CurrentlyRunningMainloop: self.ParentForm.TKroot.quit() - def TabGroupSelectHandler(self, event): + def _TabGroupSelectHandler(self, event): + """ + + :param event: + + """ if self.Key is not None: self.ParentForm.LastButtonClicked = self.Key else: @@ -514,6 +584,11 @@ class Element(object): self.ParentForm.TKroot.quit() def _KeyboardHandler(self, event): + """ + + :param event: + + """ if self.Key is not None: self.ParentForm.LastButtonClicked = self.Key else: @@ -523,6 +598,11 @@ class Element(object): self.ParentForm.TKroot.quit() def _ClickHandler(self, event): + """ + + :param event: + + """ if self.Key is not None: self.ParentForm.LastButtonClicked = self.Key else: @@ -532,9 +612,15 @@ class Element(object): self.ParentForm.TKroot.quit() def SetTooltip(self, tooltip_text): + """ + + :param tooltip_text: + + """ self.TooltipObject = ToolTip(self.Widget, text=tooltip_text, timeout=DEFAULT_TOOLTIP_TIME) def __del__(self): + """ """ try: self.TKStringVar.__del__() except: @@ -557,30 +643,31 @@ class Element(object): # Input Class # # ---------------------------------------------------------------------- # class InputText(Element): + """Shows a single line of input.""" def __init__(self, default_text='', size=(None, None), disabled=False, password_char='', - justification=None, background_color=None, text_color=None, font=None, tooltip=None, + justification=None, background_color=None, text_color=None, font=None, tooltip=None, change_submits=False, enable_events=False, do_not_clear=True, key=None, focus=False, pad=None, right_click_menu=None, visible=True): - ''' - InputText - :param default_text: - :param size: - :param disabled: - :param password_char: - :param justification: - :param background_color: - :param text_color: - :param font: - :param tooltip: - :param change_submits: - :param enable_events: - :param do_not_clear: - :param key: - :param focus: - :param pad: - :param right_click_menu: - :param visible: - ''' + """ + + :param default_text: (Default value = '') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param disabled: set disable state for element (Default value = False) + :param password_char: Passwork character if this is a password field (Default value = '') + :param justification: justification for data display (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param change_submits: If True, pressing Enter key submits window- DEPRICATED DO NOT USE! (Default value = False) + :param enable_events: Turns on the element specific events. Use this instead of change_submits (Default value = False) + :param do_not_clear: see docx (Default value = True) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param visible: set visibility state of the element (Default value = True) + """ self.DefaultText = default_text self.PasswordCharacter = password_char bg = background_color if background_color is not None else DEFAULT_INPUT_ELEMENTS_COLOR @@ -591,10 +678,20 @@ class InputText(Element): self.Disabled = disabled self.ChangeSubmits = change_submits or enable_events self.RightClickMenu = right_click_menu + self.TKEntry = self.Widget = None # type: tk.Entry super().__init__(ELEM_TYPE_INPUT_TEXT, size=size, background_color=bg, text_color=fg, key=key, pad=pad, font=font, tooltip=tooltip, visible=visible) def Update(self, value=None, disabled=None, select=None, visible=None): + """ + + :param value: (Default value = None) + :param disabled: disable or enable state of the element (Default value = None) + :param select: (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ + #NOTE - Read or Finalize must be called on Window prior to Update call if disabled is True: self.TKEntry['state'] = 'readonly' elif disabled is False: @@ -613,19 +710,25 @@ class InputText(Element): self.TKEntry.pack() def Get(self): + """ """ try: text = self.TKStringVar.get() except: text = '' return text - def SetFocus(self): + def SetFocus(self, force=False): + """ """ try: - self.TKEntry.focus_set() + if force: + self.TKEntry.focus_force() + else: + self.TKEntry.focus_set() except: pass def __del__(self): + """ """ super().__del__() @@ -639,27 +742,29 @@ I = InputText # Combo # # ---------------------------------------------------------------------- # class Combo(Element): + """ComboBox Element""" def __init__(self, values, default_value=None, size=(None, None), auto_size_text=None, background_color=None, text_color=None, change_submits=False, enable_events=False, disabled=False, key=None, pad=None, tooltip=None, readonly=False, font=None, visible=True): - ''' - Combo - :param values: - :param default_value: - :param size: - :param auto_size_text: - :param background_color: - :param text_color: - :param change_submits: - :param enable_events: - :param disabled: - :param key: - :param pad: - :param tooltip: - :param readonly: - :param font: - :param visible: - ''' + """Combo Element - Combo Box, Drop Down Menu, ... + + :param values: + :param default_value: (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param disabled: set disable state for element (Default value = False) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param readonly: make element readonly (Default value = False) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ self.Values = values self.DefaultValue = default_value self.ChangeSubmits = change_submits or enable_events @@ -674,6 +779,16 @@ class Combo(Element): text_color=fg, key=key, pad=pad, tooltip=tooltip, font=font or DEFAULT_FONT, visible=visible) def Update(self, value=None, values=None, set_to_index=None, disabled=None, readonly=None, font=None, visible=None): + """ + :param value: (Default value = None) + :param values: (Default value = None) + :param set_to_index: (Default value = None) + :param disabled: disable or enable state of the element (Default value = None) + :param readonly: make element readonly (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ if values is not None: try: self.TKCombo['values'] = values @@ -712,6 +827,7 @@ class Combo(Element): self.TKCombo.pack() def __del__(self): + """ """ try: self.TKCombo.__del__() except: @@ -729,22 +845,30 @@ Drop = InputCombo # Option Menu # # ---------------------------------------------------------------------- # class OptionMenu(Element): + """Option Menu is an Element available ONLY on the tkinter port of PySimpleGUI. It's is a widget that is unique + to tkinter. However, it looks much like a ComboBox. Instead of an arrow to click to pull down the list of + choices, another little graphic is shown on the widget to indicate where you click. After clicking to activate, + it looks like a Combo Box that you scroll to select a choice. + + + """ def __init__(self, values, default_value=None, size=(None, None), disabled=False, auto_size_text=None, background_color=None, text_color=None, key=None, pad=None, tooltip=None, visible=True): - ''' - OptionMenu - :param values: - :param default_value: - :param size: - :param disabled: - :param auto_size_text: - :param background_color: - :param text_color: - :param key: - :param pad: - :param tooltip: - :param visible: - ''' + """Option Menu Element + + :param values: + :param default_value: (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param disabled: set disable state for element (Default value = False) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ self.Values = values self.DefaultValue = default_value self.TKOptionMenu = None # type: tk.OptionMenu @@ -756,6 +880,14 @@ class OptionMenu(Element): text_color=fg, key=key, pad=pad, tooltip=tooltip, visible=visible) def Update(self, value=None, values=None, disabled=None, visible=None): + """OptionMenu Element Update + + :param value: (Default value = None) + :param values: (Default value = None) + :param disabled: disable or enable state of the element (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ if values is not None: self.Values = values self.TKOptionMenu['menu'].delete(0, 'end') @@ -784,6 +916,7 @@ class OptionMenu(Element): self.TKOptionMenu.pack() def __del__(self): + """ """ try: self.TKOptionMenu.__del__() except: @@ -799,30 +932,36 @@ InputOptionMenu = OptionMenu # Listbox # # ---------------------------------------------------------------------- # class Listbox(Element): + """A List Box. Provide a list of values for the user to choose one or more of. Returns a list of selected rows + when a window.Read() is executed. + + + """ def __init__(self, values, default_values=None, select_mode=None, change_submits=False, enable_events=False, bind_return_key=False, size=(None, None), disabled=False, auto_size_text=None, font=None, background_color=None, text_color=None, key=None, pad=None, tooltip=None, right_click_menu=None, visible=True): - ''' - Listbox - :param values: - :param default_values: - :param select_mode: - :param change_submits: - :param enable_events: - :param bind_return_key: - :param size: - :param disabled: - :param auto_size_text: - :param font: - :param background_color: - :param text_color: - :param key: - :param pad: - :param tooltip: - :param right_click_menu: - :param visible: - ''' + """Listbox Element + + :param values: + :param default_values: (Default value = None) + :param select_mode: (Default value = None) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param bind_return_key: (Default value = False) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param disabled: set disable state for element (Default value = False) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ self.Values = values self.DefaultValues = default_values self.TKListbox = None @@ -843,10 +982,21 @@ class Listbox(Element): fg = text_color if text_color is not None else DEFAULT_INPUT_TEXT_COLOR self.RightClickMenu = right_click_menu self.vsb = None # type: tk.Scrollbar + self.TKListbox = self.Widget = None # type: tk.Listbox super().__init__(ELEM_TYPE_INPUT_LISTBOX, size=size, auto_size_text=auto_size_text, font=font, background_color=bg, text_color=fg, key=key, pad=pad, tooltip=tooltip, visible=visible) - def Update(self, values=None, disabled=None, set_to_index=None, visible=None): + def Update(self, values=None, disabled=None, set_to_index=None, scroll_to_index=None, visible=None): + """ + + :param values: (Default value = None) + :param disabled: disable or enable state of the element (Default value = None) + :param set_to_index: highlights the item at this index as if user clicked (Default value = None) + :param scroll_to_index: scroll the listbox so that this index is the first shown (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ + if disabled == True: self.TKListbox.configure(state='disabled') elif disabled == False: @@ -869,8 +1019,15 @@ class Listbox(Element): elif visible is True: self.TKListbox.pack() self.vsb.pack() + if scroll_to_index is not None and len(self.Values): + self.TKListbox.yview_moveto(scroll_to_index/len(self.Values)) def SetValue(self, values): + """ + + :param values: + + """ for index, item in enumerate(self.Values): try: if item in values: @@ -882,15 +1039,21 @@ class Listbox(Element): self.DefaultValues = values def GetListValues(self): + """ """ return self.Values - def SetFocus(self): + def SetFocus(self, force=False): + """ """ try: - self.TKListbox.focus_set() + if force: + self.TKListbox.focus_force() + else: + self.TKListbox.focus_set() except: pass def __del__(self): + """ """ try: self.TKListBox.__del__() except: @@ -902,27 +1065,34 @@ class Listbox(Element): # Radio # # ---------------------------------------------------------------------- # class Radio(Element): + """Radio Button Element - Used in a group of other Radio Elements to provide user with ability to select only + 1 choice in a list of choices. + + + """ def __init__(self, text, group_id, default=False, disabled=False, size=(None, None), auto_size_text=None, background_color=None, text_color=None, font=None, key=None, pad=None, tooltip=None, change_submits=False, enable_events=False, visible=True): - ''' - Radio - :param text: - :param group_id: - :param default: - :param disabled: - :param size: - :param auto_size_text: - :param background_color: - :param text_color: - :param font: - :param key: - :param pad: - :param tooltip: - :param change_submits: - :param enable_events: - :param visible: - ''' + """ + + :param text: + :param group_id: + :param default: (Default value = False) + :param disabled: set disable state for element (Default value = False) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param visible: set visibility state of the element (Default value = True) + + """ + self.InitialState = default self.Text = text self.TKRadio = None @@ -937,6 +1107,14 @@ class Radio(Element): tooltip=tooltip, visible=visible) def Update(self, value=None, disabled=None, visible=None): + """ + + :param value: (Default value = None) + :param disabled: disable or enable state of the element (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ + if value is not None: try: self.TKIntVar.set(self.EncodedRadioValue) @@ -953,9 +1131,19 @@ class Radio(Element): self.TKRadio.pack() def ResetGroup(self): + """ """ self.TKIntVar.set(0) + def Get(self): + """ + A snapshot of the value of Radio Button (True/False) + :return: Bool - True is radio button is chosen + """ + return self.TKIntVar.get() == self.EncodedRadioValue + + def __del__(self): + """ """ try: self.TKRadio.__del__() except: @@ -966,27 +1154,35 @@ class Radio(Element): # ---------------------------------------------------------------------- # # Checkbox # # ---------------------------------------------------------------------- # +########################################################################## +# June 15, 2019 - This is the last element that has been converted to use the new +# Doc strings +# Note - The renaming of the member function to have _ if internal only has NOT yet been done! class Checkbox(Element): + """ """ + def __init__(self, text, default=False, size=(None, None), auto_size_text=None, font=None, background_color=None, text_color=None, change_submits=False, enable_events=False, disabled=False, key=None, pad=None, tooltip=None, visible=True): - ''' - Checkbox - :param text: - :param default: - :param size: - :param auto_size_text: - :param font: - :param background_color: - :param text_color: - :param change_submits: - :param enable_events: - :param disabled: - :param key: - :param pad: - :param tooltip: - :param visible: - ''' + """ + + :param text: + :param default: (Default value = False) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param disabled: set disable state for element (Default value = False) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.Text = text self.InitialState = default self.Value = None @@ -1000,9 +1196,18 @@ class Checkbox(Element): tooltip=tooltip, visible=visible) def Get(self): + """ """ return self.TKIntVar.get() def Update(self, value=None, disabled=None, visible=None): + """ + + :param value: (Default value = None) + :param disabled: disable or enable state of the element (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ + if value is not None: try: self.TKIntVar.set(value) @@ -1019,6 +1224,7 @@ class Checkbox(Element): self.TKCheckbutton.pack() def __del__(self): + """ """ super().__del__() @@ -1033,28 +1239,30 @@ Check = Checkbox # ---------------------------------------------------------------------- # class Spin(Element): - # Values = None - # TKSpinBox = None + """ """ + def __init__(self, values, initial_value=None, disabled=False, change_submits=False, enable_events=False, size=(None, None), auto_size_text=None, font=None, background_color=None, text_color=None, key=None, pad=None, tooltip=None, visible=True): - ''' - Spin - :param values: - :param initial_value: - :param disabled: - :param change_submits: - :param enable_events: - :param size: - :param auto_size_text: - :param font: - :param background_color: - :param text_color: - :param key: - :param pad: - :param tooltip: - :param visible: - ''' + """ + + :param values: + :param initial_value: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.Values = values self.DefaultValue = initial_value self.ChangeSubmits = change_submits or enable_events @@ -1068,6 +1276,15 @@ class Spin(Element): return def Update(self, value=None, values=None, disabled=None, visible=None): + """ + + :param value: (Default value = None) + :param values: (Default value = None) + :param disabled: disable or enable state of the element (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ + if values != None: old_value = self.TKStringVar.get() self.Values = values @@ -1089,6 +1306,11 @@ class Spin(Element): self.TKSpinBox.pack() def SpinChangedHandler(self, event): + """ + + :param event: + + """ # first, get the results table built # modify the Results table in the parent FlexForm object if self.Key is not None: @@ -1100,9 +1322,11 @@ class Spin(Element): self.ParentForm.TKroot.quit() # kick the users out of the mainloop def Get(self): + """ """ return self.TKStringVar.get() def __del__(self): + """ """ try: self.TKSpinBox.__del__() except: @@ -1114,31 +1338,36 @@ class Spin(Element): # Multiline # # ---------------------------------------------------------------------- # class Multiline(Element): + """ """ + def __init__(self, default_text='', enter_submits=False, disabled=False, autoscroll=False, border_width=None, size=(None, None), auto_size_text=None, background_color=None, text_color=None, change_submits=False, enable_events=False, do_not_clear=True, key=None, focus=False, font=None, pad=None, tooltip=None, right_click_menu=None, visible=True): - ''' - Multiline - :param default_text: - :param enter_submits: - :param disabled: - :param autoscroll: - :param size: - :param auto_size_text: - :param background_color: - :param text_color: - :param change_submits: - :param enable_events: - :param do_not_clear: - :param key: - :param focus: - :param font: - :param pad: - :param tooltip: - :param right_click_menu: - :param visible: - ''' + """ + + :param default_text: (Default value = '') + :param enter_submits: (Default value = False) + :param disabled: set disable state for element (Default value = False) + :param autoscroll: (Default value = False) + :param border_width: (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param do_not_clear: see docx (Default value = True) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param focus: if focus should be set to this (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.DefaultText = default_text self.EnterSubmits = enter_submits @@ -1158,6 +1387,19 @@ class Multiline(Element): def Update(self, value=None, disabled=None, append=False, font=None, text_color=None, background_color=None, visible=None, autoscroll=None): + """ + + :param value: (Default value = None) + :param disabled: disable or enable state of the element (Default value = None) + :param append: (Default value = False) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param text_color: color of the text (Default value = None) + :param background_color: color of background (Default value = None) + :param visible: change visibility of element (Default value = None) + :param autoscroll: (Default value = None) + + """ + if autoscroll is not None: self.Autoscroll = autoscroll if value is not None: @@ -1190,15 +1432,22 @@ class Multiline(Element): self.TKText.pack() def Get(self): + """ """ return self.TKText.get(1.0, tk.END) - def SetFocus(self): + def SetFocus(self, force=False): + """ """ try: - self.TKText.focus_set() + if force: + self.TKText.focus_force() + else: + self.TKText.focus_set() except: pass + def __del__(self): + """ """ super().__del__() @@ -1206,27 +1455,31 @@ class Multiline(Element): # Text # # ---------------------------------------------------------------------- # class Text(Element): + """ """ + def __init__(self, text, size=(None, None), auto_size_text=None, click_submits=False, enable_events=False, relief=None, font=None, text_color=None, background_color=None, justification=None, pad=None, key=None, right_click_menu=None, tooltip=None, visible=True): - ''' - Text - :param text: - :param size: - :param auto_size_text: - :param click_submits: - :param enable_events: - :param relief: - :param font: - :param text_color: - :param background_color: - :param justification: - :param pad: - :param key: - :param right_click_menu: - :param tooltip: - :param visible: - ''' + """ + + :param text: + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param click_submits: (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param relief: (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param text_color: color of the text (Default value = None) + :param background_color: color of background (Default value = None) + :param justification: justification for data display (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.DisplayText = str(text) self.TextColor = text_color if text_color else DEFAULT_TEXT_COLOR self.Justification = justification @@ -1244,6 +1497,16 @@ class Text(Element): return def Update(self, value=None, background_color=None, text_color=None, font=None, visible=None): + """ + + :param value: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ + if value is not None: self.DisplayText = value stringvar = self.TKStringVar @@ -1260,6 +1523,7 @@ class Text(Element): self.TKText.pack() def __del__(self): + """ """ super().__del__() @@ -1272,26 +1536,30 @@ T = Text # StatusBar # # ---------------------------------------------------------------------- # class StatusBar(Element): + """ """ + def __init__(self, text, size=(None, None), auto_size_text=None, click_submits=None, enable_events=False, relief=RELIEF_SUNKEN, font=None, text_color=None, background_color=None, justification=None, pad=None, key=None, tooltip=None, visible=True): - ''' - StatusBar - :param text: - :param size: - :param auto_size_text: - :param click_submits: - :param enable_events: - :param relief: - :param font: - :param text_color: - :param background_color: - :param justification: - :param pad: - :param key: - :param tooltip: - :param visible: - ''' + """ + + :param text: + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param click_submits: (Default value = None) + :param enable_events: Turns on the element specific events.(Default value = False) + :param relief: (Default value = RELIEF_SUNKEN) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param text_color: color of the text (Default value = None) + :param background_color: color of background (Default value = None) + :param justification: justification for data display (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.DisplayText = text self.TextColor = text_color if text_color else DEFAULT_TEXT_COLOR self.Justification = justification @@ -1307,6 +1575,16 @@ class StatusBar(Element): return def Update(self, value=None, background_color=None, text_color=None, font=None, visible=None): + """ + + :param value: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ + if value is not None: self.DisplayText = value stringvar = self.TKStringVar @@ -1323,6 +1601,7 @@ class StatusBar(Element): self.TKText.pack() def __del__(self): + """ """ super().__del__() @@ -1332,9 +1611,24 @@ class StatusBar(Element): # ---------------------------------------------------------------------- # class TKProgressBar(object): + """ """ def __init__(self, root, max, length=400, width=DEFAULT_PROGRESS_BAR_SIZE[1], style=DEFAULT_PROGRESS_BAR_STYLE, relief=DEFAULT_PROGRESS_BAR_RELIEF, border_width=DEFAULT_PROGRESS_BAR_BORDER_WIDTH, orientation='horizontal', BarColor=(None, None), key=None): + """ + + :param root: + :param max: + :param length: (Default value = 400) + :param width: (Default value = DEFAULT_PROGRESS_BAR_SIZE[1]) + :param style: (Default value = DEFAULT_PROGRESS_BAR_STYLE) + :param relief: (Default value = DEFAULT_PROGRESS_BAR_RELIEF) + :param border_width: (Default value = DEFAULT_PROGRESS_BAR_BORDER_WIDTH) + :param orientation: (Default value = 'horizontal') + :param BarColor: (Default value = (None, None)) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ self.Length = length self.Width = width @@ -1370,6 +1664,12 @@ class TKProgressBar(object): length=length, orient=tk.VERTICAL, mode='determinate') def Update(self, count=None, max=None): + """ + + :param count: (Default value = None) + :param max: (Default value = None) + + """ if max is not None: self.Max = max try: @@ -1385,6 +1685,7 @@ class TKProgressBar(object): return True def __del__(self): + """ """ try: self.TKProgressBarForReal.__del__() except: @@ -1398,7 +1699,20 @@ class TKProgressBar(object): # Scroll bar will span the length of the frame # # ---------------------------------------------------------------------- # class TKOutput(tk.Frame): + """ """ def __init__(self, parent, width, height, bd, background_color=None, text_color=None, font=None, pad=None): + """ + + :param parent: + :param width: + :param height: + :param bd: + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + + """ self.frame = tk.Frame(parent) tk.Frame.__init__(self, self.frame) self.output = tk.Text(self.frame, width=width, height=height, bd=bd, font=font) @@ -1420,6 +1734,11 @@ class TKOutput(tk.Frame): self.pack() def write(self, txt): + """ + + :param txt: + + """ try: self.output.insert(tk.END, str(txt)) self.output.see(tk.END) @@ -1427,14 +1746,17 @@ class TKOutput(tk.Frame): pass def Close(self): + """ """ sys.stdout = self.previous_stdout sys.stderr = self.previous_stderr def flush(self): + """ """ sys.stdout = self.previous_stdout sys.stderr = self.previous_stderr def __del__(self): + """ """ sys.stdout = self.previous_stdout sys.stderr = self.previous_stderr @@ -1444,20 +1766,24 @@ class TKOutput(tk.Frame): # Routes stdout, stderr to a scrolled window # # ---------------------------------------------------------------------- # class Output(Element): + """ """ + def __init__(self, size=(None, None), background_color=None, text_color=None, pad=None, font=None, tooltip=None, key=None, right_click_menu=None, visible=True): - ''' - Output - :param size: - :param background_color: - :param text_color: - :param pad: - :param font: - :param tooltip: - :param key: - :param right_click_menu: - :param visible: - ''' + """ + + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self._TKOut = None bg = background_color if background_color else DEFAULT_INPUT_ELEMENTS_COLOR fg = text_color if text_color is not None else DEFAULT_INPUT_TEXT_COLOR @@ -1468,12 +1794,20 @@ class Output(Element): @property def TKOut(self): + """ """ if self._TKOut is None: print('*** Did you forget to call Finalize()? Your code should look something like: ***') print('*** form = sg.Window("My Form").Layout(layout).Finalize() ***') return self._TKOut def Update(self, value=None, visible=None): + """ + + :param value: (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ + if value is not None: self._TKOut.output.delete('1.0', tk.END) self._TKOut.output.insert(tk.END, value) @@ -1483,6 +1817,7 @@ class Output(Element): self._TKOut.frame.pack() def __del__(self): + """ """ try: self._TKOut.__del__() except: @@ -1494,37 +1829,41 @@ class Output(Element): # Button Class # # ---------------------------------------------------------------------- # class Button(Element): + """ """ + def __init__(self, button_text='', button_type=BUTTON_TYPE_READ_FORM, target=(None, None), tooltip=None, file_types=(("ALL Files", "*.*"),), initial_folder=None, disabled=False, change_submits=False, enable_events=False, image_filename=None, image_data=None, image_size=(None, None), image_subsample=None, border_width=None, size=(None, None), auto_size_button=None, button_color=None, font=None, bind_return_key=False, focus=False, pad=None, key=None, visible=True): - ''' - Button - :param button_text: - :param button_type: - :param target: - :param tooltip: - :param file_types: - :param initial_folder: - :param disabled: - :param change_submits: - :param enable_events: - :param image_filename: - :param image_data: - :param image_size: - :param image_subsample: - :param border_width: - :param size: - :param auto_size_button: - :param button_color: - :param font: - :param bind_return_key: - :param focus: - :param pad: - :param key: - :param visible: - ''' + """ + + :param button_text: (Default value = '') + :param button_type: (Default value = BUTTON_TYPE_READ_FORM) + :param target: + :param tooltip: text, that will appear the you hover on (Default value = None) + :param file_types: (Default value = (("ALL Files", "*.*"),)) + :param initial_folder: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param image_filename: (Default value = None) + :param image_data: (Default value = None) + :param image_size: (Default value = (None) + :param image_subsample: (Default value = None) + :param border_width: (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param bind_return_key: (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.AutoSizeButton = auto_size_button self.BType = button_type self.FileTypes = file_types @@ -1554,11 +1893,21 @@ class Button(Element): # Realtime button release callback def ButtonReleaseCallBack(self, parm): + """ + + :param parm: + + """ self.LastButtonClickedWasRealtime = False self.ParentForm.LastButtonClicked = None # Realtime button callback def ButtonPressCallBack(self, parm): + """ + + :param parm: + + """ self.ParentForm.LastButtonClickedWasRealtime = True if self.Key is not None: self.ParentForm.LastButtonClicked = self.Key @@ -1569,6 +1918,7 @@ class Button(Element): # ------- Button Callback ------- # def ButtonCallBack(self): + """ """ # global _my_windows # print('Button callback') @@ -1695,7 +2045,7 @@ class Button(Element): pass elif self.BType == BUTTON_TYPE_SHOW_DEBUGGER: if self.ParentForm.DebuggerEnabled: - Debugger.debugger._build_floating_window() + _Debugger.debugger._build_floating_window() # show_debugger_window() if should_submit_window: @@ -1708,6 +2058,19 @@ class Button(Element): def Update(self, text=None, button_color=(None, None), disabled=None, image_data=None, image_filename=None, visible=None, image_subsample=None, image_size=None): + """ + + :param text: (Default value = None) + :param button_color: (Default value = (None) + :param disabled: disable or enable state of the element (Default value = None) + :param image_data: (Default value = None) + :param image_filename: (Default value = None) + :param visible: change visibility of element (Default value = None) + :param image_subsample: (Default value = None) + :param image_size: (Default value = None) + + """ + try: if text is not None: self.TKButton.configure(text=text) @@ -1716,6 +2079,7 @@ class Button(Element): print('Button.Update *** WARNING - Button colors are not supported on the Mac***') if button_color != (None, None): self.TKButton.config(foreground=button_color[0], background=button_color[1]) + self.ButtonColor = button_color except: return if disabled == True: @@ -1749,18 +2113,25 @@ class Button(Element): self.TKButton.pack() def GetText(self): + """ """ return self.ButtonText - def SetFocus(self): + def SetFocus(self, force=False): + """ """ try: - self.TKButton.focus_set() + if force: + self.TKButton.focus_force() + else: + self.TKButton.focus_set() except: pass + def Click(self): - """ - Generates a click of the button as if the user clicked the button + """Generates a click of the button as if the user clicked the button :return: + + """ try: self.TKButton.invoke() @@ -1768,6 +2139,7 @@ class Button(Element): print('Exception clicking button') def __del__(self): + """ """ try: self.TKButton.__del__() except: @@ -1785,30 +2157,34 @@ Butt = Button # ButtonMenu Class # # ---------------------------------------------------------------------- # class ButtonMenu(Element): + """ """ + def __init__(self, button_text, menu_def, tooltip=None, disabled=False, image_filename=None, image_data=None, image_size=(None, None), image_subsample=None, border_width=None, size=(None, None), auto_size_button=None, button_color=None, font=None, pad=None, key=None, tearoff=False, visible=True): - ''' - ButtonMenu - :param button_text: - :param menu_def: - :param tooltip: - :param disabled: - :param image_filename: - :param image_data: - :param image_size: - :param image_subsample: - :param border_width: - :param size: - :param auto_size_button: - :param button_color: - :param font: - :param pad: - :param key: - :param tearoff: - :param visible: - ''' + """ + + :param button_text: + :param menu_def: + :param tooltip: text, that will appear the you hover on (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param image_filename: (Default value = None) + :param image_data: (Default value = None) + :param image_size: (Default value = (None, None)) + :param image_subsample: (Default value = None) + :param border_width: (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param tearoff: (Default value = False) + :param visible: set visibility state of the element (Default value = True) + + """ + self.MenuDefinition = menu_def self.AutoSizeButton = auto_size_button self.ButtonText = button_text @@ -1832,7 +2208,12 @@ class ButtonMenu(Element): text_color=self.TextColor, background_color=self.BackgroundColor, visible=visible) return - def MenuItemChosenCallback(self, item_chosen): # ButtonMenu Menu Item Chosen Callback + def _MenuItemChosenCallback(self, item_chosen): # ButtonMenu Menu Item Chosen Callback + """ + + :param item_chosen: + + """ # print('IN MENU ITEM CALLBACK', item_chosen) self.MenuItemChosen = item_chosen.replace('&', '') self.ParentForm.LastButtonClicked = self.Key @@ -1841,6 +2222,13 @@ class ButtonMenu(Element): self.ParentForm.TKroot.quit() # kick the users out of the mainloop def Update(self, menu_definition, visible=None): + """ + + :param menu_definition: + :param visible: change visibility of element (Default value = None) + + """ + self.MenuDefinition = menu_definition if menu_definition is not None: self.TKMenu = tk.Menu(self.TKButtonMenu, tearoff=self.Tearoff) # create the menubar @@ -1852,6 +2240,7 @@ class ButtonMenu(Element): self.TKButtonMenu.pack() def __del__(self): + """ """ try: self.TKButton.__del__() except: @@ -1863,22 +2252,25 @@ class ButtonMenu(Element): # ProgreessBar # # ---------------------------------------------------------------------- # class ProgressBar(Element): + """ """ def __init__(self, max_value, orientation=None, size=(None, None), auto_size_text=None, bar_color=(None, None), style=None, border_width=None, relief=None, key=None, pad=None, visible=True): - ''' - ProgressBar - :param max_value: - :param orientation: - :param size: - :param auto_size_text: - :param bar_color: - :param style: - :param border_width: - :param relief: - :param key: - :param pad: - :param visible: - ''' + """ + + :param max_value: + :param orientation: (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param bar_color: (Default value = (None) + :param style: (Default value = None) + :param border_width: (Default value = None) + :param relief: (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.MaxValue = max_value self.TKProgressBar = None self.Cancelled = False @@ -1894,6 +2286,13 @@ class ProgressBar(Element): # returns False if update failed def UpdateBar(self, current_count, max=None): + """ + + :param current_count: + :param max: (Default value = None) + + """ + if self.ParentForm.TKrootDestroyed: return False self.TKProgressBar.Update(current_count, max=max) @@ -1906,12 +2305,19 @@ class ProgressBar(Element): return True def Update(self, visible=None): + """ + + :param visible: change visibility of element (Default value = None) + + """ + if visible is False: self.TKProgressBar.TKProgressBarForReal.pack_forget() elif visible is True: self.TKProgressBar.TKProgressBarForReal.pack() def __del__(self): + """ """ try: self.TKProgressBar.__del__() except: @@ -1923,21 +2329,25 @@ class ProgressBar(Element): # Image # # ---------------------------------------------------------------------- # class Image(Element): + """ """ + def __init__(self, filename=None, data=None, background_color=None, size=(None, None), pad=None, key=None, tooltip=None, right_click_menu=None, visible=True, enable_events=False): - ''' - Image - :param filename: - :param data: - :param background_color: - :param size: - :param pad: - :param key: - :param tooltip: - :param right_click_menu: - :param visible: - :param enable_events: - ''' + """ + + :param filename: (Default value = None) + :param data: (Default value = None) + :param background_color: color of background (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param visible: set visibility state of the element (Default value = True) + :param enable_events: Turns on the element specific events.(Default value = False) + + """ + self.Filename = filename self.Data = data self.tktext_label = None @@ -1957,6 +2367,15 @@ class Image(Element): return def Update(self, filename=None, data=None, size=(None, None), visible=None): + """ + + :param filename: (Default value = None) + :param data: (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param visible: change visibility of element (Default value = None) + + """ + if filename is not None: image = tk.PhotoImage(file=filename) elif data is not None: @@ -1979,6 +2398,13 @@ class Image(Element): self.tktext_label.pack() def UpdateAnimation(self, source, time_between_frames=0): + """ + + :param source: + :param time_between_frames: (Default value = 0) + + """ + if self.Source != source: self.AnimatedFrames = None self.Source = source @@ -2019,6 +2445,7 @@ class Image(Element): pass def __del__(self): + """ """ super().__del__() @@ -2026,19 +2453,23 @@ class Image(Element): # Canvas # # ---------------------------------------------------------------------- # class Canvas(Element): + """ """ + def __init__(self, canvas=None, background_color=None, size=(None, None), pad=None, key=None, tooltip=None, right_click_menu=None, visible=True): - ''' - Canvas - :param canvas: - :param background_color: - :param size: - :param pad: - :param key: - :param tooltip: - :param right_click_menu: - :param visible: - ''' + """ + + :param canvas: (Default value = None) + :param background_color: color of background (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.BackgroundColor = background_color if background_color is not None else DEFAULT_BACKGROUND_COLOR self._TKCanvas = canvas self.RightClickMenu = right_click_menu @@ -2049,12 +2480,14 @@ class Canvas(Element): @property def TKCanvas(self): + """ """ if self._TKCanvas is None: print('*** Did you forget to call Finalize()? Your code should look something like: ***') print('*** form = sg.Window("My Form").Layout(layout).Finalize() ***') return self._TKCanvas def __del__(self): + """ """ super().__del__() @@ -2062,24 +2495,28 @@ class Canvas(Element): # Graph # # ---------------------------------------------------------------------- # class Graph(Element): + """ """ + def __init__(self, canvas_size, graph_bottom_left, graph_top_right, background_color=None, pad=None, change_submits=False, drag_submits=False, enable_events=False, key=None, tooltip=None, right_click_menu=None, visible=True): - ''' - Graph - :param canvas_size: - :param graph_bottom_left: - :param graph_top_right: - :param background_color: - :param pad: - :param change_submits: - :param drag_submits: - :param enable_events: - :param key: - :param tooltip: - :param right_click_menu: - :param visible: - ''' + """ + + :param canvas_size: + :param graph_bottom_left: + :param graph_top_right: + :param background_color: color of background (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param drag_submits: (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.CanvasSize = canvas_size self.BottomLeft = graph_bottom_left self.TopRight = graph_top_right @@ -2097,6 +2534,12 @@ class Graph(Element): return def _convert_xy_to_canvas_xy(self, x_in, y_in): + """ + + :param x_in: + :param y_in: + + """ if None in (x_in, y_in): return None, None scale_x = (self.CanvasSize[0] - 0) / (self.TopRight[0] - self.BottomLeft[0]) @@ -2106,6 +2549,12 @@ class Graph(Element): return new_x, new_y def _convert_canvas_xy_to_xy(self, x_in, y_in): + """ + + :param x_in: + :param y_in: + + """ if None in (x_in, y_in): return None, None scale_x = (self.CanvasSize[0] - 0) / (self.TopRight[0] - self.BottomLeft[0]) @@ -2116,6 +2565,14 @@ class Graph(Element): return int(new_x), int(new_y) def DrawLine(self, point_from, point_to, color='black', width=1): + """ + + :param point_from: + :param point_to: + :param color: (Default value = 'black') + :param width: (Default value = 1) + + """ if point_from == (None, None): return converted_point_from = self._convert_xy_to_canvas_xy(point_from[0], point_from[1]) @@ -2131,6 +2588,13 @@ class Graph(Element): return id def DrawPoint(self, point, size=2, color='black'): + """ + + :param point: + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = 2) + :param color: (Default value = 'black') + + """ if point == (None, None): return converted_point = self._convert_xy_to_canvas_xy(point[0], point[1]) @@ -2147,6 +2611,14 @@ class Graph(Element): return def DrawCircle(self, center_location, radius, fill_color=None, line_color='black'): + """ + + :param center_location: + :param radius: + :param fill_color: (Default value = None) + :param line_color: (Default value = 'black') + + """ if center_location == (None, None): return converted_point = self._convert_xy_to_canvas_xy(center_location[0], center_location[1]) @@ -2163,6 +2635,14 @@ class Graph(Element): return id def DrawOval(self, top_left, bottom_right, fill_color=None, line_color=None): + """ + + :param top_left: + :param bottom_right: + :param fill_color: (Default value = None) + :param line_color: (Default value = None) + + """ converted_top_left = self._convert_xy_to_canvas_xy(top_left[0], top_left[1]) converted_bottom_right = self._convert_xy_to_canvas_xy(bottom_right[0], bottom_right[1]) if self._TKCanvas2 is None: @@ -2178,6 +2658,16 @@ class Graph(Element): return id def DrawArc(self, top_left, bottom_right, extent, start_angle, style=None, arc_color='black'): + """ + + :param top_left: + :param bottom_right: + :param extent: + :param start_angle: + :param style: (Default value = None) + :param arc_color: (Default value = 'black') + + """ converted_top_left = self._convert_xy_to_canvas_xy(top_left[0], top_left[1]) converted_bottom_right = self._convert_xy_to_canvas_xy(bottom_right[0], bottom_right[1]) tkstyle = tk.PIESLICE if style is None else style @@ -2194,6 +2684,14 @@ class Graph(Element): return id def DrawRectangle(self, top_left, bottom_right, fill_color=None, line_color=None): + """ + + :param top_left: + :param bottom_right: + :param fill_color: (Default value = None) + :param line_color: (Default value = None) + + """ converted_top_left = self._convert_xy_to_canvas_xy(top_left[0], top_left[1]) converted_bottom_right = self._convert_xy_to_canvas_xy(bottom_right[0], bottom_right[1]) if self._TKCanvas2 is None: @@ -2209,6 +2707,15 @@ class Graph(Element): return id def DrawText(self, text, location, color='black', font=None, angle=0): + """ + + :param text: + :param location: + :param color: (Default value = 'black') + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param angle: (Default value = 0) + + """ if location == (None, None): return converted_point = self._convert_xy_to_canvas_xy(location[0], location[1]) @@ -2224,6 +2731,16 @@ class Graph(Element): return id def DrawImage(self, filename=None, data=None, location=(None, None), color='black', font=None, angle=0): + """ + + :param filename: (Default value = None) + :param data: (Default value = None) + :param location: (Default value = (None, None)) + :param color: (Default value = 'black') + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param angle: (Default value = 0) + + """ if location == (None, None): return if filename is not None: @@ -2247,6 +2764,7 @@ class Graph(Element): return id def Erase(self): + """ """ if self._TKCanvas2 is None: print('*** WARNING - The Graph element has not been finalized and cannot be drawn upon ***') print('Call Window.Finalize() prior to this operation') @@ -2258,6 +2776,11 @@ class Graph(Element): pass def DeleteFigure(self, id): + """ + + :param id: + + """ try: self._TKCanvas2.delete(id) except: @@ -2267,6 +2790,12 @@ class Graph(Element): except: pass def Update(self, background_color, visible=None): + """ + + :param background_color: color of background + :param visible: change visibility of element (Default value = None) + + """ if self._TKCanvas2 is None: print('*** WARNING - The Graph element has not been finalized and cannot be drawn upon ***') print('Call Window.Finalize() prior to this operation') @@ -2278,6 +2807,12 @@ class Graph(Element): self._TKCanvas2.pack() def Move(self, x_direction, y_direction): + """ + + :param x_direction: + :param y_direction: + + """ zero_converted = self._convert_xy_to_canvas_xy(0, 0) shift_converted = self._convert_xy_to_canvas_xy(x_direction, y_direction) shift_amount = (shift_converted[0] - zero_converted[0], shift_converted[1] - zero_converted[1]) @@ -2288,6 +2823,13 @@ class Graph(Element): self._TKCanvas2.move('all', shift_amount[0], shift_amount[1]) def MoveFigure(self, figure, x_direction, y_direction): + """ + + :param figure: + :param x_direction: + :param y_direction: + + """ zero_converted = self._convert_xy_to_canvas_xy(0, 0) shift_converted = self._convert_xy_to_canvas_xy(x_direction, y_direction) shift_amount = (shift_converted[0] - zero_converted[0], shift_converted[1] - zero_converted[1]) @@ -2298,6 +2840,13 @@ class Graph(Element): self._TKCanvas2.move(figure, shift_amount[0], shift_amount[1]) def RelocateFigure(self, figure, x, y): + """ + + :param figure: + :param x: + :param y: + + """ zero_converted = self._convert_xy_to_canvas_xy(0, 0) shift_converted = self._convert_xy_to_canvas_xy(x, y) shift_amount = (shift_converted[0] - zero_converted[0], shift_converted[1] - zero_converted[1]) @@ -2310,6 +2859,7 @@ class Graph(Element): @property def TKCanvas(self): + """ """ if self._TKCanvas2 is None: print('*** Did you forget to call Finalize()? Your code should look something like: ***') print('*** form = sg.Window("My Form").Layout(layout).Finalize() ***') @@ -2317,6 +2867,11 @@ class Graph(Element): # Realtime button release callback def ButtonReleaseCallBack(self, event): + """ + + :param event: + + """ self.ClickPosition = (None, None) self.LastButtonClickedWasRealtime = not self.DragSubmits if self.Key is not None: @@ -2331,6 +2886,11 @@ class Graph(Element): # Realtime button callback def ButtonPressCallBack(self, event): + """ + + :param event: + + """ self.ClickPosition = self._convert_canvas_xy_to_xy(event.x, event.y) self.ParentForm.LastButtonClickedWasRealtime = self.DragSubmits if self.Key is not None: @@ -2343,6 +2903,11 @@ class Graph(Element): # Realtime button callback def MotionCallBack(self, event): + """ + + :param event: + + """ if not self.MouseButtonDown: return self.ClickPosition = self._convert_canvas_xy_to_xy(event.x, event.y) @@ -2354,11 +2919,19 @@ class Graph(Element): if self.ParentForm.CurrentlyRunningMainloop: self.ParentForm.TKroot.quit() # kick out of loop if read was called - def SetFocus(self): - self._TKCanvas2.focus_set() - # self._TKCanvas2.focus_force() + def SetFocus(self, force=False): + """ """ + try: + if force: + self._TKCanvas2.focus_force() + else: + self._TKCanvas2.focus_set() + except: + pass + def __del__(self): + """ """ super().__del__() @@ -2366,26 +2939,30 @@ class Graph(Element): # Frame # # ---------------------------------------------------------------------- # class Frame(Element): + """ """ + def __init__(self, title, layout, title_color=None, background_color=None, title_location=None, relief=DEFAULT_FRAME_RELIEF, size=(None, None), font=None, pad=None, border_width=None, key=None, tooltip=None, right_click_menu=None, visible=True): - ''' - Frame - :param title: - :param layout: - :param title_color: - :param background_color: - :param title_location: - :param relief: - :param size: - :param font: - :param pad: - :param border_width: - :param key: - :param tooltip: - :param right_click_menu: - :param visible: - ''' + """ + + :param title: + :param layout: + :param title_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param title_location: (Default value = None) + :param relief: (Default value = DEFAULT_FRAME_RELIEF) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param border_width: (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.UseDictionary = False self.ReturnValues = None self.ReturnValuesList = [] @@ -2410,7 +2987,11 @@ class Frame(Element): return def AddRow(self, *args): - ''' Parms are a variable number of Elements ''' + """Parms are a variable number of Elements + + :param *args: + + """ NumRows = len(self.Rows) # number of existing rows is our row number CurrentRowNumber = NumRows # this row's number CurrentRow = [] # start with a blank row and build up @@ -2425,22 +3006,39 @@ class Frame(Element): self.Rows.append(CurrentRow) def Layout(self, rows): + """ + + :param rows: + + """ for row in rows: self.AddRow(*row) def _GetElementAtLocation(self, location): + """ + + :param location: + + """ (row_num, col_num) = location row = self.Rows[row_num] element = row[col_num] return element def Update(self, visible=None): + """ + + :param visible: change visibility of element (Default value = None) + + """ + if visible is False: self.TKFrame.pack_forget() elif visible is True: self.TKFrame.pack() def __del__(self): + """ """ for row in self.Rows: for element in row: element.__del__() @@ -2452,16 +3050,21 @@ class Frame(Element): # Routes stdout, stderr to a scrolled window # # ---------------------------------------------------------------------- # class VerticalSeparator(Element): + """ """ + def __init__(self, pad=None): - ''' - VerticalSeperator - A separator that spans only 1 row in a vertical fashion - :param pad: - ''' + """ + + :param pad: (common_key) Amount of padding to put around element (Default value = None) + + """ + self.Orientation = 'vertical' # for now only vertical works super().__init__(ELEM_TYPE_SEPARATOR, pad=pad) def __del__(self): + """ """ super().__del__() @@ -2473,23 +3076,27 @@ VSep = VerticalSeparator # Tab # # ---------------------------------------------------------------------- # class Tab(Element): + """ """ + def __init__(self, title, layout, title_color=None, background_color=None, font=None, pad=None, disabled=False, border_width=None, key=None, tooltip=None, right_click_menu=None, visible=True): - ''' - Tab - :param title: - :param layout: - :param title_color: - :param background_color: - :param font: - :param pad: - :param disabled: - :param border_width: - :param key: - :param tooltip: - :param right_click_menu: - :param visible: - ''' + """ + + :param title: + :param layout: + :param title_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param border_width: (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.UseDictionary = False self.ReturnValues = None self.ReturnValuesList = [] @@ -2514,7 +3121,11 @@ class Tab(Element): return def AddRow(self, *args): - ''' Parms are a variable number of Elements ''' + """Parms are a variable number of Elements + + :param *args: + + """ NumRows = len(self.Rows) # number of existing rows is our row number CurrentRowNumber = NumRows # this row's number CurrentRow = [] # start with a blank row and build up @@ -2529,11 +3140,23 @@ class Tab(Element): self.Rows.append(CurrentRow) def Layout(self, rows): + """ + + :param rows: + + """ for row in rows: self.AddRow(*row) return self def Update(self, disabled=None, visible=None): # TODO Disable / enable of tabs is not complete + """ + + :param disabled: disable or enable state of the element (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ + if disabled is None: return self.Disabled = disabled @@ -2546,12 +3169,18 @@ class Tab(Element): return self def _GetElementAtLocation(self, location): + """ + + :param location: + + """ (row_num, col_num) = location row = self.Rows[row_num] element = row[col_num] return element def __del__(self): + """ """ for row in self.Rows: for element in row: element.__del__() @@ -2562,26 +3191,30 @@ class Tab(Element): # TabGroup # # ---------------------------------------------------------------------- # class TabGroup(Element): + """ """ + def __init__(self, layout, tab_location=None, title_color=None, selected_title_color=None, background_color=None, font=None, change_submits=False, enable_events=False, pad=None, border_width=None, theme=None, key=None, tooltip=None, visible=True): - ''' - TabGroup - :param layout: - :param tab_location: - :param title_color: - :param selected_title_color: - :param background_color: - :param font: - :param change_submits: - :param enable_events: - :param pad: - :param border_width: - :param theme: - :param key: - :param tooltip: - :param visible: - ''' + """ + + :param layout: + :param tab_location: (Default value = None) + :param title_color: (Default value = None) + :param selected_title_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param border_width: (Default value = None) + :param theme: (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.UseDictionary = False self.ReturnValues = None self.ReturnValuesList = [] @@ -2606,7 +3239,11 @@ class TabGroup(Element): return def AddRow(self, *args): - ''' Parms are a variable number of Elements ''' + """Parms are a variable number of Elements + + :param *args: + + """ NumRows = len(self.Rows) # number of existing rows is our row number CurrentRowNumber = NumRows # this row's number CurrentRow = [] # start with a blank row and build up @@ -2621,16 +3258,31 @@ class TabGroup(Element): self.Rows.append(CurrentRow) def Layout(self, rows): + """ + + :param rows: + + """ for row in rows: self.AddRow(*row) def _GetElementAtLocation(self, location): + """ + + :param location: + + """ (row_num, col_num) = location row = self.Rows[row_num] element = row[col_num] return element def FindKeyFromTabName(self, tab_name): + """ + + :param tab_name: + + """ for row in self.Rows: for element in row: if element.Title == tab_name: @@ -2638,12 +3290,18 @@ class TabGroup(Element): return None def SelectTab(self, index): + """ + + :param index: + + """ try: self.TKNotebook.select(index) except Exception as e: print('Exception Selecting Tab {}'.format(e)) def __del__(self): + """ """ for row in self.Rows: for element in row: element.__del__() @@ -2654,32 +3312,36 @@ class TabGroup(Element): # Slider # # ---------------------------------------------------------------------- # class Slider(Element): + """ """ + def __init__(self, range=(None, None), default_value=None, resolution=None, tick_interval=None, orientation=None, disable_number_display=False, border_width=None, relief=None, change_submits=False, enable_events=False, disabled=False, size=(None, None), font=None, background_color=None, text_color=None, key=None, pad=None, tooltip=None, visible=True): - ''' - Slider - :param range: - :param default_value: - :param resolution: - :param tick_interval: - :param orientation: - :param disable_number_display: - :param border_width: - :param relief: - :param change_submits: - :param enable_events: - :param disabled: - :param size: - :param font: - :param background_color: - :param text_color: - :param key: - :param pad: - :param tooltip: - :param visible: - ''' + """ + + :param range: (Default value = (None, None)) + :param default_value: (Default value = None) + :param resolution: (Default value = None) + :param tick_interval: (Default value = None) + :param orientation: (Default value = None) + :param disable_number_display: (Default value = False) + :param border_width: (Default value = None) + :param relief: (Default value = None) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param disabled: set disable state for element (Default value = False) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.TKScale = None # type: tk.Scale self.Range = (1, 10) if range == (None, None) else range self.DefaultValue = self.Range[0] if default_value is None else default_value @@ -2700,6 +3362,15 @@ class Slider(Element): return def Update(self, value=None, range=(None, None), disabled=None, visible=None): + """ + + :param value: (Default value = None) + :param range: (Default value = (None, None)) + :param disabled: disable or enable state of the element (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ + if value is not None: try: self.TKIntVar.set(value) @@ -2717,7 +3388,12 @@ class Slider(Element): elif visible is True: self.TKScale.pack() - def SliderChangedHandler(self, event): + def _SliderChangedHandler(self, event): + """ + + :param event: + + """ # first, get the results table built # modify the Results table in the parent FlexForm object if self.Key is not None: @@ -2729,6 +3405,7 @@ class Slider(Element): self.ParentForm.TKroot.quit() # kick the users out of the mainloop def __del__(self): + """ """ super().__del__() @@ -2736,7 +3413,14 @@ class Slider(Element): # TkScrollableFrame (Used by Column) # # ---------------------------------------------------------------------- # class TkFixedFrame(tk.Frame): + """ """ def __init__(self, master, **kwargs): + """ + + :param master: + :param **kwargs: + + """ tk.Frame.__init__(self, master, **kwargs) self.canvas = tk.Canvas(self) @@ -2759,7 +3443,15 @@ class TkFixedFrame(tk.Frame): # TkScrollableFrame (Used by Column) # # ---------------------------------------------------------------------- # class TkScrollableFrame(tk.Frame): + """ """ def __init__(self, master, vertical_only, **kwargs): + """ + + :param master: + :param vertical_only: + :param **kwargs: + + """ tk.Frame.__init__(self, master, **kwargs) # create a canvas object and a vertical scrollbar for scrolling it @@ -2804,21 +3496,42 @@ class TkScrollableFrame(tk.Frame): self.bind_mouse_scroll(self.vscrollbar, self.yscroll) def resize_frame(self, e): + """ + + :param e: + + """ self.canvas.itemconfig(self.frame_id, height=e.height, width=e.width) def yscroll(self, event): + """ + + :param event: + + """ if event.num == 5 or event.delta < 0: self.canvas.yview_scroll(1, "unit") elif event.num == 4 or event.delta > 0: self.canvas.yview_scroll(-1, "unit") def xscroll(self, event): + """ + + :param event: + + """ if event.num == 5 or event.delta < 0: self.canvas.xview_scroll(1, "unit") elif event.num == 4 or event.delta > 0: self.canvas.xview_scroll(-1, "unit") def bind_mouse_scroll(self, parent, mode): + """ + + :param parent: + :param mode: + + """ # ~~ Windows only parent.bind("", mode) # ~~ Unix only @@ -2826,7 +3539,11 @@ class TkScrollableFrame(tk.Frame): parent.bind("", mode) def set_scrollregion(self, event=None): - """ Set the scroll region on the canvas""" + """Set the scroll region on the canvas + + :param event: (Default value = None) + + """ self.canvas.configure(scrollregion=self.canvas.bbox('all')) @@ -2834,20 +3551,24 @@ class TkScrollableFrame(tk.Frame): # Column # # ---------------------------------------------------------------------- # class Column(Element): + """ """ + def __init__(self, layout, background_color=None, size=(None, None), pad=None, scrollable=False, vertical_scroll_only=False, right_click_menu=None, key=None, visible=True): - ''' - Column - :param layout: - :param background_color: - :param size: - :param pad: - :param scrollable: - :param vertical_scroll_only: - :param right_click_menu: - :param key: - :param visible: - ''' + """ + + :param layout: + :param background_color: color of background (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param scrollable: (Default value = False) + :param vertical_scroll_only: (Default value = False) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.UseDictionary = False self.ReturnValues = None self.ReturnValuesList = [] @@ -2870,7 +3591,11 @@ class Column(Element): return def AddRow(self, *args): - ''' Parms are a variable number of Elements ''' + """Parms are a variable number of Elements + + :param *args: + + """ NumRows = len(self.Rows) # number of existing rows is our row number CurrentRowNumber = NumRows # this row's number CurrentRow = [] # start with a blank row and build up @@ -2885,16 +3610,32 @@ class Column(Element): self.Rows.append(CurrentRow) def Layout(self, rows): + """ + + :param rows: + + """ for row in rows: self.AddRow(*row) def _GetElementAtLocation(self, location): + """ + + :param location: + + """ (row_num, col_num) = location row = self.Rows[row_num] element = row[col_num] return element def Update(self, visible=None): + """ + + :param visible: change visibility of element (Default value = None) + + """ + if visible is False: if self.TKColFrame: self.TKColFrame.pack_forget() @@ -2907,6 +3648,7 @@ class Column(Element): self.ParentPanedWindow.add(self.TKColFrame) def __del__(self): + """ """ for row in self.Rows: for element in row: element.__del__() @@ -2921,22 +3663,26 @@ class Column(Element): # Pane # # ---------------------------------------------------------------------- # class Pane(Element): + """ """ + def __init__(self, pane_list, background_color=None, size=(None, None), pad=None, orientation='vertical', show_handle=True, relief=RELIEF_RAISED, handle_size=None, border_width=None, key=None, visible=True): - ''' - Pane - :param pane_list: - :param background_color: - :param size: - :param pad: - :param orientation: - :param show_handle: - :param relief: - :param handle_size: - :param border_width: - :param key: - :param visible: - ''' + """ + + :param pane_list: + :param background_color: color of background (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param orientation: (Default value = 'vertical') + :param show_handle: (Default value = True) + :param relief: (Default value = RELIEF_RAISED) + :param handle_size: (Default value = None) + :param border_width: (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.UseDictionary = False self.ReturnValues = None self.ReturnValuesList = [] @@ -2960,6 +3706,12 @@ class Pane(Element): return def Update(self, visible=None): + """ + + :param visible: change visibility of element (Default value = None) + + """ + if visible is False: self.PanedWindow.pack_forget() elif visible is True: @@ -2970,20 +3722,25 @@ class Pane(Element): # Calendar # # ---------------------------------------------------------------------- # class TKCalendar(tkinter.ttk.Frame): - """ - This code was shamelessly lifted from moshekaplan's repository - moshekaplan/tkinter_components - """ + """This code was shamelessly lifted from moshekaplan's repository - moshekaplan/tkinter_components""" # XXX ToDo: cget and configure datetime = calendar.datetime.datetime timedelta = calendar.datetime.timedelta def __init__(self, master=None, target_element=None, close_when_chosen=True, default_date=(None, None, None), **kw): - """ - WIDGET-SPECIFIC OPTIONS - + """WIDGET-SPECIFIC OPTIONS + locale, firstweekday, year, month, selectbackground, selectforeground + + :param master: (Default value = None) + :param target_element: (Default value = None) + :param close_when_chosen: (Default value = True) + :param default_date: (Default value = (None) + :param None:, None)) + :param **kw: + """ self._TargetElement = target_element default_mon, default_day, default_year = default_date @@ -3023,6 +3780,12 @@ class TKCalendar(tkinter.ttk.Frame): self._build_calendar() def __setitem__(self, item, value): + """ + + :param item: + :param value: + + """ if item in ('year', 'month'): raise AttributeError("attribute '%s' is not writeable" % item) elif item == 'selectbackground': @@ -3033,6 +3796,11 @@ class TKCalendar(tkinter.ttk.Frame): tkinter.ttk.Frame.__setitem__(self, item, value) def __getitem__(self, item): + """ + + :param item: + + """ if item in ('year', 'month'): return getattr(self._date, item) elif item == 'selectbackground': @@ -3044,6 +3812,7 @@ class TKCalendar(tkinter.ttk.Frame): return r[item] def __setup_styles(self): + """ """ # custom ttk styles style = tkinter.ttk.Style(self.master) arrow_layout = lambda dir: ( @@ -3053,6 +3822,7 @@ class TKCalendar(tkinter.ttk.Frame): style.layout('R.TButton', arrow_layout('right')) def __place_widgets(self): + """ """ # header frame and its widgets hframe = tkinter.ttk.Frame(self) lbtn = tkinter.ttk.Button(hframe, style='L.TButton', command=self._prev_month) @@ -3069,6 +3839,7 @@ class TKCalendar(tkinter.ttk.Frame): self._calendar.pack(in_=self, expand=1, fill='both', side='bottom') def __config_calendar(self): + """ """ cols = self._cal.formatweekheader(3).split() self._calendar['columns'] = cols self._calendar.tag_configure('header', background='grey90') @@ -3081,6 +3852,12 @@ class TKCalendar(tkinter.ttk.Frame): anchor='e') def __setup_selection(self, sel_bg, sel_fg): + """ + + :param sel_bg: + :param sel_fg: + + """ self._font = tkinter.font.Font() self._canvas = canvas = tk.Canvas(self._calendar, background=sel_bg, borderwidth=0, highlightthickness=0) @@ -3091,11 +3868,17 @@ class TKCalendar(tkinter.ttk.Frame): self._calendar.bind('', self._pressed) def __minsize(self, evt): + """ + + :param evt: + + """ width, height = self._calendar.master.geometry().split('x') height = height[:height.index('+')] self._calendar.master.minsize(width, height) def _build_calendar(self): + """ """ year, month = self._date.year, self._date.month # update header text (Month, YEAR) @@ -3110,7 +3893,12 @@ class TKCalendar(tkinter.ttk.Frame): self._calendar.item(item, values=fmt_week) def _show_selection(self, text, bbox): - """Configure canvas for a new selection.""" + """Configure canvas for a new selection. + + :param text: + :param bbox: + + """ x, y, width, height = bbox textw = self._font.measure(text) @@ -3124,7 +3912,11 @@ class TKCalendar(tkinter.ttk.Frame): # Callbacks def _pressed(self, evt): - """Clicked somewhere in the calendar.""" + """Clicked somewhere in the calendar. + + :param evt: + + """ x, y, widget = evt.x, evt.y, evt.widget item = widget.identify_row(y) column = widget.identify_column(x) @@ -3186,7 +3978,7 @@ class TKCalendar(tkinter.ttk.Frame): @property def selection(self): - """Return a datetime representing the current selected date.""" + """ """ if not self._selection: return None @@ -3198,21 +3990,25 @@ class TKCalendar(tkinter.ttk.Frame): # Menu # # ---------------------------------------------------------------------- # class Menu(Element): + """ """ + def __init__(self, menu_definition, background_color=None, size=(None, None), tearoff=False, pad=None, key=None, visible=True): - ''' - Menu - :param menu_definition: - :param background_color: - :param size: - :param tearoff: - :param pad: - :param key: - :param visible: - ''' + """ + + :param menu_definition: + :param background_color: color of background (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param tearoff: (Default value = False) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.BackgroundColor = background_color if background_color is not None else DEFAULT_BACKGROUND_COLOR self.MenuDefinition = menu_definition - self.TKMenu = None + self.TKMenu = None # type: tk.Menu self.Tearoff = tearoff self.MenuItemChosen = None @@ -3220,7 +4016,12 @@ class Menu(Element): visible=visible) return - def MenuItemChosenCallback(self, item_chosen): # Menu Menu Item Chosen Callback + def _MenuItemChosenCallback(self, item_chosen): # Menu Menu Item Chosen Callback + """ + + :param item_chosen: + + """ # print('IN MENU ITEM CALLBACK', item_chosen) self.MenuItemChosen = item_chosen self.ParentForm.LastButtonClicked = item_chosen @@ -3229,6 +4030,13 @@ class Menu(Element): self.ParentForm.TKroot.quit() # kick the users out of the mainloop def Update(self, menu_definition, visible=None): + """ + + :param menu_definition: + :param visible: change visibility of element (Default value = None) + + """ + self.MenuDefinition = menu_definition self.TKMenu = tk.Menu(self.ParentForm.TKroot, tearoff=self.Tearoff) # create the menubar menubar = self.TKMenu @@ -3252,6 +4060,7 @@ class Menu(Element): # TODO add visible code for menus def __del__(self): + """ """ super().__del__() @@ -3262,40 +4071,47 @@ MenuBar = Menu # another name for Menu to make it clear it's the Menu Bar # Table # # ---------------------------------------------------------------------- # class Table(Element): + """ """ + def __init__(self, values, headings=None, visible_column_map=None, col_widths=None, def_col_width=10, auto_size_columns=True, max_col_width=20, select_mode=None, display_row_numbers=False, num_rows=None, row_height=None, font=None, justification='right', text_color=None, background_color=None, alternating_row_color=None, row_colors=None, vertical_scroll_only=True, hide_vertical_scroll=False, size=(None, None), change_submits=False, enable_events=False, bind_return_key=False, pad=None, key=None, tooltip=None, right_click_menu=None, visible=True): - ''' - Table - :param values: - :param headings: - :param visible_column_map: - :param col_widths: - :param def_col_width: - :param auto_size_columns: - :param max_col_width: - :param select_mode: - :param display_row_numbers: - :param num_rows: - :param row_height: - :param font: - :param justification: - :param text_color: - :param background_color: - :param alternating_row_color: - :param size: - :param change_submits: - :param enable_events: - :param bind_return_key: - :param pad: - :param key: - :param tooltip: - :param right_click_menu: - :param visible: - ''' + """ + + :param values: + :param headings: (Default value = None) + :param visible_column_map: (Default value = None) + :param col_widths: (Default value = None) + :param def_col_width: (Default value = 10) + :param auto_size_columns: (Default value = True) + :param max_col_width: (Default value = 20) + :param select_mode: (Default value = None) + :param display_row_numbers: (Default value = False) + :param num_rows: (Default value = None) + :param row_height: (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param justification: (Default value = 'right') + :param text_color: color of the text (Default value = None) + :param background_color: color of background (Default value = None) + :param alternating_row_color: (Default value = None) + :param row_colors: (Default value = None) + :param vertical_scroll_only: (Default value = True) + :param hide_vertical_scroll: (Default value = False) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param bind_return_key: (Default value = False) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.Values = values self.ColumnHeadings = headings self.ColumnsToDisplay = visible_column_map @@ -3328,6 +4144,15 @@ class Table(Element): return def Update(self, values=None, num_rows=None, visible=None, select_rows=None): + """ + + :param values: (Default value = None) + :param num_rows: (Default value = None) + :param visible: change visibility of element (Default value = None) + :param select_rows: (Default value = None) + + """ + if values is not None: children = self.TKTreeview.get_children() for i in children: @@ -3354,6 +4179,11 @@ class Table(Element): self.TKTreeview.selection_set(rows_to_select) def treeview_selected(self, event): + """ + + :param event: + + """ selections = self.TKTreeview.selection() self.SelectedRows = [int(x) - 1 for x in selections] if self.ChangeSubmits: @@ -3367,6 +4197,11 @@ class Table(Element): self.ParentForm.TKroot.quit() def treeview_double_click(self, event): + """ + + :param event: + + """ selections = self.TKTreeview.selection() self.SelectedRows = [int(x) - 1 for x in selections] if self.BindReturnKey: @@ -3380,6 +4215,7 @@ class Table(Element): self.ParentForm.TKroot.quit() def __del__(self): + """ """ super().__del__() @@ -3387,36 +4223,41 @@ class Table(Element): # Tree # # ---------------------------------------------------------------------- # class Tree(Element): + """ """ + 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, show_expanded=False, change_submits=False, enable_events=False, font=None, justification='right', text_color=None, background_color=None, num_rows=None, row_height=None, pad=None, key=None, tooltip=None, right_click_menu=None, visible=True): - ''' - Tree - :param data: - :param headings: - :param visible_column_map: - :param col_widths: - :param col0_width: - :param def_col_width: - :param auto_size_columns: - :param max_col_width: - :param select_mode: - :param show_expanded: - :param change_submits: - :param enable_events: - :param font: - :param justification: - :param text_color: - :param background_color: - :param num_rows: - :param pad: - :param key: - :param tooltip: - :param right_click_menu: - :param visible: - ''' + """ + + :param data: (Default value = None) + :param headings: (Default value = None) + :param visible_column_map: (Default value = None) + :param col_widths: (Default value = None) + :param col0_width: (Default value = 10) + :param def_col_width: (Default value = 10) + :param auto_size_columns: (Default value = True) + :param max_col_width: (Default value = 20) + :param select_mode: (Default value = None) + :param show_expanded: (Default value = False) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param justification: (Default value = 'right') + :param text_color: color of the text (Default value = None) + :param background_color: color of background (Default value = None) + :param num_rows: (Default value = None) + :param row_height: (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param visible: set visibility state of the element (Default value = True) + + """ + self.TreeData = data self.ColumnHeadings = headings self.ColumnsToDisplay = visible_column_map @@ -3444,6 +4285,11 @@ class Tree(Element): return def treeview_selected(self, event): + """ + + :param event: + + """ selections = self.TKTreeview.selection() self.SelectedRows = [x for x in selections] if self.ChangeSubmits: @@ -3457,6 +4303,11 @@ class Tree(Element): self.ParentForm.TKroot.quit() def add_treeview_data(self, node): + """ + + :param node: + + """ # print(f'Inserting {node.key} under parent {node.parent}') if node.key != '': if node.icon: @@ -3478,6 +4329,17 @@ class Tree(Element): self.add_treeview_data(node) def Update(self, values=None, key=None, value=None, text=None, icon=None, visible=None): + """ + + :param values: (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param value: (Default value = None) + :param text: (Default value = None) + :param icon: (Default value = None) + :param visible: change visibility of element (Default value = None) + + """ + if values is not None: children = self.TKTreeview.get_children() for i in children: @@ -3511,12 +4373,24 @@ class Tree(Element): return self def __del__(self): + """ """ super().__del__() class TreeData(object): + """ """ class Node(object): + """ """ def __init__(self, parent, key, text, values, icon=None): + """ + + :param parent: + :param key: (common_key) Used with window.FindElement and with return values + :param text: + :param values: + :param icon: (Default value = None) + + """ self.parent = parent self.children = [] self.key = key @@ -3525,26 +4399,54 @@ class TreeData(object): self.icon = icon def _Add(self, node): + """ + + :param node: + + """ self.children.append(node) def __init__(self): + """ """ self.tree_dict = {} self.root_node = self.Node("", "", 'root', [], None) self.tree_dict[""] = self.root_node def _AddNode(self, key, node): + """ + + :param key: (common_key) Used with window.FindElement and with return values + :param node: + + """ self.tree_dict[key] = node def Insert(self, parent, key, text, values, icon=None): + """ + + :param parent: + :param key: (common_key) Used with window.FindElement and with return values + :param text: + :param values: + :param icon: (Default value = None) + + """ node = self.Node(parent, key, text, values, icon) self.tree_dict[key] = node parent_node = self.tree_dict[parent] parent_node._Add(node) def __repr__(self): + """ """ return self._NodeStr(self.root_node, 1) def _NodeStr(self, node, level): + """ + + :param node: + :param level: + + """ return '\n'.join( [str(node.key) + ' : ' + str(node.text)] + [' ' * 4 * level + self._NodeStr(child, level + 1) for child in node.children]) @@ -3554,17 +4456,26 @@ class TreeData(object): # Error Element # # ---------------------------------------------------------------------- # class ErrorElement(Element): + """ """ def __init__(self, key=None): - ''' - Error Element - :param key: - ''' + """Error Element + + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ self.Key = key super().__init__(ELEM_TYPE_ERROR, key=key) return def Update(self, silent_on_error=True, *args, **kwargs): + """ + + :param silent_on_error: (Default value = True) + :param *args: + :param **kwargs: + + """ if not silent_on_error: PopupError('Keyword error in Update', 'You need to stop this madness and check your spelling', @@ -3574,9 +4485,11 @@ class ErrorElement(Element): return self def Get(self): + """ """ return 'This is NOT a valid Element!\nSTOP trying to do things with it or I will have to crash at some point!' def __del__(self): + """ """ super().__del__() @@ -3591,6 +4504,8 @@ Stretch = ErrorElement # Window CLASS # # ------------------------------------------------------------------------- # class Window(object): + """ """ + NumOpenWindows = 0 user_defined_icon = None hidden_master_root = None @@ -3607,37 +4522,43 @@ class Window(object): alpha_channel=1, return_keyboard_events=False, use_default_focus=True, text_justification=None, no_titlebar=False, grab_anywhere=False, keep_on_top=False, resizable=False, disable_close=False, disable_minimize=False, right_click_menu=None, transparent_color=None, debugger_enabled=True): - ''' - Window - :param title: - :param default_element_size: - :param default_button_element_size: - :param auto_size_text: - :param auto_size_buttons: - :param location: - :param size: - :param element_padding: - :param button_color: - :param font: - :param progress_bar_color: - :param background_color: - :param border_depth: - :param auto_close: - :param auto_close_duration: - :param icon: - :param force_toplevel: - :param alpha_channel: - :param return_keyboard_events: - :param use_default_focus: - :param text_justification: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param resizable: - :param disable_close: - :param disable_minimize: - :param right_click_menu: - ''' + """ + + :param title: + :param layout: (Default value = None) + :param default_element_size: (Default value = DEFAULT_ELEMENT_SIZE) + :param default_button_element_size: (Default value = (None, None)) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param auto_size_buttons: (Default value = None) + :param location: (Default value = (None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None) + :param element_padding: (Default value = None) + :param margins: (Default value = (None) + :param button_color: (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param progress_bar_color: (Default value = (None) + :param background_color: color of background (Default value = None) + :param border_depth: (Default value = None) + :param auto_close: (Default value = False) + :param auto_close_duration: (Default value = DEFAULT_AUTOCLOSE_TIME) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param force_toplevel: (Default value = False) + :param alpha_channel: (Default value = 1) + :param return_keyboard_events: (Default value = False) + :param use_default_focus: (Default value = True) + :param text_justification: (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param resizable: (Default value = False) + :param disable_close: (Default value = False) + :param disable_minimize: (Default value = False) + :param right_click_menu: see "Right Click Menus" (Default value = None) + :param transparent_color: (Default value = None) + :param debugger_enabled: (Default value = True) + + """ + self.AutoSizeText = auto_size_text if auto_size_text is not None else DEFAULT_AUTOSIZE_TEXT self.AutoSizeButtons = auto_size_buttons if auto_size_buttons is not None else DEFAULT_AUTOSIZE_BUTTONS self.Title = title @@ -3702,22 +4623,29 @@ class Window(object): @classmethod def GetAContainerNumber(cls): + """ """ cls.container_element_counter += 1 return cls.container_element_counter @classmethod def IncrementOpenCount(self): + """ """ self.NumOpenWindows += 1 # print('+++++ INCREMENTING Num Open Windows = {} ---'.format(Window.NumOpenWindows)) @classmethod def DecrementOpenCount(self): + """ """ self.NumOpenWindows -= 1 * (self.NumOpenWindows != 0) # decrement if not 0 # print('----- DECREMENTING Num Open Windows = {} ---'.format(Window.NumOpenWindows)) # ------------------------- Add ONE Row to Form ------------------------- # def AddRow(self, *args): - ''' Parms are a variable number of Elements ''' + """Parms are a variable number of Elements + + :param *args: + + """ NumRows = len(self.Rows) # number of existing rows is our row number CurrentRowNumber = NumRows # this row's number CurrentRow = [] # start with a blank row and build up @@ -3750,15 +4678,31 @@ class Window(object): # ------------------------- Add Multiple Rows to Form ------------------------- # def AddRows(self, rows): + """ + + :param rows: + + """ for row in rows: self.AddRow(*row) def Layout(self, rows): + """ + + :param rows: + + """ self.AddRows(rows) self.BuildKeyDict() return self def LayoutAndRead(self, rows, non_blocking=False): + """ + + :param rows: + :param non_blocking: (Default value = False) + + """ raise DeprecationWarning( 'LayoutAndRead is no longer supported... change your call window.Layout(layout).Read()') # self.AddRows(rows) @@ -3766,10 +4710,20 @@ class Window(object): # return self.ReturnValues def LayoutAndShow(self, rows): + """ + + :param rows: + + """ raise DeprecationWarning('LayoutAndShow is no longer supported... ') # ------------------------- ShowForm THIS IS IT! ------------------------- # def Show(self, non_blocking=False): + """ + + :param non_blocking: (Default value = False) + + """ self.Shown = True # Compute num rows & num cols (it'll come in handy debugging) self.NumRows = len(self.Rows) @@ -3805,6 +4759,12 @@ class Window(object): # ------------------------- SetIcon - set the window's fav icon ------------------------- # def SetIcon(self, icon=None, pngbase64=None): + """ + + :param icon: (Default value = None) + :param pngbase64: (Default value = None) + + """ if type(icon) is bytes: wicon = tkinter.PhotoImage(data=icon) try: @@ -3827,15 +4787,22 @@ class Window(object): pass def _GetElementAtLocation(self, location): + """ + + :param location: + + """ (row_num, col_num) = location row = self.Rows[row_num] element = row[col_num] return element def _GetDefaultElementSize(self): + """ """ return self.DefaultElementSize def _AutoCloseAlarmCallback(self): + """ """ try: window = self if window: @@ -3849,6 +4816,7 @@ class Window(object): pass def _TimeoutAlarmCallback(self): + """ """ # first, get the results table built # modify the Results table in the parent FlexForm object # print('TIMEOUT CALLBACK') @@ -3860,6 +4828,12 @@ class Window(object): self.TKroot.quit() # kick the users out of the mainloop def Read(self, timeout=None, timeout_key=TIMEOUT_KEY): + """ + + :param timeout: (Default value = None) + :param timeout_key: (Default value = TIMEOUT_KEY) + + """ # ensure called only 1 time through a single read cycle if not Window.read_call_from_debugger: refresh_debugger() @@ -3922,8 +4896,8 @@ class Window(object): self.TKAfterID = self.TKroot.after(timeout, self._TimeoutAlarmCallback) self.CurrentlyRunningMainloop = True # print(f'In main {self.Title} {self.TKroot}') - self.TKroot.protocol("WM_DESTROY_WINDOW", self.OnClosingCallback) - self.TKroot.protocol("WM_DELETE_WINDOW", self.OnClosingCallback) + # self.TKroot.protocol("WM_DESTROY_WINDOW", self.OnClosingCallback) + # self.TKroot.protocol("WM_DELETE_WINDOW", self.OnClosingCallback) self.TKroot.mainloop() # print('Out main') self.CurrentlyRunningMainloop = False @@ -3963,6 +4937,7 @@ class Window(object): return self.ReturnValues def ReadNonBlocking(self): + """ """ if self.TKrootDestroyed: try: self.TKroot.quit() @@ -3992,6 +4967,7 @@ class Window(object): return BuildResults(self, False, self) def Finalize(self): + """ """ if self.TKrootDestroyed: return self if not self.Shown: @@ -4006,6 +4982,7 @@ class Window(object): return self def Refresh(self): + """ """ if self.TKrootDestroyed: return self try: @@ -4015,10 +4992,21 @@ class Window(object): return self def Fill(self, values_dict): + """ + + :param values_dict: + + """ FillFormWithValues(self, values_dict) return self def FindElement(self, key, silent_on_error=False): + """ + + :param key: (common_key) Used with window.FindElement and with return values + :param silent_on_error: (Default value = False) + + """ # print(f'In find elem key={key}', self.AllKeysDict) try: @@ -4041,15 +5029,24 @@ class Window(object): Find = FindElement def FindElementWithFocus(self): + """ """ element = _FindElementWithFocusInSubForm(self) return element def BuildKeyDict(self): + """ """ dict = {} self.AllKeysDict = self._BuildKeyDictForWindow(self, self, dict) # print(f'keys built = {self.AllKeysDict}') def _BuildKeyDictForWindow(self, top_window, window, key_dict): + """ + + :param top_window: + :param window: + :param key_dict: + + """ for row_num, row in enumerate(window.Rows): for col_num, element in enumerate(row): if element.Type == ELEM_TYPE_COLUMN: @@ -4084,6 +5081,11 @@ class Window(object): return key_dict def SaveToDisk(self, filename): + """ + + :param filename: + + """ try: results = BuildResults(self, False, self) with open(filename, 'wb') as sf: @@ -4092,6 +5094,11 @@ class Window(object): print('*** Error saving form to disk ***') def LoadFromDisk(self, filename): + """ + + :param filename: + + """ try: with open(filename, 'rb') as df: self.Fill(pickle.load(df)) @@ -4099,6 +5106,7 @@ class Window(object): print('*** Error loading form to disk ***') def GetScreenDimensions(self): + """ """ if self.TKrootDestroyed: return None, None screen_width = self.TKroot.winfo_screenwidth() # get window info to move to middle of screen @@ -4106,15 +5114,23 @@ class Window(object): return screen_width, screen_height def Move(self, x, y): + """ + + :param x: + :param y: + + """ try: self.TKroot.geometry("+%s+%s" % (x, y)) except: pass def Minimize(self): + """ """ self.TKroot.iconify() def Maximize(self): + """ """ if sys.platform != 'linux': self.TKroot.state('zoomed') else: @@ -4123,36 +5139,60 @@ class Window(object): # self.TKroot.attributes('-fullscreen', True) def Normal(self): + """ """ if sys.platform != 'linux': self.TKroot.state('normal') else: self.TKroot.attributes('-fullscreen', False) def StartMove(self, event): + """ + + :param event: + + """ try: self.TKroot.x = event.x self.TKroot.y = event.y except: pass + # print('Start move {},{}'.format(event.x,event.y)) def StopMove(self, event): + """ + + :param event: + + """ try: self.TKroot.x = None self.TKroot.y = None except: pass + # print('-Stop- move {},{}'.format(event.x,event.y)) def OnMotion(self, event): + """ + + :param event: + + """ try: deltax = event.x - self.TKroot.x deltay = event.y - self.TKroot.y x = self.TKroot.winfo_x() + deltax y = self.TKroot.winfo_y() + deltay self.TKroot.geometry("+%s+%s" % (x, y)) + # print('{},{}'.format(x,y)) except: pass def _KeyboardCallback(self, event): + """ + + :param event: + + """ self.LastButtonClicked = None self.FormRemainedOpen = True if event.char != '': @@ -4165,6 +5205,11 @@ class Window(object): self.TKroot.quit() def _MouseWheelCallback(self, event): + """ + + :param event: + + """ self.LastButtonClicked = None self.FormRemainedOpen = True self.LastKeyboardEvent = 'MouseWheel:Down' if event.delta < 0 else 'MouseWheel:Up' @@ -4174,6 +5219,7 @@ class Window(object): self.TKroot.quit() def _Close(self): + """ """ try: self.TKroot.update() except: @@ -4187,6 +5233,7 @@ class Window(object): return None def Close(self): + """ """ if self.TKrootDestroyed: return try: @@ -4208,6 +5255,7 @@ class Window(object): # IT FINALLY WORKED! 29-Oct-2018 was the first time this damned thing got called def OnClosingCallback(self): + """ """ # global _my_windows # print('Got closing callback', self.DisableClose) if self.DisableClose: @@ -4226,63 +5274,85 @@ class Window(object): return def Disable(self): + """ """ self.TKroot.attributes('-disabled', 1) # self.TKroot.grab_set_global() def Enable(self): + """ """ self.TKroot.attributes('-disabled', 0) # self.TKroot.grab_release() def Hide(self): + """ """ self._Hidden = True self.TKroot.withdraw() def UnHide(self): + """ """ if self._Hidden: self.TKroot.deiconify() self._Hidden = False def Disappear(self): + """ """ self.TKroot.attributes('-alpha', 0) def Reappear(self): + """ """ self.TKroot.attributes('-alpha', 255) def SetAlpha(self, alpha): - ''' - Change the window's transparency - :param alpha: From 0 to 1 with 0 being completely transparent - :return: - ''' + """ + + :param alpha: + + """ + # Change the window's transparency + # :param alpha: From 0 to 1 with 0 being completely transparent self._AlphaChannel = alpha self.TKroot.attributes('-alpha', alpha) @property def AlphaChannel(self): + """ """ return self._AlphaChannel @AlphaChannel.setter def AlphaChannel(self, alpha): + """ + + :param alpha: + + """ self._AlphaChannel = alpha self.TKroot.attributes('-alpha', alpha) def BringToFront(self): + """ """ try: self.TKroot.lift() except: pass def CurrentLocation(self): + """ """ return int(self.TKroot.winfo_x()), int(self.TKroot.winfo_y()) @property def Size(self): + """ """ win_width = self.TKroot.winfo_width() win_height = self.TKroot.winfo_height() return win_width, win_height @Size.setter def Size(self, size): + """ + + :param size: (common_key) + + """ try: self.TKroot.geometry("%sx%s" % (size[0], size[1])) self.TKroot.update_idletasks() @@ -4290,50 +5360,77 @@ class Window(object): pass def VisibilityChanged(self): + """ """ # A dummy function. Needed in Qt but not tkinter return def SetTransparentColor(self, color): + """ + + :param color: + + """ try: self.TKroot.attributes('-transparentcolor', color) except: print('Transparent color not supported on this platform (windows only)') def GrabAnyWhereOn(self): + """ """ self.TKroot.bind("", self.StartMove) self.TKroot.bind("", self.StopMove) self.TKroot.bind("", self.OnMotion) def GrabAnyWhereOff(self): + """ """ self.TKroot.unbind("") self.TKroot.unbind("") self.TKroot.unbind("") def _callback_main_debugger_window_create_keystroke(self, event): - Debugger.debugger._build_main_debugger_window() + """ + + :param event: + + """ + _Debugger.debugger._build_main_debugger_window() def _callback_popout_window_create_keystroke(self, event): - Debugger.debugger._build_floating_window() + """ + + :param event: + + """ + _Debugger.debugger._build_floating_window() def EnableDebugger(self): + """ """ self.TKroot.bind('', self._callback_main_debugger_window_create_keystroke) self.TKroot.bind('', self._callback_popout_window_create_keystroke) self.DebuggerEnabled = True def DisableDebugger(self): + """ """ self.TKroot.unbind("") self.TKroot.unbind("") self.DebuggerEnabled = False def __enter__(self): + """ """ return self def __exit__(self, *a): + """ + + :param *a: + + """ self.__del__() return False def __del__(self): + """ """ # print('DELETING WINDOW') for row in self.Rows: for element in row: @@ -4359,6 +5456,24 @@ FlexForm = Window def FolderBrowse(button_text='Browse', target=(ThisRow, -1), initial_folder=None, tooltip=None, size=(None, None), auto_size_button=None, button_color=None, disabled=False, change_submits=False, enable_events=False, font=None, pad=None, key=None): + """ + + :param button_text: (Default value = 'Browse') + :param target: (Default value = (ThisRow, -1)) + :param initial_folder: (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ + return Button(button_text=button_text, button_type=BUTTON_TYPE_BROWSE_FOLDER, target=target, initial_folder=initial_folder, tooltip=tooltip, size=size, auto_size_button=auto_size_button, disabled=disabled, button_color=button_color, change_submits=change_submits, @@ -4370,6 +5485,24 @@ def FileBrowse(button_text='Browse', target=(ThisRow, -1), file_types=(("ALL Fil tooltip=None, size=(None, None), auto_size_button=None, button_color=None, change_submits=False, enable_events=False, font=None, disabled=False, pad=None, key=None): + """ + + :param button_text: (Default value = 'Browse') + :param target: (Default value = (ThisRow, -1)) + :param file_types: (Default value = (("ALL Files", "*.*"))) + :param initial_folder: (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_BROWSE_FILE, target=target, file_types=file_types, initial_folder=initial_folder, tooltip=tooltip, size=size, auto_size_button=auto_size_button, change_submits=change_submits, enable_events=enable_events, disabled=disabled, @@ -4381,6 +5514,24 @@ def FilesBrowse(button_text='Browse', target=(ThisRow, -1), file_types=(("ALL Fi initial_folder=None, tooltip=None, size=(None, None), auto_size_button=None, button_color=None, change_submits=False, enable_events=False, font=None, pad=None, key=None): + """ + + :param button_text: (Default value = 'Browse') + :param target: (Default value = (ThisRow, -1)) + :param file_types: (Default value = (("ALL Files", "*.*"))) + :param disabled: set disable state for element (Default value = False) + :param initial_folder: (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_BROWSE_FILES, target=target, file_types=file_types, initial_folder=initial_folder, change_submits=change_submits, enable_events=enable_events, tooltip=tooltip, size=size, auto_size_button=auto_size_button, @@ -4392,6 +5543,24 @@ def FileSaveAs(button_text='Save As...', target=(ThisRow, -1), file_types=(("ALL disabled=False, tooltip=None, size=(None, None), auto_size_button=None, button_color=None, change_submits=False, enable_events=False, font=None, pad=None, key=None): + """ + + :param button_text: (Default value = 'Save As...') + :param target: (Default value = (ThisRow, -1)) + :param file_types: (Default value = (("ALL Files", "*.*"))) + :param initial_folder: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_SAVEAS_FILE, target=target, file_types=file_types, initial_folder=initial_folder, tooltip=tooltip, size=size, disabled=disabled, auto_size_button=auto_size_button, button_color=button_color, change_submits=change_submits, @@ -4403,6 +5572,24 @@ def SaveAs(button_text='Save As...', target=(ThisRow, -1), file_types=(("ALL Fil disabled=False, tooltip=None, size=(None, None), auto_size_button=None, button_color=None, change_submits=False, enable_events=False, font=None, pad=None, key=None): + """ + + :param button_text: (Default value = 'Save As...') + :param target: (Default value = (ThisRow, -1)) + :param file_types: (Default value = (("ALL Files", "*.*"))) + :param initial_folder: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param change_submits: If True, pressing Enter key submits window (Default value = False) + :param enable_events: Turns on the element specific events.(Default value = False) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_SAVEAS_FILE, target=target, file_types=file_types, initial_folder=initial_folder, tooltip=tooltip, size=size, disabled=disabled, auto_size_button=auto_size_button, button_color=button_color, change_submits=change_submits, @@ -4412,6 +5599,21 @@ def SaveAs(button_text='Save As...', target=(ThisRow, -1), file_types=(("ALL Fil # ------------------------- SAVE BUTTON Element lazy function ------------------------- # def Save(button_text='Save', size=(None, None), auto_size_button=None, button_color=None, bind_return_key=True, disabled=False, tooltip=None, font=None, focus=False, pad=None, key=None): + """ + + :param button_text: (Default value = 'Save') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param bind_return_key: (Default value = True) + :param disabled: set disable state for element (Default value = False) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_READ_FORM, tooltip=tooltip, size=size, auto_size_button=auto_size_button, button_color=button_color, font=font, disabled=disabled, bind_return_key=bind_return_key, focus=focus, pad=pad, key=key) @@ -4420,6 +5622,21 @@ def Save(button_text='Save', size=(None, None), auto_size_button=None, button_co # ------------------------- SUBMIT BUTTON Element lazy function ------------------------- # def Submit(button_text='Submit', size=(None, None), auto_size_button=None, button_color=None, disabled=False, bind_return_key=True, tooltip=None, font=None, focus=False, pad=None, key=None): + """ + + :param button_text: (Default value = 'Submit') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param bind_return_key: (Default value = True) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_READ_FORM, tooltip=tooltip, size=size, auto_size_button=auto_size_button, button_color=button_color, font=font, disabled=disabled, bind_return_key=bind_return_key, focus=focus, pad=pad, key=key) @@ -4429,6 +5646,21 @@ def Submit(button_text='Submit', size=(None, None), auto_size_button=None, butto # ------------------------- OPEN BUTTON Element lazy function ------------------------- # def Open(button_text='Open', size=(None, None), auto_size_button=None, button_color=None, disabled=False, bind_return_key=True, tooltip=None, font=None, focus=False, pad=None, key=None): + """ + + :param button_text: (Default value = 'Open') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param bind_return_key: (Default value = True) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_READ_FORM, tooltip=tooltip, size=size, auto_size_button=auto_size_button, button_color=button_color, font=font, disabled=disabled, bind_return_key=bind_return_key, focus=focus, pad=pad, key=key) @@ -4437,6 +5669,21 @@ def Open(button_text='Open', size=(None, None), auto_size_button=None, button_co # ------------------------- OK BUTTON Element lazy function ------------------------- # def OK(button_text='OK', size=(None, None), auto_size_button=None, button_color=None, disabled=False, bind_return_key=True, tooltip=None, font=None, focus=False, pad=None, key=None): + """ + + :param button_text: (Default value = 'OK') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param bind_return_key: (Default value = True) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_READ_FORM, tooltip=tooltip, size=size, auto_size_button=auto_size_button, button_color=button_color, font=font, disabled=disabled, bind_return_key=bind_return_key, focus=focus, pad=pad, key=key) @@ -4445,6 +5692,21 @@ def OK(button_text='OK', size=(None, None), auto_size_button=None, button_color= # ------------------------- YES BUTTON Element lazy function ------------------------- # def Ok(button_text='Ok', size=(None, None), auto_size_button=None, button_color=None, disabled=False, bind_return_key=True, tooltip=None, font=None, focus=False, pad=None, key=None): + """ + + :param button_text: (Default value = 'Ok') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param bind_return_key: (Default value = True) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_READ_FORM, tooltip=tooltip, size=size, auto_size_button=auto_size_button, button_color=button_color, font=font, disabled=disabled, bind_return_key=bind_return_key, focus=focus, pad=pad, key=key) @@ -4453,6 +5715,21 @@ def Ok(button_text='Ok', size=(None, None), auto_size_button=None, button_color= # ------------------------- CANCEL BUTTON Element lazy function ------------------------- # def Cancel(button_text='Cancel', size=(None, None), auto_size_button=None, button_color=None, disabled=False, tooltip=None, font=None, bind_return_key=False, focus=False, pad=None, key=None): + """ + + :param button_text: (Default value = 'Cancel') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param bind_return_key: (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_READ_FORM, tooltip=tooltip, size=size, auto_size_button=auto_size_button, button_color=button_color, font=font, disabled=disabled, bind_return_key=bind_return_key, focus=focus, pad=pad, key=key) @@ -4461,6 +5738,21 @@ def Cancel(button_text='Cancel', size=(None, None), auto_size_button=None, butto # ------------------------- QUIT BUTTON Element lazy function ------------------------- # def Quit(button_text='Quit', size=(None, None), auto_size_button=None, button_color=None, disabled=False, tooltip=None, font=None, bind_return_key=False, focus=False, pad=None, key=None): + """ + + :param button_text: (Default value = 'Quit') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param bind_return_key: (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_READ_FORM, tooltip=tooltip, size=size, auto_size_button=auto_size_button, button_color=button_color, font=font, disabled=disabled, bind_return_key=bind_return_key, focus=focus, pad=pad, key=key) @@ -4469,6 +5761,21 @@ def Quit(button_text='Quit', size=(None, None), auto_size_button=None, button_co # ------------------------- Exit BUTTON Element lazy function ------------------------- # def Exit(button_text='Exit', size=(None, None), auto_size_button=None, button_color=None, disabled=False, tooltip=None, font=None, bind_return_key=False, focus=False, pad=None, key=None): + """ + + :param button_text: (Default value = 'Exit') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param bind_return_key: (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_READ_FORM, tooltip=tooltip, size=size, auto_size_button=auto_size_button, button_color=button_color, font=font, disabled=disabled, bind_return_key=bind_return_key, focus=focus, pad=pad, key=key) @@ -4477,6 +5784,21 @@ def Exit(button_text='Exit', size=(None, None), auto_size_button=None, button_co # ------------------------- YES BUTTON Element lazy function ------------------------- # def Yes(button_text='Yes', size=(None, None), auto_size_button=None, button_color=None, disabled=False, tooltip=None, font=None, bind_return_key=True, focus=False, pad=None, key=None): + """ + + :param button_text: (Default value = 'Yes') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param bind_return_key: (Default value = True) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_READ_FORM, tooltip=tooltip, size=size, auto_size_button=auto_size_button, button_color=button_color, font=font, disabled=disabled, bind_return_key=bind_return_key, focus=focus, pad=pad, key=key) @@ -4485,6 +5807,21 @@ def Yes(button_text='Yes', size=(None, None), auto_size_button=None, button_colo # ------------------------- NO BUTTON Element lazy function ------------------------- # def No(button_text='No', size=(None, None), auto_size_button=None, button_color=None, disabled=False, tooltip=None, font=None, bind_return_key=False, focus=False, pad=None, key=None): + """ + + :param button_text: (Default value = 'No') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param bind_return_key: (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_READ_FORM, tooltip=tooltip, size=size, auto_size_button=auto_size_button, button_color=button_color, font=font, disabled=disabled, bind_return_key=bind_return_key, focus=focus, pad=pad, key=key) @@ -4493,6 +5830,21 @@ def No(button_text='No', size=(None, None), auto_size_button=None, button_color= # ------------------------- NO BUTTON Element lazy function ------------------------- # def Help(button_text='Help', size=(None, None), auto_size_button=None, button_color=None, disabled=False, font=None, tooltip=None, bind_return_key=False, focus=False, pad=None, key=None): + """ + + :param button_text: (Default value = 'Help') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param bind_return_key: (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_READ_FORM, tooltip=tooltip, size=size, auto_size_button=auto_size_button, button_color=button_color, font=font, disabled=disabled, bind_return_key=bind_return_key, focus=focus, pad=pad, key=key) @@ -4501,6 +5853,21 @@ def Help(button_text='Help', size=(None, None), auto_size_button=None, button_co # ------------------------- NO BUTTON Element lazy function ------------------------- # def Debug(button_text='', size=(None, None), auto_size_button=None, button_color=None, disabled=False, font=None, tooltip=None, bind_return_key=False, focus=False, pad=None, key=None): + """ + + :param button_text: (Default value = '') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param bind_return_key: (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_SHOW_DEBUGGER, tooltip=tooltip, size=size, auto_size_button=auto_size_button, button_color=COLOR_SYSTEM_DEFAULT, font=font, disabled=disabled, bind_return_key=bind_return_key, focus=focus, pad=pad, key=key, image_data=PSG_DEBUGGER_LOGO, @@ -4511,6 +5878,26 @@ def Debug(button_text='', size=(None, None), auto_size_button=None, button_color def SimpleButton(button_text, image_filename=None, image_data=None, image_size=(None, None), image_subsample=None, border_width=None, tooltip=None, size=(None, None), auto_size_button=None, button_color=None, font=None, bind_return_key=False, disabled=False, focus=False, pad=None, key=None): + """ + + :param button_text: + :param image_filename: (Default value = None) + :param image_data: (Default value = None) + :param image_size: (Default value = (None, None)) + :param image_subsample: (Default value = None) + :param border_width: (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param bind_return_key: (Default value = False) + :param disabled: set disable state for element (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_CLOSES_WIN, image_filename=image_filename, image_data=image_data, image_size=image_size, image_subsample=image_subsample, border_width=border_width, tooltip=tooltip, disabled=disabled, size=size, @@ -4522,6 +5909,26 @@ def SimpleButton(button_text, image_filename=None, image_data=None, image_size=( def CloseButton(button_text, image_filename=None, image_data=None, image_size=(None, None), image_subsample=None, border_width=None, tooltip=None, size=(None, None), auto_size_button=None, button_color=None, font=None, bind_return_key=False, disabled=False, focus=False, pad=None, key=None): + """ + + :param button_text: + :param image_filename: (Default value = None) + :param image_data: (Default value = None) + :param image_size: (Default value = (None, None)) + :param image_subsample: (Default value = None) + :param border_width: (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param bind_return_key: (Default value = False) + :param disabled: set disable state for element (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_CLOSES_WIN, image_filename=image_filename, image_data=image_data, image_size=image_size, image_subsample=image_subsample, border_width=border_width, tooltip=tooltip, disabled=disabled, size=size, @@ -4536,6 +5943,26 @@ CButton = CloseButton def ReadButton(button_text, image_filename=None, image_data=None, image_size=(None, None), image_subsample=None, border_width=None, tooltip=None, size=(None, None), auto_size_button=None, button_color=None, font=None, bind_return_key=False, disabled=False, focus=False, pad=None, key=None): + """ + + :param button_text: + :param image_filename: (Default value = None) + :param image_data: (Default value = None) + :param image_size: (Default value = (None, None)) + :param image_subsample: (Default value = None) + :param border_width: (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param bind_return_key: (Default value = False) + :param disabled: set disable state for element (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_READ_FORM, image_filename=image_filename, image_data=image_data, image_size=image_size, image_subsample=image_subsample, border_width=border_width, tooltip=tooltip, size=size, disabled=disabled, @@ -4551,6 +5978,26 @@ RButton = ReadFormButton def RealtimeButton(button_text, image_filename=None, image_data=None, image_size=(None, None), image_subsample=None, border_width=None, tooltip=None, size=(None, None), auto_size_button=None, button_color=None, font=None, disabled=False, bind_return_key=False, focus=False, pad=None, key=None): + """ + + :param button_text: + :param image_filename: (Default value = None) + :param image_data: (Default value = None) + :param image_size: (Default value = (None, None)) + :param image_subsample: (Default value = None) + :param border_width: (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param bind_return_key: (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_REALTIME, image_filename=image_filename, image_data=image_data, image_size=image_size, image_subsample=image_subsample, border_width=border_width, tooltip=tooltip, disabled=disabled, size=size, @@ -4562,6 +6009,26 @@ def RealtimeButton(button_text, image_filename=None, image_data=None, image_size def DummyButton(button_text, image_filename=None, image_data=None, image_size=(None, None), image_subsample=None, border_width=None, tooltip=None, size=(None, None), auto_size_button=None, button_color=None, font=None, disabled=False, bind_return_key=False, focus=False, pad=None, key=None): + """ + + :param button_text: + :param image_filename: (Default value = None) + :param image_data: (Default value = None) + :param image_size: (Default value = (None, None)) + :param image_subsample: (Default value = None) + :param border_width: (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param bind_return_key: (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_CLOSES_WIN_ONLY, image_filename=image_filename, image_data=image_data, image_size=image_size, image_subsample=image_subsample, border_width=border_width, tooltip=tooltip, size=size, auto_size_button=auto_size_button, @@ -4575,6 +6042,32 @@ def CalendarButton(button_text, target=(None, None), close_when_date_chosen=True image_subsample=None, tooltip=None, border_width=None, size=(None, None), auto_size_button=None, button_color=None, disabled=False, font=None, bind_return_key=False, focus=False, pad=None, key=None, locale=None, format=None): + """ + + :param button_text: + :param target: (Default value = (None, None)) + :param close_when_date_chosen: (Default value = True) + :param default_date_m_d_y: (Default value = (None) + :param None: + :param image_filename: (Default value = None) + :param image_data: (Default value = None) + :param image_size: (Default value = (None) + :param image_subsample: (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param border_width: (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param bind_return_key: (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + :param locale: (Default value = None) + :param format: (Default value = None) + + """ button = Button(button_text=button_text, button_type=BUTTON_TYPE_CALENDAR_CHOOSER, target=target, image_filename=image_filename, image_data=image_data, image_size=image_size, image_subsample=image_subsample, border_width=border_width, tooltip=tooltip, size=size, @@ -4592,6 +6085,27 @@ def ColorChooserButton(button_text, target=(None, None), image_filename=None, im image_subsample=None, tooltip=None, border_width=None, size=(None, None), auto_size_button=None, button_color=None, disabled=False, font=None, bind_return_key=False, focus=False, pad=None, key=None): + """ + + :param button_text: + :param target: (Default value = (None, None)) + :param image_filename: (Default value = None) + :param image_data: (Default value = None) + :param image_size: (Default value = (None) + :param image_subsample: (Default value = None) + :param tooltip: text, that will appear the you hover on (Default value = None) + :param border_width: (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None) + :param auto_size_button: (Default value = None) + :param button_color: (Default value = None) + :param disabled: set disable state for element (Default value = False) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param bind_return_key: (Default value = False) + :param focus: if focus should be set to this (Default value = None) + :param pad: (common_key) Amount of padding to put around element (Default value = None) + :param key: (common_key) Used with window.FindElement and with return values (Default value = None) + + """ return Button(button_text=button_text, button_type=BUTTON_TYPE_COLOR_CHOOSER, target=target, image_filename=image_filename, image_data=image_data, image_size=image_size, image_subsample=image_subsample, border_width=border_width, tooltip=tooltip, size=size, @@ -4602,6 +6116,13 @@ def ColorChooserButton(button_text, target=(None, None), image_filename=None, im ##################################### ----- RESULTS ------ ################################################## def AddToReturnDictionary(form, element, value): + """ + + :param form: + :param element: + :param value: + + """ form.ReturnValuesDictionary[element.Key] = value return # if element.Key is None: @@ -4613,12 +6134,23 @@ def AddToReturnDictionary(form, element, value): def AddToReturnList(form, value): + """ + + :param form: + :param value: + + """ form.ReturnValuesList.append(value) # ----------------------------------------------------------------------------# # ------- FUNCTION InitializeResults. Sets up form results matrix --------# def InitializeResults(form): + """ + + :param form: + + """ BuildResults(form, True, form) return @@ -4626,6 +6158,11 @@ def InitializeResults(form): # ===== Radio Button RadVar encoding and decoding =====# # ===== The value is simply the row * 1000 + col =====# def DecodeRadioRowCol(RadValue): + """ + + :param RadValue: + + """ container = RadValue // 100000 row = RadValue // 1000 col = RadValue % 1000 @@ -4633,6 +6170,13 @@ def DecodeRadioRowCol(RadValue): def EncodeRadioRowCol(container, row, col): + """ + + :param container: + :param row: + :param col: + + """ RadValue = container * 100000 + row * 1000 + col return RadValue @@ -4641,6 +6185,13 @@ def EncodeRadioRowCol(container, row, col): # format of return values is # (Button Pressed, input_values) def BuildResults(form, initialize_only, top_level_form): + """ + + :param form: + :param initialize_only: + :param top_level_form: + + """ # Results for elements are: # TEXT - Nothing # INPUT - Read value from TK @@ -4657,6 +6208,13 @@ def BuildResults(form, initialize_only, top_level_form): def BuildResultsForSubform(form, initialize_only, top_level_form): + """ + + :param form: + :param initialize_only: + :param top_level_form: + + """ button_pressed_text = top_level_form.LastButtonClicked for row_num, row in enumerate(form.Rows): for col_num, element in enumerate(row): @@ -4857,10 +6415,22 @@ def BuildResultsForSubform(form, initialize_only, top_level_form): def FillFormWithValues(form, values_dict): + """ + + :param form: + :param values_dict: + + """ FillSubformWithValues(form, values_dict) def FillSubformWithValues(form, values_dict): + """ + + :param form: + :param values_dict: + + """ for row_num, row in enumerate(form.Rows): for col_num, element in enumerate(row): value = None @@ -4899,6 +6469,12 @@ def FillSubformWithValues(form, values_dict): def _FindElementFromKeyInSubForm(form, key): + """ + + :param form: + :param key: (common_key) Used with window.FindElement and with return values + + """ for row_num, row in enumerate(form.Rows): for col_num, element in enumerate(row): if element.Type == ELEM_TYPE_COLUMN: @@ -4926,6 +6502,11 @@ def _FindElementFromKeyInSubForm(form, key): def _FindElementWithFocusInSubForm(form): + """ + + :param form: + + """ for row_num, row in enumerate(form.Rows): for col_num, element in enumerate(row): if element.Type == ELEM_TYPE_COLUMN: @@ -4960,6 +6541,15 @@ def _FindElementWithFocusInSubForm(form): if sys.version_info[0] >= 3: def AddMenuItem(top_menu, sub_menu_info, element, is_sub_menu=False, skip=False): + """ + + :param top_menu: + :param sub_menu_info: + :param element: + :param is_sub_menu: (Default value = False) + :param skip: (Default value = False) + + """ return_val = None if type(sub_menu_info) is str: if not is_sub_menu and not skip: @@ -5010,6 +6600,15 @@ if sys.version_info[0] >= 3: return return_val else: def AddMenuItem(top_menu, sub_menu_info, element, is_sub_menu=False, skip=False): + """ + + :param top_menu: + :param sub_menu_info: + :param element: + :param is_sub_menu: (Default value = False) + :param skip: (Default value = False) + + """ if not isinstance(sub_menu_info, list): if not is_sub_menu and not skip: # print(f'Adding command {sub_menu_info}') @@ -5081,7 +6680,15 @@ else: # ======================== TK CODE STARTS HERE ========================================= # def PackFormIntoFrame(form, containing_frame, toplevel_form): + """ + + :param form: + :param containing_frame: + :param toplevel_form: + + """ def CharWidthInPixels(): + """ """ return tkinter.font.Font().measure('A') # single character width border_depth = toplevel_form.BorderDepth if toplevel_form.BorderDepth is not None else DEFAULT_BORDER_WIDTH @@ -5256,7 +6863,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): tktext_label.pack_forget() element.TKText = tktext_label if element.ClickSubmits: - tktext_label.bind('', element.TextClickedHandler) + tktext_label.bind('', element._TextClickedHandler) if element.Tooltip is not None: element.TooltipObject = ToolTip(element.TKText, text=element.Tooltip, timeout=DEFAULT_TOOLTIP_TIME) if element.RightClickMenu or toplevel_form.RightClickMenu: @@ -5336,10 +6943,10 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): if element.Visible is False: tkbutton.pack_forget() if element.BindReturnKey: - element.TKButton.bind('', element.ReturnKeyHandler) + element.TKButton.bind('', element._ReturnKeyHandler) if element.Focus is True or (toplevel_form.UseDefaultFocus and not focus_set): focus_set = True - element.TKButton.bind('', element.ReturnKeyHandler) + element.TKButton.bind('', element._ReturnKeyHandler) element.TKButton.focus_set() toplevel_form.TKroot.focus_force() if element.Disabled == True: @@ -5439,7 +7046,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): font=font, show=show, justify=justify) if element.ChangeSubmits: element.TKEntry.bind('', element._KeyboardHandler) - element.TKEntry.bind('', element.ReturnKeyHandler) + element.TKEntry.bind('', element._ReturnKeyHandler) if element.BackgroundColor is not None and element.BackgroundColor != COLOR_SYSTEM_DEFAULT: element.TKEntry.configure(background=element.BackgroundColor) if text_color is not None and text_color != COLOR_SYSTEM_DEFAULT: @@ -5531,7 +7138,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): elif element.Values: element.TKCombo.current(0) if element.ChangeSubmits: - element.TKCombo.bind('<>', element.ComboboxSelectHandler) + element.TKCombo.bind('<>', element._ComboboxSelectHandler) if element.Readonly: element.TKCombo['state'] = 'readonly' if element.Disabled is True: # note overrides readonly if disabled @@ -5586,7 +7193,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): if text_color is not None and text_color != COLOR_SYSTEM_DEFAULT: element.TKListbox.configure(fg=text_color) if element.ChangeSubmits: - element.TKListbox.bind('<>', element.ListboxSelectHandler) + element.TKListbox.bind('<>', element._ListboxSelectHandler) element.vsb = tk.Scrollbar(listbox_frame, orient="vertical", command=element.TKListbox.yview) element.TKListbox.configure(yscrollcommand=element.vsb.set) element.TKListbox.pack(side=tk.LEFT) @@ -5596,8 +7203,8 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): listbox_frame.pack_forget() element.vsb.pack_forget() if element.BindReturnKey: - element.TKListbox.bind('', element.ListboxSelectHandler) - element.TKListbox.bind('', element.ListboxSelectHandler) + element.TKListbox.bind('', element._ListboxSelectHandler) + element.TKListbox.bind('', element._ListboxSelectHandler) if element.Disabled == True: element.TKListbox['state'] = 'disabled' if element.Tooltip is not None: @@ -5629,7 +7236,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): if element.ChangeSubmits: element.TKText.bind('', element._KeyboardHandler) if element.EnterSubmits: - element.TKText.bind('', element.ReturnKeyHandler) + element.TKText.bind('', element._ReturnKeyHandler) if element.Focus is True or (toplevel_form.UseDefaultFocus and not focus_set): focus_set = True element.TKText.focus_set() @@ -5657,7 +7264,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): text=element.Text, width=width, variable=element.TKIntVar, bd=border_depth, font=font, - command=element.CheckboxHandler) + command=element._CheckboxHandler) else: element.TKCheckbutton = element.Widget = tk.Checkbutton(tk_row_frame, anchor=tk.NW, text=element.Text, width=width, @@ -5721,7 +7328,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): width=width, variable=element.TKIntVar, value=value, bd=border_depth, font=font, - command=element.RadioHandler) + command=element._RadioHandler) else: element.TKRadio = element.Widget = tk.Radiobutton(tk_row_frame, anchor=tk.NW, text=element.Text, width=width, @@ -6009,7 +7616,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): PackFormIntoFrame(element, toplevel_form.TKroot, toplevel_form) if element.ChangeSubmits: - element.TKNotebook.bind('<>', element.TabGroupSelectHandler) + element.TKNotebook.bind('<>', element._TabGroupSelectHandler) if element.BorderWidth is not None: element.TKNotebook.configure(borderwidth=element.BorderWidth) if element.Tooltip is not None: @@ -6036,7 +7643,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): bd=element.BorderWidth, relief=element.Relief, font=font, tickinterval=element.TickInterval, - command=element.SliderChangedHandler) + command=element._SliderChangedHandler) else: tkscale = element.Widget = tk.Scale(tk_row_frame, orient=element.Orientation, variable=element.TKIntVar, @@ -6208,6 +7815,11 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): treeview.column(heading, width=width * CharWidthInPixels(), anchor=anchor) def add_treeview_data(node): + """ + + :param node: + + """ # print(f'Inserting {node.key} under parent {node.parent}') if node.key != '': if node.icon: @@ -6311,7 +7923,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): tktext_label.pack_forget() element.TKText = tktext_label if element.ClickSubmits: - tktext_label.bind('', element.TextClickedHandler) + tktext_label.bind('', element._TextClickedHandler) if element.Tooltip is not None: element.TooltipObject = ToolTip(element.TKText, text=element.Tooltip, timeout=DEFAULT_TOOLTIP_TIME) @@ -6327,6 +7939,11 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form): def ConvertFlexToTK(MyFlexForm): + """ + + :param MyFlexForm: + + """ MyFlexForm # type: Window master = MyFlexForm.TKroot master.title(MyFlexForm.Title) @@ -6370,6 +7987,11 @@ def ConvertFlexToTK(MyFlexForm): # ----====----====----====----====----==== STARTUP TK ====----====----====----====----====----# def StartupTK(my_flex_form): + """ + + :param my_flex_form: Window: + + """ # global _my_windows # ow = _my_windows.NumOpenWindows ow = Window.NumOpenWindows @@ -6471,6 +8093,12 @@ def StartupTK(my_flex_form): # Helper function for determining how to wrap text # # ===================================================# def _GetNumLinesNeeded(text, max_line_width): + """ + + :param text: + :param max_line_width: + + """ if max_line_width == 0: return 1 lines = text.split('\n') @@ -6486,6 +8114,11 @@ def _GetNumLinesNeeded(text, max_line_width): # ============================== PROGRESS METER ========================================== # def ConvertArgsToSingleString(*args): + """ + + :param *args: + + """ max_line_total, width_used, total_lines, = 0, 0, 0 single_line_message = '' # loop through args and built a SINGLE string from them @@ -6510,6 +8143,7 @@ METER_STOPPED = False class QuickMeter(object): + """ """ active_meters = {} exit_reasons = {} @@ -6526,6 +8160,21 @@ class QuickMeter(object): else: bar_color = (None, None) if 'orientation' in _3to2kwargs: orientation = _3to2kwargs['orientation']; del _3to2kwargs['orientation'] else: orientation = 'v' + """ + + :param title: + :param current_value: + :param max_value: + :param key: (common_key) Used with window.FindElement and with return values + :param *args: + :param orientation: (Default value = 'v') + :param bar_color: (Default value = (None, None)) + :param button_color: (Default value = (None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = DEFAULT_PROGRESS_BAR_SIZE) + :param border_width: (Default value = None) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + + """ self.start_time = datetime.datetime.utcnow() self.key = key self.orientation = orientation @@ -6541,6 +8190,11 @@ class QuickMeter(object): self.window = self.BuildWindow(*args) def BuildWindow(self, *args): + """ + + :param *args: + + """ layout = [] if self.orientation.lower().startswith('h'): col = [] @@ -6566,6 +8220,13 @@ class QuickMeter(object): return self.window def UpdateMeter(self, current_value, max_value, *args): ### support for *args when updating + """ + + :param current_value: + :param max_value: + :param *args: + + """ self.current_value = current_value self.max_value = max_value self.window.Element('_PROG_').UpdateBar(self.current_value, self.max_value) @@ -6582,6 +8243,7 @@ class QuickMeter(object): return METER_OK def ComputeProgressStats(self): + """ """ utc = datetime.datetime.utcnow() time_delta = utc - self.start_time total_seconds = time_delta.total_seconds() @@ -6623,6 +8285,21 @@ def OneLineProgressMeter(title, current_value, max_value, key, *args, **_3to2kwa else: bar_color = (None, None) if 'orientation' in _3to2kwargs: orientation = _3to2kwargs['orientation']; del _3to2kwargs['orientation'] else: orientation = 'v' + """ + + :param title: + :param current_value: + :param max_value: + :param key: (common_key) Used with window.FindElement and with return values + :param *args: + :param orientation: (Default value = 'v') + :param bar_color: (Default value = (None, None)) + :param button_color: (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = DEFAULT_PROGRESS_BAR_SIZE) + :param border_width: (Default value = None) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + + """ if key not in QuickMeter.active_meters: meter = QuickMeter(title, current_value, max_value, key, *args, orientation=orientation, bar_color=bar_color, button_color=button_color, size=size, border_width=border_width, grab_anywhere=grab_anywhere) @@ -6636,6 +8313,11 @@ def OneLineProgressMeter(title, current_value, max_value, key, *args, **_3to2kwa def OneLineProgressMeterCancel(key): + """ + + :param key: (common_key) Used with window.FindElement and with return values + + """ try: meter = QuickMeter.active_meters[key] meter.window.Close() @@ -6648,6 +8330,11 @@ def OneLineProgressMeterCancel(key): # input is #RRGGBB # output is #RRGGBB def GetComplimentaryHex(color): + """ + + :param color: + + """ # strip the # from the beginning color = color[1:] # convert the string into hex @@ -6663,10 +8350,23 @@ def GetComplimentaryHex(color): # ======================== EasyPrint =====# # ===================================================# class DebugWin(object): + """ """ debug_window = None def __init__(self, size=(None, None), location=(None, None), font=None, no_titlebar=False, no_button=False, grab_anywhere=False, keep_on_top=False, do_not_reroute_stdout=True): + """ + + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param location: (Default value = (None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param no_button: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param do_not_reroute_stdout: (Default value = True) + + """ # Show a form that's a running counter self.size = size self.location = location @@ -6699,6 +8399,13 @@ class DebugWin(object): else: sep = None if 'end' in _3to2kwargs: end = _3to2kwargs['end']; del _3to2kwargs['end'] else: end = None + """ + + :param *args: + :param end: (Default value = None) + :param sep: (Default value = None) + + """ sepchar = sep if sep is not None else ' ' endchar = end if end is not None else '\n' @@ -6722,18 +8429,20 @@ class DebugWin(object): print(*args, sep=sepchar, end=endchar) def Close(self): + """ """ self.window.Close() self.window.__del__() self.window = None def PrintClose(): + """ """ EasyPrintClose() def EasyPrint(*args, **_3to2kwargs): if 'do_not_reroute_stdout' in _3to2kwargs: do_not_reroute_stdout = _3to2kwargs['do_not_reroute_stdout']; del _3to2kwargs['do_not_reroute_stdout'] - else: do_not_reroute_stdout = False + else: do_not_reroute_stdout = True if 'keep_on_top' in _3to2kwargs: keep_on_top = _3to2kwargs['keep_on_top']; del _3to2kwargs['keep_on_top'] else: keep_on_top = False if 'grab_anywhere' in _3to2kwargs: grab_anywhere = _3to2kwargs['grab_anywhere']; del _3to2kwargs['grab_anywhere'] @@ -6752,6 +8461,21 @@ def EasyPrint(*args, **_3to2kwargs): else: end = None if 'size' in _3to2kwargs: size = _3to2kwargs['size']; del _3to2kwargs['size'] else: size = (None, None) + """ + + :param *args: + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param end: (Default value = None) + :param sep: (Default value = None) + :param location: (Default value = (None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param no_button: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param do_not_reroute_stdout: (Default value = True) + + """ if DebugWin.debug_window is None: DebugWin.debug_window = DebugWin(size=size, location=location, font=font, no_titlebar=no_titlebar, no_button=no_button, grab_anywhere=grab_anywhere, keep_on_top=keep_on_top, @@ -6764,6 +8488,7 @@ eprint = EasyPrint def EasyPrintClose(): + """ """ if DebugWin.debug_window is not None: DebugWin.debug_window.Close() DebugWin.debug_window = None @@ -6788,6 +8513,19 @@ def PopupScrolled(*args, **_3to2kwargs): else: yes_no = False if 'button_color' in _3to2kwargs: button_color = _3to2kwargs['button_color']; del _3to2kwargs['button_color'] else: button_color = None + """ + + :param *args: + :param button_color: (Default value = None) + :param yes_no: (Default value = False) + :param auto_close: (Default value = False) + :param auto_close_duration: (Default value = None) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param location: (Default value = (None) + :param title: (Default value = None) + :param non_blocking: (Default value = False) + + """ if not args: return width, height = size width = width if width else MESSAGE_BOX_LINE_WIDTH @@ -6835,6 +8573,11 @@ ScrolledTextBox = PopupScrolled # Sets the icon to be used by default # # ===================================================# def SetGlobalIcon(icon): + """ + + :param icon: + + """ # global _my_windows try: @@ -6861,6 +8604,43 @@ def SetOptions(icon=None, button_color=None, element_size=(None, None), button_e text_element_background_color=None, input_elements_background_color=None, input_text_color=None, scrollbar_color=None, text_color=None, element_text_color=None, debug_win_size=(None, None), window_location=(None, None), error_button_color=(None, None), tooltip_time=None): + """ + + :param icon: (Default value = None) + :param button_color: (Default value = None) + :param element_size: (Default value = (None, None)) + :param button_element_size: (Default value = (None) + :param margins: (Default value = (None) + :param element_padding: (Default value = (None) + :param auto_size_text: True if size should fit the text length (Default value = None) + :param auto_size_buttons: (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param border_width: (Default value = None) + :param slider_border_width: (Default value = None) + :param slider_relief: (Default value = None) + :param slider_orientation: (Default value = None) + :param autoclose_time: (Default value = None) + :param message_box_line_width: (Default value = None) + :param progress_meter_border_depth: (Default value = None) + :param progress_meter_style: (Default value = None) + :param progress_meter_relief: (Default value = None) + :param progress_meter_color: (Default value = None) + :param progress_meter_size: (Default value = None) + :param text_justification: (Default value = None) + :param background_color: color of background (Default value = None) + :param element_background_color: (Default value = None) + :param text_element_background_color: (Default value = None) + :param input_elements_background_color: (Default value = None) + :param input_text_color: (Default value = None) + :param scrollbar_color: (Default value = None) + :param text_color: color of the text (Default value = None) + :param element_text_color: (Default value = None) + :param debug_win_size: (Default value = (None) + :param window_location: (Default value = (None) + :param error_button_color: (Default value = (None) + :param tooltip_time: (Default value = None) + + """ global DEFAULT_ELEMENT_SIZE global DEFAULT_BUTTON_ELEMENT_SIZE global DEFAULT_MARGINS # Margins for each LEFT/RIGHT margin is first term @@ -7282,10 +9062,16 @@ LOOK_AND_FEEL_TABLE = {'SystemDefault': def ListOfLookAndFeelValues(): + """ """ return list(LOOK_AND_FEEL_TABLE.keys()) def ChangeLookAndFeel(index): + """ + + :param index: + + """ # global LOOK_AND_FEEL_TABLE if sys.platform == 'darwin': @@ -7324,6 +9110,11 @@ sprint = ScrolledTextBox # Converts an object's contents into a nice printable string. Great for dumping debug data def ObjToStringSingleObj(obj): + """ + + :param obj: + + """ if obj is None: return 'None' return str(obj.__class__) + '\n' + '\n'.join( @@ -7331,6 +9122,12 @@ def ObjToStringSingleObj(obj): def ObjToString(obj, extra=' '): + """ + + :param obj: + :param extra: (Default value = ' ') + + """ if obj is None: return 'None' return str(obj.__class__) + '\n' + '\n'.join( @@ -7387,24 +9184,27 @@ def Popup(*args, **_3to2kwargs): else: button_color = None if 'title' in _3to2kwargs: title = _3to2kwargs['title']; del _3to2kwargs['title'] else: title = None - """ - Popup - Display a popup box with as many parms as you wish to include - :param args: - :param button_color: - :param background_color: - :param text_color: - :param button_type: - :param auto_close: - :param auto_close_duration: - :param non_blocking: - :param icon: - :param line_width: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Popup - Display a popup box with as many parms as you wish to include :return: + + :param *args: + :param title: (Default value = None) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param button_type: (Default value = POPUP_BUTTONS_OK) + :param auto_close: (Default value = False) + :param auto_close_duration: (Default value = None) + :param custom_text: (Default value = (None, None)) + :param non_blocking: (Default value = False) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param line_width: Width of lines in characters (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None) + """ if not args: args_to_print = [''] @@ -7486,6 +9286,11 @@ def Popup(*args, **_3to2kwargs): # ==================================================# # MsgBox is the legacy call and should not be used any longer def MsgBox(*args): + """ + + :param *args: + + """ raise DeprecationWarning('MsgBox is no longer supported... change your call to Popup') @@ -7519,23 +9324,25 @@ def PopupNoButtons(*args, **_3to2kwargs): else: button_color = None if 'title' in _3to2kwargs: title = _3to2kwargs['title']; del _3to2kwargs['title'] else: title = None - """ - Show a Popup but without any buttons - :param args: - :param button_color: - :param background_color: - :param text_color: - :param auto_close: - :param auto_close_duration: - :param non_blocking: - :param icon: - :param line_width: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Show a Popup but without any buttons :return: + + :param *args: + :param title: (Default value = None) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param auto_close: (Default value = False) + :param auto_close_duration: (Default value = None) + :param non_blocking: (Default value = False) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param line_width: Width of lines in characters (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None, None)) + """ Popup(*args, title=title, button_color=button_color, background_color=background_color, text_color=text_color, button_type=POPUP_BUTTONS_NO_BUTTONS, @@ -7576,24 +9383,26 @@ def PopupNonBlocking(*args, **_3to2kwargs): else: button_type = POPUP_BUTTONS_OK if 'title' in _3to2kwargs: title = _3to2kwargs['title']; del _3to2kwargs['title'] else: title = None - """ - Show Popup box and immediately return (does not block) - :param args: - :param button_type: - :param button_color: - :param background_color: - :param text_color: - :param auto_close: - :param auto_close_duration: - :param non_blocking: - :param icon: - :param line_width: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Show Popup box and immediately return (does not block) :return: + + :param *args: + :param title: (Default value = None) + :param button_type: (Default value = POPUP_BUTTONS_OK) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param auto_close: (Default value = False) + :param auto_close_duration: (Default value = None) + :param non_blocking: (Default value = True) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param line_width: Width of lines in characters (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None, None)) + """ Popup(*args, title=title, button_color=button_color, background_color=background_color, text_color=text_color, button_type=button_type, @@ -7637,24 +9446,26 @@ def PopupQuick(*args, **_3to2kwargs): else: button_type = POPUP_BUTTONS_OK if 'title' in _3to2kwargs: title = _3to2kwargs['title']; del _3to2kwargs['title'] else: title = None - """ - Show Popup box that doesn't block and closes itself - :param args: - :param button_type: - :param button_color: - :param background_color: - :param text_color: - :param auto_close: - :param auto_close_duration: - :param non_blocking: - :param icon: - :param line_width: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Show Popup box that doesn't block and closes itself :return: + + :param *args: + :param title: (Default value = None) + :param button_type: (Default value = POPUP_BUTTONS_OK) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param auto_close: (Default value = True) + :param auto_close_duration: (Default value = 2) + :param non_blocking: (Default value = True) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param line_width: Width of lines in characters (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None, None)) + """ Popup(*args, title=title, button_color=button_color, background_color=background_color, text_color=text_color, button_type=button_type, @@ -7695,24 +9506,26 @@ def PopupQuickMessage(*args, **_3to2kwargs): else: button_type = POPUP_BUTTONS_NO_BUTTONS if 'title' in _3to2kwargs: title = _3to2kwargs['title']; del _3to2kwargs['title'] else: title = None - """ - Show Popup box that doesn't block and closes itself - :param args: - :param button_type: - :param button_color: - :param background_color: - :param text_color: - :param auto_close: - :param auto_close_duration: - :param non_blocking: - :param icon: - :param line_width: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Show Popup box that doesn't block and closes itself :return: + + :param *args: + :param title: (Default value = None) + :param button_type: (Default value = POPUP_BUTTONS_NO_BUTTONS) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param auto_close: (Default value = True) + :param auto_close_duration: (Default value = 2) + :param non_blocking: (Default value = True) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param line_width: Width of lines in characters (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = True) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None, None)) + """ Popup(*args, title=title, button_color=button_color, background_color=background_color, text_color=text_color, button_type=button_type, @@ -7751,23 +9564,25 @@ def PopupNoTitlebar(*args, **_3to2kwargs): else: button_type = POPUP_BUTTONS_OK if 'title' in _3to2kwargs: title = _3to2kwargs['title']; del _3to2kwargs['title'] else: title = None - """ - Display a Popup without a titlebar. Enables grab anywhere so you can move it - :param args: - :param button_type: - :param button_color: - :param background_color: - :param text_color: - :param auto_close: - :param auto_close_duration: - :param non_blocking: - :param icon: - :param line_width: - :param font: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Display a Popup without a titlebar. Enables grab anywhere so you can move it :return: + + :param *args: + :param title: (Default value = None) + :param button_type: (Default value = POPUP_BUTTONS_OK) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param auto_close: (Default value = False) + :param auto_close_duration: (Default value = None) + :param non_blocking: (Default value = False) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param line_width: Width of lines in characters (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param grab_anywhere: (Default value = True) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None, None)) + """ Popup(*args, title=title, button_color=button_color, background_color=background_color, text_color=text_color, button_type=button_type, @@ -7813,24 +9628,26 @@ def PopupAutoClose(*args, **_3to2kwargs): else: button_type = POPUP_BUTTONS_OK if 'title' in _3to2kwargs: title = _3to2kwargs['title']; del _3to2kwargs['title'] else: title = None - """ - Popup that closes itself after some time period - :param args: - :param button_type: - :param button_color: - :param background_color: - :param text_color: - :param auto_close: - :param auto_close_duration: - :param non_blocking: - :param icon: - :param line_width: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Popup that closes itself after some time period :return: + + :param *args: + :param title: (Default value = None) + :param button_type: (Default value = POPUP_BUTTONS_OK) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param auto_close: (Default value = True) + :param auto_close_duration: (Default value = None) + :param non_blocking: (Default value = False) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param line_width: Width of lines in characters (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None, None)) + """ Popup(*args, title=title, button_color=button_color, background_color=background_color, text_color=text_color, button_type=button_type, @@ -7872,23 +9689,25 @@ def PopupError(*args, **_3to2kwargs): else: button_color = (None, None) if 'title' in _3to2kwargs: title = _3to2kwargs['title']; del _3to2kwargs['title'] else: title = None - """ - Popup with colored button and 'Error' as button text - :param args: - :param button_color: - :param background_color: - :param text_color: - :param auto_close: - :param auto_close_duration: - :param non_blocking: - :param icon: - :param line_width: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Popup with colored button and 'Error' as button text :return: + + :param *args: + :param title: (Default value = None) + :param button_color: (Default value = (None, None)) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param auto_close: (Default value = False) + :param auto_close_duration: (Default value = None) + :param non_blocking: (Default value = False) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param line_width: Width of lines in characters (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None) + """ tbutton_color = DEFAULT_ERROR_BUTTON_COLOR if button_color == (None, None) else button_color Popup(*args, title=title, button_type=POPUP_BUTTONS_ERROR, background_color=background_color, text_color=text_color, @@ -7928,23 +9747,25 @@ def PopupCancel(*args, **_3to2kwargs): else: button_color = None if 'title' in _3to2kwargs: title = _3to2kwargs['title']; del _3to2kwargs['title'] else: title = None - """ - Display Popup with "cancelled" button text - :param args: - :param button_color: - :param background_color: - :param text_color: - :param auto_close: - :param auto_close_duration: - :param non_blocking: - :param icon: - :param line_width: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Display Popup with "cancelled" button text :return: + + :param *args: + :param title: (Default value = None) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param auto_close: (Default value = False) + :param auto_close_duration: (Default value = None) + :param non_blocking: (Default value = False) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param line_width: Width of lines in characters (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None, None)) + """ Popup(*args, title=title, button_type=POPUP_BUTTONS_CANCELLED, background_color=background_color, text_color=text_color, @@ -7983,23 +9804,25 @@ def PopupOK(*args, **_3to2kwargs): else: button_color = None if 'title' in _3to2kwargs: title = _3to2kwargs['title']; del _3to2kwargs['title'] else: title = None - """ - Display Popup with OK button only - :param args: - :param button_color: - :param background_color: - :param text_color: - :param auto_close: - :param auto_close_duration: - :param non_blocking: - :param icon: - :param line_width: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Display Popup with OK button only :return: + + :param *args: + :param title: (Default value = None) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param auto_close: (Default value = False) + :param auto_close_duration: (Default value = None) + :param non_blocking: (Default value = False) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param line_width: Width of lines in characters (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None, None)) + """ Popup(*args, title=title, button_type=POPUP_BUTTONS_OK, background_color=background_color, text_color=text_color, non_blocking=non_blocking, icon=icon, line_width=line_width, button_color=button_color, auto_close=auto_close, @@ -8037,23 +9860,25 @@ def PopupOKCancel(*args, **_3to2kwargs): else: button_color = None if 'title' in _3to2kwargs: title = _3to2kwargs['title']; del _3to2kwargs['title'] else: title = None - """ - Display popup with OK and Cancel buttons - :param args: - :param button_color: - :param background_color: - :param text_color: - :param auto_close: - :param auto_close_duration: - :param non_blocking: - :param icon: - :param line_width: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Display popup with OK and Cancel buttons :return: OK, Cancel or None + + :param *args: + :param title: (Default value = None) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param auto_close: (Default value = False) + :param auto_close_duration: (Default value = None) + :param non_blocking: (Default value = False) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param line_width: Width of lines in characters (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None, None)) + """ return Popup(*args, title=title, button_type=POPUP_BUTTONS_OK_CANCEL, background_color=background_color, text_color=text_color, @@ -8092,23 +9917,25 @@ def PopupYesNo(*args, **_3to2kwargs): else: button_color = None if 'title' in _3to2kwargs: title = _3to2kwargs['title']; del _3to2kwargs['title'] else: title = None - """ - Display Popup with Yes and No buttons - :param args: - :param button_color: - :param background_color: - :param text_color: - :param auto_close: - :param auto_close_duration: - :param non_blocking: - :param icon: - :param line_width: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Display Popup with Yes and No buttons :return: Yes, No or None + + :param *args: + :param title: (Default value = None) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param auto_close: (Default value = False) + :param auto_close_duration: (Default value = None) + :param non_blocking: (Default value = False) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param line_width: Width of lines in characters (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None, None)) + """ return Popup(*args, title=title, button_type=POPUP_BUTTONS_YES_NO, background_color=background_color, text_color=text_color, @@ -8127,22 +9954,25 @@ def PopupYesNo(*args, **_3to2kwargs): def PopupGetFolder(message, title=None, default_path='', no_window=False, size=(None, None), button_color=None, background_color=None, text_color=None, icon=DEFAULT_WINDOW_ICON, font=None, no_titlebar=False, grab_anywhere=False, keep_on_top=False, location=(None, None), initial_folder=None): - """ - Display popup with text entry field and browse button. Browse for folder - :param message: - :param default_path: - :param no_window: - :param size: - :param button_color: - :param background_color: - :param text_color: - :param icon: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Display popup with text entry field and browse button. Browse for folder :return: Contents of text field. None if closed using X or cancelled + + :param message: + :param title: (Default value = None) + :param default_path: (Default value = '') + :param no_window: (Default value = False) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None) + :param initial_folder: (Default value = None) + """ # global _my_windows @@ -8199,25 +10029,29 @@ def PopupGetFile(message, title=None, default_path='', default_extension='', sav no_window=False, size=(None, None), button_color=None, background_color=None, text_color=None, icon=DEFAULT_WINDOW_ICON, font=None, no_titlebar=False, grab_anywhere=False, keep_on_top=False, location=(None, None), initial_folder=None): - """ - Display popup with text entry field and browse button. Browse for file - :param message: - :param default_path: - :param default_extension: - :param save_as: - :param file_types: - :param no_window: - :param size: - :param button_color: - :param background_color: - :param text_color: - :param icon: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Display popup with text entry field and browse button. Browse for file :return: string representing the path chosen, None if cancelled or window closed with X + + :param message: + :param title: (Default value = None) + :param default_path: (Default value = '') + :param default_extension: (Default value = '') + :param save_as: (Default value = False) + :param multiple_files: (Default value = False) + :param file_types: (Default value = (("ALL Files", "*.*"))) + :param no_window: (Default value = False) + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None) + :param initial_folder: (Default value = None) + """ if no_window: @@ -8287,22 +10121,24 @@ def PopupGetFile(message, title=None, default_path='', default_extension='', sav def PopupGetText(message, title=None, default_text='', password_char='', size=(None, None), button_color=None, background_color=None, text_color=None, icon=DEFAULT_WINDOW_ICON, font=None, no_titlebar=False, grab_anywhere=False, keep_on_top=False, location=(None, None)): - """ - Display Popup with text entry field - :param message: - :param default_text: - :param password_char: - :param size: - :param button_color: - :param background_color: - :param text_color: - :param icon: - :param font: - :param no_titlebar: - :param grab_anywhere: - :param keep_on_top: - :param location: + """Display Popup with text entry field :return: Text entered or None if window was closed + + :param message: + :param title: (Default value = None) + :param default_text: (Default value = '') + :param password_char: Passwork character if this is a password field (Default value = '') + :param size: (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) + :param button_color: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param icon: Icon to display (Default value = DEFAULT_WINDOW_ICON) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = False) + :param grab_anywhere: If True can grab anywhere to move the window (Default value = False) + :param location: Location on screen to display (Default value = (None, None)) + :param location: (Default value = (None) + """ layout = [[Text(message, auto_size_text=True, text_color=text_color, background_color=background_color, font=font)], @@ -8328,6 +10164,22 @@ def PopupGetText(message, title=None, default_text='', password_char='', size=(N def PopupAnimated(image_source, message=None, background_color=None, text_color=None, font=None, no_titlebar=True, grab_anywhere=True, keep_on_top=True, location=(None, None), alpha_channel=None, time_between_frames=0, transparent_color=None): + """ + + :param image_source: + :param message: (Default value = None) + :param background_color: color of background (Default value = None) + :param text_color: color of the text (Default value = None) + :param font: (common_key) specifies the font family, size, etc (Default value = None) + :param no_titlebar: (Default value = True) + :param grab_anywhere: (Default value = True) + :param keep_on_top: (Default value = True) + :param location: (Default value = (None, None)) + :param alpha_channel: (Default value = None) + :param time_between_frames: (Default value = 0) + :param transparent_color: (Default value = None) + + """ if image_source is None: for image in Window.animated_popup_dict: window = Window.animated_popup_dict[image] @@ -8380,7 +10232,8 @@ MAX_LINES_PER_RESULT_MAIN = 3 POPOUT_WINDOW_FONT = 'Sans 8' -class Debugger(object): +class _Debugger(object): + """ """ debugger = None @@ -8393,6 +10246,7 @@ class Debugger(object): # # # # # # # ###### ###### ##### #### #### #### ###### # # def __init__(self): + """ """ self.watcher_window = None # type: Window self.popout_window = None # type: Window self.local_choices = {} @@ -8403,15 +10257,21 @@ class Debugger(object): self.popout_choices = {} - - def _build_main_debugger_window_callback(self, events): - self._build_main_debugger_window() - # Includes the DUAL PANE (now 2 tabs)! Don't forget REPL is there too! def _build_main_debugger_window(self, location=(None, None)): + """ + + :param location: (Default value = (None, None)) + + """ ChangeLookAndFeel(COLOR_SCHEME) def InVar(key1): + """ + + :param key1: + + """ row1 = [T(' '), I(key=key1, size=(WIDTH_VARIABLES, 1)), T('', key=key1 + 'CHANGED_', size=(WIDTH_RESULTS, 1)), B('Detail', key=key1 + 'DETAIL_'), @@ -8470,6 +10330,12 @@ class Debugger(object): # # # # # # # ####### ## ###### # # # ####### #### #### # def _refresh_main_debugger_window(self, mylocals, myglobals): + """ + + :param mylocals: + :param myglobals: + + """ if not self.watcher_window: # if there is no window setup, nothing to do return False event, values = self.watcher_window.Read(timeout=1) @@ -8485,18 +10351,15 @@ class Debugger(object): if event == 'Go': # GO BUTTON self.watcher_window.Element('_REPL_').Update('') self.watcher_window.Element('_OUTPUT_').Update(">>> {}\n".format(cmd), append=True, autoscroll=True) - # REMOVED ONLY IN 2.7 version!!! HAND EDITED! - - # if sys.version_info[0] >= 3: - # try: - # result = eval('{}'.format(cmd), myglobals, mylocals) - # except Exception as e: - # try: - # result = exec('{}'.format(cmd), myglobals, mylocals) - # except Exception as e: - # result = 'Exception {}\n'.format(e) - # else: - result = '' + + try: + result = eval('{}'.format(cmd), myglobals, mylocals) + except Exception as e: + # try: + # result = exec('{}'.format(cmd), myglobals, mylocals) + # except Exception as e: + # result = 'Exception {}\n'.format(e) + result = '' self.watcher_window.Element('_OUTPUT_').Update('{}\n'.format(result), append=True, autoscroll=True) # BUTTON - DETAIL elif event.endswith('_DETAIL_'): # DETAIL BUTTON @@ -8602,6 +10465,11 @@ class Debugger(object): # displays them into a single text box def _display_all_vars(self, dict): + """ + + :param dict: + + """ num_cols = 3 output_text = '' num_lines = 2 @@ -8641,6 +10509,11 @@ class Debugger(object): # # # # # # # # ##### ###### ###### #### ## ## # # # def _choose_auto_watches(self, my_locals): + """ + + :param my_locals: + + """ ChangeLookAndFeel(COLOR_SCHEME) num_cols = 3 output_text = '' @@ -8713,6 +10586,11 @@ class Debugger(object): ## ## # # # ##### #### # # def _build_floating_window(self, location=(None, None)): + """ + + :param location: (Default value = (None, None)) + + """ if self.popout_window: # if floating window already exists, close it first self.popout_window.Close() ChangeLookAndFeel('Topanga') @@ -8783,6 +10661,7 @@ class Debugger(object): ## ## # # # ##### #### # # def _refresh_floating_window(self): + """ """ if not self.popout_window: return for key in self.popout_choices: @@ -8819,9 +10698,15 @@ class Debugger(object): def show_debugger_window(location=(None, None), *args): - if Debugger.debugger is None: - Debugger.debugger = Debugger() - debugger = Debugger.debugger + """ + + :param location: (Default value = (None, None)) + :param *args: + + """ + if _Debugger.debugger is None: + _Debugger.debugger = _Debugger() + debugger = _Debugger.debugger frame = inspect.currentframe() prev_frame = inspect.currentframe().f_back # frame, *others = inspect.stack()[1] @@ -8837,9 +10722,15 @@ def show_debugger_window(location=(None, None), *args): def show_debugger_popout_window(location=(None, None), *args): - if Debugger.debugger is None: - Debugger.debugger = Debugger() - debugger = Debugger.debugger + """ + + :param location: (Default value = (None, None)) + :param *args: + + """ + if _Debugger.debugger is None: + _Debugger.debugger = _Debugger() + debugger = _Debugger.debugger frame = inspect.currentframe() prev_frame = inspect.currentframe().f_back # frame = inspect.getframeinfo(prev_frame) @@ -8856,9 +10747,10 @@ def show_debugger_popout_window(location=(None, None), *args): def refresh_debugger(): - if Debugger.debugger is None: - Debugger.debugger = Debugger() - debugger = Debugger.debugger + """ """ + if _Debugger.debugger is None: + _Debugger.debugger = _Debugger() + debugger = _Debugger.debugger Window.read_call_from_debugger = True frame = inspect.currentframe() frame = inspect.currentframe().f_back @@ -8893,6 +10785,7 @@ def refresh_debugger(): def main(): + """ """ from random import randint ChangeLookAndFeel('GreenTan') @@ -8988,8 +10881,7 @@ def main(): graph_elem.DrawCircle((200, 200), 50, 'blue') i = 0 while True: # Event Loop - # TimerStart() - event, values = window.Read(timeout=0) + event, values = window.Read(timeout=1) if event != TIMEOUT_KEY: print(event, values) if event is None or event == 'Exit': @@ -9016,18 +10908,6 @@ def main(): # TimerStop() window.Close() - # layout = [[Text('You are running the PySimpleGUI.py file itself')], - # [Text('You should be importing it rather than running it', size=(50, 2))], - # [Text('Here is your sample input window....')], - # [Text('Source Folder', size=(15, 1), justification='right'), InputText('Source', focus=True), - # FolderBrowse(tooltip='Browse for a folder')], - # [Text('Destination Folder', size=(15, 1), justification='right'), InputText('Dest'), FolderBrowse()], - # [Ok(bind_return_key=True), Cancel()]] - # - # window = Window('Demo window..').Layout(layout) - # event, values = window.Read() - # window.Close() - if __name__ == '__main__': main() diff --git a/docs/index.md b/docs/index.md index 7a1aadc1..024be149 100644 --- a/docs/index.md +++ b/docs/index.md @@ -1,5 +1,4 @@ - - + + ```python def PopupAnimated(image_source, message=None, @@ -1811,20 +1794,27 @@ The code is a crude representation of the GUI, laid out in text. Many of the main method calls and Element names have shortcuts. This enables you to code much quicker once you are used to using the SDK. The Text Element, for example, has 3 different names `Text`, `Txt` or`T`. InputText can also be written `Input` or `In` . `FindElement` was recently renamed to `Element` because it's a commonly used function. - -## Text Element -``` -layout = [[sg.Text('This is what a Text Element looks like')]] -``` + + + + + +## Text Element | `T == Txt == Text` +Basic Element. It displays text. That's it. + + + + +```python +layout = [ + [sg.Text('This is what a Text Element looks like')], + [sg.T('Second label')], + ] +``` ![simple text](https://user-images.githubusercontent.com/13696193/44959877-e9d97b00-aec3-11e8-9d24-b4405ee4a148.jpg) -The most basic element is the Text element. It simply displays text. Many of the 'options' that can be set for a Text element are shared by other elements. - - - - ```python Text(text, size=(None, None), @@ -1843,9 +1833,9 @@ Text(text, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | text | | | size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | @@ -1867,75 +1857,32 @@ Parameters explained: +### Methods - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -Some commonly used elements have 'shorthand' versions of the functions to make the code more compact. The functions `T` and `Txt` are the same as calling `Text`. + Parameters explained: +| Name | Meaning | +|-|-| +| value | (Default value = None) | +| background_color | color of background (Default value = None) | +| text_color | color of the text (Default value = None) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + +--- ### Fonts @@ -1959,39 +1906,27 @@ The shorthand functions for `Text` are `Txt` and `T` If you set the parameter `enable_events` or `click_submits` then you will get an event if the user clicks on the Text. -### Text Methods - -#### Update -If you want to change the text, font, or colors after the element has been created, then use the `Update` method. - -```python -Update(value = None, background_color=None, text_color=None, font=None, visible=None) -``` -| name | meaning | -| - | - | -| value | new value to set text element to | -| background_color | new background color | -| text_color | text color to display | -| font | font to use to display | -| visible | visibility | -| | | + ## Multiline Element +This Element doubles as both an input and output Element. + + +```python layout = [[sg.Multiline('This is what a Multi-line Text Element looks like', size=(45,5))]] +``` ![multiline](https://user-images.githubusercontent.com/13696193/44959853-b139a180-aec3-11e8-972f-f52188510c88.jpg) -This Element doubles as both an input and output Element. - ```python Multiline(default_text="", @@ -2015,9 +1950,9 @@ Multiline(default_text="", visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | default_text | (Default value = '') | | enter_submits | (Default value = False) | @@ -2041,42 +1976,10 @@ Parameters explained: ||| - - -### Multiline Methods - +### Methods + ```python Update(value=None, @@ -2084,91 +1987,45 @@ Update(value=None, append=False, font=None, text_color=None, - background_color=None) - -Update(self, value=None, disabled=None, append=False, font=None, text_color=None, background_color=None)value=None, disabled=None, append=False): + background_color=None, + visible=None, + autoscroll=None) ``` -|name|meaning| + + + Parameters explained: +| Name | Meaning | |-|-| -| value | string to set the text field to| -| disabled | set to True to disable the element| -| append | rather than replacing the current text with new text, add the new text onto the end| +| value | (Default value = None) | +| disabled | disable or enable state of the element (Default value = None) | +| append | (Default value = False) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| text_color | color of the text (Default value = None) | +| background_color | color of background (Default value = None) | +| visible | change visibility of element (Default value = None) | +| autoscroll | (Default value = None) | ||| + + + + + + + + + + -## Output Element -Output re-routes `Stdout` to a scrolled text box. - -Whatever you print will show up in this window. - -Note that you will NOT see what you print until you call either window.Read or window.Refresh. If you want to immediately see what was printed, call window.Refresh() immediately after your print statement. - -```python -layout = [[sg.Output(size=(80,10)]] -``` - -![output](https://user-images.githubusercontent.com/13696193/44959863-b72f8280-aec3-11e8-8caa-7bc743149953.jpg) - - - -```python -Output(size=(None, None), - background_color=None, - text_color=None, - pad=None, - font=None, - tooltip=None, - key=None, - right_click_menu=None, - visible=True) -``` - -Parameters explained: - -|Name|Meaning| -|-|-| -| size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | -| background_color | color of background (Default value = None) | -| text_color | color of the text (Default value = None) | -| pad | (common_key) Amount of padding to put around element (Default value = None) | -| font | (common_key) specifies the font family, size, etc (Default value = None) | -| tooltip | text, that will appear the you hover on (Default value = None) | -| key | (common_key) Used with window.FindElement and with return values (Default value = None) | -| right_click_menu | see "Right Click Menus" (Default value = None) | -| visible | set visibility state of the element (Default value = True) | -||| - - - - - -## Input Elements -These make up the majority of the window definition. Optional variables at the Element level override the window level values (e.g. `size` is specified in the Element). All input Elements create an entry in the list of return values. A Text Input Element creates a string in the list of items returned. - -## Text Input Element Shortcuts (Input, In) +## Text Input Element | `Input == In` +Shows a single line of input. ```python layout = [[sg.InputText('Default text')]] ``` ![inputtext 2](https://user-images.githubusercontent.com/13696193/44959861-b5fe5580-aec3-11e8-8040-53ec241b5079.jpg) -Shows a single line of input. ```python InputText(default_text="", @@ -2190,9 +2047,9 @@ InputText(default_text="", visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | default_text | (Default value = '') | | size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | @@ -2214,6 +2071,8 @@ Parameters explained: ||| + + + + + + + + + + -## Combo Element +## Combo Element | `Combo == DropDown == Drop` Also known as a drop-down list. Only required parameter is the list of choices. The return value is a string matching what's visible on the GUI. +ComboBox Element + ```python layout = [[sg.InputCombo(['choice 1', 'choice 2'])]] ``` @@ -2293,8 +2193,6 @@ layout = [[sg.InputCombo(['choice 1', 'choice 2'])]] ![combobox](https://user-images.githubusercontent.com/13696193/44959860-b565bf00-aec3-11e8-82fe-dbe41252458b.jpg) -ComboBox Element - ```python Combo(values, default_value=None, @@ -2313,9 +2211,9 @@ Combo(values, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | values | | | default_value | (Default value = None) | @@ -2335,6 +2233,8 @@ Parameters explained: ||| + + -Shortcut functions - Combo, DropDown, Drop -### Combo Methods +### Methods + + ```python +Update(value=None, + values=None, + set_to_index=None, + disabled=None, + readonly=None, + font=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| value | (Default value = None) | +| values | (Default value = None) | +| set_to_index | (Default value = None) | +| disabled | disable or enable state of the element (Default value = None) | +| readonly | make element readonly (Default value = None) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + + + + + + + ## Listbox Element The standard listbox like you'll find in most GUIs. Note that the return values from this element will be a ***list of results, not a single result***. This is because the user can select more than 1 item from the list (if you set the right mode). -```python - layout = [[sg.Listbox(values=['Listbox 1', 'Listbox 2', 'Listbox 3'], size=(30, 6))]] -``` - -![listbox 2](https://user-images.githubusercontent.com/13696193/44959859-b4cd2880-aec3-11e8-881c-1e369d5c6337.jpg) - - A List Box. Provide a list of values for the user to choose one or more of. Returns a list of selected rows when a window.Read() is executed. +```python +layout = [[sg.Listbox(values=['Listbox 1', 'Listbox 2', 'Listbox 3'], size=(30, 6))]] +``` + +![listbox 2](https://user-images.githubusercontent.com/13696193/44959859-b4cd2880-aec3-11e8-881c-1e369d5c6337.jpg) ```python @@ -2418,9 +2350,9 @@ Listbox(values, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | values | | | default_values | (Default value = None) | @@ -2442,59 +2374,59 @@ Parameters explained: ||| - - The `select_mode` option can be a string or a constant value defined as a variable. Generally speaking strings are used for these kinds of options. -ListBoxes can cause a window to return from a Read call. If the flag change_submits is set, then when a user makes a selection, the Read immediately returns. -Another way ListBoxes can cause Reads to return is if the flag bind_return_key is set. If True, then if the user presses the return key while an entry is selected, then the Read returns. Also, if this flag is set, if the user double-clicks an entry it will return from the Read. -### Listbox Methods +### Methods + + +```python +Update(values=None, + disabled=None, + set_to_index=None, + scroll_to_index=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| values | (Default value = None) | +| disabled | disable or enable state of the element (Default value = None) | +| set_to_index | highlights the item at this index as if user clicked (Default value = None) | +| scroll_to_index | scroll the listbox so that this index is the first shown (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + +```python +SetValue(values) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| values | | +||| + + + + + +
+GetListValues()
+ + + + + +--- + +ListBoxes can cause a window to return from a Read call. If the flag change_submits is set, then when a user makes a selection, the Read immediately returns. +Another way ListBoxes can cause Reads to return is if the flag bind_return_key is set. If True, then if the user presses the return key while an entry is selected, then the Read returns. Also, if this flag is set, if the user double-clicks an entry it will return from the Read. + + + + + + + + + + + ## Slider Element Sliders have a couple of slider-specific settings as well as appearance settings. Examples include the `orientation` and `range` settings. - layout = [[sg.Slider(range=(1,500), default_value=222, size=(20,15), orientation='horizontal', font=('Helvetica', 12))]] + +```python +layout = [[sg.Slider(range=(1,500), default_value=222, size=(20,15), orientation='horizontal', font=('Helvetica', 12))]] +``` ![slider](https://user-images.githubusercontent.com/13696193/44959858-b4349200-aec3-11e8-9e25-c0fcf025d19e.jpg) - ```python Slider(range=(None, None), @@ -2544,9 +2495,9 @@ Slider(range=(None, None), visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | range | (Default value = (None, None)) | | default_value | (Default value = None) | @@ -2570,6 +2521,8 @@ Parameters explained: ||| + + + + ### Qt Sliders There is an important difference between Qt and tkinter sliders. On Qt, the slider values must be integer, not float. If you want your slider to go from 0.1 to 1.0, then make your slider go from 1 to 10 and divide by 10. It's an easy math thing to do and not a big deal. Just deal with it.... you're writing software after all. Presumably you know how to do these things. ;-) ### Slider Methods + + ```python +Update(value=None, + range=(None, None), + disabled=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| value | (Default value = None) | +| range | (Default value = (None, None)) | +| disabled | disable or enable state of the element (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + + + + + + + + + + + + + + + + ## Radio Element - Creates one radio button that is assigned to a group of radio buttons. Only 1 of the buttons in the group can be selected at any one time. +Radio Button Element - Used in a group of other Radio Elements to provide user with ability to select only +1 choice in a list of choices. ```python layout = [ @@ -2640,11 +2634,6 @@ layout = [ ![radio](https://user-images.githubusercontent.com/13696193/44959857-b4349200-aec3-11e8-8e2d-e6a49ffbd0b6.jpg) -Radio Button Element - Used in a group of other Radio Elements to provide user with ability to select only -1 choice in a list of choices. - - - ```python Radio(text, @@ -2664,9 +2653,9 @@ Radio(text, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | text | | | group_id | | @@ -2686,6 +2675,8 @@ Parameters explained: ||| + + -### Radio Button Methods +### Methods + + ```python +Update(value=None, + disabled=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| value | (Default value = None) | +| disabled | disable or enable state of the element (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + + -## Checkbox Element +## Checkbox Element | `CBox == CB == Check` Checkbox elements are like Radio Button elements. They return a bool indicating whether or not they are checked. - layout = [[sg.Checkbox('My first Checkbox!', default=True), sg.Checkbox('My second Checkbox!')]] - +```python +layout = [[sg.Checkbox('My first Checkbox!', default=True), sg.Checkbox('My second Checkbox!')]] +``` ![checkbox](https://user-images.githubusercontent.com/13696193/44959906-6f5d2b00-aec4-11e8-9c8a-962c787f0286.jpg) - + ```python Checkbox(text, @@ -2756,9 +2770,9 @@ Checkbox(text, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | text | | | default | (Default value = False) | @@ -2777,19 +2791,21 @@ Parameters explained: ||| - -Shortcut functions - CBox, CB, Check -### Checkbox Methods +### Methods ```python Update(value=None, disabled=None, visible=None) Get() @@ -2827,14 +2842,16 @@ Get() ## Spin Element - An up/down spinner control. The valid values are passed in as a list. - layout = [[sg.Spin([i for i in range(1,11)], initial_value=1), sg.Text('Volume level')]] + + +```python +layout = [[sg.Spin([i for i in range(1,11)], initial_value=1), sg.Text('Volume level')]] +``` ![spinner](https://user-images.githubusercontent.com/13696193/44959855-b1d23800-aec3-11e8-9f51-afb2109879da.jpg) - ```python Spin(values, @@ -2853,9 +2870,9 @@ Spin(values, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | values | | | initial_value | (Default value = None) | @@ -2874,37 +2891,39 @@ Parameters explained: ||| + + + values - List of valid values + initial_value - String with initial value + size - (width, height) size of element in characters + auto_size_text - Bool. True if should size width to fit text + font - Font type and size for text display + disabled - Bool. If True element is disabled + background_color - color to use for the background + text_color - color to use for the typed text + change_submits - causes window read to immediately return if the spinner value changes + key = Dictionary key to use for return values + pad - padding around element in the window + tooltip - text to show when mouse hovered over element --> ### Qt Differences - `values` is a range! @@ -2914,13 +2933,56 @@ Note that Qt does not allow arbitrary spinner values. With PySimpleGUI-tkinter On Qt values is a tuple representing a range. On plain PySimpleGUI this value is a list of items. Make sure on the plain version you specify items as a list using [] and not a generator using (). -### Spin Methods +### Methods + + ```python +Update(value=None, + values=None, + disabled=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| value | (Default value = None) | +| values | (Default value = None) | +| disabled | disable or enable state of the element (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + + + + + + + + + + + + + + + + + + + + + ## Image Element @@ -2928,7 +2990,9 @@ disabled - if True disables the element Images can be placed in your window provide they are in PNG, GIF, PPM/PGM format. JPGs cannot be shown because tkinter does not naively support JPGs. You can use the Python Imaging Library (PIL) package to convert your image to PNG prior to calling PySimpleGUI if your images are in JPG format. - + +????????????? add image here + ```python Image(filename=None, @@ -2943,9 +3007,9 @@ Image(filename=None, enable_events=False) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | filename | (Default value = None) | | data | (Default value = None) | @@ -2960,67 +3024,110 @@ Parameters explained: ||| + + -### `Update` Method +### Methods + + +```python +Update(filename=None, + data=None, + size=(None, None), + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| filename | (Default value = None) | +| data | (Default value = None) | +| size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | +| visible | change visibility of element (Default value = None) | +||| -Like other Elements, the Image Element has an `Update` method. Call Update if you want to change the image. - def Update(self, filename=None, data=None, visible=None): Choose **either** a filename or in-ram data image to use to replace current image -### `UpdateAnimation` Method for Animated GIFs + +##### `UpdateAnimation` Method for Animated GIFs + +You can specify an animated GIF as an image and can animate the GIF by calling `UpdateAnimation`. Exciting stuff! ![loading animation](https://user-images.githubusercontent.com/13696193/51280871-d2041e80-19ae-11e9-8757-802eb95352ed.gif) -Starting in version 3.23 you can specify an animated GIF as an image and can animate the GIF by calling `UpdateAnimation`. Exciting stuff! - ```python UpdateAnimation(source, - time_between_frames=0) + time_between_frames=0) ``` -`source` can be a filename ***or*** a base64 bytes variable (unlike other calls that split out the filename parameter and base64 parameter into 2 parameters. -`time_between_frames` is an optional parameter. It will keep track of the amount of time between frame changes for you to give you a smooth animation. With this parameter you can call the function as often as you want and it will advance to the next frame only after the correct amount of time has lapsed. + + Parameters explained: +| Name | Meaning | +|-|-| +| source | | +| time_between_frames | (Default value = 0) | +||| + + + You can call the method without setting the `time_between_frames` value and it will show a frame and immediately move on to the next frame. This enables you to do the inter-frame timing. + + + + + + + + + + + + + + + + ## Button Element @@ -3072,7 +3179,14 @@ In Oct 2018, the definition of Button changed. Previously Button would CLOSE th The most basic Button element call to use is `Button` - + + +```python +layout = [[sg.OK(), sg.Cancel()]] +``` + +![ok cancel 3](https://user-images.githubusercontent.com/13696193/44959927-aa5f5e80-aec4-11e8-86e1-5dc0b3a2b803.jpg) + ```python Button(button_text="", @@ -3100,9 +3214,9 @@ Button(button_text="", visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | button_text | (Default value = '') | | button_type | (Default value = BUTTON_TYPE_READ_FORM) | @@ -3130,71 +3244,75 @@ Parameters explained: ||| - ### Shortcut, Pre-defined Buttons These Pre-made buttons are some of the most important elements of all because they are used so much. They all basically do the same thing, set the button text to match the function name and set the parameters to commonly used values. If you find yourself needing to create a custom button often because it's not on this list, please post a request on GitHub. . They include: - OK - Ok - Submit - Cancel - Yes - No - Exit - Quit - Help - Save - SaveAs - FileBrowse - FilesBrowse - FileSaveAs - FolderBrowse + OK + Ok + Submit + Cancel + Yes + No + Exit + Quit + Help + Save + SaveAs + FileBrowse + FilesBrowse + FileSaveAs + FolderBrowse **IMPORT NOTE ABOUT SHORTCUT BUTTONS** Prior to release 3.11.0, these buttons closed the window. Starting with 3.11 they will not close the window. They act like RButtons (return the button text and do not close the window) @@ -3203,13 +3321,6 @@ If you are having trouble with these buttons closing your window, please check y Using older versions, if you want a Submit() button that does not close the window, then you would instead use RButton('Submit'). Using the new version, if you want a Submit button that closes the window like the sold Submit() call did, you would write that as `CloseButton('Submit')` or `CButton('Submit')` - - layout = [[sg.OK(), sg.Cancel()]] - -![ok cancel 3](https://user-images.githubusercontent.com/13696193/44959927-aa5f5e80-aec4-11e8-86e1-5dc0b3a2b803.jpg) - - - ### Button targets The `FileBrowse`, `FolderBrowse`, `FileSaveAs` , `FilesSaveAs`, `CalendarButton`, `ColorChooserButton` buttons all fill-in values into another element located on the window. The target can be a Text Element or an InputText Element. The location of the element is specified by the `target` variable in the function call. @@ -3236,20 +3347,24 @@ Let's examine this window as an example: The `InputText` element is located at (1,0)... row 1, column 0. The `Browse` button is located at position (2,0). The Target for the button could be any of these values: - Target = (1,0) - Target = (-1,0) + Target = (1,0) + Target = (-1,0) The code for the entire window could be: - layout = [[sg.T('Source Folder')], - [sg.In()], - [sg.FolderBrowse(target=(-1, 0)), sg.OK()]] +```python +layout = [[sg.T('Source Folder')], + [sg.In()], + [sg.FolderBrowse(target=(-1, 0)), sg.OK()]] +``` or if using keys, then the code would be: - layout = [[sg.T('Source Folder')], - [sg.In(key='input')], - [sg.FolderBrowse(target='input'), sg.OK()]] +```python +layout = [[sg.T('Source Folder')], + [sg.In(key='input')], + [sg.FolderBrowse(target='input'), sg.OK()]] +``` See how much easier the key method is? @@ -3298,14 +3413,14 @@ Your button images need to be in PNG or GIF format. When you make a button with This example comes from the `Demo Media Player.py` example program. Because it's a non-blocking button, it's defined as `RButton`. You also put images on blocking buttons by using `Button`. - sg.RButton('Restart Song', button_color=sg.TRANSPARENT_BUTTON, - image_filename=image_restart, image_size=(50, 50), image_subsample=2, border_width=0) + sg.RButton('Restart Song', button_color=sg.TRANSPARENT_BUTTON, + image_filename=image_restart, image_size=(50, 50), image_subsample=2, border_width=0) Three parameters are used for button images. - image_filename - Filename. Can be a relative path - image_size - Size of image file in pixels - image_subsample - Amount to divide the size by. 2 means your image will be 1/2 the size. 3 means 1/3 + image_filename - Filename. Can be a relative path + image_size - Size of image file in pixels + image_subsample - Amount to divide the size by. 2 means your image will be 1/2 the size. 3 means 1/3 Here's an example window made with button images. @@ -3313,12 +3428,12 @@ Here's an example window made with button images. You'll find the source code in the file Demo Media Player. Here is what the button calls look like to create media player window ```python - sg.RButton('Pause', button_color=sg.TRANSPARENT_BUTTON, - image_filename=image_pause, - image_size=(50, 50), - image_subsample=2, - border_width=0) - ``` +sg.RButton('Pause', button_color=sg.TRANSPARENT_BUTTON, + image_filename=image_pause, + image_size=(50, 50), + image_subsample=2, + border_width=0) +``` This is one you'll have to experiment with at this point. Not up for an exhaustive explanation. **Realtime Buttons** @@ -3336,12 +3451,12 @@ Here is the code to make, show and get results from this window: import PySimpleGUI as sg gui_rows = [[sg.Text('Robotics Remote Control')], - [sg.T(' ' * 10), sg.RealtimeButton('Forward')], - [sg.RealtimeButton('Left'), sg.T(' ' * 15), sg.RealtimeButton('Right')], - [sg.T(' ' * 10), sg.RealtimeButton('Reverse')], - [sg.T('')], - [sg.Quit(button_color=('black', 'orange'))] - ] + [sg.T(' ' * 10), sg.RealtimeButton('Forward')], + [sg.RealtimeButton('Left'), sg.T(' ' * 15), sg.RealtimeButton('Right')], + [sg.T(' ' * 10), sg.RealtimeButton('Reverse')], + [sg.T('')], + [sg.Quit(button_color=('black', 'orange'))] + ] window = sg.Window('Robotics Remote Control', auto_size_text=True).Layout(gui_rows) @@ -3352,12 +3467,12 @@ window = sg.Window('Robotics Remote Control', auto_size_text=True).Layout(gui_ro # # your program's main loop while (True): - # This is the code that reads and updates your window - event, values = window.Read(timeout=0) - if event is not None: - print(event) - if event == 'Quit' or values is None: - break + # This is the code that reads and updates your window + event, values = window.Read(timeout=0) + if event is not None: + print(event) + if event == 'Quit' or values is None: + break window.Close() # Don't forget to close your window! ``` @@ -3367,38 +3482,89 @@ This loop will read button values and print them. When one of the Realtime butt **File Types** The `FileBrowse` & `SaveAs` buttons have an additional setting named `file_types`. This variable is used to filter the files shown in the file dialog box. The default value for this setting is - FileTypes=(("ALL Files", "*.*"),) + FileTypes=(("ALL Files", "*.*"),) This code produces a window where the Browse button only shows files of type .TXT - layout = [[sg.In() ,sg.FileBrowse(file_types=(("Text Files", "*.txt"),))]] + layout = [[sg.In() ,sg.FileBrowse(file_types=(("Text Files", "*.txt"),))]] NOTE - Mac users will not be able to use the file_types parameter. tkinter has a bug on Macs that will crash the program is a file_type is attempted so that feature had to be removed. Sorry about that! ***The ENTER key*** - The ENTER key is an important part of data entry for windows. There's a long tradition of the enter key being used to quickly submit windows. PySimpleGUI implements this by tying the ENTER key to the first button that closes or reads a window. + The ENTER key is an important part of data entry for windows. There's a long tradition of the enter key being used to quickly submit windows. PySimpleGUI implements this by tying the ENTER key to the first button that closes or reads a window. The Enter Key can be "bound" to a particular button so that when the key is pressed, it causes the window to return as if the button was clicked. This is done using the `bind_return_key` parameter in the button calls. If there are more than 1 button on a window, the FIRST button that is of type Close window or Read window is used. First is determined by scanning the window, top to bottom and left to right. -### Button Methods +### Methods + + ```python +Update(text=None, + button_color=(None, None), + disabled=None, + image_data=None, + image_filename=None, + visible=None, + image_subsample=None, + image_size=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| text | (Default value = None) | +| button_color | (Default value = (None) | +| disabled | disable or enable state of the element (Default value = None) | +| image_data | (Default value = None) | +| image_filename | (Default value = None) | +| visible | change visibility of element (Default value = None) | +| image_subsample | (Default value = None) | +| image_size | (Default value = None) | +||| + + + + + GetText - Returns the current text shown on a button +
+GetText()
+ + + + + + + + + + + + + + + + + ## ButtonMenu Element @@ -3411,7 +3577,8 @@ Here is a sample definition: The very first string normally specifies what is shown on the menu bar. In this case, the value is **not used**. You set the text for the button using a different parameter, the `button_text` parm. - + + ```python ButtonMenu(button_text, @@ -3433,9 +3600,9 @@ ButtonMenu(button_text, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | button_text | | | menu_def | | @@ -3457,25 +3624,27 @@ Parameters explained: ||| + + @@ -3489,19 +3658,57 @@ Return values for ButtonMenus are sent via the return values dictionary. If a s + + + + + + + + + + + + + + + + + ## VerticalSeparator Element This element has limited usefulness and is being included more for completeness than anything else. It will draw a line between elements. +It works best when placed between columns or elements that span multiple rows. If on a "normal" row with elements that are only 1 row high, then it will only span that one row. + + + ```python VerticalSeparator(pad=None) ``` ![snag-0129](https://user-images.githubusercontent.com/13696193/47376041-a92a0100-d6bf-11e8-8f5b-0c0df56cf0f3.jpg) -It works best when placed between columns or elements that span multiple rows. If on a "normal" row with elements that are only 1 row high, then it will only span that one row. + +```python +VerticalSeparator(pad=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| pad | (common_key) Amount of padding to put around element (Default value = None) | +||| + + + + + + + @@ -3513,7 +3720,9 @@ The `ProgressBar` element is used to build custom Progress Bar windows. It is H The **easiest** way to get progress meters into your code is to use the `OneLineProgressMeter` API. This consists of a pair of functions, `OneLineProgressMeter` and `OneLineProgressMeterCancel`. You can easily cancel any progress meter by calling it with the current value = max value. This will mark the meter as expired and close the window. You've already seen OneLineProgressMeter calls presented earlier in this readme. - sg.OneLineProgressMeter('My Meter', i+1, 1000, 'key', 'Optional message') +```python +sg.OneLineProgressMeter('My Meter', i+1, 1000, 'key', 'Optional message') +``` The return value for `OneLineProgressMeter` is: `True` if meter updated correctly @@ -3522,65 +3731,240 @@ The return value for `OneLineProgressMeter` is: #### Progress Mater in Your window Another way of using a Progress Meter with PySimpleGUI is to build a custom window with a `ProgressBar` Element in the window. You will need to run your window as a non-blocking window. When you are ready to update your progress bar, you call the `UpdateBar` method for the `ProgressBar` element itself. + + +```python +import PySimpleGUI as sg + +# layout the window +layout = [[sg.Text('A custom progress meter')], + [sg.ProgressBar(10000, orientation='h', size=(20, 20), key='progressbar')], + [sg.Cancel()]] + +# create the window` +window = sg.Window('Custom Progress Meter').Layout(layout) +progress_bar = window.FindElement('progressbar') +# loop that would normally do something useful +for i in range(10000): + # check to see if the cancel button was clicked and exit loop if clicked + event, values = window.Read(timeout=0) + if event == 'Cancel' or event is None: + break + # update bar with loop value +1 so that bar eventually reaches the maximum + progress_bar.UpdateBar(i + 1) +# done with loop... need to destroy the window as it's still open +window.Close()) +``` + ![progress custom](https://user-images.githubusercontent.com/13696193/45243969-c3508100-b2c3-11e8-82bc-927d0307e093.jpg) - import PySimpleGUI as sg - - # layout the window - layout = [[sg.Text('A custom progress meter')], - [sg.ProgressBar(10000, orientation='h', size=(20, 20), key='progressbar')], - [sg.Cancel()]] - - # create the window` - window = sg.Window('Custom Progress Meter').Layout(layout) - progress_bar = window.FindElement('progressbar') - # loop that would normally do something useful - for i in range(10000): - # check to see if the cancel button was clicked and exit loop if clicked - event, values = window.Read(timeout=0) - if event == 'Cancel' or event is None: - break - # update bar with loop value +1 so that bar eventually reaches the maximum - progress_bar.UpdateBar(i + 1) - # done with loop... need to destroy the window as it's still open - window.Close()) -#### Output +```python +ProgressBar(max_value, + orientation=None, + size=(None, None), + auto_size_text=None, + bar_color=(None, None), + style=None, + border_width=None, + relief=None, + key=None, + pad=None, + visible=True) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| max_value | | +| orientation | (Default value = None) | +| size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | +| auto_size_text | True if size should fit the text length (Default value = None) | +| bar_color | (Default value = (None) | +| style | (Default value = None) | +| border_width | (Default value = None) | +| relief | (Default value = None) | +| key | (common_key) Used with window.FindElement and with return values (Default value = None) | +| pad | (common_key) Amount of padding to put around element (Default value = None) | +| visible | set visibility state of the element (Default value = True) | +||| + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +## Output Elements The Output Element is a re-direction of Stdout. Anything "printed" will be displayed in this element. - Output(size=(None, None)) +Note that you will NOT see what you `print` until you call either window.Read or window.Refresh. If you want to immediately see what was printed, call window.Refresh() immediately after your print statement. -Here's a complete solution for a chat-window using an Async window with an Output Element - import PySimpleGUI as sg - # Blocking window that doesn't close - def ChatBot(): - layout = [[(sg.Text('This is where standard out is being routed', size=[40, 1]))], - [sg.Output(size=(80, 20))], - [sg.Multiline(size=(70, 5), enter_submits=True), - sg.RButton('SEND', button_color=(sg.YELLOWS[0], sg.BLUES[0])), - sg.Button('EXIT', button_color=(sg.YELLOWS[0], sg.GREENS[0]))]] - window = sg.Window('Chat Window', default_element_size=(30, 2)).Layout(layout) - # ---===--- Loop taking in user input and using it to query HowDoI web oracle --- # - while True: - event, value = window.Read() - if event == 'SEND': - print(value) - else: - break - - ChatBot() - -### ProgressBar Methods ```python +Output(size=(None, None)) +``` + +![output](https://user-images.githubusercontent.com/13696193/44959863-b72f8280-aec3-11e8-8caa-7bc743149953.jpg) + + +```python +Output(size=(None, None), + background_color=None, + text_color=None, + pad=None, + font=None, + tooltip=None, + key=None, + right_click_menu=None, + visible=True) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | +| background_color | color of background (Default value = None) | +| text_color | color of the text (Default value = None) | +| pad | (common_key) Amount of padding to put around element (Default value = None) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| tooltip | text, that will appear the you hover on (Default value = None) | +| key | (common_key) Used with window.FindElement and with return values (Default value = None) | +| right_click_menu | see "Right Click Menus" (Default value = None) | +| visible | set visibility state of the element (Default value = True) | +||| + + + + +### Methods + + +```python +Update(value=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| value | (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + + + +---- + +UpdateBar - ? + + + +---- + +Here's a complete solution for a chat-window using an Async window with an Output Element + +```python +import PySimpleGUI as sg + +# Blocking window that doesn't close +def ChatBot(): + layout = [[(sg.Text('This is where standard out is being routed', size=[40, 1]))], + [sg.Output(size=(80, 20))], + [sg.Multiline(size=(70, 5), enter_submits=True), + sg.RButton('SEND', button_color=(sg.YELLOWS[0], sg.BLUES[0])), + sg.Button('EXIT', button_color=(sg.YELLOWS[0], sg.GREENS[0]))]] + + window = sg.Window('Chat Window', default_element_size=(30, 2)).Layout(layout) + + # ---===--- Loop taking in user input and using it to query HowDoI web oracle --- # + while True: + event, value = window.Read() + if event == 'SEND': + print(value) + else: + break + +ChatBot() +``` + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + @@ -3589,7 +3973,14 @@ Starting in version 2.9 you'll be able to do more complex layouts by using the C Columns are specified in exactly the same way as a window is, as a list of lists. - +Columns are needed when you have an element that has a height > 1 line on the left, with single-line elements on the right. Here's an example of this kind of layout: + + + +![column](https://user-images.githubusercontent.com/13696193/44959988-66b92480-aec5-11e8-9c26-316ed24a68c0.jpg) + +The Column Element has 1 required parameter and 1 optional (the layout and the background color). Setting the background color has the same effect as setting the window's background color, except it only affects the column rectangle. + ```python Column(layout, @@ -3603,9 +3994,9 @@ Column(layout, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | layout | | | background_color | color of background (Default value = None) | @@ -3620,129 +4011,137 @@ Parameters explained: - - - -Columns are needed when you have an element that has a height > 1 line on the left, with single-line elements on the right. Here's an example of this kind of layout: - - -![column](https://user-images.githubusercontent.com/13696193/44959988-66b92480-aec5-11e8-9c26-316ed24a68c0.jpg) - - -This code produced the above window. - - - import PySimpleGUI as sg - - # Demo of how columns work - # window has on row 1 a vertical slider followed by a COLUMN with 7 rows - # Prior to the Column element, this layout was not possible - # Columns layouts look identical to window layouts, they are a list of lists of elements. - - window = sg.Window('Columns') # blank window - - # Column layout - col = [[sg.Text('col Row 1')], - [sg.Text('col Row 2'), sg.Input('col input 1')], - [sg.Text('col Row 3'), sg.Input('col input 2')], - [sg.Text('col Row 4'), sg.Input('col input 3')], - [sg.Text('col Row 5'), sg.Input('col input 4')], - [sg.Text('col Row 6'), sg.Input('col input 5')], - [sg.Text('col Row 7'), sg.Input('col input 6')]] - - layout = [[sg.Slider(range=(1,100), default_value=10, orientation='v', size=(8,20)), sg.Column(col)], - [sg.In('Last input')], - [sg.OK()]] - - # Display the window and get values - # If you're willing to not use the "context manager" design pattern, then it's possible - # to collapse the window display and read down to a single line of code. - event, values = sg.Window('Compact 1-line window with column').Layout(layout).Read() - - sg.Popup(event, values, line_width=200) - -The Column Element has 1 required parameter and 1 optional (the layout and the background color). Setting the background color has the same effect as setting the window's background color, except it only affects the column rectangle. - - Column(layout, background_color=None) - -The default background color for Columns is the same as the default window background color. If you change the look and feel of the window, the column background will match the window background automatically. - ---- + ## Frame Element (Labelled Frames, Frames with a title) Frames work exactly the same way as Columns. You create layout that is then used to initialize the Frame. -```python -Frame( title, - layout, - title_color=None, - background_color=None, - title_location=None, - relief=DEFAULT_FRAME_RELIEF, - size=(None, None), - font=None, - pad=None, - border_width=None, - key=None, - tooltip=None, - right_click_menu=None, - visible=True) -``` - def Frame(title - the label / title to put on frame - layout - list of rows of elements the frame contains - title_color - color of the title text - background_color - color of background - title_location - locations to put the title - relief - type of relief to use - size - size of Frame in characters. Do not use if you want frame to autosize - font - font to use for title - pad - element padding to use when packing - border_width - how thick the line going around frame should be - key - key used to location the element - tooltip - tooltip text -This code creates a window with a Frame and 2 buttons. - - frame_layout = [ - [sg.T('Text inside of a frame')], - [sg.CB('Check 1'), sg.CB('Check 2')], - ] - layout = [ - [sg.Frame('My Frame Title', frame_layout, font='Any 12', title_color='blue')], - [sg.Submit(), sg.Cancel()] - ] - - window = sg.Window('Frame with buttons', font=("Helvetica", 12)).Layout(layout) - - ![frame element](https://user-images.githubusercontent.com/13696193/45889173-c2245700-bd8d-11e8-8f73-1e5f1be3ddb1.jpg) - - -Notice how the Frame layout looks identical to a window layout. A window works exactly the same way as a Column and a Frame. They all are "container elements". Elements that contain other elements. +Notice how the Frame layout looks identical to a window layout. A window works exactly the same way as a Column and a Frame. They all are "container elements" - elements that contain other elements. *These container Elements can be nested as deep as you want.* That's a pretty spiffy feature, right? Took a lot of work so be appreciative. Recursive code isn't trivial. + + +```python +Frame(title, + layout, + title_color=None, + background_color=None, + title_location=None, + relief="groove", + size=(None, None), + font=None, + pad=None, + border_width=None, + key=None, + tooltip=None, + right_click_menu=None, + visible=True) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| title | | +| layout | | +| title_color | (Default value = None) | +| background_color | color of background (Default value = None) | +| title_location | (Default value = None) | +| relief | (Default value = DEFAULT_FRAME_RELIEF) | +| size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| pad | (common_key) Amount of padding to put around element (Default value = None) | +| border_width | (Default value = None) | +| key | (common_key) Used with window.FindElement and with return values (Default value = None) | +| tooltip | text, that will appear the you hover on (Default value = None) | +| right_click_menu | see "Right Click Menus" (Default value = None) | +| visible | set visibility state of the element (Default value = True) | +||| + + + + +--- + +This code creates a window with a Frame and 2 buttons. + +```python +frame_layout = [ + [sg.T('Text inside of a frame')], + [sg.CB('Check 1'), sg.CB('Check 2')], + ] +layout = [ + [sg.Frame('My Frame Title', frame_layout, font='Any 12', title_color='blue')], + [sg.Submit(), sg.Cancel()] + ] + +window = sg.Window('Frame with buttons', font=("Helvetica", 12)).Layout(layout) +``` + + + + + + + + + + + + + + ## Canvas Element +In my opinion, the tkinter Canvas Widget is the most powerful of the tkinter widget. While I try my best to completely isolate the user from anything that is tkinter related, the Canvas Element is the one exception. It enables integration with a number of other packages, often with spectacular results. + + - ```python Canvas(canvas=None, @@ -3755,9 +4154,9 @@ Canvas(canvas=None, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | canvas | (Default value = None) | | background_color | color of background (Default value = None) | @@ -3770,49 +4169,37 @@ Parameters explained: ||| - -In my opinion, the tkinter Canvas Widget is the most powerful of the tkinter widget. While I try my best to completely isolate the user from anything that is tkinter related, the Canvas Element is the one exception. It enables integration with a number of other packages, often with spectacular results. + ### Matplotlib, Pyplot Integration One such integration is with Matploplib and Pyplot. There is a Demo program written that you can use as a design pattern to get an understanding of how to use the Canvas Widget once you get it. - def Canvas(canvas - a tkinter canvasf if you created one. Normally not set - background_color - canvas color - size - size in pixels - pad - element padding for packing - key - key used to lookup element - tooltip - tooltip text + def Canvas(canvas - a tkinter canvasf if you created one. Normally not set + background_color - canvas color + size - size in pixels + pad - element padding for packing + key - key used to lookup element + tooltip - tooltip text The order of operations to obtain a tkinter Canvas Widget is: - figure_x, figure_y, figure_w, figure_h = fig.bbox.bounds - # define the window layout - layout = [[sg.Text('Plot test')], - [sg.Canvas(size=(figure_w, figure_h), key='canvas')], - [sg.OK(pad=((figure_w / 2, 0), 3), size=(4, 2))]] + figure_x, figure_y, figure_w, figure_h = fig.bbox.bounds + # define the window layout + layout = [[sg.Text('Plot test')], + [sg.Canvas(size=(figure_w, figure_h), key='canvas')], + [sg.OK(pad=((figure_w / 2, 0), 3), size=(4, 2))]] - # create the window and show it without the plot - window = sg.Window('Demo Application - Embedding Matplotlib In PySimpleGUI').Layout(layout).Finalize() + # create the window and show it without the plot + window = sg.Window('Demo Application - Embedding Matplotlib In PySimpleGUI').Layout(layout).Finalize() - # add the plot to the window - fig_photo = draw_figure(window.FindElement('canvas').TKCanvas, fig) + # add the plot to the window + fig_photo = draw_figure(window.FindElement('canvas').TKCanvas, fig) - # show it all again and get buttons - event, values = window.Read() + # show it all again and get buttons + event, values = window.Read() To get a tkinter Canvas Widget from PySimpleGUI, follow these steps: * Add Canvas Element to your window @@ -3826,12 +4213,23 @@ To get a tkinter Canvas Widget from PySimpleGUI, follow these steps: See `Demo_Matplotlib.py` for a Recipe you can copy. -### Canvas Methods +### Methods TKCanvas - not a method but a property. Returns the tkinter Canvas Widget + + + + + + + + + + + ## Graph Element @@ -3856,12 +4254,13 @@ After you supply those values you can scribble all of over your graph by creatin You can move your figures around on the canvas by supplying the Figure ID the x,y amount to move. - graph.MoveFigure(my_circle, 10, 10) + graph.MoveFigure(my_circle, 10, 10) This Element is relatively new and may have some parameter additions or deletions. It shouldn't break your code however. - + + ```python Graph(canvas_size, @@ -3878,9 +4277,9 @@ Graph(canvas_size, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | canvas_size | | | graph_bottom_left | | @@ -3898,55 +4297,293 @@ Parameters explained: - -### Graph Methods + +### Methods + +`DrawLine` - draws a point, line, circle, oval, arc, rectangle, text
+`DrawImage` - places an image onto the graph
+`Erase` - erases entire graph
+`Update` - changes background color
+`Move` - moves everything an x,y direction
+`MoveFigure` - moves an individual figure by some delta
+`RelocateFigure` - moves figure to an absolute location
+`DeleteFigure` - delete an individual figure
+ +
+ +All of the Drawing methods return a "***figure***" that can be used move and delete the figure
+ + ```python -DrawLine(self, point_from, point_to, color='black', width=1) -DrawPoint(self, point, size=2, color='black') -DrawCircle(self, center_location, radius, fill_color=None, line_color='black') -DrawOval(self, top_left, bottom_right, fill_color=None, line_color=None) -DrawArc(self, top_left, bottom_right, extent, start_angle, style=None, arc_color='black') -DrawRectangle(self, top_left, bottom_right, fill_color=None, line_color=None) -DrawText(self, text, location, color='black', font=None, angle=0) -Erase(background_color) -DeleteFigure(figure_id) -Update() -Move(self, x_direction, y_direction) -MoveFigure(self, figure, x_direction, y_direction) -RelocateFigure(self, figure, x, y) -TKCanvas +DrawLine(point_from, + point_to, + color="black", + width=1) ``` -All of the Drawing methods return a "***figure***" that can be used move and delete the figure -DrawLine - draws a line -DrawPoint - draws a single point -DrawCircle - draws a circle -DrawOval - draws an oval -DrawArc - draws an arc -DrawRectangle - draws a rectangle -DrawText - draws text -DrawImage - places an image onto the graph -Erase - erases entire graph -Update - changes background color -Move - moves everything an x,y direction -MoveFigure - moves an individual figure by some delta -RelocateFigure - moves figure to an absolute location -DeleteFigure - delete an individual figure + + Parameters explained: +| Name | Meaning | +|-|-| +| point_from | | +| point_to | | +| color | (Default value = 'black') | +| width | (Default value = 1) | +||| + + + + + +```python +DrawPoint(point, + size=2, + color="black") +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| point | | +| size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = 2) | +| color | (Default value = 'black') | +||| + + + + + +```python +DrawCircle(center_location, + radius, + fill_color=None, + line_color="black") +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| center_location | | +| radius | | +| fill_color | (Default value = None) | +| line_color | (Default value = 'black') | +||| + + + + + +```python +DrawOval(top_left, + bottom_right, + fill_color=None, + line_color=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| top_left | | +| bottom_right | | +| fill_color | (Default value = None) | +| line_color | (Default value = None) | +||| + + + + + +```python +DrawArc(top_left, + bottom_right, + extent, + start_angle, + style=None, + arc_color="black") +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| top_left | | +| bottom_right | | +| extent | | +| start_angle | | +| style | (Default value = None) | +| arc_color | (Default value = 'black') | +||| + + + + + +```python +DrawRectangle(top_left, + bottom_right, + fill_color=None, + line_color=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| top_left | | +| bottom_right | | +| fill_color | (Default value = None) | +| line_color | (Default value = None) | +||| + + + + + +```python +DrawText(text, + location, + color="black", + font=None, + angle=0) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| text | | +| location | | +| color | (Default value = 'black') | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| angle | (Default value = 0) | +||| + + + + + +
+Erase()
+ + + + +```python +DeleteFigure(id) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| id | | +||| + + + + + +```python +Update(background_color, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| background_color | color of background | +| visible | change visibility of element (Default value = None) | +||| + + + + + +```python +Move(x_direction, + y_direction) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| x_direction | | +| y_direction | | +||| + + + + + +```python +MoveFigure(figure, + x_direction, + y_direction) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| figure | | +| x_direction | | +| y_direction | | +||| + + + + + +```python +RelocateFigure(figure, + x, + y) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| figure | | +| x | | +| y | | +||| + + + + + + + + + + + + + + + @@ -3954,12 +4591,8 @@ DeleteFigure - delete an individual figure Out of all of the Elements, it's the Table and the Tree that are the most "problematic" in the tkinter inter and Qt implementations. They're hard is my only defense. -### Known visualization problem.... - -If you click on the header, it can go into spasms for some tables. I don't understand what's causing it and it's been there evidently since the first release of Tables. - ```python Table(values, @@ -3992,9 +4625,9 @@ Table(values, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | values | | | headings | (Default value = None) | @@ -4028,83 +4661,50 @@ Parameters explained: - - ### Read return values from Table Element The values returned from a `Window.Read` call for the Tree Element are a list of row numbers that are currently highlighted. -### Update Call + +### Methods The Update method can be used to make changes to a table that's already been displayed. The call takes a single parameter, values, which is the new table to display. The entire table is replaced. + ```python -def Update(self, values=None): +Update(values=None, + num_rows=None, + visible=None, + select_rows=None) ``` -`values` is a table containing your rows just like you passed in when creating the Table Element. + + + Parameters explained: +| Name | Meaning | +|-|-| +| values | (Default value = None) | +| num_rows | (Default value = None) | +| visible | change visibility of element (Default value = None) | +| select_rows | (Default value = None) | +||| + + + + +--- +### Known visualization problem.... + +If you click on the header, it can go into spasms for some tables. I don't understand what's causing it and it's been there evidently since the first release of Tables. + ## 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. - + + ```python Tree(data=None, @@ -4132,9 +4732,9 @@ Tree(data=None, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | data | (Default value = None) | | headings | (Default value = None) | @@ -4163,53 +4763,6 @@ Parameters explained: - 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 @@ -4224,7 +4777,9 @@ def Insert(self, parent, key, text, values, icon=None) To "insert" data into the tree the TreeData method Insert is called. -`Insert(parent_key, key, display_text, values)` +```python +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 = "" @@ -4274,15 +4829,21 @@ View of second tab: First we have the Tab layout definitions. They mirror what you see in the screen shots. Tab 1 has 1 Text Element in it. Tab 2 has a Text and an Input Element. - tab1_layout = [[sg.T('This is inside tab 1')]] - tab2_layout = [[sg.T('This is inside tab 2')], - [sg.In(key='in')]] +```python +tab1_layout = [[sg.T('This is inside tab 1')]] + +tab2_layout = [[sg.T('This is inside tab 2')], + [sg.In(key='in')]] + +``` The layout for the entire window looks like this: - layout = [[sg.TabGroup([[sg.Tab('Tab 1', tab1_layout), sg.Tab('Tab 2', tab2_layout)]])], - [sg.RButton('Read')]] +```python +layout = [[sg.TabGroup([[sg.Tab('Tab 1', tab1_layout), sg.Tab('Tab 2', tab2_layout)]])], + [sg.RButton('Read')]] +``` The Window layout has the TabGroup and within the tab Group are the two Tab elements. @@ -4301,51 +4862,175 @@ Check out Tabs 7 and 8. We've got a Window with a Column containing Tabs 5 and As of Release 3.8.0, not all of *options* shown in the API definitions of the Tab and TabGroup Elements are working. They are there as placeholders. -The definition of a TabGroup is - TabGroup(layout, - title_color=None - background_color=None - font=None - pad=None - border_width=None - change_submits = False - key=None - tooltip=None) -The definition of a Tab Element is +```python +TabGroup(layout, + tab_location=None, + title_color=None, + selected_title_color=None, + background_color=None, + font=None, + change_submits=False, + enable_events=False, + pad=None, + border_width=None, + theme=None, + key=None, + tooltip=None, + visible=True) +``` - Tab(title, - layout, - title_color=None, - background_color=None, - font=None, - pad=None - disabled=False - border_width=None - key=None - tooltip=None) + Parameters explained: +| Name | Meaning | +|-|-| +| layout | | +| tab_location | (Default value = None) | +| title_color | (Default value = None) | +| selected_title_color | (Default value = None) | +| background_color | color of background (Default value = None) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| change_submits | If True, pressing Enter key submits window (Default value = False) | +| enable_events | Turns on the element specific events.(Default value = False) | +| pad | (common_key) Amount of padding to put around element (Default value = None) | +| border_width | (Default value = None) | +| theme | (Default value = None) | +| key | (common_key) Used with window.FindElement and with return values (Default value = None) | +| tooltip | text, that will appear the you hover on (Default value = None) | +| visible | set visibility state of the element (Default value = True) | +||| + + + + + + + + + + + + + + +```python +Tab(title, + layout, + title_color=None, + background_color=None, + font=None, + pad=None, + disabled=False, + border_width=None, + key=None, + tooltip=None, + right_click_menu=None, + visible=True) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| title | | +| layout | | +| title_color | (Default value = None) | +| background_color | color of background (Default value = None) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| pad | (common_key) Amount of padding to put around element (Default value = None) | +| disabled | set disable state for element (Default value = False) | +| border_width | (Default value = None) | +| key | (common_key) Used with window.FindElement and with return values (Default value = None) | +| tooltip | text, that will appear the you hover on (Default value = None) | +| right_click_menu | see "Right Click Menus" (Default value = None) | +| visible | set visibility state of the element (Default value = True) | +||| + + + + + ### Reading Tab Groups Tab Groups now return a value when a Read returns. They return which tab is currently selected. There is also a change_submits parameter that can be set that causes a Read to return if a Tab in that group is selected / changed. The key or title belonging to the Tab that was switched to will be returned as the value -### Tab Element Methods +### Methods + + ```python +Update(values=None, + key=None, + value=None, + text=None, + icon=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| values | (Default value = None) | +| key | (common_key) Used with window.FindElement and with return values (Default value = None) | +| value | (Default value = None) | +| text | (Default value = None) | +| icon | (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + + + + + + + + + + + + + + ## Pane Element New in version 3.20 is the Pane Element, a super-cool tkinter feature. You won't find this one in PySimpleGUIQt, only PySimpleGUI. It's difficult to describe one of these things. Think of them as "Tabs without labels" that you can slide. + + ![pane3](https://user-images.githubusercontent.com/13696193/50035040-fcd50e80-ffcd-11e8-939c-df8ab8d64712.gif) - ```python Pane(pane_list, @@ -4361,9 +5046,9 @@ Pane(pane_list, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | pane_list | | | background_color | color of background (Default value = None) | @@ -4379,6 +5064,8 @@ Parameters explained: ||| + + + + + + + + + + + + # Global Settings **Global Settings** Let's have some fun customizing! Make PySimpleGUI look the way you want it to look. You can set the global settings using the function `PySimpleGUI.SetOptions`. Each option has an optional parameter that's used to set it. - SetOptions(icon=None - button_color=(None,None) - element_size=(None,None), - margins=(None,None), - element_padding=(None,None) - auto_size_text=None - auto_size_buttons=None - font=None - border_width=None - slider_border_width=None - slider_relief=None - slider_orientation=None - autoclose_time=None - message_box_line_width=None - progress_meter_border_depth=None - progress_meter_style=None - progress_meter_relief=None - progress_meter_color=None - progress_meter_size=None - text_justification=None - text_color=None - background_color=None - element_background_color=None - text_element_background_color=None - input_elements_background_color=None - element_text_color=None - input_text_color=None - scrollbar_color=None, text_color=None - debug_win_size=(None,None) - window_location=(None,None) - tooltip_time = None + SetOptions(icon=None + button_color=(None,None) + element_size=(None,None), + margins=(None,None), + element_padding=(None,None) + auto_size_text=None + auto_size_buttons=None + font=None + border_width=None + slider_border_width=None + slider_relief=None + slider_orientation=None + autoclose_time=None + message_box_line_width=None + progress_meter_border_depth=None + progress_meter_style=None + progress_meter_relief=None + progress_meter_color=None + progress_meter_size=None + text_justification=None + text_color=None + background_color=None + element_background_color=None + text_element_background_color=None + input_elements_background_color=None + element_text_color=None + input_text_color=None + scrollbar_color=None, text_color=None + debug_win_size=(None,None) + window_location=(None,None) + tooltip_time = None Explanation of parameters - icon - filename of icon used for taskbar and title bar - button_color - button color (foreground, background) - element_size - element size (width, height) in characters - margins - tkinter margins around outsize - element_padding - tkinter padding around each element - auto_size_text - autosize the elements to fit their text - auto_size_buttons - autosize the buttons to fit their text - font - font used for elements - border_width - amount of bezel or border around sunken or raised elements - slider_border_width - changes the way sliders look - slider_relief - changes the way sliders look - slider_orientation - changes orientation of slider - autoclose_time - time in seconds for autoclose boxes - message_box_line_width - number of characers in a line of text in message boxes - progress_meter_border_depth - amount of border around raised or lowered progress meters - progress_meter_style - style of progress meter as defined by tkinter - progress_meter_relief - relief style - progress_meter_color - color of the bar and background of progress meters - progress_meter_size - size in (characters, pixels) - background_color - Color of the main window's background - element_background_color - Background color of the elements - text_element_background_color - Text element background color - input_elements_background_color - Input fields background color - element_text_color - Text color of elements that have text, like Radio Buttons - input_text_color - Color of the text that you type in - scrollbar_color - Color for scrollbars (may not always work) - text_color - Text element default text color - text_justification - justification to use on Text Elements. Values are strings - 'left', 'right', 'center' - debug_win_size - size of the Print output window - window_location - location on the screen (x,y) of window's top left cornder - tooltip_time - time in milliseconds to wait before showing a tooltip. Default is 400ms +| Name | Meaning | +| - | - | +| icon | filename of icon used for taskbar and title bar | +| button_color | button color (foreground, background) | +| element_size | element size (width, height) in characters | +| margins | tkinter margins around outsize | +| element_padding | tkinter padding around each element | +| auto_size_text | autosize the elements to fit their text | +| auto_size_buttons | autosize the buttons to fit their text | +| font | font used for elements | +| border_width | amount of bezel or border around sunken or raised elements | +| slider_border_width | changes the way sliders look | +| slider_relief | changes the way sliders look | +| slider_orientation | changes orientation of slider | +| autoclose_time | time in seconds for autoclose boxes | +| message_box_line_width | number of characers in a line of text in message boxes | +| progress_meter_border_depth | amount of border around raised or lowered progress meters | +| progress_meter_style | style of progress meter as defined by tkinter | +| progress_meter_relief | relief style | +| progress_meter_color | color of the bar and background of progress meters | +| progress_meter_size | size in (characters, pixels) | +| background_color | Color of the main window's background | +| element_background_color | Background color of the elements | +| text_element_background_color | Text element background color | +| input_elements_background_color | Input fields background color | +| element_text_color | Text color of elements that have text, like Radio Buttons | +| input_text_color | Color of the text that you type in | +| scrollbar_color | Color for scrollbars (may not always work) | +| text_color | Text element default text color | +| text_justification | justification to use on Text Elements. Values are strings - 'left', 'right', 'center' | +| debug_win_size | size of the Print output window | +| window_location | location on the screen (x,y) of window's top left cornder | +| tooltip_time | time in milliseconds to wait before showing a tooltip. Default is 400ms | +| | | These settings apply to all windows `SetOptions`. The Row options and Element options will take precedence over these settings. Settings can be thought of as levels of settings with the window-level being the highest and the Element-level the lowest. Thus the levels are: @@ -4714,16 +5418,16 @@ The design pattern for Persistent Windows was already shown to you earlier in th import PySimpleGUI as sg layout = [[sg.Text('Persistent window')], - [sg.Input()], - [sg.RButton('Read'), sg.Exit()]] + [sg.Input()], + [sg.RButton('Read'), sg.Exit()]] window = sg.Window('Window that stays open').Layout(layout) while True: - event, values = window.Read() - if event is None or event == 'Exit': - break - print(event, values) + event, values = window.Read() + if event is None or event == 'Exit': + break + print(event, values) window.Close() ``` @@ -4743,9 +5447,9 @@ Let's say you had a device that you want to "poll" every 100ms. The "easy way ```python # YOU SHOULD NOT DO THIS.... while True: # Event Loop - event, values = window.ReadNonBlocking() # DO NOT USE THIS CALL ANYMORE - read_my_hardware() # process my device here - time.sleep(.1) # sleep 1/10 second + event, values = window.ReadNonBlocking() # DO NOT USE THIS CALL ANYMORE + read_my_hardware() # process my device here + time.sleep(.1) # sleep 1/10 second ``` This program will quickly test for user input, then deal with the hardware. Then it'll sleep for 100ms, while your gui is non-responsive, then it'll check in with your GUI again. I fully realize this is a crude way of doing things. We're talking dirt simple stuff without trying to use threads, etc to 'get it right'. It's for demonstration purposes. @@ -4756,8 +5460,8 @@ using the Read Timeout mechanism, the sleep goes away. ```python # This is the right way to poll for hardware while True: # Event Loop - event, values = window.Read(timeout = 100) - read_my_hardware() # process my device here + event, values = window.Read(timeout = 100) + read_my_hardware() # process my device here ``` This event loop will run every 100 ms. You're making a Read call, so anything that the use does will return back to you immediately, and you're waiting up to 100ms for the user to do something. If the user doesn't do anything, then the read will timeout and execution will return to the program. @@ -4788,11 +5492,11 @@ TIMEOUT_KEY = '__timeout__' If you wanted to test for "no event" in your loop, it would be written like this: ```python while True: - event, value = window.Read(timeout=0) - if event is None: - break # the use has closed the window - if event == sg.TIMEOUT_KEY: - print("Nothing happened") + event, value = window.Read(timeout=0) + if event is None: + break # the use has closed the window + if event == sg.TIMEOUT_KEY: + print("Nothing happened") ``` @@ -4832,33 +5536,33 @@ window.Close() See the sample code on the GitHub named Demo Media Player for another example of Async windows. We're going to make a window and update one of the elements of that window every .01 seconds. Here's the entire code to do that. ```python - import PySimpleGUI as sg - import time +import PySimpleGUI as sg +import time - # ---------------- Create Form ---------------- - sg.ChangeLookAndFeel('Black') - sg.SetOptions(element_padding=(0, 0)) +# ---------------- Create Form ---------------- +sg.ChangeLookAndFeel('Black') +sg.SetOptions(element_padding=(0, 0)) - layout = [[sg.Text('')], - [sg.Text('', size=(8, 2), font=('Helvetica', 20), justification='center', key='text')], - [sg.ReadButton('Pause', key='button', button_color=('white', '#001480')), - sg.ReadButton('Reset', button_color=('white', '#007339'), key='Reset'), - sg.Exit(button_color=('white', 'firebrick4'), key='Exit')]] +layout = [[sg.Text('')], + [sg.Text('', size=(8, 2), font=('Helvetica', 20), justification='center', key='text')], + [sg.ReadButton('Pause', key='button', button_color=('white', '#001480')), + sg.ReadButton('Reset', button_color=('white', '#007339'), key='Reset'), + sg.Exit(button_color=('white', 'firebrick4'), key='Exit')]] - window = sg.Window('Running Timer', no_titlebar=True, auto_size_buttons=False, keep_on_top=True, grab_anywhere=True).Layout(layout) +window = sg.Window('Running Timer', no_titlebar=True, auto_size_buttons=False, keep_on_top=True, grab_anywhere=True).Layout(layout) - # ---------------- main loop ---------------- - current_time = 0 - paused = False - start_time = int(round(time.time() * 100)) - while (True): - # --------- Read and update window -------- - event, values = window.Read(timeout=10) - current_time = int(round(time.time() * 100)) - start_time - # --------- Display timer in window -------- - window.FindElement('text').Update('{:02d}:{:02d}.{:02d}'.format((current_time // 100) // 60, - (current_time // 100) % 60, - current_time % 100)) +# ---------------- main loop ---------------- +current_time = 0 +paused = False +start_time = int(round(time.time() * 100)) +while (True): + # --------- Read and update window -------- + event, values = window.Read(timeout=10) + current_time = int(round(time.time() * 100)) - start_time + # --------- Display timer in window -------- + window.FindElement('text').Update('{:02d}:{:02d}.{:02d}'.format((current_time // 100) // 60, + (current_time // 100) % 60, + current_time % 100)) ``` @@ -4892,14 +5596,14 @@ Here is an example of updating a Text Element import PySimpleGUI as sg layout = [ [sg.Text('My layout', key='_TEXT_')], - [sg.Button('Read')]] + [sg.Button('Read')]] window = sg.Window('My new window').Layout(layout) while True: # Event Loop event, values = window.Read() - if event is None: - break + if event is None: + break window.Element('_TEXT_').Update('My new text value') ``` @@ -4910,8 +5614,8 @@ In this example, the Update is done prior the Read. Because of this, the Finali import PySimpleGUI as sg layout = [ [sg.Text('My layout', key='_TEXT_')], - [sg.Button('Read')] - ] + [sg.Button('Read')] + ] window = sg.Window('My new window').Layout(layout).Finalize() @@ -4919,8 +5623,8 @@ window.Element('_TEXT_').Update('My new text value') while True: # Event Loop event, values = window.Read() - if event is None: - break + if event is None: + break ``` @@ -4943,34 +5647,36 @@ The way this is done is via an Update method that is available for nearly all of In some programs these updates happen in response to another Element. This program takes a Spinner and a Slider's input values and uses them to resize a Text Element. The Spinner and Slider are on the left, the Text element being changed is on the right. - # Testing async window, see if can have a slider - # that adjusts the size of text displayed +```python +# Testing async window, see if can have a slider +# that adjusts the size of text displayed - import PySimpleGUI as sg - fontSize = 12 - layout = [[sg.Spin([sz for sz in range(6, 172)], font=('Helvetica 20'), initial_value=fontSize, change_submits=True, key='spin'), - sg.Slider(range=(6,172), orientation='h', size=(10,20), - change_submits=True, key='slider', font=('Helvetica 20')), - sg.Text("Aa", size=(2, 1), font="Helvetica " + str(fontSize), key='text')]] +import PySimpleGUI as sg +fontSize = 12 +layout = [[sg.Spin([sz for sz in range(6, 172)], font=('Helvetica 20'), initial_value=fontSize, change_submits=True, key='spin'), + sg.Slider(range=(6,172), orientation='h', size=(10,20), + change_submits=True, key='slider', font=('Helvetica 20')), + sg.Text("Aa", size=(2, 1), font="Helvetica " + str(fontSize), key='text')]] - sz = fontSize - window = sg.Window("Font size selector", grab_anywhere=False).Layout(layout) - # Event Loop - while True: - event, values= window.Read() - if event is None: - break - sz_spin = int(values['spin']) - sz_slider = int(values['slider']) - sz = sz_spin if sz_spin != fontSize else sz_slider - if sz != fontSize: - fontSize = sz - font = "Helvetica " + str(fontSize) - window.FindElement('text').Update(font=font) - window.FindElement('slider').Update(sz) - window.FindElement('spin').Update(sz) +sz = fontSize +window = sg.Window("Font size selector", grab_anywhere=False).Layout(layout) +# Event Loop +while True: + event, values= window.Read() + if event is None: + break + sz_spin = int(values['spin']) + sz_slider = int(values['slider']) + sz = sz_spin if sz_spin != fontSize else sz_slider + if sz != fontSize: + fontSize = sz + font = "Helvetica " + str(fontSize) + window.FindElement('text').Update(font=font) + window.FindElement('slider').Update(sz) + window.FindElement('spin').Update(sz) - print("Done.") +print("Done.") +``` Inside the event loop we read the value of the Spinner and the Slider using those Elements' keys. @@ -4978,16 +5684,18 @@ For example, `values['slider']` is the value of the Slider Element. This program changes all 3 elements if either the Slider or the Spinner changes. This is done with these statements: - window.FindElement('text').Update(font=font) - window.FindElement('slider').Update(sz) - window.FindElement('spin').Update(sz) +```python +window.FindElement('text').Update(font=font) +window.FindElement('slider').Update(sz) +window.FindElement('spin').Update(sz) +``` Remember this design pattern because you will use it OFTEN if you use persistent windows. It works as follows. The call to `window.FindElement` returns the Element object represented by they provided `key`. This element is then updated by calling it's `Update` method. This is another example of Python's "chaining" feature. We could write this code using the long-form: - text_element = window.FindElement('text') - text_element.Update(font=font) + text_element = window.FindElement('text') + text_element.Update(font=font) The takeaway from this exercise is that keys are key in PySimpleGUI's design. They are used to both read the values of the window and also to identify elements. As already mentioned, they are used as targets in Button calls. @@ -5016,30 +5724,30 @@ For scroll-wheel events, if the mouse is scrolled up, then the `button` text wil Keyboard keys return 2 types of key events. For "normal" keys (a,b,c, etc), a single character is returned that represents that key. Modifier and special keys are returned as a string with 2 parts: - Key Sym:Key Code + Key Sym:Key Code Key Sym is a string such as 'Control_L'. The Key Code is a numeric representation of that key. The left control key, when pressed will return the value 'Control_L:17' - import PySimpleGUI as sg + import PySimpleGUI as sg - # Recipe for getting keys, one at a time as they are released - # If want to use the space bar, then be sure and disable the "default focus" + # Recipe for getting keys, one at a time as they are released + # If want to use the space bar, then be sure and disable the "default focus" - with sg.Window("Keyboard Test", return_keyboard_events=True, use_default_focus=False) as window: - text_elem = sg.Text("", size=(18, 1)) - layout = [[sg.Text("Press a key or scroll mouse")], - [text_elem], - [sg.Button("OK")]] + with sg.Window("Keyboard Test", return_keyboard_events=True, use_default_focus=False) as window: + text_elem = sg.Text("", size=(18, 1)) + layout = [[sg.Text("Press a key or scroll mouse")], + [text_elem], + [sg.Button("OK")]] - window.Layout(layout) - # ---===--- Loop taking in user input --- # - while True: - event, value = window.Read() + window.Layout(layout) + # ---===--- Loop taking in user input --- # + while True: + event, value = window.Read() - if event == "OK" or event is None: - print(event, "exiting") - break - text_elem.Update(event) + if event == "OK" or event is None: + print(event, "exiting") + break + text_elem.Update(event) @@ -5048,21 +5756,21 @@ You want to turn off the default focus so that there no buttons that will be sel ### Realtime Keyboard Capture Use realtime keyboard capture by calling - import PySimpleGUI as sg + import PySimpleGUI as sg - with sg.Window("Realtime Keyboard Test", return_keyboard_events=True, use_default_focus=False) as window: - layout = [[sg.Text("Hold down a key")], - [sg.Button("OK")]] + with sg.Window("Realtime Keyboard Test", return_keyboard_events=True, use_default_focus=False) as window: + layout = [[sg.Text("Hold down a key")], + [sg.Button("OK")]] - window.Layout(layout) + window.Layout(layout) - while True: - event, value = window.Read(timeout=0) - if event == "OK" or event is None: - print(event, value, "exiting") - break - if event != sg.TIMEOUT_KEY: - print(event) + while True: + event, value = window.Read(timeout=0) + if event == "OK" or event is None: + print(event, value, "exiting") + break + if event != sg.TIMEOUT_KEY: + print(event) # Menus @@ -5072,22 +5780,40 @@ Use realtime keyboard capture by calling Beginning in version 3.01 you can add a MenuBar to your window. You specify the menus in much the same way as you do window layouts, with lists. Menu selections are returned as events and as of 3.17, also as in the values dictionary. The value returned will be the entire menu entry, including the key if you specified one. -This definition: - menu_def = [['File', ['Open', 'Save', 'Exit',]], - ['Edit', ['Paste', ['Special', 'Normal',], 'Undo'],], - ['Help', 'About...'],] - -Note the placement of ',' and of []. It's tricky to get the nested menus correct that implement cascading menus. See how paste has Special and Normal as a list after it. This means that Paste has a cascading menu with items Special and Normal. - -They menu_def layout produced this window: +```python + menu_def = [['File', ['Open', 'Save', 'Exit',]], + ['Edit', ['Paste', ['Special', 'Normal',], 'Undo'],], + ['Help', 'About...'],] +``` ![menu](https://user-images.githubusercontent.com/13696193/45306723-56b7cb00-b4eb-11e8-8cbd-faef0c90f8b4.jpg) +Note the placement of ',' and of []. It's tricky to get the nested menus correct that implement cascading menus. See how paste has Special and Normal as a list after it. This means that Paste has a cascading menu with items Special and Normal. + +## Methods + +```python +Update(menu_definition, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| menu_definition | | +| visible | change visibility of element (Default value = None) | +||| + + + + +--- To add a menu to a Window place the `Menu` or `MenuBar` element into your layout. - layout = [[sg.Menu(menu_def)]] + layout = [[sg.Menu(menu_def)]] It doesn't really matter where you place the Menu Element in your layout as it will always be located at the top of the window. @@ -5122,8 +5848,8 @@ This is an example Menu with underlines and a separator. ``` # ------ Menu Definition ------ # menu_def = [['&File', ['&Open', '&Save', '---', 'Properties', 'E&xit' ]], - ['&Edit', ['Paste', ['Special', 'Normal',], 'Undo'],], - ['&Help', '&About...'],] + ['&Edit', ['Paste', ['Special', 'Normal',], 'Undo'],], + ['&Help', '&About...'],] ``` And this is the spiffy menu it produced: ![menus with shortcuts](https://user-images.githubusercontent.com/13696193/46251674-f5b74f00-c427-11e8-95c6-547adc59041b.jpg) @@ -5173,31 +5899,31 @@ import PySimpleGUI as sg # Design pattern 2 - First window remains active layout = [[ sg.Text('Window 1'),], - [sg.Input(do_not_clear=True)], - [sg.Text('', key='_OUTPUT_')], - [sg.Button('Launch 2'), sg.Button('Exit')]] + [sg.Input(do_not_clear=True)], + [sg.Text('', key='_OUTPUT_')], + [sg.Button('Launch 2'), sg.Button('Exit')]] win1 = sg.Window('Window 1').Layout(layout) win2_active = False while True: - ev1, vals1 = win1.Read(timeout=100) - win1.FindElement('_OUTPUT_').Update(vals1[0]) - if ev1 is None or ev1 == 'Exit': - break + ev1, vals1 = win1.Read(timeout=100) + win1.FindElement('_OUTPUT_').Update(vals1[0]) + if ev1 is None or ev1 == 'Exit': + break - if not win2_active and ev1 == 'Launch 2': - win2_active = True - layout2 = [[sg.Text('Window 2')], - [sg.Button('Exit')]] + if not win2_active and ev1 == 'Launch 2': + win2_active = True + layout2 = [[sg.Text('Window 2')], + [sg.Button('Exit')]] - win2 = sg.Window('Window 2').Layout(layout2) + win2 = sg.Window('Window 2').Layout(layout2) - if win2_active: - ev2, vals2 = win2.Read(timeout=100) - if ev2 is None or ev2 == 'Exit': - win2_active = False - win2.Close() + if win2_active: + ev2, vals2 = win2.Read(timeout=100) + if ev2 is None or ev2 == 'Exit': + win2_active = False + win2.Close() ``` @@ -5209,36 +5935,37 @@ import PySimpleGUIQt as sg # Design pattern 1 - First window does not remain active layout = [[ sg.Text('Window 1'),], - [sg.Input(do_not_clear=True)], - [sg.Text('', key='_OUTPUT_')], - [sg.Button('Launch 2')]] + [sg.Input(do_not_clear=True)], + [sg.Text('', key='_OUTPUT_')], + [sg.Button('Launch 2')]] win1 = sg.Window('Window 1').Layout(layout) win2_active=False while True: - ev1, vals1 = win1.Read(timeout=100) - if ev1 is None: - break - win1.FindElement('_OUTPUT_').Update(vals1[0]) + ev1, vals1 = win1.Read(timeout=100) + if ev1 is None: + break + win1.FindElement('_OUTPUT_').Update(vals1[0]) - if ev1 == 'Launch 2' and not win2_active: - win2_active = True - win1.Hide() - layout2 = [[sg.Text('Window 2')], # note must create a layout from scratch every time. No reuse - [sg.Button('Exit')]] + if ev1 == 'Launch 2' and not win2_active: + win2_active = True + win1.Hide() + layout2 = [[sg.Text('Window 2')], # note must create a layout from scratch every time. No reuse + [sg.Button('Exit')]] - win2 = sg.Window('Window 2').Layout(layout2) - while True: - ev2, vals2 = win2.Read() - if ev2 is None or ev2 == 'Exit': - win2.Close() - win2_active = False - win1.UnHide() - break + win2 = sg.Window('Window 2').Layout(layout2) + while True: + ev2, vals2 = win2.Read() + if ev2 is None or ev2 == 'Exit': + win2.Close() + win2_active = False + win1.UnHide() + break ``` + # Sample Applications There are too many to list!! @@ -6024,6 +6751,25 @@ A combination of user requests, and needs of new `imwatchingyou` debugger * Was crashing * Enables application to fill these in later +# 4.0.0 PySimpleGUI & 2.0.0 PySimpleGUI27 19-June-2019 + +* DOC STRINGS DOCS STRINGS DOC STRINGS! + * Your IDE is about to become very happy + * All Elements have actual documentation in the call signature + * The Readme and ReadTheDocs will be generated going forward using the CODE + * HUGE Thanks for @nngogol for both copying & adding all those strings, but also for making an entire document creation system. +* New __version__ string for PySimpleGUI.py +* New parameter to ALL `SetFocus` calls. + * def SetFocus(self, force=False) + * If force is True, then a call to `focus_force` is made instead of `focus_set` +* Get - New Radio Button Method. Returns True is the Radio Button is set +* Rename of Debugger class to _Debugger so IDEs don't get confused +* User read access to last Button Color set now available via property `Button.ButtonColor` +* Rename of a number of callback handlers to start with _ +* Fix for memory leak in Read call. Every call to read lost a little memory due to root.protocol calls +* Listbox.Update - New parameter - scroll_to_index - scroll view so that index is shown at the top +* First PyPI release to use new documentation! + ### Upcoming diff --git a/readme.md b/readme.md index 7a1aadc1..024be149 100644 --- a/readme.md +++ b/readme.md @@ -1,5 +1,4 @@ - - + + ```python def PopupAnimated(image_source, message=None, @@ -1811,20 +1794,27 @@ The code is a crude representation of the GUI, laid out in text. Many of the main method calls and Element names have shortcuts. This enables you to code much quicker once you are used to using the SDK. The Text Element, for example, has 3 different names `Text`, `Txt` or`T`. InputText can also be written `Input` or `In` . `FindElement` was recently renamed to `Element` because it's a commonly used function. - -## Text Element -``` -layout = [[sg.Text('This is what a Text Element looks like')]] -``` + + + + + +## Text Element | `T == Txt == Text` +Basic Element. It displays text. That's it. + + + + +```python +layout = [ + [sg.Text('This is what a Text Element looks like')], + [sg.T('Second label')], + ] +``` ![simple text](https://user-images.githubusercontent.com/13696193/44959877-e9d97b00-aec3-11e8-9d24-b4405ee4a148.jpg) -The most basic element is the Text element. It simply displays text. Many of the 'options' that can be set for a Text element are shared by other elements. - - - - ```python Text(text, size=(None, None), @@ -1843,9 +1833,9 @@ Text(text, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | text | | | size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | @@ -1867,75 +1857,32 @@ Parameters explained: +### Methods - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -Some commonly used elements have 'shorthand' versions of the functions to make the code more compact. The functions `T` and `Txt` are the same as calling `Text`. + Parameters explained: +| Name | Meaning | +|-|-| +| value | (Default value = None) | +| background_color | color of background (Default value = None) | +| text_color | color of the text (Default value = None) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + +--- ### Fonts @@ -1959,39 +1906,27 @@ The shorthand functions for `Text` are `Txt` and `T` If you set the parameter `enable_events` or `click_submits` then you will get an event if the user clicks on the Text. -### Text Methods - -#### Update -If you want to change the text, font, or colors after the element has been created, then use the `Update` method. - -```python -Update(value = None, background_color=None, text_color=None, font=None, visible=None) -``` -| name | meaning | -| - | - | -| value | new value to set text element to | -| background_color | new background color | -| text_color | text color to display | -| font | font to use to display | -| visible | visibility | -| | | + ## Multiline Element +This Element doubles as both an input and output Element. + + +```python layout = [[sg.Multiline('This is what a Multi-line Text Element looks like', size=(45,5))]] +``` ![multiline](https://user-images.githubusercontent.com/13696193/44959853-b139a180-aec3-11e8-972f-f52188510c88.jpg) -This Element doubles as both an input and output Element. - ```python Multiline(default_text="", @@ -2015,9 +1950,9 @@ Multiline(default_text="", visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | default_text | (Default value = '') | | enter_submits | (Default value = False) | @@ -2041,42 +1976,10 @@ Parameters explained: ||| - - -### Multiline Methods - +### Methods + ```python Update(value=None, @@ -2084,91 +1987,45 @@ Update(value=None, append=False, font=None, text_color=None, - background_color=None) - -Update(self, value=None, disabled=None, append=False, font=None, text_color=None, background_color=None)value=None, disabled=None, append=False): + background_color=None, + visible=None, + autoscroll=None) ``` -|name|meaning| + + + Parameters explained: +| Name | Meaning | |-|-| -| value | string to set the text field to| -| disabled | set to True to disable the element| -| append | rather than replacing the current text with new text, add the new text onto the end| +| value | (Default value = None) | +| disabled | disable or enable state of the element (Default value = None) | +| append | (Default value = False) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| text_color | color of the text (Default value = None) | +| background_color | color of background (Default value = None) | +| visible | change visibility of element (Default value = None) | +| autoscroll | (Default value = None) | ||| + + + + + + + + + + -## Output Element -Output re-routes `Stdout` to a scrolled text box. - -Whatever you print will show up in this window. - -Note that you will NOT see what you print until you call either window.Read or window.Refresh. If you want to immediately see what was printed, call window.Refresh() immediately after your print statement. - -```python -layout = [[sg.Output(size=(80,10)]] -``` - -![output](https://user-images.githubusercontent.com/13696193/44959863-b72f8280-aec3-11e8-8caa-7bc743149953.jpg) - - - -```python -Output(size=(None, None), - background_color=None, - text_color=None, - pad=None, - font=None, - tooltip=None, - key=None, - right_click_menu=None, - visible=True) -``` - -Parameters explained: - -|Name|Meaning| -|-|-| -| size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | -| background_color | color of background (Default value = None) | -| text_color | color of the text (Default value = None) | -| pad | (common_key) Amount of padding to put around element (Default value = None) | -| font | (common_key) specifies the font family, size, etc (Default value = None) | -| tooltip | text, that will appear the you hover on (Default value = None) | -| key | (common_key) Used with window.FindElement and with return values (Default value = None) | -| right_click_menu | see "Right Click Menus" (Default value = None) | -| visible | set visibility state of the element (Default value = True) | -||| - - - - - -## Input Elements -These make up the majority of the window definition. Optional variables at the Element level override the window level values (e.g. `size` is specified in the Element). All input Elements create an entry in the list of return values. A Text Input Element creates a string in the list of items returned. - -## Text Input Element Shortcuts (Input, In) +## Text Input Element | `Input == In` +Shows a single line of input. ```python layout = [[sg.InputText('Default text')]] ``` ![inputtext 2](https://user-images.githubusercontent.com/13696193/44959861-b5fe5580-aec3-11e8-8040-53ec241b5079.jpg) -Shows a single line of input. ```python InputText(default_text="", @@ -2190,9 +2047,9 @@ InputText(default_text="", visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | default_text | (Default value = '') | | size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | @@ -2214,6 +2071,8 @@ Parameters explained: ||| + + + + + + + + + + -## Combo Element +## Combo Element | `Combo == DropDown == Drop` Also known as a drop-down list. Only required parameter is the list of choices. The return value is a string matching what's visible on the GUI. +ComboBox Element + ```python layout = [[sg.InputCombo(['choice 1', 'choice 2'])]] ``` @@ -2293,8 +2193,6 @@ layout = [[sg.InputCombo(['choice 1', 'choice 2'])]] ![combobox](https://user-images.githubusercontent.com/13696193/44959860-b565bf00-aec3-11e8-82fe-dbe41252458b.jpg) -ComboBox Element - ```python Combo(values, default_value=None, @@ -2313,9 +2211,9 @@ Combo(values, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | values | | | default_value | (Default value = None) | @@ -2335,6 +2233,8 @@ Parameters explained: ||| + + -Shortcut functions - Combo, DropDown, Drop -### Combo Methods +### Methods + + ```python +Update(value=None, + values=None, + set_to_index=None, + disabled=None, + readonly=None, + font=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| value | (Default value = None) | +| values | (Default value = None) | +| set_to_index | (Default value = None) | +| disabled | disable or enable state of the element (Default value = None) | +| readonly | make element readonly (Default value = None) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + + + + + + + ## Listbox Element The standard listbox like you'll find in most GUIs. Note that the return values from this element will be a ***list of results, not a single result***. This is because the user can select more than 1 item from the list (if you set the right mode). -```python - layout = [[sg.Listbox(values=['Listbox 1', 'Listbox 2', 'Listbox 3'], size=(30, 6))]] -``` - -![listbox 2](https://user-images.githubusercontent.com/13696193/44959859-b4cd2880-aec3-11e8-881c-1e369d5c6337.jpg) - - A List Box. Provide a list of values for the user to choose one or more of. Returns a list of selected rows when a window.Read() is executed. +```python +layout = [[sg.Listbox(values=['Listbox 1', 'Listbox 2', 'Listbox 3'], size=(30, 6))]] +``` + +![listbox 2](https://user-images.githubusercontent.com/13696193/44959859-b4cd2880-aec3-11e8-881c-1e369d5c6337.jpg) ```python @@ -2418,9 +2350,9 @@ Listbox(values, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | values | | | default_values | (Default value = None) | @@ -2442,59 +2374,59 @@ Parameters explained: ||| - - The `select_mode` option can be a string or a constant value defined as a variable. Generally speaking strings are used for these kinds of options. -ListBoxes can cause a window to return from a Read call. If the flag change_submits is set, then when a user makes a selection, the Read immediately returns. -Another way ListBoxes can cause Reads to return is if the flag bind_return_key is set. If True, then if the user presses the return key while an entry is selected, then the Read returns. Also, if this flag is set, if the user double-clicks an entry it will return from the Read. -### Listbox Methods +### Methods + + +```python +Update(values=None, + disabled=None, + set_to_index=None, + scroll_to_index=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| values | (Default value = None) | +| disabled | disable or enable state of the element (Default value = None) | +| set_to_index | highlights the item at this index as if user clicked (Default value = None) | +| scroll_to_index | scroll the listbox so that this index is the first shown (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + +```python +SetValue(values) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| values | | +||| + + + + + +
+GetListValues()
+ + + + + +--- + +ListBoxes can cause a window to return from a Read call. If the flag change_submits is set, then when a user makes a selection, the Read immediately returns. +Another way ListBoxes can cause Reads to return is if the flag bind_return_key is set. If True, then if the user presses the return key while an entry is selected, then the Read returns. Also, if this flag is set, if the user double-clicks an entry it will return from the Read. + + + + + + + + + + + ## Slider Element Sliders have a couple of slider-specific settings as well as appearance settings. Examples include the `orientation` and `range` settings. - layout = [[sg.Slider(range=(1,500), default_value=222, size=(20,15), orientation='horizontal', font=('Helvetica', 12))]] + +```python +layout = [[sg.Slider(range=(1,500), default_value=222, size=(20,15), orientation='horizontal', font=('Helvetica', 12))]] +``` ![slider](https://user-images.githubusercontent.com/13696193/44959858-b4349200-aec3-11e8-9e25-c0fcf025d19e.jpg) - ```python Slider(range=(None, None), @@ -2544,9 +2495,9 @@ Slider(range=(None, None), visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | range | (Default value = (None, None)) | | default_value | (Default value = None) | @@ -2570,6 +2521,8 @@ Parameters explained: ||| + + + + ### Qt Sliders There is an important difference between Qt and tkinter sliders. On Qt, the slider values must be integer, not float. If you want your slider to go from 0.1 to 1.0, then make your slider go from 1 to 10 and divide by 10. It's an easy math thing to do and not a big deal. Just deal with it.... you're writing software after all. Presumably you know how to do these things. ;-) ### Slider Methods + + ```python +Update(value=None, + range=(None, None), + disabled=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| value | (Default value = None) | +| range | (Default value = (None, None)) | +| disabled | disable or enable state of the element (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + + + + + + + + + + + + + + + + ## Radio Element - Creates one radio button that is assigned to a group of radio buttons. Only 1 of the buttons in the group can be selected at any one time. +Radio Button Element - Used in a group of other Radio Elements to provide user with ability to select only +1 choice in a list of choices. ```python layout = [ @@ -2640,11 +2634,6 @@ layout = [ ![radio](https://user-images.githubusercontent.com/13696193/44959857-b4349200-aec3-11e8-8e2d-e6a49ffbd0b6.jpg) -Radio Button Element - Used in a group of other Radio Elements to provide user with ability to select only -1 choice in a list of choices. - - - ```python Radio(text, @@ -2664,9 +2653,9 @@ Radio(text, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | text | | | group_id | | @@ -2686,6 +2675,8 @@ Parameters explained: ||| + + -### Radio Button Methods +### Methods + + ```python +Update(value=None, + disabled=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| value | (Default value = None) | +| disabled | disable or enable state of the element (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + + -## Checkbox Element +## Checkbox Element | `CBox == CB == Check` Checkbox elements are like Radio Button elements. They return a bool indicating whether or not they are checked. - layout = [[sg.Checkbox('My first Checkbox!', default=True), sg.Checkbox('My second Checkbox!')]] - +```python +layout = [[sg.Checkbox('My first Checkbox!', default=True), sg.Checkbox('My second Checkbox!')]] +``` ![checkbox](https://user-images.githubusercontent.com/13696193/44959906-6f5d2b00-aec4-11e8-9c8a-962c787f0286.jpg) - + ```python Checkbox(text, @@ -2756,9 +2770,9 @@ Checkbox(text, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | text | | | default | (Default value = False) | @@ -2777,19 +2791,21 @@ Parameters explained: ||| - -Shortcut functions - CBox, CB, Check -### Checkbox Methods +### Methods ```python Update(value=None, disabled=None, visible=None) Get() @@ -2827,14 +2842,16 @@ Get() ## Spin Element - An up/down spinner control. The valid values are passed in as a list. - layout = [[sg.Spin([i for i in range(1,11)], initial_value=1), sg.Text('Volume level')]] + + +```python +layout = [[sg.Spin([i for i in range(1,11)], initial_value=1), sg.Text('Volume level')]] +``` ![spinner](https://user-images.githubusercontent.com/13696193/44959855-b1d23800-aec3-11e8-9f51-afb2109879da.jpg) - ```python Spin(values, @@ -2853,9 +2870,9 @@ Spin(values, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | values | | | initial_value | (Default value = None) | @@ -2874,37 +2891,39 @@ Parameters explained: ||| + + + values - List of valid values + initial_value - String with initial value + size - (width, height) size of element in characters + auto_size_text - Bool. True if should size width to fit text + font - Font type and size for text display + disabled - Bool. If True element is disabled + background_color - color to use for the background + text_color - color to use for the typed text + change_submits - causes window read to immediately return if the spinner value changes + key = Dictionary key to use for return values + pad - padding around element in the window + tooltip - text to show when mouse hovered over element --> ### Qt Differences - `values` is a range! @@ -2914,13 +2933,56 @@ Note that Qt does not allow arbitrary spinner values. With PySimpleGUI-tkinter On Qt values is a tuple representing a range. On plain PySimpleGUI this value is a list of items. Make sure on the plain version you specify items as a list using [] and not a generator using (). -### Spin Methods +### Methods + + ```python +Update(value=None, + values=None, + disabled=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| value | (Default value = None) | +| values | (Default value = None) | +| disabled | disable or enable state of the element (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + + + + + + + + + + + + + + + + + + + + + ## Image Element @@ -2928,7 +2990,9 @@ disabled - if True disables the element Images can be placed in your window provide they are in PNG, GIF, PPM/PGM format. JPGs cannot be shown because tkinter does not naively support JPGs. You can use the Python Imaging Library (PIL) package to convert your image to PNG prior to calling PySimpleGUI if your images are in JPG format. - + +????????????? add image here + ```python Image(filename=None, @@ -2943,9 +3007,9 @@ Image(filename=None, enable_events=False) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | filename | (Default value = None) | | data | (Default value = None) | @@ -2960,67 +3024,110 @@ Parameters explained: ||| + + -### `Update` Method +### Methods + + +```python +Update(filename=None, + data=None, + size=(None, None), + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| filename | (Default value = None) | +| data | (Default value = None) | +| size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | +| visible | change visibility of element (Default value = None) | +||| -Like other Elements, the Image Element has an `Update` method. Call Update if you want to change the image. - def Update(self, filename=None, data=None, visible=None): Choose **either** a filename or in-ram data image to use to replace current image -### `UpdateAnimation` Method for Animated GIFs + +##### `UpdateAnimation` Method for Animated GIFs + +You can specify an animated GIF as an image and can animate the GIF by calling `UpdateAnimation`. Exciting stuff! ![loading animation](https://user-images.githubusercontent.com/13696193/51280871-d2041e80-19ae-11e9-8757-802eb95352ed.gif) -Starting in version 3.23 you can specify an animated GIF as an image and can animate the GIF by calling `UpdateAnimation`. Exciting stuff! - ```python UpdateAnimation(source, - time_between_frames=0) + time_between_frames=0) ``` -`source` can be a filename ***or*** a base64 bytes variable (unlike other calls that split out the filename parameter and base64 parameter into 2 parameters. -`time_between_frames` is an optional parameter. It will keep track of the amount of time between frame changes for you to give you a smooth animation. With this parameter you can call the function as often as you want and it will advance to the next frame only after the correct amount of time has lapsed. + + Parameters explained: +| Name | Meaning | +|-|-| +| source | | +| time_between_frames | (Default value = 0) | +||| + + + You can call the method without setting the `time_between_frames` value and it will show a frame and immediately move on to the next frame. This enables you to do the inter-frame timing. + + + + + + + + + + + + + + + + ## Button Element @@ -3072,7 +3179,14 @@ In Oct 2018, the definition of Button changed. Previously Button would CLOSE th The most basic Button element call to use is `Button` - + + +```python +layout = [[sg.OK(), sg.Cancel()]] +``` + +![ok cancel 3](https://user-images.githubusercontent.com/13696193/44959927-aa5f5e80-aec4-11e8-86e1-5dc0b3a2b803.jpg) + ```python Button(button_text="", @@ -3100,9 +3214,9 @@ Button(button_text="", visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | button_text | (Default value = '') | | button_type | (Default value = BUTTON_TYPE_READ_FORM) | @@ -3130,71 +3244,75 @@ Parameters explained: ||| - ### Shortcut, Pre-defined Buttons These Pre-made buttons are some of the most important elements of all because they are used so much. They all basically do the same thing, set the button text to match the function name and set the parameters to commonly used values. If you find yourself needing to create a custom button often because it's not on this list, please post a request on GitHub. . They include: - OK - Ok - Submit - Cancel - Yes - No - Exit - Quit - Help - Save - SaveAs - FileBrowse - FilesBrowse - FileSaveAs - FolderBrowse + OK + Ok + Submit + Cancel + Yes + No + Exit + Quit + Help + Save + SaveAs + FileBrowse + FilesBrowse + FileSaveAs + FolderBrowse **IMPORT NOTE ABOUT SHORTCUT BUTTONS** Prior to release 3.11.0, these buttons closed the window. Starting with 3.11 they will not close the window. They act like RButtons (return the button text and do not close the window) @@ -3203,13 +3321,6 @@ If you are having trouble with these buttons closing your window, please check y Using older versions, if you want a Submit() button that does not close the window, then you would instead use RButton('Submit'). Using the new version, if you want a Submit button that closes the window like the sold Submit() call did, you would write that as `CloseButton('Submit')` or `CButton('Submit')` - - layout = [[sg.OK(), sg.Cancel()]] - -![ok cancel 3](https://user-images.githubusercontent.com/13696193/44959927-aa5f5e80-aec4-11e8-86e1-5dc0b3a2b803.jpg) - - - ### Button targets The `FileBrowse`, `FolderBrowse`, `FileSaveAs` , `FilesSaveAs`, `CalendarButton`, `ColorChooserButton` buttons all fill-in values into another element located on the window. The target can be a Text Element or an InputText Element. The location of the element is specified by the `target` variable in the function call. @@ -3236,20 +3347,24 @@ Let's examine this window as an example: The `InputText` element is located at (1,0)... row 1, column 0. The `Browse` button is located at position (2,0). The Target for the button could be any of these values: - Target = (1,0) - Target = (-1,0) + Target = (1,0) + Target = (-1,0) The code for the entire window could be: - layout = [[sg.T('Source Folder')], - [sg.In()], - [sg.FolderBrowse(target=(-1, 0)), sg.OK()]] +```python +layout = [[sg.T('Source Folder')], + [sg.In()], + [sg.FolderBrowse(target=(-1, 0)), sg.OK()]] +``` or if using keys, then the code would be: - layout = [[sg.T('Source Folder')], - [sg.In(key='input')], - [sg.FolderBrowse(target='input'), sg.OK()]] +```python +layout = [[sg.T('Source Folder')], + [sg.In(key='input')], + [sg.FolderBrowse(target='input'), sg.OK()]] +``` See how much easier the key method is? @@ -3298,14 +3413,14 @@ Your button images need to be in PNG or GIF format. When you make a button with This example comes from the `Demo Media Player.py` example program. Because it's a non-blocking button, it's defined as `RButton`. You also put images on blocking buttons by using `Button`. - sg.RButton('Restart Song', button_color=sg.TRANSPARENT_BUTTON, - image_filename=image_restart, image_size=(50, 50), image_subsample=2, border_width=0) + sg.RButton('Restart Song', button_color=sg.TRANSPARENT_BUTTON, + image_filename=image_restart, image_size=(50, 50), image_subsample=2, border_width=0) Three parameters are used for button images. - image_filename - Filename. Can be a relative path - image_size - Size of image file in pixels - image_subsample - Amount to divide the size by. 2 means your image will be 1/2 the size. 3 means 1/3 + image_filename - Filename. Can be a relative path + image_size - Size of image file in pixels + image_subsample - Amount to divide the size by. 2 means your image will be 1/2 the size. 3 means 1/3 Here's an example window made with button images. @@ -3313,12 +3428,12 @@ Here's an example window made with button images. You'll find the source code in the file Demo Media Player. Here is what the button calls look like to create media player window ```python - sg.RButton('Pause', button_color=sg.TRANSPARENT_BUTTON, - image_filename=image_pause, - image_size=(50, 50), - image_subsample=2, - border_width=0) - ``` +sg.RButton('Pause', button_color=sg.TRANSPARENT_BUTTON, + image_filename=image_pause, + image_size=(50, 50), + image_subsample=2, + border_width=0) +``` This is one you'll have to experiment with at this point. Not up for an exhaustive explanation. **Realtime Buttons** @@ -3336,12 +3451,12 @@ Here is the code to make, show and get results from this window: import PySimpleGUI as sg gui_rows = [[sg.Text('Robotics Remote Control')], - [sg.T(' ' * 10), sg.RealtimeButton('Forward')], - [sg.RealtimeButton('Left'), sg.T(' ' * 15), sg.RealtimeButton('Right')], - [sg.T(' ' * 10), sg.RealtimeButton('Reverse')], - [sg.T('')], - [sg.Quit(button_color=('black', 'orange'))] - ] + [sg.T(' ' * 10), sg.RealtimeButton('Forward')], + [sg.RealtimeButton('Left'), sg.T(' ' * 15), sg.RealtimeButton('Right')], + [sg.T(' ' * 10), sg.RealtimeButton('Reverse')], + [sg.T('')], + [sg.Quit(button_color=('black', 'orange'))] + ] window = sg.Window('Robotics Remote Control', auto_size_text=True).Layout(gui_rows) @@ -3352,12 +3467,12 @@ window = sg.Window('Robotics Remote Control', auto_size_text=True).Layout(gui_ro # # your program's main loop while (True): - # This is the code that reads and updates your window - event, values = window.Read(timeout=0) - if event is not None: - print(event) - if event == 'Quit' or values is None: - break + # This is the code that reads and updates your window + event, values = window.Read(timeout=0) + if event is not None: + print(event) + if event == 'Quit' or values is None: + break window.Close() # Don't forget to close your window! ``` @@ -3367,38 +3482,89 @@ This loop will read button values and print them. When one of the Realtime butt **File Types** The `FileBrowse` & `SaveAs` buttons have an additional setting named `file_types`. This variable is used to filter the files shown in the file dialog box. The default value for this setting is - FileTypes=(("ALL Files", "*.*"),) + FileTypes=(("ALL Files", "*.*"),) This code produces a window where the Browse button only shows files of type .TXT - layout = [[sg.In() ,sg.FileBrowse(file_types=(("Text Files", "*.txt"),))]] + layout = [[sg.In() ,sg.FileBrowse(file_types=(("Text Files", "*.txt"),))]] NOTE - Mac users will not be able to use the file_types parameter. tkinter has a bug on Macs that will crash the program is a file_type is attempted so that feature had to be removed. Sorry about that! ***The ENTER key*** - The ENTER key is an important part of data entry for windows. There's a long tradition of the enter key being used to quickly submit windows. PySimpleGUI implements this by tying the ENTER key to the first button that closes or reads a window. + The ENTER key is an important part of data entry for windows. There's a long tradition of the enter key being used to quickly submit windows. PySimpleGUI implements this by tying the ENTER key to the first button that closes or reads a window. The Enter Key can be "bound" to a particular button so that when the key is pressed, it causes the window to return as if the button was clicked. This is done using the `bind_return_key` parameter in the button calls. If there are more than 1 button on a window, the FIRST button that is of type Close window or Read window is used. First is determined by scanning the window, top to bottom and left to right. -### Button Methods +### Methods + + ```python +Update(text=None, + button_color=(None, None), + disabled=None, + image_data=None, + image_filename=None, + visible=None, + image_subsample=None, + image_size=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| text | (Default value = None) | +| button_color | (Default value = (None) | +| disabled | disable or enable state of the element (Default value = None) | +| image_data | (Default value = None) | +| image_filename | (Default value = None) | +| visible | change visibility of element (Default value = None) | +| image_subsample | (Default value = None) | +| image_size | (Default value = None) | +||| + + + + + GetText - Returns the current text shown on a button +
+GetText()
+ + + + + + + + + + + + + + + + + ## ButtonMenu Element @@ -3411,7 +3577,8 @@ Here is a sample definition: The very first string normally specifies what is shown on the menu bar. In this case, the value is **not used**. You set the text for the button using a different parameter, the `button_text` parm. - + + ```python ButtonMenu(button_text, @@ -3433,9 +3600,9 @@ ButtonMenu(button_text, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | button_text | | | menu_def | | @@ -3457,25 +3624,27 @@ Parameters explained: ||| + + @@ -3489,19 +3658,57 @@ Return values for ButtonMenus are sent via the return values dictionary. If a s + + + + + + + + + + + + + + + + + ## VerticalSeparator Element This element has limited usefulness and is being included more for completeness than anything else. It will draw a line between elements. +It works best when placed between columns or elements that span multiple rows. If on a "normal" row with elements that are only 1 row high, then it will only span that one row. + + + ```python VerticalSeparator(pad=None) ``` ![snag-0129](https://user-images.githubusercontent.com/13696193/47376041-a92a0100-d6bf-11e8-8f5b-0c0df56cf0f3.jpg) -It works best when placed between columns or elements that span multiple rows. If on a "normal" row with elements that are only 1 row high, then it will only span that one row. + +```python +VerticalSeparator(pad=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| pad | (common_key) Amount of padding to put around element (Default value = None) | +||| + + + + + + + @@ -3513,7 +3720,9 @@ The `ProgressBar` element is used to build custom Progress Bar windows. It is H The **easiest** way to get progress meters into your code is to use the `OneLineProgressMeter` API. This consists of a pair of functions, `OneLineProgressMeter` and `OneLineProgressMeterCancel`. You can easily cancel any progress meter by calling it with the current value = max value. This will mark the meter as expired and close the window. You've already seen OneLineProgressMeter calls presented earlier in this readme. - sg.OneLineProgressMeter('My Meter', i+1, 1000, 'key', 'Optional message') +```python +sg.OneLineProgressMeter('My Meter', i+1, 1000, 'key', 'Optional message') +``` The return value for `OneLineProgressMeter` is: `True` if meter updated correctly @@ -3522,65 +3731,240 @@ The return value for `OneLineProgressMeter` is: #### Progress Mater in Your window Another way of using a Progress Meter with PySimpleGUI is to build a custom window with a `ProgressBar` Element in the window. You will need to run your window as a non-blocking window. When you are ready to update your progress bar, you call the `UpdateBar` method for the `ProgressBar` element itself. + + +```python +import PySimpleGUI as sg + +# layout the window +layout = [[sg.Text('A custom progress meter')], + [sg.ProgressBar(10000, orientation='h', size=(20, 20), key='progressbar')], + [sg.Cancel()]] + +# create the window` +window = sg.Window('Custom Progress Meter').Layout(layout) +progress_bar = window.FindElement('progressbar') +# loop that would normally do something useful +for i in range(10000): + # check to see if the cancel button was clicked and exit loop if clicked + event, values = window.Read(timeout=0) + if event == 'Cancel' or event is None: + break + # update bar with loop value +1 so that bar eventually reaches the maximum + progress_bar.UpdateBar(i + 1) +# done with loop... need to destroy the window as it's still open +window.Close()) +``` + ![progress custom](https://user-images.githubusercontent.com/13696193/45243969-c3508100-b2c3-11e8-82bc-927d0307e093.jpg) - import PySimpleGUI as sg - - # layout the window - layout = [[sg.Text('A custom progress meter')], - [sg.ProgressBar(10000, orientation='h', size=(20, 20), key='progressbar')], - [sg.Cancel()]] - - # create the window` - window = sg.Window('Custom Progress Meter').Layout(layout) - progress_bar = window.FindElement('progressbar') - # loop that would normally do something useful - for i in range(10000): - # check to see if the cancel button was clicked and exit loop if clicked - event, values = window.Read(timeout=0) - if event == 'Cancel' or event is None: - break - # update bar with loop value +1 so that bar eventually reaches the maximum - progress_bar.UpdateBar(i + 1) - # done with loop... need to destroy the window as it's still open - window.Close()) -#### Output +```python +ProgressBar(max_value, + orientation=None, + size=(None, None), + auto_size_text=None, + bar_color=(None, None), + style=None, + border_width=None, + relief=None, + key=None, + pad=None, + visible=True) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| max_value | | +| orientation | (Default value = None) | +| size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | +| auto_size_text | True if size should fit the text length (Default value = None) | +| bar_color | (Default value = (None) | +| style | (Default value = None) | +| border_width | (Default value = None) | +| relief | (Default value = None) | +| key | (common_key) Used with window.FindElement and with return values (Default value = None) | +| pad | (common_key) Amount of padding to put around element (Default value = None) | +| visible | set visibility state of the element (Default value = True) | +||| + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +## Output Elements The Output Element is a re-direction of Stdout. Anything "printed" will be displayed in this element. - Output(size=(None, None)) +Note that you will NOT see what you `print` until you call either window.Read or window.Refresh. If you want to immediately see what was printed, call window.Refresh() immediately after your print statement. -Here's a complete solution for a chat-window using an Async window with an Output Element - import PySimpleGUI as sg - # Blocking window that doesn't close - def ChatBot(): - layout = [[(sg.Text('This is where standard out is being routed', size=[40, 1]))], - [sg.Output(size=(80, 20))], - [sg.Multiline(size=(70, 5), enter_submits=True), - sg.RButton('SEND', button_color=(sg.YELLOWS[0], sg.BLUES[0])), - sg.Button('EXIT', button_color=(sg.YELLOWS[0], sg.GREENS[0]))]] - window = sg.Window('Chat Window', default_element_size=(30, 2)).Layout(layout) - # ---===--- Loop taking in user input and using it to query HowDoI web oracle --- # - while True: - event, value = window.Read() - if event == 'SEND': - print(value) - else: - break - - ChatBot() - -### ProgressBar Methods ```python +Output(size=(None, None)) +``` + +![output](https://user-images.githubusercontent.com/13696193/44959863-b72f8280-aec3-11e8-8caa-7bc743149953.jpg) + + +```python +Output(size=(None, None), + background_color=None, + text_color=None, + pad=None, + font=None, + tooltip=None, + key=None, + right_click_menu=None, + visible=True) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | +| background_color | color of background (Default value = None) | +| text_color | color of the text (Default value = None) | +| pad | (common_key) Amount of padding to put around element (Default value = None) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| tooltip | text, that will appear the you hover on (Default value = None) | +| key | (common_key) Used with window.FindElement and with return values (Default value = None) | +| right_click_menu | see "Right Click Menus" (Default value = None) | +| visible | set visibility state of the element (Default value = True) | +||| + + + + +### Methods + + +```python +Update(value=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| value | (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + + + +---- + +UpdateBar - ? + + + +---- + +Here's a complete solution for a chat-window using an Async window with an Output Element + +```python +import PySimpleGUI as sg + +# Blocking window that doesn't close +def ChatBot(): + layout = [[(sg.Text('This is where standard out is being routed', size=[40, 1]))], + [sg.Output(size=(80, 20))], + [sg.Multiline(size=(70, 5), enter_submits=True), + sg.RButton('SEND', button_color=(sg.YELLOWS[0], sg.BLUES[0])), + sg.Button('EXIT', button_color=(sg.YELLOWS[0], sg.GREENS[0]))]] + + window = sg.Window('Chat Window', default_element_size=(30, 2)).Layout(layout) + + # ---===--- Loop taking in user input and using it to query HowDoI web oracle --- # + while True: + event, value = window.Read() + if event == 'SEND': + print(value) + else: + break + +ChatBot() +``` + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + @@ -3589,7 +3973,14 @@ Starting in version 2.9 you'll be able to do more complex layouts by using the C Columns are specified in exactly the same way as a window is, as a list of lists. - +Columns are needed when you have an element that has a height > 1 line on the left, with single-line elements on the right. Here's an example of this kind of layout: + + + +![column](https://user-images.githubusercontent.com/13696193/44959988-66b92480-aec5-11e8-9c26-316ed24a68c0.jpg) + +The Column Element has 1 required parameter and 1 optional (the layout and the background color). Setting the background color has the same effect as setting the window's background color, except it only affects the column rectangle. + ```python Column(layout, @@ -3603,9 +3994,9 @@ Column(layout, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | layout | | | background_color | color of background (Default value = None) | @@ -3620,129 +4011,137 @@ Parameters explained: - - - -Columns are needed when you have an element that has a height > 1 line on the left, with single-line elements on the right. Here's an example of this kind of layout: - - -![column](https://user-images.githubusercontent.com/13696193/44959988-66b92480-aec5-11e8-9c26-316ed24a68c0.jpg) - - -This code produced the above window. - - - import PySimpleGUI as sg - - # Demo of how columns work - # window has on row 1 a vertical slider followed by a COLUMN with 7 rows - # Prior to the Column element, this layout was not possible - # Columns layouts look identical to window layouts, they are a list of lists of elements. - - window = sg.Window('Columns') # blank window - - # Column layout - col = [[sg.Text('col Row 1')], - [sg.Text('col Row 2'), sg.Input('col input 1')], - [sg.Text('col Row 3'), sg.Input('col input 2')], - [sg.Text('col Row 4'), sg.Input('col input 3')], - [sg.Text('col Row 5'), sg.Input('col input 4')], - [sg.Text('col Row 6'), sg.Input('col input 5')], - [sg.Text('col Row 7'), sg.Input('col input 6')]] - - layout = [[sg.Slider(range=(1,100), default_value=10, orientation='v', size=(8,20)), sg.Column(col)], - [sg.In('Last input')], - [sg.OK()]] - - # Display the window and get values - # If you're willing to not use the "context manager" design pattern, then it's possible - # to collapse the window display and read down to a single line of code. - event, values = sg.Window('Compact 1-line window with column').Layout(layout).Read() - - sg.Popup(event, values, line_width=200) - -The Column Element has 1 required parameter and 1 optional (the layout and the background color). Setting the background color has the same effect as setting the window's background color, except it only affects the column rectangle. - - Column(layout, background_color=None) - -The default background color for Columns is the same as the default window background color. If you change the look and feel of the window, the column background will match the window background automatically. - ---- + ## Frame Element (Labelled Frames, Frames with a title) Frames work exactly the same way as Columns. You create layout that is then used to initialize the Frame. -```python -Frame( title, - layout, - title_color=None, - background_color=None, - title_location=None, - relief=DEFAULT_FRAME_RELIEF, - size=(None, None), - font=None, - pad=None, - border_width=None, - key=None, - tooltip=None, - right_click_menu=None, - visible=True) -``` - def Frame(title - the label / title to put on frame - layout - list of rows of elements the frame contains - title_color - color of the title text - background_color - color of background - title_location - locations to put the title - relief - type of relief to use - size - size of Frame in characters. Do not use if you want frame to autosize - font - font to use for title - pad - element padding to use when packing - border_width - how thick the line going around frame should be - key - key used to location the element - tooltip - tooltip text -This code creates a window with a Frame and 2 buttons. - - frame_layout = [ - [sg.T('Text inside of a frame')], - [sg.CB('Check 1'), sg.CB('Check 2')], - ] - layout = [ - [sg.Frame('My Frame Title', frame_layout, font='Any 12', title_color='blue')], - [sg.Submit(), sg.Cancel()] - ] - - window = sg.Window('Frame with buttons', font=("Helvetica", 12)).Layout(layout) - - ![frame element](https://user-images.githubusercontent.com/13696193/45889173-c2245700-bd8d-11e8-8f73-1e5f1be3ddb1.jpg) - - -Notice how the Frame layout looks identical to a window layout. A window works exactly the same way as a Column and a Frame. They all are "container elements". Elements that contain other elements. +Notice how the Frame layout looks identical to a window layout. A window works exactly the same way as a Column and a Frame. They all are "container elements" - elements that contain other elements. *These container Elements can be nested as deep as you want.* That's a pretty spiffy feature, right? Took a lot of work so be appreciative. Recursive code isn't trivial. + + +```python +Frame(title, + layout, + title_color=None, + background_color=None, + title_location=None, + relief="groove", + size=(None, None), + font=None, + pad=None, + border_width=None, + key=None, + tooltip=None, + right_click_menu=None, + visible=True) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| title | | +| layout | | +| title_color | (Default value = None) | +| background_color | color of background (Default value = None) | +| title_location | (Default value = None) | +| relief | (Default value = DEFAULT_FRAME_RELIEF) | +| size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = (None, None)) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| pad | (common_key) Amount of padding to put around element (Default value = None) | +| border_width | (Default value = None) | +| key | (common_key) Used with window.FindElement and with return values (Default value = None) | +| tooltip | text, that will appear the you hover on (Default value = None) | +| right_click_menu | see "Right Click Menus" (Default value = None) | +| visible | set visibility state of the element (Default value = True) | +||| + + + + +--- + +This code creates a window with a Frame and 2 buttons. + +```python +frame_layout = [ + [sg.T('Text inside of a frame')], + [sg.CB('Check 1'), sg.CB('Check 2')], + ] +layout = [ + [sg.Frame('My Frame Title', frame_layout, font='Any 12', title_color='blue')], + [sg.Submit(), sg.Cancel()] + ] + +window = sg.Window('Frame with buttons', font=("Helvetica", 12)).Layout(layout) +``` + + + + + + + + + + + + + + ## Canvas Element +In my opinion, the tkinter Canvas Widget is the most powerful of the tkinter widget. While I try my best to completely isolate the user from anything that is tkinter related, the Canvas Element is the one exception. It enables integration with a number of other packages, often with spectacular results. + + - ```python Canvas(canvas=None, @@ -3755,9 +4154,9 @@ Canvas(canvas=None, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | canvas | (Default value = None) | | background_color | color of background (Default value = None) | @@ -3770,49 +4169,37 @@ Parameters explained: ||| - -In my opinion, the tkinter Canvas Widget is the most powerful of the tkinter widget. While I try my best to completely isolate the user from anything that is tkinter related, the Canvas Element is the one exception. It enables integration with a number of other packages, often with spectacular results. + ### Matplotlib, Pyplot Integration One such integration is with Matploplib and Pyplot. There is a Demo program written that you can use as a design pattern to get an understanding of how to use the Canvas Widget once you get it. - def Canvas(canvas - a tkinter canvasf if you created one. Normally not set - background_color - canvas color - size - size in pixels - pad - element padding for packing - key - key used to lookup element - tooltip - tooltip text + def Canvas(canvas - a tkinter canvasf if you created one. Normally not set + background_color - canvas color + size - size in pixels + pad - element padding for packing + key - key used to lookup element + tooltip - tooltip text The order of operations to obtain a tkinter Canvas Widget is: - figure_x, figure_y, figure_w, figure_h = fig.bbox.bounds - # define the window layout - layout = [[sg.Text('Plot test')], - [sg.Canvas(size=(figure_w, figure_h), key='canvas')], - [sg.OK(pad=((figure_w / 2, 0), 3), size=(4, 2))]] + figure_x, figure_y, figure_w, figure_h = fig.bbox.bounds + # define the window layout + layout = [[sg.Text('Plot test')], + [sg.Canvas(size=(figure_w, figure_h), key='canvas')], + [sg.OK(pad=((figure_w / 2, 0), 3), size=(4, 2))]] - # create the window and show it without the plot - window = sg.Window('Demo Application - Embedding Matplotlib In PySimpleGUI').Layout(layout).Finalize() + # create the window and show it without the plot + window = sg.Window('Demo Application - Embedding Matplotlib In PySimpleGUI').Layout(layout).Finalize() - # add the plot to the window - fig_photo = draw_figure(window.FindElement('canvas').TKCanvas, fig) + # add the plot to the window + fig_photo = draw_figure(window.FindElement('canvas').TKCanvas, fig) - # show it all again and get buttons - event, values = window.Read() + # show it all again and get buttons + event, values = window.Read() To get a tkinter Canvas Widget from PySimpleGUI, follow these steps: * Add Canvas Element to your window @@ -3826,12 +4213,23 @@ To get a tkinter Canvas Widget from PySimpleGUI, follow these steps: See `Demo_Matplotlib.py` for a Recipe you can copy. -### Canvas Methods +### Methods TKCanvas - not a method but a property. Returns the tkinter Canvas Widget + + + + + + + + + + + ## Graph Element @@ -3856,12 +4254,13 @@ After you supply those values you can scribble all of over your graph by creatin You can move your figures around on the canvas by supplying the Figure ID the x,y amount to move. - graph.MoveFigure(my_circle, 10, 10) + graph.MoveFigure(my_circle, 10, 10) This Element is relatively new and may have some parameter additions or deletions. It shouldn't break your code however. - + + ```python Graph(canvas_size, @@ -3878,9 +4277,9 @@ Graph(canvas_size, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | canvas_size | | | graph_bottom_left | | @@ -3898,55 +4297,293 @@ Parameters explained: - -### Graph Methods + +### Methods + +`DrawLine` - draws a point, line, circle, oval, arc, rectangle, text
+`DrawImage` - places an image onto the graph
+`Erase` - erases entire graph
+`Update` - changes background color
+`Move` - moves everything an x,y direction
+`MoveFigure` - moves an individual figure by some delta
+`RelocateFigure` - moves figure to an absolute location
+`DeleteFigure` - delete an individual figure
+ +
+ +All of the Drawing methods return a "***figure***" that can be used move and delete the figure
+ + ```python -DrawLine(self, point_from, point_to, color='black', width=1) -DrawPoint(self, point, size=2, color='black') -DrawCircle(self, center_location, radius, fill_color=None, line_color='black') -DrawOval(self, top_left, bottom_right, fill_color=None, line_color=None) -DrawArc(self, top_left, bottom_right, extent, start_angle, style=None, arc_color='black') -DrawRectangle(self, top_left, bottom_right, fill_color=None, line_color=None) -DrawText(self, text, location, color='black', font=None, angle=0) -Erase(background_color) -DeleteFigure(figure_id) -Update() -Move(self, x_direction, y_direction) -MoveFigure(self, figure, x_direction, y_direction) -RelocateFigure(self, figure, x, y) -TKCanvas +DrawLine(point_from, + point_to, + color="black", + width=1) ``` -All of the Drawing methods return a "***figure***" that can be used move and delete the figure -DrawLine - draws a line -DrawPoint - draws a single point -DrawCircle - draws a circle -DrawOval - draws an oval -DrawArc - draws an arc -DrawRectangle - draws a rectangle -DrawText - draws text -DrawImage - places an image onto the graph -Erase - erases entire graph -Update - changes background color -Move - moves everything an x,y direction -MoveFigure - moves an individual figure by some delta -RelocateFigure - moves figure to an absolute location -DeleteFigure - delete an individual figure + + Parameters explained: +| Name | Meaning | +|-|-| +| point_from | | +| point_to | | +| color | (Default value = 'black') | +| width | (Default value = 1) | +||| + + + + + +```python +DrawPoint(point, + size=2, + color="black") +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| point | | +| size | (common_key) (w,h) w=characters-wide, h=rows-high (Default value = 2) | +| color | (Default value = 'black') | +||| + + + + + +```python +DrawCircle(center_location, + radius, + fill_color=None, + line_color="black") +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| center_location | | +| radius | | +| fill_color | (Default value = None) | +| line_color | (Default value = 'black') | +||| + + + + + +```python +DrawOval(top_left, + bottom_right, + fill_color=None, + line_color=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| top_left | | +| bottom_right | | +| fill_color | (Default value = None) | +| line_color | (Default value = None) | +||| + + + + + +```python +DrawArc(top_left, + bottom_right, + extent, + start_angle, + style=None, + arc_color="black") +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| top_left | | +| bottom_right | | +| extent | | +| start_angle | | +| style | (Default value = None) | +| arc_color | (Default value = 'black') | +||| + + + + + +```python +DrawRectangle(top_left, + bottom_right, + fill_color=None, + line_color=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| top_left | | +| bottom_right | | +| fill_color | (Default value = None) | +| line_color | (Default value = None) | +||| + + + + + +```python +DrawText(text, + location, + color="black", + font=None, + angle=0) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| text | | +| location | | +| color | (Default value = 'black') | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| angle | (Default value = 0) | +||| + + + + + +
+Erase()
+ + + + +```python +DeleteFigure(id) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| id | | +||| + + + + + +```python +Update(background_color, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| background_color | color of background | +| visible | change visibility of element (Default value = None) | +||| + + + + + +```python +Move(x_direction, + y_direction) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| x_direction | | +| y_direction | | +||| + + + + + +```python +MoveFigure(figure, + x_direction, + y_direction) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| figure | | +| x_direction | | +| y_direction | | +||| + + + + + +```python +RelocateFigure(figure, + x, + y) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| figure | | +| x | | +| y | | +||| + + + + + + + + + + + + + + + @@ -3954,12 +4591,8 @@ DeleteFigure - delete an individual figure Out of all of the Elements, it's the Table and the Tree that are the most "problematic" in the tkinter inter and Qt implementations. They're hard is my only defense. -### Known visualization problem.... - -If you click on the header, it can go into spasms for some tables. I don't understand what's causing it and it's been there evidently since the first release of Tables. - ```python Table(values, @@ -3992,9 +4625,9 @@ Table(values, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | values | | | headings | (Default value = None) | @@ -4028,83 +4661,50 @@ Parameters explained: - - ### Read return values from Table Element The values returned from a `Window.Read` call for the Tree Element are a list of row numbers that are currently highlighted. -### Update Call + +### Methods The Update method can be used to make changes to a table that's already been displayed. The call takes a single parameter, values, which is the new table to display. The entire table is replaced. + ```python -def Update(self, values=None): +Update(values=None, + num_rows=None, + visible=None, + select_rows=None) ``` -`values` is a table containing your rows just like you passed in when creating the Table Element. + + + Parameters explained: +| Name | Meaning | +|-|-| +| values | (Default value = None) | +| num_rows | (Default value = None) | +| visible | change visibility of element (Default value = None) | +| select_rows | (Default value = None) | +||| + + + + +--- +### Known visualization problem.... + +If you click on the header, it can go into spasms for some tables. I don't understand what's causing it and it's been there evidently since the first release of Tables. + ## 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. - + + ```python Tree(data=None, @@ -4132,9 +4732,9 @@ Tree(data=None, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | data | (Default value = None) | | headings | (Default value = None) | @@ -4163,53 +4763,6 @@ Parameters explained: - 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 @@ -4224,7 +4777,9 @@ def Insert(self, parent, key, text, values, icon=None) To "insert" data into the tree the TreeData method Insert is called. -`Insert(parent_key, key, display_text, values)` +```python +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 = "" @@ -4274,15 +4829,21 @@ View of second tab: First we have the Tab layout definitions. They mirror what you see in the screen shots. Tab 1 has 1 Text Element in it. Tab 2 has a Text and an Input Element. - tab1_layout = [[sg.T('This is inside tab 1')]] - tab2_layout = [[sg.T('This is inside tab 2')], - [sg.In(key='in')]] +```python +tab1_layout = [[sg.T('This is inside tab 1')]] + +tab2_layout = [[sg.T('This is inside tab 2')], + [sg.In(key='in')]] + +``` The layout for the entire window looks like this: - layout = [[sg.TabGroup([[sg.Tab('Tab 1', tab1_layout), sg.Tab('Tab 2', tab2_layout)]])], - [sg.RButton('Read')]] +```python +layout = [[sg.TabGroup([[sg.Tab('Tab 1', tab1_layout), sg.Tab('Tab 2', tab2_layout)]])], + [sg.RButton('Read')]] +``` The Window layout has the TabGroup and within the tab Group are the two Tab elements. @@ -4301,51 +4862,175 @@ Check out Tabs 7 and 8. We've got a Window with a Column containing Tabs 5 and As of Release 3.8.0, not all of *options* shown in the API definitions of the Tab and TabGroup Elements are working. They are there as placeholders. -The definition of a TabGroup is - TabGroup(layout, - title_color=None - background_color=None - font=None - pad=None - border_width=None - change_submits = False - key=None - tooltip=None) -The definition of a Tab Element is +```python +TabGroup(layout, + tab_location=None, + title_color=None, + selected_title_color=None, + background_color=None, + font=None, + change_submits=False, + enable_events=False, + pad=None, + border_width=None, + theme=None, + key=None, + tooltip=None, + visible=True) +``` - Tab(title, - layout, - title_color=None, - background_color=None, - font=None, - pad=None - disabled=False - border_width=None - key=None - tooltip=None) + Parameters explained: +| Name | Meaning | +|-|-| +| layout | | +| tab_location | (Default value = None) | +| title_color | (Default value = None) | +| selected_title_color | (Default value = None) | +| background_color | color of background (Default value = None) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| change_submits | If True, pressing Enter key submits window (Default value = False) | +| enable_events | Turns on the element specific events.(Default value = False) | +| pad | (common_key) Amount of padding to put around element (Default value = None) | +| border_width | (Default value = None) | +| theme | (Default value = None) | +| key | (common_key) Used with window.FindElement and with return values (Default value = None) | +| tooltip | text, that will appear the you hover on (Default value = None) | +| visible | set visibility state of the element (Default value = True) | +||| + + + + + + + + + + + + + + +```python +Tab(title, + layout, + title_color=None, + background_color=None, + font=None, + pad=None, + disabled=False, + border_width=None, + key=None, + tooltip=None, + right_click_menu=None, + visible=True) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| title | | +| layout | | +| title_color | (Default value = None) | +| background_color | color of background (Default value = None) | +| font | (common_key) specifies the font family, size, etc (Default value = None) | +| pad | (common_key) Amount of padding to put around element (Default value = None) | +| disabled | set disable state for element (Default value = False) | +| border_width | (Default value = None) | +| key | (common_key) Used with window.FindElement and with return values (Default value = None) | +| tooltip | text, that will appear the you hover on (Default value = None) | +| right_click_menu | see "Right Click Menus" (Default value = None) | +| visible | set visibility state of the element (Default value = True) | +||| + + + + + ### Reading Tab Groups Tab Groups now return a value when a Read returns. They return which tab is currently selected. There is also a change_submits parameter that can be set that causes a Read to return if a Tab in that group is selected / changed. The key or title belonging to the Tab that was switched to will be returned as the value -### Tab Element Methods +### Methods + + ```python +Update(values=None, + key=None, + value=None, + text=None, + icon=None, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| values | (Default value = None) | +| key | (common_key) Used with window.FindElement and with return values (Default value = None) | +| value | (Default value = None) | +| text | (Default value = None) | +| icon | (Default value = None) | +| visible | change visibility of element (Default value = None) | +||| + + + + + + + + + + + + + + + + + + ## Pane Element New in version 3.20 is the Pane Element, a super-cool tkinter feature. You won't find this one in PySimpleGUIQt, only PySimpleGUI. It's difficult to describe one of these things. Think of them as "Tabs without labels" that you can slide. + + ![pane3](https://user-images.githubusercontent.com/13696193/50035040-fcd50e80-ffcd-11e8-939c-df8ab8d64712.gif) - ```python Pane(pane_list, @@ -4361,9 +5046,9 @@ Pane(pane_list, visible=True) ``` -Parameters explained: -|Name|Meaning| + Parameters explained: +| Name | Meaning | |-|-| | pane_list | | | background_color | color of background (Default value = None) | @@ -4379,6 +5064,8 @@ Parameters explained: ||| + + + + + + + + + + + + # Global Settings **Global Settings** Let's have some fun customizing! Make PySimpleGUI look the way you want it to look. You can set the global settings using the function `PySimpleGUI.SetOptions`. Each option has an optional parameter that's used to set it. - SetOptions(icon=None - button_color=(None,None) - element_size=(None,None), - margins=(None,None), - element_padding=(None,None) - auto_size_text=None - auto_size_buttons=None - font=None - border_width=None - slider_border_width=None - slider_relief=None - slider_orientation=None - autoclose_time=None - message_box_line_width=None - progress_meter_border_depth=None - progress_meter_style=None - progress_meter_relief=None - progress_meter_color=None - progress_meter_size=None - text_justification=None - text_color=None - background_color=None - element_background_color=None - text_element_background_color=None - input_elements_background_color=None - element_text_color=None - input_text_color=None - scrollbar_color=None, text_color=None - debug_win_size=(None,None) - window_location=(None,None) - tooltip_time = None + SetOptions(icon=None + button_color=(None,None) + element_size=(None,None), + margins=(None,None), + element_padding=(None,None) + auto_size_text=None + auto_size_buttons=None + font=None + border_width=None + slider_border_width=None + slider_relief=None + slider_orientation=None + autoclose_time=None + message_box_line_width=None + progress_meter_border_depth=None + progress_meter_style=None + progress_meter_relief=None + progress_meter_color=None + progress_meter_size=None + text_justification=None + text_color=None + background_color=None + element_background_color=None + text_element_background_color=None + input_elements_background_color=None + element_text_color=None + input_text_color=None + scrollbar_color=None, text_color=None + debug_win_size=(None,None) + window_location=(None,None) + tooltip_time = None Explanation of parameters - icon - filename of icon used for taskbar and title bar - button_color - button color (foreground, background) - element_size - element size (width, height) in characters - margins - tkinter margins around outsize - element_padding - tkinter padding around each element - auto_size_text - autosize the elements to fit their text - auto_size_buttons - autosize the buttons to fit their text - font - font used for elements - border_width - amount of bezel or border around sunken or raised elements - slider_border_width - changes the way sliders look - slider_relief - changes the way sliders look - slider_orientation - changes orientation of slider - autoclose_time - time in seconds for autoclose boxes - message_box_line_width - number of characers in a line of text in message boxes - progress_meter_border_depth - amount of border around raised or lowered progress meters - progress_meter_style - style of progress meter as defined by tkinter - progress_meter_relief - relief style - progress_meter_color - color of the bar and background of progress meters - progress_meter_size - size in (characters, pixels) - background_color - Color of the main window's background - element_background_color - Background color of the elements - text_element_background_color - Text element background color - input_elements_background_color - Input fields background color - element_text_color - Text color of elements that have text, like Radio Buttons - input_text_color - Color of the text that you type in - scrollbar_color - Color for scrollbars (may not always work) - text_color - Text element default text color - text_justification - justification to use on Text Elements. Values are strings - 'left', 'right', 'center' - debug_win_size - size of the Print output window - window_location - location on the screen (x,y) of window's top left cornder - tooltip_time - time in milliseconds to wait before showing a tooltip. Default is 400ms +| Name | Meaning | +| - | - | +| icon | filename of icon used for taskbar and title bar | +| button_color | button color (foreground, background) | +| element_size | element size (width, height) in characters | +| margins | tkinter margins around outsize | +| element_padding | tkinter padding around each element | +| auto_size_text | autosize the elements to fit their text | +| auto_size_buttons | autosize the buttons to fit their text | +| font | font used for elements | +| border_width | amount of bezel or border around sunken or raised elements | +| slider_border_width | changes the way sliders look | +| slider_relief | changes the way sliders look | +| slider_orientation | changes orientation of slider | +| autoclose_time | time in seconds for autoclose boxes | +| message_box_line_width | number of characers in a line of text in message boxes | +| progress_meter_border_depth | amount of border around raised or lowered progress meters | +| progress_meter_style | style of progress meter as defined by tkinter | +| progress_meter_relief | relief style | +| progress_meter_color | color of the bar and background of progress meters | +| progress_meter_size | size in (characters, pixels) | +| background_color | Color of the main window's background | +| element_background_color | Background color of the elements | +| text_element_background_color | Text element background color | +| input_elements_background_color | Input fields background color | +| element_text_color | Text color of elements that have text, like Radio Buttons | +| input_text_color | Color of the text that you type in | +| scrollbar_color | Color for scrollbars (may not always work) | +| text_color | Text element default text color | +| text_justification | justification to use on Text Elements. Values are strings - 'left', 'right', 'center' | +| debug_win_size | size of the Print output window | +| window_location | location on the screen (x,y) of window's top left cornder | +| tooltip_time | time in milliseconds to wait before showing a tooltip. Default is 400ms | +| | | These settings apply to all windows `SetOptions`. The Row options and Element options will take precedence over these settings. Settings can be thought of as levels of settings with the window-level being the highest and the Element-level the lowest. Thus the levels are: @@ -4714,16 +5418,16 @@ The design pattern for Persistent Windows was already shown to you earlier in th import PySimpleGUI as sg layout = [[sg.Text('Persistent window')], - [sg.Input()], - [sg.RButton('Read'), sg.Exit()]] + [sg.Input()], + [sg.RButton('Read'), sg.Exit()]] window = sg.Window('Window that stays open').Layout(layout) while True: - event, values = window.Read() - if event is None or event == 'Exit': - break - print(event, values) + event, values = window.Read() + if event is None or event == 'Exit': + break + print(event, values) window.Close() ``` @@ -4743,9 +5447,9 @@ Let's say you had a device that you want to "poll" every 100ms. The "easy way ```python # YOU SHOULD NOT DO THIS.... while True: # Event Loop - event, values = window.ReadNonBlocking() # DO NOT USE THIS CALL ANYMORE - read_my_hardware() # process my device here - time.sleep(.1) # sleep 1/10 second + event, values = window.ReadNonBlocking() # DO NOT USE THIS CALL ANYMORE + read_my_hardware() # process my device here + time.sleep(.1) # sleep 1/10 second ``` This program will quickly test for user input, then deal with the hardware. Then it'll sleep for 100ms, while your gui is non-responsive, then it'll check in with your GUI again. I fully realize this is a crude way of doing things. We're talking dirt simple stuff without trying to use threads, etc to 'get it right'. It's for demonstration purposes. @@ -4756,8 +5460,8 @@ using the Read Timeout mechanism, the sleep goes away. ```python # This is the right way to poll for hardware while True: # Event Loop - event, values = window.Read(timeout = 100) - read_my_hardware() # process my device here + event, values = window.Read(timeout = 100) + read_my_hardware() # process my device here ``` This event loop will run every 100 ms. You're making a Read call, so anything that the use does will return back to you immediately, and you're waiting up to 100ms for the user to do something. If the user doesn't do anything, then the read will timeout and execution will return to the program. @@ -4788,11 +5492,11 @@ TIMEOUT_KEY = '__timeout__' If you wanted to test for "no event" in your loop, it would be written like this: ```python while True: - event, value = window.Read(timeout=0) - if event is None: - break # the use has closed the window - if event == sg.TIMEOUT_KEY: - print("Nothing happened") + event, value = window.Read(timeout=0) + if event is None: + break # the use has closed the window + if event == sg.TIMEOUT_KEY: + print("Nothing happened") ``` @@ -4832,33 +5536,33 @@ window.Close() See the sample code on the GitHub named Demo Media Player for another example of Async windows. We're going to make a window and update one of the elements of that window every .01 seconds. Here's the entire code to do that. ```python - import PySimpleGUI as sg - import time +import PySimpleGUI as sg +import time - # ---------------- Create Form ---------------- - sg.ChangeLookAndFeel('Black') - sg.SetOptions(element_padding=(0, 0)) +# ---------------- Create Form ---------------- +sg.ChangeLookAndFeel('Black') +sg.SetOptions(element_padding=(0, 0)) - layout = [[sg.Text('')], - [sg.Text('', size=(8, 2), font=('Helvetica', 20), justification='center', key='text')], - [sg.ReadButton('Pause', key='button', button_color=('white', '#001480')), - sg.ReadButton('Reset', button_color=('white', '#007339'), key='Reset'), - sg.Exit(button_color=('white', 'firebrick4'), key='Exit')]] +layout = [[sg.Text('')], + [sg.Text('', size=(8, 2), font=('Helvetica', 20), justification='center', key='text')], + [sg.ReadButton('Pause', key='button', button_color=('white', '#001480')), + sg.ReadButton('Reset', button_color=('white', '#007339'), key='Reset'), + sg.Exit(button_color=('white', 'firebrick4'), key='Exit')]] - window = sg.Window('Running Timer', no_titlebar=True, auto_size_buttons=False, keep_on_top=True, grab_anywhere=True).Layout(layout) +window = sg.Window('Running Timer', no_titlebar=True, auto_size_buttons=False, keep_on_top=True, grab_anywhere=True).Layout(layout) - # ---------------- main loop ---------------- - current_time = 0 - paused = False - start_time = int(round(time.time() * 100)) - while (True): - # --------- Read and update window -------- - event, values = window.Read(timeout=10) - current_time = int(round(time.time() * 100)) - start_time - # --------- Display timer in window -------- - window.FindElement('text').Update('{:02d}:{:02d}.{:02d}'.format((current_time // 100) // 60, - (current_time // 100) % 60, - current_time % 100)) +# ---------------- main loop ---------------- +current_time = 0 +paused = False +start_time = int(round(time.time() * 100)) +while (True): + # --------- Read and update window -------- + event, values = window.Read(timeout=10) + current_time = int(round(time.time() * 100)) - start_time + # --------- Display timer in window -------- + window.FindElement('text').Update('{:02d}:{:02d}.{:02d}'.format((current_time // 100) // 60, + (current_time // 100) % 60, + current_time % 100)) ``` @@ -4892,14 +5596,14 @@ Here is an example of updating a Text Element import PySimpleGUI as sg layout = [ [sg.Text('My layout', key='_TEXT_')], - [sg.Button('Read')]] + [sg.Button('Read')]] window = sg.Window('My new window').Layout(layout) while True: # Event Loop event, values = window.Read() - if event is None: - break + if event is None: + break window.Element('_TEXT_').Update('My new text value') ``` @@ -4910,8 +5614,8 @@ In this example, the Update is done prior the Read. Because of this, the Finali import PySimpleGUI as sg layout = [ [sg.Text('My layout', key='_TEXT_')], - [sg.Button('Read')] - ] + [sg.Button('Read')] + ] window = sg.Window('My new window').Layout(layout).Finalize() @@ -4919,8 +5623,8 @@ window.Element('_TEXT_').Update('My new text value') while True: # Event Loop event, values = window.Read() - if event is None: - break + if event is None: + break ``` @@ -4943,34 +5647,36 @@ The way this is done is via an Update method that is available for nearly all of In some programs these updates happen in response to another Element. This program takes a Spinner and a Slider's input values and uses them to resize a Text Element. The Spinner and Slider are on the left, the Text element being changed is on the right. - # Testing async window, see if can have a slider - # that adjusts the size of text displayed +```python +# Testing async window, see if can have a slider +# that adjusts the size of text displayed - import PySimpleGUI as sg - fontSize = 12 - layout = [[sg.Spin([sz for sz in range(6, 172)], font=('Helvetica 20'), initial_value=fontSize, change_submits=True, key='spin'), - sg.Slider(range=(6,172), orientation='h', size=(10,20), - change_submits=True, key='slider', font=('Helvetica 20')), - sg.Text("Aa", size=(2, 1), font="Helvetica " + str(fontSize), key='text')]] +import PySimpleGUI as sg +fontSize = 12 +layout = [[sg.Spin([sz for sz in range(6, 172)], font=('Helvetica 20'), initial_value=fontSize, change_submits=True, key='spin'), + sg.Slider(range=(6,172), orientation='h', size=(10,20), + change_submits=True, key='slider', font=('Helvetica 20')), + sg.Text("Aa", size=(2, 1), font="Helvetica " + str(fontSize), key='text')]] - sz = fontSize - window = sg.Window("Font size selector", grab_anywhere=False).Layout(layout) - # Event Loop - while True: - event, values= window.Read() - if event is None: - break - sz_spin = int(values['spin']) - sz_slider = int(values['slider']) - sz = sz_spin if sz_spin != fontSize else sz_slider - if sz != fontSize: - fontSize = sz - font = "Helvetica " + str(fontSize) - window.FindElement('text').Update(font=font) - window.FindElement('slider').Update(sz) - window.FindElement('spin').Update(sz) +sz = fontSize +window = sg.Window("Font size selector", grab_anywhere=False).Layout(layout) +# Event Loop +while True: + event, values= window.Read() + if event is None: + break + sz_spin = int(values['spin']) + sz_slider = int(values['slider']) + sz = sz_spin if sz_spin != fontSize else sz_slider + if sz != fontSize: + fontSize = sz + font = "Helvetica " + str(fontSize) + window.FindElement('text').Update(font=font) + window.FindElement('slider').Update(sz) + window.FindElement('spin').Update(sz) - print("Done.") +print("Done.") +``` Inside the event loop we read the value of the Spinner and the Slider using those Elements' keys. @@ -4978,16 +5684,18 @@ For example, `values['slider']` is the value of the Slider Element. This program changes all 3 elements if either the Slider or the Spinner changes. This is done with these statements: - window.FindElement('text').Update(font=font) - window.FindElement('slider').Update(sz) - window.FindElement('spin').Update(sz) +```python +window.FindElement('text').Update(font=font) +window.FindElement('slider').Update(sz) +window.FindElement('spin').Update(sz) +``` Remember this design pattern because you will use it OFTEN if you use persistent windows. It works as follows. The call to `window.FindElement` returns the Element object represented by they provided `key`. This element is then updated by calling it's `Update` method. This is another example of Python's "chaining" feature. We could write this code using the long-form: - text_element = window.FindElement('text') - text_element.Update(font=font) + text_element = window.FindElement('text') + text_element.Update(font=font) The takeaway from this exercise is that keys are key in PySimpleGUI's design. They are used to both read the values of the window and also to identify elements. As already mentioned, they are used as targets in Button calls. @@ -5016,30 +5724,30 @@ For scroll-wheel events, if the mouse is scrolled up, then the `button` text wil Keyboard keys return 2 types of key events. For "normal" keys (a,b,c, etc), a single character is returned that represents that key. Modifier and special keys are returned as a string with 2 parts: - Key Sym:Key Code + Key Sym:Key Code Key Sym is a string such as 'Control_L'. The Key Code is a numeric representation of that key. The left control key, when pressed will return the value 'Control_L:17' - import PySimpleGUI as sg + import PySimpleGUI as sg - # Recipe for getting keys, one at a time as they are released - # If want to use the space bar, then be sure and disable the "default focus" + # Recipe for getting keys, one at a time as they are released + # If want to use the space bar, then be sure and disable the "default focus" - with sg.Window("Keyboard Test", return_keyboard_events=True, use_default_focus=False) as window: - text_elem = sg.Text("", size=(18, 1)) - layout = [[sg.Text("Press a key or scroll mouse")], - [text_elem], - [sg.Button("OK")]] + with sg.Window("Keyboard Test", return_keyboard_events=True, use_default_focus=False) as window: + text_elem = sg.Text("", size=(18, 1)) + layout = [[sg.Text("Press a key or scroll mouse")], + [text_elem], + [sg.Button("OK")]] - window.Layout(layout) - # ---===--- Loop taking in user input --- # - while True: - event, value = window.Read() + window.Layout(layout) + # ---===--- Loop taking in user input --- # + while True: + event, value = window.Read() - if event == "OK" or event is None: - print(event, "exiting") - break - text_elem.Update(event) + if event == "OK" or event is None: + print(event, "exiting") + break + text_elem.Update(event) @@ -5048,21 +5756,21 @@ You want to turn off the default focus so that there no buttons that will be sel ### Realtime Keyboard Capture Use realtime keyboard capture by calling - import PySimpleGUI as sg + import PySimpleGUI as sg - with sg.Window("Realtime Keyboard Test", return_keyboard_events=True, use_default_focus=False) as window: - layout = [[sg.Text("Hold down a key")], - [sg.Button("OK")]] + with sg.Window("Realtime Keyboard Test", return_keyboard_events=True, use_default_focus=False) as window: + layout = [[sg.Text("Hold down a key")], + [sg.Button("OK")]] - window.Layout(layout) + window.Layout(layout) - while True: - event, value = window.Read(timeout=0) - if event == "OK" or event is None: - print(event, value, "exiting") - break - if event != sg.TIMEOUT_KEY: - print(event) + while True: + event, value = window.Read(timeout=0) + if event == "OK" or event is None: + print(event, value, "exiting") + break + if event != sg.TIMEOUT_KEY: + print(event) # Menus @@ -5072,22 +5780,40 @@ Use realtime keyboard capture by calling Beginning in version 3.01 you can add a MenuBar to your window. You specify the menus in much the same way as you do window layouts, with lists. Menu selections are returned as events and as of 3.17, also as in the values dictionary. The value returned will be the entire menu entry, including the key if you specified one. -This definition: - menu_def = [['File', ['Open', 'Save', 'Exit',]], - ['Edit', ['Paste', ['Special', 'Normal',], 'Undo'],], - ['Help', 'About...'],] - -Note the placement of ',' and of []. It's tricky to get the nested menus correct that implement cascading menus. See how paste has Special and Normal as a list after it. This means that Paste has a cascading menu with items Special and Normal. - -They menu_def layout produced this window: +```python + menu_def = [['File', ['Open', 'Save', 'Exit',]], + ['Edit', ['Paste', ['Special', 'Normal',], 'Undo'],], + ['Help', 'About...'],] +``` ![menu](https://user-images.githubusercontent.com/13696193/45306723-56b7cb00-b4eb-11e8-8cbd-faef0c90f8b4.jpg) +Note the placement of ',' and of []. It's tricky to get the nested menus correct that implement cascading menus. See how paste has Special and Normal as a list after it. This means that Paste has a cascading menu with items Special and Normal. + +## Methods + +```python +Update(menu_definition, + visible=None) +``` + + + Parameters explained: +| Name | Meaning | +|-|-| +| menu_definition | | +| visible | change visibility of element (Default value = None) | +||| + + + + +--- To add a menu to a Window place the `Menu` or `MenuBar` element into your layout. - layout = [[sg.Menu(menu_def)]] + layout = [[sg.Menu(menu_def)]] It doesn't really matter where you place the Menu Element in your layout as it will always be located at the top of the window. @@ -5122,8 +5848,8 @@ This is an example Menu with underlines and a separator. ``` # ------ Menu Definition ------ # menu_def = [['&File', ['&Open', '&Save', '---', 'Properties', 'E&xit' ]], - ['&Edit', ['Paste', ['Special', 'Normal',], 'Undo'],], - ['&Help', '&About...'],] + ['&Edit', ['Paste', ['Special', 'Normal',], 'Undo'],], + ['&Help', '&About...'],] ``` And this is the spiffy menu it produced: ![menus with shortcuts](https://user-images.githubusercontent.com/13696193/46251674-f5b74f00-c427-11e8-95c6-547adc59041b.jpg) @@ -5173,31 +5899,31 @@ import PySimpleGUI as sg # Design pattern 2 - First window remains active layout = [[ sg.Text('Window 1'),], - [sg.Input(do_not_clear=True)], - [sg.Text('', key='_OUTPUT_')], - [sg.Button('Launch 2'), sg.Button('Exit')]] + [sg.Input(do_not_clear=True)], + [sg.Text('', key='_OUTPUT_')], + [sg.Button('Launch 2'), sg.Button('Exit')]] win1 = sg.Window('Window 1').Layout(layout) win2_active = False while True: - ev1, vals1 = win1.Read(timeout=100) - win1.FindElement('_OUTPUT_').Update(vals1[0]) - if ev1 is None or ev1 == 'Exit': - break + ev1, vals1 = win1.Read(timeout=100) + win1.FindElement('_OUTPUT_').Update(vals1[0]) + if ev1 is None or ev1 == 'Exit': + break - if not win2_active and ev1 == 'Launch 2': - win2_active = True - layout2 = [[sg.Text('Window 2')], - [sg.Button('Exit')]] + if not win2_active and ev1 == 'Launch 2': + win2_active = True + layout2 = [[sg.Text('Window 2')], + [sg.Button('Exit')]] - win2 = sg.Window('Window 2').Layout(layout2) + win2 = sg.Window('Window 2').Layout(layout2) - if win2_active: - ev2, vals2 = win2.Read(timeout=100) - if ev2 is None or ev2 == 'Exit': - win2_active = False - win2.Close() + if win2_active: + ev2, vals2 = win2.Read(timeout=100) + if ev2 is None or ev2 == 'Exit': + win2_active = False + win2.Close() ``` @@ -5209,36 +5935,37 @@ import PySimpleGUIQt as sg # Design pattern 1 - First window does not remain active layout = [[ sg.Text('Window 1'),], - [sg.Input(do_not_clear=True)], - [sg.Text('', key='_OUTPUT_')], - [sg.Button('Launch 2')]] + [sg.Input(do_not_clear=True)], + [sg.Text('', key='_OUTPUT_')], + [sg.Button('Launch 2')]] win1 = sg.Window('Window 1').Layout(layout) win2_active=False while True: - ev1, vals1 = win1.Read(timeout=100) - if ev1 is None: - break - win1.FindElement('_OUTPUT_').Update(vals1[0]) + ev1, vals1 = win1.Read(timeout=100) + if ev1 is None: + break + win1.FindElement('_OUTPUT_').Update(vals1[0]) - if ev1 == 'Launch 2' and not win2_active: - win2_active = True - win1.Hide() - layout2 = [[sg.Text('Window 2')], # note must create a layout from scratch every time. No reuse - [sg.Button('Exit')]] + if ev1 == 'Launch 2' and not win2_active: + win2_active = True + win1.Hide() + layout2 = [[sg.Text('Window 2')], # note must create a layout from scratch every time. No reuse + [sg.Button('Exit')]] - win2 = sg.Window('Window 2').Layout(layout2) - while True: - ev2, vals2 = win2.Read() - if ev2 is None or ev2 == 'Exit': - win2.Close() - win2_active = False - win1.UnHide() - break + win2 = sg.Window('Window 2').Layout(layout2) + while True: + ev2, vals2 = win2.Read() + if ev2 is None or ev2 == 'Exit': + win2.Close() + win2_active = False + win1.UnHide() + break ``` + # Sample Applications There are too many to list!! @@ -6024,6 +6751,25 @@ A combination of user requests, and needs of new `imwatchingyou` debugger * Was crashing * Enables application to fill these in later +# 4.0.0 PySimpleGUI & 2.0.0 PySimpleGUI27 19-June-2019 + +* DOC STRINGS DOCS STRINGS DOC STRINGS! + * Your IDE is about to become very happy + * All Elements have actual documentation in the call signature + * The Readme and ReadTheDocs will be generated going forward using the CODE + * HUGE Thanks for @nngogol for both copying & adding all those strings, but also for making an entire document creation system. +* New __version__ string for PySimpleGUI.py +* New parameter to ALL `SetFocus` calls. + * def SetFocus(self, force=False) + * If force is True, then a call to `focus_force` is made instead of `focus_set` +* Get - New Radio Button Method. Returns True is the Radio Button is set +* Rename of Debugger class to _Debugger so IDEs don't get confused +* User read access to last Button Color set now available via property `Button.ButtonColor` +* Rename of a number of callback handlers to start with _ +* Fix for memory leak in Read call. Every call to read lost a little memory due to root.protocol calls +* Listbox.Update - New parameter - scroll_to_index - scroll view so that index is shown at the top +* First PyPI release to use new documentation! + ### Upcoming