right_click_menu_tearoff parm added to Window, expand_x & expand_y added to vtop vbottom vcenter, listbox has new listbox_frame member variable needed if wanting listbox to resize with the window, changed print in Image element creation if error happens to an error popup (cannot use stdout/stderr because they've been rerouted and will also cause errors), error popup wraps text better
This commit is contained in:
parent
c890ba0b63
commit
8955b9db77
|
@ -1,6 +1,6 @@
|
||||||
#!/usr/bin/python3
|
#!/usr/bin/python3
|
||||||
|
|
||||||
version = __version__ = "4.38.0.3 Unreleased\nAdded Element.block_focus to allow blocking an element from getting focus, Listbox now sets the selected colors to be opposite of normal text/background colors, added highlight parms to Listbox so that they can be directly set, gave Mac users the abliity to override the TTK-Buttons-Only rule for Macs so that if forced, a Button CAN use tk buttons on a Mac"
|
version = __version__ = "4.38.0.4 Unreleased\nAdded Element.block_focus to allow blocking an element from getting focus, Listbox now sets the selected colors to be opposite of normal text/background colors, added highlight parms to Listbox so that they can be directly set, gave Mac users the abliity to override the TTK-Buttons-Only rule for Macs so that if forced, a Button CAN use tk buttons on a Mac, exposed listbox_frame for Listbox so can expand a listbox, new parameter right_click_menu_tearoff parm added to Window, better line wrapping for error windows, show an error window if a bad Image specified in the Image element, expand_x & expand_y parms for vtop"
|
||||||
|
|
||||||
__version__ = version.split()[0] # For PEP 396 and PEP 345
|
__version__ = version.split()[0] # For PEP 396 and PEP 345
|
||||||
|
|
||||||
|
@ -843,6 +843,22 @@ class Element():
|
||||||
if self.Type == ELEM_TYPE_GRAPH:
|
if self.Type == ELEM_TYPE_GRAPH:
|
||||||
self._update_position_for_returned_values(event)
|
self._update_position_for_returned_values(event)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def _tearoff_menu_callback(self, parent, menu):
|
||||||
|
"""
|
||||||
|
Callback function that's called when a right click menu is torn off.
|
||||||
|
The reason for this fuction is to relocate the torn-off menu. It will default to 0,0 otherwise
|
||||||
|
This callback moves the right click menu window to the clocation of the current window
|
||||||
|
|
||||||
|
:param parent: information provided by tkinter - the parent of the Meny
|
||||||
|
:param menu: information provided by tkinter - the menu window
|
||||||
|
|
||||||
|
"""
|
||||||
|
winx, winy = self.ParentForm.current_location()
|
||||||
|
self.ParentForm.TKroot.tk.call('wm', 'geometry', menu, "+{}+{}".format(winx, winy))
|
||||||
|
|
||||||
|
|
||||||
def _MenuItemChosenCallback(self, item_chosen): # TEXT Menu item callback
|
def _MenuItemChosenCallback(self, item_chosen): # TEXT Menu item callback
|
||||||
"""
|
"""
|
||||||
Callback function called when user chooses a menu item from menubar, Button Menu or right click menu
|
Callback function called when user chooses a menu item from menubar, Button Menu or right click menu
|
||||||
|
@ -1881,6 +1897,7 @@ class Listbox(Element):
|
||||||
self.RightClickMenu = right_click_menu
|
self.RightClickMenu = right_click_menu
|
||||||
self.vsb = None # type: tk.Scrollbar
|
self.vsb = None # type: tk.Scrollbar
|
||||||
self.TKListbox = self.Widget = None # type: tk.Listbox
|
self.TKListbox = self.Widget = None # type: tk.Listbox
|
||||||
|
self.listbox_frame = None # type: tk.Frame
|
||||||
self.NoScrollbar = no_scrollbar
|
self.NoScrollbar = no_scrollbar
|
||||||
key = key if key is not None else k
|
key = key if key is not None else k
|
||||||
|
|
||||||
|
@ -7591,7 +7608,7 @@ class Window:
|
||||||
no_titlebar=False, grab_anywhere=False, keep_on_top=False, resizable=False, disable_close=False,
|
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,
|
disable_minimize=False, right_click_menu=None, transparent_color=None, debugger_enabled=True,
|
||||||
right_click_menu_background_color=None, right_click_menu_text_color=None, right_click_menu_disabled_text_color=None,
|
right_click_menu_background_color=None, right_click_menu_text_color=None, right_click_menu_disabled_text_color=None,
|
||||||
right_click_menu_font=None,
|
right_click_menu_font=None, right_click_menu_tearoff=False,
|
||||||
finalize=False, element_justification='left', ttk_theme=None, use_ttk_buttons=None, modal=False, enable_close_attempted_event=False,
|
finalize=False, element_justification='left', ttk_theme=None, use_ttk_buttons=None, modal=False, enable_close_attempted_event=False,
|
||||||
titlebar_background_color=None, titlebar_text_color=None, titlebar_font=None, titlebar_icon=None,
|
titlebar_background_color=None, titlebar_text_color=None, titlebar_font=None, titlebar_icon=None,
|
||||||
use_custom_titlebar=None, metadata=None):
|
use_custom_titlebar=None, metadata=None):
|
||||||
|
@ -7668,6 +7685,8 @@ class Window:
|
||||||
:type right_click_menu_disabled_text_color: (str)
|
:type right_click_menu_disabled_text_color: (str)
|
||||||
:param right_click_menu_font: Font for right click menus
|
:param right_click_menu_font: Font for right click menus
|
||||||
:type right_click_menu_font: str | Tuple[str, int]
|
:type right_click_menu_font: str | Tuple[str, int]
|
||||||
|
:param right_click_menu_tearoff: If True then all right click menus can be torn off
|
||||||
|
:type right_click_menu_tearoff: bool
|
||||||
:param finalize: If True then the Finalize method will be called. Use this rather than chaining .Finalize for cleaner code
|
:param finalize: If True then the Finalize method will be called. Use this rather than chaining .Finalize for cleaner code
|
||||||
:type finalize: (bool)
|
:type finalize: (bool)
|
||||||
:param element_justification: All elements in the Window itself will have this justification 'left', 'right', 'center' are valid values
|
:param element_justification: All elements in the Window itself will have this justification 'left', 'right', 'center' are valid values
|
||||||
|
@ -7784,6 +7803,7 @@ class Window:
|
||||||
self.right_click_menu_text_color = right_click_menu_text_color if right_click_menu_text_color is not None else theme_input_text_color()
|
self.right_click_menu_text_color = right_click_menu_text_color if right_click_menu_text_color is not None else theme_input_text_color()
|
||||||
self.right_click_menu_disabled_text_color = right_click_menu_disabled_text_color if right_click_menu_disabled_text_color is not None else COLOR_SYSTEM_DEFAULT
|
self.right_click_menu_disabled_text_color = right_click_menu_disabled_text_color if right_click_menu_disabled_text_color is not None else COLOR_SYSTEM_DEFAULT
|
||||||
self.right_click_menu_font = right_click_menu_font if right_click_menu_font is not None else self.Font
|
self.right_click_menu_font = right_click_menu_font if right_click_menu_font is not None else self.Font
|
||||||
|
self.right_click_menu_tearoff = right_click_menu_tearoff
|
||||||
self.auto_close_timer_needs_starting = False
|
self.auto_close_timer_needs_starting = False
|
||||||
self.finalize_in_progress = False
|
self.finalize_in_progress = False
|
||||||
self.close_destroys_window = not enable_close_attempted_event if enable_close_attempted_event is not None else None
|
self.close_destroys_window = not enable_close_attempted_event if enable_close_attempted_event is not None else None
|
||||||
|
@ -10109,6 +10129,10 @@ def pin(elem, vertical_alignment=None, shrink=True, expand_x=None, expand_y=None
|
||||||
:type vertical_alignment: str | None
|
:type vertical_alignment: str | None
|
||||||
:param shrink: If True, then the space will shrink down to a single pixel when hidden. False leaves the area large and blank
|
:param shrink: If True, then the space will shrink down to a single pixel when hidden. False leaves the area large and blank
|
||||||
:type shrink: bool
|
:type shrink: bool
|
||||||
|
:param expand_x: If True/False the value will be passed to the Column Elements used to make this feature
|
||||||
|
:type expand_x: (bool)
|
||||||
|
:param expand_y: If True/False the value will be passed to the Column Elements used to make this feature
|
||||||
|
:type expand_y: (bool)
|
||||||
:return: A column element containing the provided element
|
:return: A column element containing the provided element
|
||||||
:rtype: Column
|
:rtype: Column
|
||||||
"""
|
"""
|
||||||
|
@ -10118,28 +10142,36 @@ def pin(elem, vertical_alignment=None, shrink=True, expand_x=None, expand_y=None
|
||||||
return Column([[elem]], pad=(0,0), vertical_alignment=vertical_alignment, expand_x=expand_x, expand_y=expand_y)
|
return Column([[elem]], pad=(0,0), vertical_alignment=vertical_alignment, expand_x=expand_x, expand_y=expand_y)
|
||||||
|
|
||||||
|
|
||||||
def vtop(elem_or_row):
|
def vtop(elem_or_row, expand_x=None, expand_y=None):
|
||||||
"""
|
"""
|
||||||
Align an element or a row of elements to the top of the row that contains it
|
Align an element or a row of elements to the top of the row that contains it
|
||||||
|
|
||||||
:param elem_or_row: the element or row of elements
|
:param elem_or_row: the element or row of elements
|
||||||
:type elem_or_row: Element | List[Element] | Tuple[Element]
|
:type elem_or_row: Element | List[Element] | Tuple[Element]
|
||||||
|
:param expand_x: If True/False the value will be passed to the Column Elements used to make this feature
|
||||||
|
:type expand_x: (bool)
|
||||||
|
:param expand_y: If True/False the value will be passed to the Column Elements used to make this feature
|
||||||
|
:type expand_y: (bool)
|
||||||
:return: A column element containing the provided element aligned to the top or list of elements (a row)
|
:return: A column element containing the provided element aligned to the top or list of elements (a row)
|
||||||
:rtype: Column | List[Column]
|
:rtype: Column | List[Column]
|
||||||
"""
|
"""
|
||||||
|
|
||||||
if isinstance(elem_or_row, list) or isinstance(elem_or_row, tuple):
|
if isinstance(elem_or_row, list) or isinstance(elem_or_row, tuple):
|
||||||
return [Column([[e]], pad=(0,0), vertical_alignment='top') for e in elem_or_row]
|
return [Column([[e]], pad=(0,0), vertical_alignment='top', expand_x=expand_x, expand_y=expand_y) for e in elem_or_row]
|
||||||
|
|
||||||
return Column([[elem_or_row]], pad=(0,0), vertical_alignment='top')
|
return Column([[elem_or_row]], pad=(0,0), vertical_alignment='top', expand_x=expand_x, expand_y=expand_y)
|
||||||
|
|
||||||
|
|
||||||
def vcenter(elem_or_row):
|
def vcenter(elem_or_row, expand_x=None, expand_y=None):
|
||||||
"""
|
"""
|
||||||
Align an element or a row of elements to the center of the row that contains it
|
Align an element or a row of elements to the center of the row that contains it
|
||||||
|
|
||||||
:param elem_or_row: the element or row of elements
|
:param elem_or_row: the element or row of elements
|
||||||
:type elem_or_row: Element | List[Element] | Tuple[Element]
|
:type elem_or_row: Element | List[Element] | Tuple[Element]
|
||||||
|
:param expand_x: If True/False the value will be passed to the Column Elements used to make this feature
|
||||||
|
:type expand_x: (bool)
|
||||||
|
:param expand_y: If True/False the value will be passed to the Column Elements used to make this feature
|
||||||
|
:type expand_y: (bool)
|
||||||
:return: A column element containing the provided element aligned to the center or list of elements (a row)
|
:return: A column element containing the provided element aligned to the center or list of elements (a row)
|
||||||
:rtype: Column | List[Column]
|
:rtype: Column | List[Column]
|
||||||
"""
|
"""
|
||||||
|
@ -10151,12 +10183,16 @@ def vcenter(elem_or_row):
|
||||||
return Column([[elem_or_row]], pad=(0,0), vertical_alignment='center')
|
return Column([[elem_or_row]], pad=(0,0), vertical_alignment='center')
|
||||||
|
|
||||||
|
|
||||||
def vbottom(elem_or_row):
|
def vbottom(elem_or_row, expand_x=None, expand_y=None):
|
||||||
"""
|
"""
|
||||||
Align an element or a row of elements to the bottom of the row that contains it
|
Align an element or a row of elements to the bottom of the row that contains it
|
||||||
|
|
||||||
:param elem_or_row: the element or row of elements
|
:param elem_or_row: the element or row of elements
|
||||||
:type elem_or_row: Element | List[Element] | Tuple[Element]
|
:type elem_or_row: Element | List[Element] | Tuple[Element]
|
||||||
|
:param expand_x: If True/False the value will be passed to the Column Elements used to make this feature
|
||||||
|
:type expand_x: (bool)
|
||||||
|
:param expand_y: If True/False the value will be passed to the Column Elements used to make this feature
|
||||||
|
:type expand_y: (bool)
|
||||||
:return: A column element containing the provided element aligned to the bottom or list of elements (a row)
|
:return: A column element containing the provided element aligned to the bottom or list of elements (a row)
|
||||||
:rtype: Column | List[Column]
|
:rtype: Column | List[Column]
|
||||||
"""
|
"""
|
||||||
|
@ -11990,7 +12026,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
|
||||||
def _add_right_click_menu(element):
|
def _add_right_click_menu(element):
|
||||||
if element.RightClickMenu or toplevel_form.RightClickMenu:
|
if element.RightClickMenu or toplevel_form.RightClickMenu:
|
||||||
menu = element.RightClickMenu or toplevel_form.RightClickMenu
|
menu = element.RightClickMenu or toplevel_form.RightClickMenu
|
||||||
top_menu = tk.Menu(toplevel_form.TKroot, tearoff=False)
|
top_menu = tk.Menu(toplevel_form.TKroot, tearoff=toplevel_form.right_click_menu_tearoff, tearoffcommand=element._tearoff_menu_callback)
|
||||||
|
|
||||||
if toplevel_form.right_click_menu_background_color not in (COLOR_SYSTEM_DEFAULT, None):
|
if toplevel_form.right_click_menu_background_color not in (COLOR_SYSTEM_DEFAULT, None):
|
||||||
top_menu.config(bg=toplevel_form.right_click_menu_background_color)
|
top_menu.config(bg=toplevel_form.right_click_menu_background_color)
|
||||||
|
@ -12729,8 +12765,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
|
||||||
listbox_frame = tk.Frame(tk_row_frame)
|
listbox_frame = tk.Frame(tk_row_frame)
|
||||||
element.TKStringVar = tk.StringVar()
|
element.TKStringVar = tk.StringVar()
|
||||||
element.TKListbox = element.Widget = tk.Listbox(listbox_frame, height=element_size[1], width=width,
|
element.TKListbox = element.Widget = tk.Listbox(listbox_frame, height=element_size[1], width=width,
|
||||||
selectmode=element.SelectMode, font=font,
|
selectmode=element.SelectMode, font=font, exportselection=False)
|
||||||
exportselection=False)
|
|
||||||
element.Widget.config(highlightthickness=0)
|
element.Widget.config(highlightthickness=0)
|
||||||
for index, item in enumerate(element.Values):
|
for index, item in enumerate(element.Values):
|
||||||
element.TKListbox.insert(tk.END, item)
|
element.TKListbox.insert(tk.END, item)
|
||||||
|
@ -12767,6 +12802,7 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
|
||||||
if element.Tooltip is not None:
|
if element.Tooltip is not None:
|
||||||
element.TooltipObject = ToolTip(element.TKListbox, text=element.Tooltip,
|
element.TooltipObject = ToolTip(element.TKListbox, text=element.Tooltip,
|
||||||
timeout=DEFAULT_TOOLTIP_TIME)
|
timeout=DEFAULT_TOOLTIP_TIME)
|
||||||
|
element.listbox_frame = listbox_frame
|
||||||
_add_right_click_menu(element)
|
_add_right_click_menu(element)
|
||||||
# ------------------------- MULTILINE placement element ------------------------- #
|
# ------------------------- MULTILINE placement element ------------------------- #
|
||||||
elif element_type == ELEM_TYPE_INPUT_MULTILINE:
|
elif element_type == ELEM_TYPE_INPUT_MULTILINE:
|
||||||
|
@ -12986,13 +13022,20 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
|
||||||
# ------------------------- IMAGE placement element ------------------------- #
|
# ------------------------- IMAGE placement element ------------------------- #
|
||||||
elif element_type == ELEM_TYPE_IMAGE:
|
elif element_type == ELEM_TYPE_IMAGE:
|
||||||
element = element # type: Image
|
element = element # type: Image
|
||||||
if element.Filename is not None:
|
try:
|
||||||
photo = tk.PhotoImage(file=element.Filename)
|
if element.Filename is not None:
|
||||||
elif element.Data is not None:
|
photo = tk.PhotoImage(file=element.Filename)
|
||||||
photo = tk.PhotoImage(data=element.Data)
|
elif element.Data is not None:
|
||||||
else:
|
photo = tk.PhotoImage(data=element.Data)
|
||||||
photo = None
|
else:
|
||||||
# print('*ERROR laying out form.... Image Element has no image specified*')
|
photo = None
|
||||||
|
# print('*ERROR laying out form.... Image Element has no image specified*')
|
||||||
|
except Exception as e:
|
||||||
|
photo=None
|
||||||
|
_error_popup_with_traceback('Your Window has an Image Element with a problem',
|
||||||
|
'The traceback will show you the Window with the problem layout',
|
||||||
|
'Look in this Window\'s layout for an Image element that has a key of {}'.format(element.Key),
|
||||||
|
'The error occuring is:', e)
|
||||||
|
|
||||||
if photo is not None:
|
if photo is not None:
|
||||||
if element_size == (
|
if element_size == (
|
||||||
|
@ -13004,23 +13047,23 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
|
||||||
element.tktext_label = tk.Label(tk_row_frame, image=photo, width=width,
|
element.tktext_label = tk.Label(tk_row_frame, image=photo, width=width,
|
||||||
height=height,
|
height=height,
|
||||||
bd=border_depth)
|
bd=border_depth)
|
||||||
else:
|
else:
|
||||||
element.tktext_label = tk.Label(tk_row_frame, width=width, height=height,
|
element.tktext_label = tk.Label(tk_row_frame, width=width, height=height,
|
||||||
bd=border_depth)
|
bd=border_depth)
|
||||||
|
|
||||||
if not element.BackgroundColor in (None, COLOR_SYSTEM_DEFAULT):
|
if not element.BackgroundColor in (None, COLOR_SYSTEM_DEFAULT):
|
||||||
element.tktext_label.config(background=element.BackgroundColor)
|
element.tktext_label.config(background=element.BackgroundColor)
|
||||||
|
|
||||||
element.tktext_label.image = photo
|
element.tktext_label.image = photo
|
||||||
# tktext_label.configure(anchor=tk.NW, image=photo)
|
# tktext_label.configure(anchor=tk.NW, image=photo)
|
||||||
element.tktext_label.pack(side=tk.LEFT, padx=elementpad[0], pady=elementpad[1])
|
element.tktext_label.pack(side=tk.LEFT, padx=elementpad[0], pady=elementpad[1])
|
||||||
|
|
||||||
if element.visible is False:
|
if element.visible is False:
|
||||||
element.tktext_label.pack_forget()
|
element.tktext_label.pack_forget()
|
||||||
if element.Tooltip is not None:
|
if element.Tooltip is not None:
|
||||||
element.TooltipObject = ToolTip(element.tktext_label, text=element.Tooltip,
|
element.TooltipObject = ToolTip(element.tktext_label, text=element.Tooltip,
|
||||||
timeout=DEFAULT_TOOLTIP_TIME)
|
timeout=DEFAULT_TOOLTIP_TIME)
|
||||||
if element.EnableEvents:
|
if element.EnableEvents and element.tktext_label is not None:
|
||||||
element.tktext_label.bind('<ButtonPress-1>', element._ClickHandler)
|
element.tktext_label.bind('<ButtonPress-1>', element._ClickHandler)
|
||||||
element.Widget = element.tktext_label
|
element.Widget = element.tktext_label
|
||||||
|
|
||||||
|
@ -17099,7 +17142,8 @@ def _error_popup_with_traceback(title, *args):
|
||||||
def _error_popup_with_code(title, filename=None, line_num=None, *args):
|
def _error_popup_with_code(title, filename=None, line_num=None, *args):
|
||||||
layout = [[Text('ERROR'), Text(title)],
|
layout = [[Text('ERROR'), Text(title)],
|
||||||
[Image(data=_random_error_emoji())]]
|
[Image(data=_random_error_emoji())]]
|
||||||
layout += [[Text(msg)] for msg in args]
|
# make max length of line be 90 chars and allow the hieight to vary based on number of needed lines
|
||||||
|
layout += [[Text(str(msg), size=(min(90, len(str(msg))),None))] for msg in args]
|
||||||
|
|
||||||
layout += [[Button('Close'), Button('Take me to error')]]
|
layout += [[Button('Close'), Button('Take me to error')]]
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue