commit
627561cda8
|
@ -56,7 +56,7 @@ DEFAULT_MARGINS = (10, 5) # Margins for each LEFT/RIGHT margin is first term
|
||||||
DEFAULT_ELEMENT_PADDING = (5, 3) # Padding between elements (row, col) in pixels
|
DEFAULT_ELEMENT_PADDING = (5, 3) # Padding between elements (row, col) in pixels
|
||||||
DEFAULT_AUTOSIZE_TEXT = True
|
DEFAULT_AUTOSIZE_TEXT = True
|
||||||
DEFAULT_AUTOSIZE_BUTTONS = True
|
DEFAULT_AUTOSIZE_BUTTONS = True
|
||||||
DEFAULT_FONT = ("Helvetica", 10)
|
DEFAULT_FONT = ("Helvetica", 15)
|
||||||
DEFAULT_TEXT_JUSTIFICATION = 'left'
|
DEFAULT_TEXT_JUSTIFICATION = 'left'
|
||||||
DEFAULT_BORDER_WIDTH = 1
|
DEFAULT_BORDER_WIDTH = 1
|
||||||
DEFAULT_AUTOCLOSE_TIME = 3 # time in seconds to show an autoclose form
|
DEFAULT_AUTOCLOSE_TIME = 3 # time in seconds to show an autoclose form
|
||||||
|
@ -263,62 +263,6 @@ POPUP_BUTTONS_OK = 0
|
||||||
POPUP_BUTTONS_NO_BUTTONS = 5
|
POPUP_BUTTONS_NO_BUTTONS = 5
|
||||||
|
|
||||||
|
|
||||||
# ------------------------------------------------------------------------- #
|
|
||||||
# ToolTip used by the Elements #
|
|
||||||
# ------------------------------------------------------------------------- #
|
|
||||||
|
|
||||||
class ToolTip:
|
|
||||||
""" Create a tooltip for a given widget
|
|
||||||
|
|
||||||
(inspired by https://stackoverflow.com/a/36221216)
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, widget, text, timeout=DEFAULT_TOOLTIP_TIME):
|
|
||||||
self.widget = widget
|
|
||||||
self.text = text
|
|
||||||
self.timeout = timeout
|
|
||||||
# self.wraplength = wraplength if wraplength else widget.winfo_screenwidth() // 2
|
|
||||||
self.tipwindow = None
|
|
||||||
self.id = None
|
|
||||||
self.x = self.y = 0
|
|
||||||
self.widget.bind("<Enter>", self.enter)
|
|
||||||
self.widget.bind("<Leave>", self.leave)
|
|
||||||
self.widget.bind("<ButtonPress>", self.leave)
|
|
||||||
|
|
||||||
def enter(self, event=None):
|
|
||||||
self.schedule()
|
|
||||||
|
|
||||||
def leave(self, event=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() + 20
|
|
||||||
y = self.widget.winfo_rooty() + self.widget.winfo_height() - 20
|
|
||||||
self.tipwindow = tk.Toplevel(self.widget)
|
|
||||||
self.tipwindow.wm_overrideredirect(True)
|
|
||||||
self.tipwindow.wm_geometry("+%d+%d" % (x, y))
|
|
||||||
label = ttk.Label(self.tipwindow, text=self.text, justify=tk.LEFT,
|
|
||||||
background="#ffffe0", relief=tk.SOLID, borderwidth=1)
|
|
||||||
label.pack()
|
|
||||||
|
|
||||||
def hidetip(self):
|
|
||||||
if self.tipwindow:
|
|
||||||
self.tipwindow.destroy()
|
|
||||||
self.tipwindow = None
|
|
||||||
|
|
||||||
|
|
||||||
# ---------------------------------------------------------------------- #
|
# ---------------------------------------------------------------------- #
|
||||||
# Cascading structure.... Objects get larger #
|
# Cascading structure.... Objects get larger #
|
||||||
# Button #
|
# Button #
|
||||||
|
@ -477,26 +421,28 @@ class Element():
|
||||||
|
|
||||||
|
|
||||||
def Update(self, widget, background_color=None, text_color=None, font=None, visible=None, disabled=None, tooltip=None):
|
def Update(self, widget, background_color=None, text_color=None, font=None, visible=None, disabled=None, tooltip=None):
|
||||||
return
|
# if font:
|
||||||
if font:
|
# widget.SetFont(font_to_wx_font(font))
|
||||||
widget.SetFont(font_to_wx_font(font))
|
# if text_color not in (None, COLOR_SYSTEM_DEFAULT):
|
||||||
if text_color not in (None, COLOR_SYSTEM_DEFAULT):
|
# widget.SetForegroundColour(text_color)
|
||||||
widget.SetForegroundColour(text_color)
|
# if background_color not in (None, COLOR_SYSTEM_DEFAULT):
|
||||||
if background_color not in (None, COLOR_SYSTEM_DEFAULT):
|
# widget.SetBackgroundColour(background_color)
|
||||||
widget.SetBackgroundColour(background_color)
|
# if visible is True:
|
||||||
if visible is True:
|
# widget.Show()
|
||||||
widget.Show()
|
# self.ParentForm.VisibilityChanged()
|
||||||
self.ParentForm.VisibilityChanged()
|
# elif visible is False:
|
||||||
elif visible is False:
|
# widget.Hide()
|
||||||
widget.Hide()
|
# self.ParentForm.VisibilityChanged()
|
||||||
self.ParentForm.VisibilityChanged()
|
# if disabled:
|
||||||
if disabled:
|
# widget.Enable(False)
|
||||||
widget.Enable(False)
|
# elif disabled is False:
|
||||||
elif disabled is False:
|
# widget.Enable(True)
|
||||||
widget.Enable(True)
|
# if tooltip is not None:
|
||||||
if tooltip is not None:
|
# widget.SetToolTip(tooltip)
|
||||||
widget.SetToolTip(tooltip)
|
if visible is False:
|
||||||
|
widget.attributes['hidden'] = 'true'
|
||||||
|
elif visible is True:
|
||||||
|
widget.attributes['hidden'] = 'false'
|
||||||
|
|
||||||
def __del__(self):
|
def __del__(self):
|
||||||
pass
|
pass
|
||||||
|
@ -531,10 +477,17 @@ class InputText(Element):
|
||||||
super().__init__(ELEM_TYPE_INPUT_TEXT, size=size, background_color=bg, text_color=fg, key=key, pad=pad,
|
super().__init__(ELEM_TYPE_INPUT_TEXT, size=size, background_color=bg, text_color=fg, key=key, pad=pad,
|
||||||
font=font, tooltip=tooltip, visible=visible, size_px=size_px)
|
font=font, tooltip=tooltip, visible=visible, size_px=size_px)
|
||||||
|
|
||||||
def InputTextCallback(self, widget:remi.Widget, value, keycode):
|
# def InputTextCallback(self, widget:remi.gui.TextInput, value, keycode):
|
||||||
|
def InputTextCallback(self,widget, key, keycode, ctrl, shift, alt):
|
||||||
# print(f'text widget value = {widget.get_value()}')
|
# print(f'text widget value = {widget.get_value()}')
|
||||||
self.ParentForm.LastButtonClicked = chr(int(keycode))
|
# widget.set_value('')
|
||||||
|
# widget.set_value(value)
|
||||||
|
self.ParentForm.LastButtonClicked = key
|
||||||
self.ParentForm.MessageQueue.put(self.ParentForm.LastButtonClicked)
|
self.ParentForm.MessageQueue.put(self.ParentForm.LastButtonClicked)
|
||||||
|
widget.set_value(widget.get_value()+key)
|
||||||
|
return (key, keycode, ctrl, shift, alt)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def Update(self, value=None, disabled=None):
|
def Update(self, value=None, disabled=None):
|
||||||
if disabled is True:
|
if disabled is True:
|
||||||
|
@ -557,6 +510,31 @@ class InputText(Element):
|
||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
class TextInput_raw_onkeyup(remi.gui.TextInput):
|
||||||
|
@remi.gui.decorate_set_on_listener("(self, emitter, key, keycode, ctrl, shift, alt)")
|
||||||
|
@remi.gui.decorate_event_js("""var params={};params['key']=event.key;
|
||||||
|
params['keycode']=(event.which||event.keyCode);
|
||||||
|
params['ctrl']=event.ctrlKey;
|
||||||
|
params['shift']=event.shiftKey;
|
||||||
|
params['alt']=event.altKey;
|
||||||
|
sendCallbackParam('%(emitter_identifier)s','%(event_name)s',params);
|
||||||
|
event.stopPropagation();event.preventDefault();return false;""")
|
||||||
|
def onkeyup(self, key, keycode, ctrl, shift, alt):
|
||||||
|
return (key, keycode, ctrl, shift, alt)
|
||||||
|
|
||||||
|
@remi.gui.decorate_set_on_listener("(self, emitter, key, keycode, ctrl, shift, alt)")
|
||||||
|
@remi.gui.decorate_event_js("""var params={};params['key']=event.key;
|
||||||
|
params['keycode']=(event.which||event.keyCode);
|
||||||
|
params['ctrl']=event.ctrlKey;
|
||||||
|
params['shift']=event.shiftKey;
|
||||||
|
params['alt']=event.altKey;
|
||||||
|
sendCallbackParam('%(emitter_identifier)s','%(event_name)s',params);
|
||||||
|
event.stopPropagation();event.preventDefault();return false;""")
|
||||||
|
def onkeydown(self, key, keycode, ctrl, shift, alt):
|
||||||
|
return (key, keycode, ctrl, shift, alt)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def __del__(self):
|
def __del__(self):
|
||||||
super().__del__()
|
super().__del__()
|
||||||
|
|
||||||
|
@ -740,16 +718,14 @@ class Listbox(Element):
|
||||||
def Update(self, values=None, disabled=None, set_to_index=None,background_color=None, text_color=None, font=None, visible=None):
|
def Update(self, values=None, disabled=None, set_to_index=None,background_color=None, text_color=None, font=None, visible=None):
|
||||||
if values is not None:
|
if values is not None:
|
||||||
self.Values = values
|
self.Values = values
|
||||||
for i in range(self.QT_ListWidget.count()):
|
self.Widget.set_value(values)
|
||||||
self.QT_ListWidget.takeItem(0)
|
# if disabled == True:
|
||||||
self.QT_ListWidget.addItems(values)
|
# self.QT_ListWidget.setDisabled(True)
|
||||||
if disabled == True:
|
# elif disabled == False:
|
||||||
self.QT_ListWidget.setDisabled(True)
|
# self.QT_ListWidget.setDisabled(False)
|
||||||
elif disabled == False:
|
# if set_to_index is not None:
|
||||||
self.QT_ListWidget.setDisabled(False)
|
# self.QT_ListWidget.setCurrentRow(set_to_index)
|
||||||
if set_to_index is not None:
|
super().Update(self.Widget, background_color=background_color, text_color=text_color, font=font, visible=visible)
|
||||||
self.QT_ListWidget.setCurrentRow(set_to_index)
|
|
||||||
super().Update(self.QT_ListWidget, background_color=background_color, text_color=text_color, font=font, visible=visible)
|
|
||||||
|
|
||||||
return
|
return
|
||||||
|
|
||||||
|
@ -1081,9 +1057,9 @@ class MultilineOutput(Element):
|
||||||
|
|
||||||
def Update(self, value=None, disabled=None, append=False, background_color=None, text_color=None, font=None, visible=None):
|
def Update(self, value=None, disabled=None, append=False, background_color=None, text_color=None, font=None, visible=None):
|
||||||
if value is not None and not append:
|
if value is not None and not append:
|
||||||
self.Widget.set_value(value)
|
self.Widget.set_value(str(value))
|
||||||
elif value is not None and append:
|
elif value is not None and append:
|
||||||
self.CurrentValue = self.CurrentValue + '\n' + value
|
self.CurrentValue = self.CurrentValue + '\n' + str(value)
|
||||||
self.Widget.set_value(self.CurrentValue)
|
self.Widget.set_value(self.CurrentValue)
|
||||||
|
|
||||||
super().Update(self.Widget, background_color=background_color, text_color=text_color, font=font, visible=visible)
|
super().Update(self.Widget, background_color=background_color, text_color=text_color, font=font, visible=visible)
|
||||||
|
@ -1099,6 +1075,9 @@ class MultilineOutput(Element):
|
||||||
super().__del__()
|
super().__del__()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# ---------------------------------------------------------------------- #
|
# ---------------------------------------------------------------------- #
|
||||||
# Text #
|
# Text #
|
||||||
# ---------------------------------------------------------------------- #
|
# ---------------------------------------------------------------------- #
|
||||||
|
@ -1146,7 +1125,7 @@ class Text(Element):
|
||||||
|
|
||||||
def Update(self, value=None, background_color=None, text_color=None, font=None, visible=None):
|
def Update(self, value=None, background_color=None, text_color=None, font=None, visible=None):
|
||||||
if value is not None:
|
if value is not None:
|
||||||
self.Widget.set_text(value)
|
self.Widget.set_text(str(value))
|
||||||
# if background_color is not None:
|
# if background_color is not None:
|
||||||
# self.TKText.configure(background=background_color)
|
# self.TKText.configure(background=background_color)
|
||||||
# if text_color is not None:
|
# if text_color is not None:
|
||||||
|
@ -1949,6 +1928,7 @@ class Frame(Element):
|
||||||
self.TitleLocation = title_location
|
self.TitleLocation = title_location
|
||||||
self.BorderWidth = border_width
|
self.BorderWidth = border_width
|
||||||
self.BackgroundColor = background_color if background_color is not None else DEFAULT_BACKGROUND_COLOR
|
self.BackgroundColor = background_color if background_color is not None else DEFAULT_BACKGROUND_COLOR
|
||||||
|
self.Justification = 'left'
|
||||||
|
|
||||||
self.Layout(layout)
|
self.Layout(layout)
|
||||||
|
|
||||||
|
@ -2244,8 +2224,7 @@ class Slider(Element):
|
||||||
# Column #
|
# Column #
|
||||||
# ---------------------------------------------------------------------- #
|
# ---------------------------------------------------------------------- #
|
||||||
class Column(Element):
|
class Column(Element):
|
||||||
def __init__(self, layout, background_color=None, size=(None, None), pad=None, scrollable=False,
|
def __init__(self, layout, background_color=None, size=(None, None), pad=None, scrollable=False, vertical_scroll_only=False, justification='left', key=None):
|
||||||
vertical_scroll_only=False, key=None):
|
|
||||||
'''
|
'''
|
||||||
Column Element
|
Column Element
|
||||||
:param layout:
|
:param layout:
|
||||||
|
@ -2265,11 +2244,12 @@ class Column(Element):
|
||||||
self.TKFrame = None
|
self.TKFrame = None
|
||||||
self.Scrollable = scrollable
|
self.Scrollable = scrollable
|
||||||
self.VerticalScrollOnly = vertical_scroll_only
|
self.VerticalScrollOnly = vertical_scroll_only
|
||||||
|
self.Justification = justification
|
||||||
# self.ImageFilename = image_filename
|
# self.ImageFilename = image_filename
|
||||||
# self.ImageData = image_data
|
# self.ImageData = image_data
|
||||||
# self.ImageSize = image_size
|
# self.ImageSize = image_size
|
||||||
# self.ImageSubsample = image_subsample
|
# self.ImageSubsample = image_subsample
|
||||||
bg = background_color if background_color is not None else DEFAULT_BACKGROUND_COLOR
|
# bg = background_color if background_color is not None else DEFAULT_BACKGROUND_COLOR
|
||||||
|
|
||||||
self.Layout(layout)
|
self.Layout(layout)
|
||||||
|
|
||||||
|
@ -2712,8 +2692,7 @@ class Window:
|
||||||
self.BackgroundImage = background_image
|
self.BackgroundImage = background_image
|
||||||
self.XFound = False
|
self.XFound = False
|
||||||
self.DisableMinimize = disable_minimize
|
self.DisableMinimize = disable_minimize
|
||||||
self.MasterFrame = None # type: wx.Frame
|
self.Justification = 'left'
|
||||||
self.MasterPanel = None # type wx.Panel
|
|
||||||
self.IgnoreClose = False
|
self.IgnoreClose = False
|
||||||
self.thread_id = None
|
self.thread_id = None
|
||||||
self.App = None # type: Window.MyApp
|
self.App = None # type: Window.MyApp
|
||||||
|
@ -3194,9 +3173,14 @@ class Window:
|
||||||
# s = remi.server.StandaloneServer(self.MyApp, width=1100, height=600)
|
# s = remi.server.StandaloneServer(self.MyApp, width=1100, height=600)
|
||||||
# s.start()
|
# s.start()
|
||||||
Window.port_number += 1
|
Window.port_number += 1
|
||||||
|
|
||||||
remi.start(self.MyApp, title=self.Title ,debug=False, address='0.0.0.0', port=Window.port_number, start_browser=True, update_interval=.00001, userdata=(self,))
|
remi.start(self.MyApp, title=self.Title ,debug=False, address='0.0.0.0', port=Window.port_number, start_browser=True, update_interval=.00001, userdata=(self,))
|
||||||
|
|
||||||
# remi.start(self.MyApp, title=self.Title ,debug=False, userdata=(self,), standalone=True) # standalone=True)
|
# remi.start(self.MyApp, title=self.Title ,debug=False, userdata=(self,), standalone=True) # standalone=True)
|
||||||
|
|
||||||
|
# remi.start(self.MyApp, standalone=True, debug=True, userdata=(self,) ) # Can't do this because of a threading problem
|
||||||
print('Returned from Remi Start command... now sending None event')
|
print('Returned from Remi Start command... now sending None event')
|
||||||
|
|
||||||
# self.App.server_starter_instance._alive = False
|
# self.App.server_starter_instance._alive = False
|
||||||
# self.App.server_starter_instance._sserver.shutdown()
|
# self.App.server_starter_instance._sserver.shutdown()
|
||||||
self.MessageQueue.put(None) # if returned from start call, then the window has been destroyed and a None event should be generated
|
self.MessageQueue.put(None) # if returned from start call, then the window has been destroyed and a None event should be generated
|
||||||
|
@ -3392,7 +3376,6 @@ def Submit(button_text='Submit', size=(None, None), auto_size_button=None, butto
|
||||||
bind_return_key=bind_return_key, focus=focus, pad=pad, key=key)
|
bind_return_key=bind_return_key, focus=focus, pad=pad, key=key)
|
||||||
|
|
||||||
|
|
||||||
# ------------------------- OPEN BUTTON Element lazy function ------------------------- #
|
|
||||||
# ------------------------- OPEN BUTTON Element lazy function ------------------------- #
|
# ------------------------- OPEN BUTTON Element lazy function ------------------------- #
|
||||||
def Open(button_text='Open', size=(None, None), auto_size_button=None, button_color=None, disabled=False,
|
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):
|
bind_return_key=True, tooltip=None, font=None, focus=False, pad=None, key=None):
|
||||||
|
@ -3985,17 +3968,6 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
|
||||||
widget.attributes['hidden'] = 'true'
|
widget.attributes['hidden'] = 'true'
|
||||||
if element.Tooltip is not None:
|
if element.Tooltip is not None:
|
||||||
widget.attributes['title'] = element.Tooltip
|
widget.attributes['title'] = element.Tooltip
|
||||||
#
|
|
||||||
# widget.SetMinSize(element_size)
|
|
||||||
# if element.Disabled:
|
|
||||||
# widget.Enable(False)
|
|
||||||
# if not element.Visible:
|
|
||||||
# widget.Hide()
|
|
||||||
# if element.Tooltip:
|
|
||||||
# widget.SetToolTip(element.Tooltip)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
border_depth = toplevel_form.BorderDepth if toplevel_form.BorderDepth is not None else DEFAULT_BORDER_WIDTH
|
border_depth = toplevel_form.BorderDepth if toplevel_form.BorderDepth is not None else DEFAULT_BORDER_WIDTH
|
||||||
# --------------------------------------------------------------------------- #
|
# --------------------------------------------------------------------------- #
|
||||||
|
@ -4009,11 +3981,16 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
|
||||||
######################### LOOP THROUGH ELEMENTS ON ROW #########################
|
######################### LOOP THROUGH ELEMENTS ON ROW #########################
|
||||||
# *********** ------- Loop through ELEMENTS ------- ***********#
|
# *********** ------- Loop through ELEMENTS ------- ***********#
|
||||||
# *********** Make TK Row ***********#
|
# *********** Make TK Row ***********#
|
||||||
tk_row_frame = remi.gui.HBox() # TODO Get horizontal ROW widget to put others into
|
tk_row_frame = remi.gui.HBox()
|
||||||
tk_row_frame.style['justify-content'] = 'flex-start'
|
if form.Justification.startswith('c'):
|
||||||
tk_row_frame.style['align-items'] = 'baseline'
|
print('Centering row')
|
||||||
if toplevel_form.BackgroundColor not in(None, COLOR_SYSTEM_DEFAULT):
|
tk_row_frame.style['align-items'] = 'center'
|
||||||
tk_row_frame.style['background-color'] = toplevel_form.BackgroundColor
|
tk_row_frame.style['justify-content'] = 'center'
|
||||||
|
else:
|
||||||
|
tk_row_frame.style['align-items'] = 'baseline'
|
||||||
|
tk_row_frame.style['justify-content'] = 'flex-start'
|
||||||
|
if form.BackgroundColor not in(None, COLOR_SYSTEM_DEFAULT):
|
||||||
|
tk_row_frame.style['background-color'] = form.BackgroundColor
|
||||||
|
|
||||||
for col_num, element in enumerate(flex_row):
|
for col_num, element in enumerate(flex_row):
|
||||||
element.ParentForm = toplevel_form # save the button's parent form object
|
element.ParentForm = toplevel_form # save the button's parent form object
|
||||||
|
@ -4055,7 +4032,22 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
|
||||||
|
|
||||||
# ------------------------- COLUMN element ------------------------- #
|
# ------------------------- COLUMN element ------------------------- #
|
||||||
if element_type == ELEM_TYPE_COLUMN:
|
if element_type == ELEM_TYPE_COLUMN:
|
||||||
pass
|
print('adding column')
|
||||||
|
element = element # type: Column
|
||||||
|
element.Widget = column_widget = remi.gui.VBox()
|
||||||
|
|
||||||
|
if element.Justification.startswith('c'):
|
||||||
|
print('CENTERING')
|
||||||
|
column_widget.style['align-items'] = 'center'
|
||||||
|
column_widget.style['justify-content'] = 'center'
|
||||||
|
else:
|
||||||
|
column_widget.style['justify-content'] = 'flex-start'
|
||||||
|
column_widget.style['align-items'] = 'baseline'
|
||||||
|
if element.BackgroundColor not in (None, COLOR_SYSTEM_DEFAULT):
|
||||||
|
column_widget.style['background-color'] = element.BackgroundColor
|
||||||
|
PackFormIntoFrame(element, column_widget, toplevel_form)
|
||||||
|
tk_row_frame.append(element.Widget)
|
||||||
|
|
||||||
# if element.Scrollable:
|
# if element.Scrollable:
|
||||||
# col_frame = TkScrollableFrame(tk_row_frame,
|
# col_frame = TkScrollableFrame(tk_row_frame,
|
||||||
# element.VerticalScrollOnly) # do not use yet! not working
|
# element.VerticalScrollOnly) # do not use yet! not working
|
||||||
|
@ -4086,6 +4078,8 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
|
||||||
element.Widget = remi.gui.Label(element.DisplayText)
|
element.Widget = remi.gui.Label(element.DisplayText)
|
||||||
element.Widget.set_layout_orientation(True)
|
element.Widget.set_layout_orientation(True)
|
||||||
do_font_and_color(element.Widget)
|
do_font_and_color(element.Widget)
|
||||||
|
if auto_size_text and element.Size == (None, None):
|
||||||
|
del(element.Widget.style['width'])
|
||||||
if element.Justification:
|
if element.Justification:
|
||||||
if element.Justification.startswith('c'):
|
if element.Justification.startswith('c'):
|
||||||
element.Widget.style['text-align'] = 'center'
|
element.Widget.style['text-align'] = 'center'
|
||||||
|
@ -4103,6 +4097,8 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
|
||||||
element.Widget = remi.gui.Button(element.ButtonText, width=size[0], height=size[1], margin='10px')
|
element.Widget = remi.gui.Button(element.ButtonText, width=size[0], height=size[1], margin='10px')
|
||||||
element.Widget.onclick.connect(element.ButtonCallBack)
|
element.Widget.onclick.connect(element.ButtonCallBack)
|
||||||
do_font_and_color(element.Widget)
|
do_font_and_color(element.Widget)
|
||||||
|
if element.AutoSizeButton or (toplevel_form.AutoSizeButtons and element.AutoSizeButton is not False) and element.Size == (None, None):
|
||||||
|
del (element.Widget.style['width'])
|
||||||
tk_row_frame.append(element.Widget)
|
tk_row_frame.append(element.Widget)
|
||||||
|
|
||||||
# stringvar = tk.StringVar()
|
# stringvar = tk.StringVar()
|
||||||
|
@ -4181,10 +4177,12 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
|
||||||
# # ------------------------- INPUT element ------------------------- #
|
# # ------------------------- INPUT element ------------------------- #
|
||||||
elif element_type == ELEM_TYPE_INPUT_TEXT:
|
elif element_type == ELEM_TYPE_INPUT_TEXT:
|
||||||
element = element # type: InputText
|
element = element # type: InputText
|
||||||
element.Widget = remi.gui.TextInput(hint=element.DefaultText)
|
element.Widget = InputText.TextInput_raw_onkeyup(hint=element.DefaultText)
|
||||||
|
# element.Widget = remi.gui.TextInput(hint=element.DefaultText)
|
||||||
do_font_and_color(element.Widget)
|
do_font_and_color(element.Widget)
|
||||||
if element.ChangeSubmits:
|
if element.ChangeSubmits:
|
||||||
element.Widget.onkeydown.connect(element.InputTextCallback)
|
element.Widget.onkeyup.connect(element.InputTextCallback)
|
||||||
|
# element.Widget.onkeydown.connect(element.InputTextCallback)
|
||||||
tk_row_frame.append(element.Widget)
|
tk_row_frame.append(element.Widget)
|
||||||
|
|
||||||
# show = element.PasswordCharacter if element.PasswordCharacter else ""
|
# show = element.PasswordCharacter if element.PasswordCharacter else ""
|
||||||
|
@ -4431,7 +4429,12 @@ def PackFormIntoFrame(form, containing_frame, toplevel_form):
|
||||||
# element.TooltipObject = ToolTip(element._TKOut, text=element.Tooltip, timeout=DEFAULT_TOOLTIP_TIME)
|
# element.TooltipObject = ToolTip(element._TKOut, text=element.Tooltip, timeout=DEFAULT_TOOLTIP_TIME)
|
||||||
# ------------------------- IMAGE element ------------------------- #
|
# ------------------------- IMAGE element ------------------------- #
|
||||||
elif element_type == ELEM_TYPE_IMAGE:
|
elif element_type == ELEM_TYPE_IMAGE:
|
||||||
pass
|
element = element # type: Image
|
||||||
|
element.Widget = remi.gui.Image(filename=element.Filename)
|
||||||
|
do_font_and_color(element.Widget)
|
||||||
|
if element.ChangeSubmits:
|
||||||
|
element.Widget.onchange.connect(element.ChangedCallback)
|
||||||
|
tk_row_frame.append(element.Widget)
|
||||||
# if element.Filename is not None:
|
# if element.Filename is not None:
|
||||||
# photo = tk.PhotoImage(file=element.Filename)
|
# photo = tk.PhotoImage(file=element.Filename)
|
||||||
# elif element.Data is not None:
|
# elif element.Data is not None:
|
||||||
|
@ -6491,11 +6494,13 @@ def main():
|
||||||
# [Text('Destination Folder', justification='right', size=(40,1)), InputText('Dest'), FolderBrowse()],
|
# [Text('Destination Folder', justification='right', size=(40,1)), InputText('Dest'), FolderBrowse()],
|
||||||
# [Ok(), Cancel(disabled=True), Exit(tooltip='Exit button'), Button('Hidden Button', visible=False)]]
|
# [Ok(), Cancel(disabled=True), Exit(tooltip='Exit button'), Button('Hidden Button', visible=False)]]
|
||||||
|
|
||||||
|
col1 = [[Text('Column 1 line 1', background_color='red')], [Text('Column 1 line 2')]]
|
||||||
|
|
||||||
layout = [
|
layout = [
|
||||||
[Text('PySimpleGUIWeb Demo of Working Elements', tooltip='text', font=('Comic sans ms', 20), text_color='red', enable_events=True, key='_PySimpleGUIWeb_')],
|
[Text('PySimpleGUIWeb Welcomes You...', tooltip='text', font=('Comic sans ms', 20),size=(40,1), text_color='red', enable_events=True, key='_PySimpleGUIWeb_')],
|
||||||
[T('Current Time '), Text('Text', key='_TEXT_', font='Arial 18', text_color='black', size=(30,1))],
|
[T('Current Time '), Text('Text', key='_TEXT_', font='Arial 18', text_color='black', size=(30,1)), Column(col1, background_color='red')],
|
||||||
[T('Up Time'), Text('Text', key='_TEXT_UPTIME_', font='Arial 18', text_color='black', size=(30,1))],
|
[T('Up Time'), Text('Text', key='_TEXT_UPTIME_', font='Arial 18', text_color='black', size=(30,1))],
|
||||||
[Input('Single Line Input', do_not_clear=True, enable_events=True, size=(30, 1))],
|
[Input('Single Line Input', do_not_clear=True, enable_events=True, size=(30, 1), text_color='red')],
|
||||||
[Multiline('Multiline Input', do_not_clear=True, size=(40, 4), enable_events=True, key='_MULTI_IN_')],
|
[Multiline('Multiline Input', do_not_clear=True, size=(40, 4), enable_events=True, key='_MULTI_IN_')],
|
||||||
[MultilineOutput('Multiline Output', size=(80, 8), text_color='blue', font='Courier 12', key='_MULTIOUT_')],
|
[MultilineOutput('Multiline Output', size=(80, 8), text_color='blue', font='Courier 12', key='_MULTIOUT_')],
|
||||||
[Checkbox('Checkbox 1', enable_events=True, key='_CB1_'), Checkbox('Checkbox 2', default=True, key='_CB2_', enable_events=True)],
|
[Checkbox('Checkbox 1', enable_events=True, key='_CB1_'), Checkbox('Checkbox 2', default=True, key='_CB2_', enable_events=True)],
|
||||||
|
@ -6504,15 +6509,14 @@ def main():
|
||||||
[Listbox(values=('Listbox 1', 'Listbox 2', 'Listbox 3'), enable_events =True, size=(10, 3), key='_LIST_')],
|
[Listbox(values=('Listbox 1', 'Listbox 2', 'Listbox 3'), enable_events =True, size=(10, 3), key='_LIST_')],
|
||||||
[Slider((1, 100), default_value=80, key='_SLIDER_', visible=True, enable_events=True)],
|
[Slider((1, 100), default_value=80, key='_SLIDER_', visible=True, enable_events=True)],
|
||||||
[Spin(values=(1, 2, 3), initial_value='2', size=(4, 1), key='_SPIN_', enable_events=True)],
|
[Spin(values=(1, 2, 3), initial_value='2', size=(4, 1), key='_SPIN_', enable_events=True)],
|
||||||
[OK(), Button('Hidden', visible=False), Button('Exit', button_color=('white', 'red'))]
|
[OK(), Button('Hidden', visible=False), Button('Values'), Button('Exit', button_color=('white', 'red'))]
|
||||||
]
|
]
|
||||||
|
|
||||||
|
window = Window('PySimpleGUIWeb Window', font='Arial 18',default_element_size=(12,1), auto_size_buttons=False).Layout(layout)
|
||||||
window = Window('PySimpleGUIWeb Window', font='Arial 18', default_element_size=(12,1)).Layout(layout)
|
|
||||||
|
|
||||||
start_time = datetime.datetime.now()
|
start_time = datetime.datetime.now()
|
||||||
while True:
|
while True:
|
||||||
event, values = window.Read(timeout=0)
|
event, values = window.Read(timeout=5)
|
||||||
window.Element('_TEXT_').Update(str(datetime.datetime.now()))
|
window.Element('_TEXT_').Update(str(datetime.datetime.now()))
|
||||||
window.Element('_TEXT_UPTIME_').Update(str(datetime.datetime.now()-start_time))
|
window.Element('_TEXT_UPTIME_').Update(str(datetime.datetime.now()-start_time))
|
||||||
print(event, values) if event != TIMEOUT_KEY else None
|
print(event, values) if event != TIMEOUT_KEY else None
|
||||||
|
@ -6520,8 +6524,12 @@ def main():
|
||||||
break
|
break
|
||||||
elif event == 'OK':
|
elif event == 'OK':
|
||||||
window.Element('_MULTIOUT_').Update('You clicked the OK button', append=True)
|
window.Element('_MULTIOUT_').Update('You clicked the OK button', append=True)
|
||||||
|
window.Element('_PySimpleGUIWeb_').Widget.style['background-image'] = "url('/my_resources:mine.png')"
|
||||||
|
|
||||||
|
elif event == 'Values':
|
||||||
|
window.Element('_MULTIOUT_').Update(str(values), append=True)
|
||||||
elif event != TIMEOUT_KEY:
|
elif event != TIMEOUT_KEY:
|
||||||
window.Element('_MULTIOUT_').Update(str(event) + '\n'+str(values), append=True)
|
window.Element('_MULTIOUT_').Update('EVENT: ' + str(event), append=True)
|
||||||
window.Close()
|
window.Close()
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -8,7 +8,7 @@
|
||||||
|
|
||||||
![Python Version](https://img.shields.io/badge/Python-3.x-yellow.svg)
|
![Python Version](https://img.shields.io/badge/Python-3.x-yellow.svg)
|
||||||
|
|
||||||
![Python Version](https://img.shields.io/badge/PySimpleGUIWeb_-0.4.0-orange.svg?longCache=true&style=for-the-badge)
|
![Python Version](https://img.shields.io/badge/PySimpleGUIWeb_-0.5.0-orange.svg?longCache=true&style=for-the-badge)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -18,6 +18,8 @@ PySimpleGUI running in your web browser!
|
||||||
|
|
||||||
Your source code will work on tkinter, Qt, WxPython and now in a browser (thanks to Remi)
|
Your source code will work on tkinter, Qt, WxPython and now in a browser (thanks to Remi)
|
||||||
|
|
||||||
|
Check out [repl.it](https://repl.it/@PySimpleGUI/PySimpleGUIWeb-Demos), the new way to run your PySimpleGUI code in a browser!
|
||||||
|
|
||||||
## Primary PySimpleGUI Documentation
|
## Primary PySimpleGUI Documentation
|
||||||
|
|
||||||
To get instructions on how use PySimpleGUI's APIs, please reference the [main documentation](http://www.PySimpleGUI.org).
|
To get instructions on how use PySimpleGUI's APIs, please reference the [main documentation](http://www.PySimpleGUI.org).
|
||||||
|
@ -28,18 +30,8 @@ This Readme is for information ***specific to*** the Web port of PySimpleGUI.
|
||||||
|
|
||||||
PySimpleGUIWeb enables you to run your PySimpleGUI programs in your web browser. It utilizes a package called Remi to achieve this amazing package.
|
PySimpleGUIWeb enables you to run your PySimpleGUI programs in your web browser. It utilizes a package called Remi to achieve this amazing package.
|
||||||
|
|
||||||
At the moment (Jan 26 2019) these elements are operational:
|
|
||||||
* Text
|
|
||||||
* Single line text input
|
|
||||||
* Multiline Input
|
|
||||||
* Multiline Output
|
|
||||||
* Listbox
|
|
||||||
* Combobox
|
|
||||||
* Checkbox
|
|
||||||
* Slider
|
|
||||||
* Spinner (numbers only...hardcoded to 0 to 100)
|
|
||||||
|
|
||||||
## Engineering Pre-Release Version 0.4.0
|
## Engineering Pre-Release Version 0.5.0
|
||||||
|
|
||||||
[Announcements of Latest Developments](https://github.com/MikeTheWatchGuy/PySimpleGUI/issues/142)
|
[Announcements of Latest Developments](https://github.com/MikeTheWatchGuy/PySimpleGUI/issues/142)
|
||||||
|
|
||||||
|
@ -85,6 +77,8 @@ PySimpleGUIWeb runs only on Python 3. Legacy Python (2.7) is not supported.
|
||||||
* Combobox Element
|
* Combobox Element
|
||||||
* Checkbox Element
|
* Checkbox Element
|
||||||
* Listbox Element
|
* Listbox Element
|
||||||
|
* Spinner Element (sorta... numbers 0 to 100 only now)
|
||||||
|
* Column Element
|
||||||
* Window background color
|
* Window background color
|
||||||
* Element padding
|
* Element padding
|
||||||
* Read with timeout
|
* Read with timeout
|
||||||
|
@ -93,12 +87,11 @@ PySimpleGUIWeb runs only on Python 3. Legacy Python (2.7) is not supported.
|
||||||
|
|
||||||
# Running online using repl.it
|
# Running online using repl.it
|
||||||
|
|
||||||
|
This is something truly unique and amazing. You can run your PySimpleGUI code in a web browser on a computer, phone, tablet without having Python installed on that computer. Through the magic of repl.it and Remi you can run PySimpleGUI code anywhere you can run a browser. Not only that, but you can embed these programs into web pages. In fact, this markdown document has one of these programs embedded in it.
|
||||||
|
|
||||||
<figure class="video_container">
|
Here is where the iframe is inserted. You may not see the page in some instances (like on GitHub).
|
||||||
<iframe src="https://repl.it/@PySimpleGUI/PySimpleGUIWeb-Demos?lite=true" frameborder="0" allowfullscreen="true"> </iframe>
|
|
||||||
</figure>
|
|
||||||
|
|
||||||
|
|
||||||
|
Here's your sample program:
|
||||||
|
|
||||||
<iframe height="400px" width="100%" src="https://repl.it/@PySimpleGUI/PySimpleGUIWeb-Demos?lite=true" scrolling="no" frameborder="no" allowtransparency="true" allowfullscreen="true" sandbox="allow-forms allow-pointer-lock allow-popups allow-same-origin allow-scripts allow-modals"></iframe>
|
<iframe height="400px" width="100%" src="https://repl.it/@PySimpleGUI/PySimpleGUIWeb-Demos?lite=true" scrolling="no" frameborder="no" allowtransparency="true" allowfullscreen="true" sandbox="allow-forms allow-pointer-lock allow-popups allow-same-origin allow-scripts allow-modals"></iframe>
|
||||||
|
|
||||||
|
@ -161,6 +154,16 @@ New features
|
||||||
* Enabled exceptions during packing operation
|
* Enabled exceptions during packing operation
|
||||||
* New test harness exercises all element types
|
* New test harness exercises all element types
|
||||||
|
|
||||||
|
## 0.5.0 PySimpleGUIWeb 1-Feb-2019
|
||||||
|
|
||||||
|
* New default font size. Was Helvetica 10, now Helvetica 15
|
||||||
|
* InputText Element single character events working! (Finally)
|
||||||
|
* Listbox simple Update (values only)
|
||||||
|
* Column Element! New!
|
||||||
|
* Column element experimental justification setting doesn't work yet
|
||||||
|
* Element background colors now picked up from container, not top level window
|
||||||
|
* Autosize Text
|
||||||
|
* Autosize Button Text
|
||||||
|
|
||||||
|
|
||||||
# Design
|
# Design
|
||||||
|
@ -173,6 +176,6 @@ New features
|
||||||
|
|
||||||
# Acknowledgments
|
# Acknowledgments
|
||||||
<!--stackedit_data:
|
<!--stackedit_data:
|
||||||
eyJoaXN0b3J5IjpbLTEwNTcxMDM2NDMsMTIxMzM1MjYzNiwtMT
|
eyJoaXN0b3J5IjpbNDc1ODY1Njc2LC0xMDU3MTAzNjQzLDEyMT
|
||||||
E2MDY4NDMzOV19
|
MzNTI2MzYsLTExNjA2ODQzMzldfQ==
|
||||||
-->
|
-->
|
Loading…
Reference in New Issue