More demo programs updates 🤦♂️ wow.....I thought for sure these were checked in....
This commit is contained in:
parent
cfe2c96a1f
commit
430d1bc77f
|
@ -85,7 +85,8 @@ def main(location):
|
||||||
margins=(1,1),
|
margins=(1,1),
|
||||||
element_padding=(0,0),
|
element_padding=(0,0),
|
||||||
border_depth=0,
|
border_depth=0,
|
||||||
location=location)
|
location=location,
|
||||||
|
right_click_menu=[[''], ['Edit Me', 'Exit',]])
|
||||||
|
|
||||||
|
|
||||||
graphs = []
|
graphs = []
|
||||||
|
@ -95,12 +96,15 @@ def main(location):
|
||||||
window[('-TXT-', (rows, cols))],
|
window[('-TXT-', (rows, cols))],
|
||||||
0, colors[(rows*NUM_COLS+cols)%len(colors)])]
|
0, colors[(rows*NUM_COLS+cols)%len(colors)])]
|
||||||
|
|
||||||
|
|
||||||
# ---------------- main loop ----------------
|
# ---------------- main loop ----------------
|
||||||
while True :
|
while True :
|
||||||
# --------- Read and update window once every Polling Frequency --------
|
# --------- Read and update window once every Polling Frequency --------
|
||||||
event, values = window.read(timeout=POLL_FREQUENCY)
|
event, values = window.read(timeout=POLL_FREQUENCY)
|
||||||
if event in (sg.WIN_CLOSED, 'Exit'): # Be nice and give an exit
|
if event in (sg.WIN_CLOSED, 'Exit'): # Be nice and give an exit
|
||||||
break
|
break
|
||||||
|
elif event == 'Edit Me':
|
||||||
|
sg.execute_editor(__file__)
|
||||||
# read CPU for each core
|
# read CPU for each core
|
||||||
stats = psutil.cpu_percent(percpu=True)
|
stats = psutil.cpu_percent(percpu=True)
|
||||||
|
|
||||||
|
|
|
@ -42,7 +42,7 @@ class Gauge():
|
||||||
: Return
|
: Return
|
||||||
Addition result for number1 and number2.
|
Addition result for number1 and number2.
|
||||||
"""
|
"""
|
||||||
return number1 + number1
|
return number1 + number2
|
||||||
|
|
||||||
def limit(number):
|
def limit(number):
|
||||||
"""
|
"""
|
||||||
|
@ -274,7 +274,7 @@ def main(location):
|
||||||
[sg.T(size=(5, 1), font='Any 20', justification='c', background_color='black', k='-gauge VALUE-')]]
|
[sg.T(size=(5, 1), font='Any 20', justification='c', background_color='black', k='-gauge VALUE-')]]
|
||||||
|
|
||||||
|
|
||||||
window = sg.Window('CPU Usage Widget Square', layout, location=location, no_titlebar=True, grab_anywhere=True, margins=(0, 0), element_padding=(0, 0), alpha_channel=ALPHA, background_color='black', element_justification='c', finalize=True, right_click_menu=sg.MENU_RIGHT_CLICK_EDITME_EXIT)
|
window = sg.Window('CPU Usage Widget Square', layout, location=location, no_titlebar=True, grab_anywhere=True, margins=(0, 0), element_padding=(0, 0), alpha_channel=ALPHA, background_color='black', element_justification='c', finalize=True, right_click_menu=sg.MENU_RIGHT_CLICK_EDITME_EXIT, enable_close_attempted_event=True)
|
||||||
|
|
||||||
gauge = Gauge(pointer_color=sg.theme_text_color(), clock_color=sg.theme_text_color(), major_tick_color=sg.theme_text_color(),
|
gauge = Gauge(pointer_color=sg.theme_text_color(), clock_color=sg.theme_text_color(), major_tick_color=sg.theme_text_color(),
|
||||||
minor_tick_color=sg.theme_input_background_color(), pointer_outer_color=sg.theme_text_color(), major_tick_start_radius=45,
|
minor_tick_color=sg.theme_input_background_color(), pointer_outer_color=sg.theme_text_color(), major_tick_start_radius=45,
|
||||||
|
@ -293,7 +293,8 @@ def main(location):
|
||||||
# ----------- update the graphics and text in the window ------------
|
# ----------- update the graphics and text in the window ------------
|
||||||
# update the window, wait for a while, then check for exit
|
# update the window, wait for a while, then check for exit
|
||||||
event, values = window.read(timeout=UPDATE_FREQUENCY_MILLISECONDS)
|
event, values = window.read(timeout=UPDATE_FREQUENCY_MILLISECONDS)
|
||||||
if event == sg.WIN_CLOSED or event == 'Exit':
|
if event in (sg.WIN_CLOSE_ATTEMPTED_EVENT, 'Exit'):
|
||||||
|
sg.user_settings_set_entry('-location-', window.current_location()) # The line of code to save the position before exiting
|
||||||
break
|
break
|
||||||
if event == 'Edit Me':
|
if event == 'Edit Me':
|
||||||
sg.execute_editor(__file__)
|
sg.execute_editor(__file__)
|
||||||
|
@ -306,5 +307,5 @@ if __name__ == '__main__':
|
||||||
location = sys.argv[1].split(',')
|
location = sys.argv[1].split(',')
|
||||||
location = (int(location[0]), int(location[1]))
|
location = (int(location[0]), int(location[1]))
|
||||||
else:
|
else:
|
||||||
location = (None, None)
|
location = sg.user_settings_get_entry('-location-', (None, None))
|
||||||
main(location)
|
main(location)
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
import PySimpleGUI as sg
|
import PySimpleGUI as sg
|
||||||
import sys
|
import sys
|
||||||
import psutil
|
import psutil
|
||||||
|
import math
|
||||||
|
|
||||||
"""
|
"""
|
||||||
Desktop floating widget - CPU Cores as Gauges
|
Desktop floating widget - CPU Cores as Gauges
|
||||||
|
@ -17,20 +18,17 @@ import psutil
|
||||||
Copyright 2020 PySimpleGUI
|
Copyright 2020 PySimpleGUI
|
||||||
"""
|
"""
|
||||||
|
|
||||||
GRAPH_WIDTH = 120 # each individual graph size in pixels
|
# gsize = (120, 75)
|
||||||
GRAPH_HEIGHT = 40
|
# gsize = (85, 40)
|
||||||
|
gsize = (55, 30)
|
||||||
|
|
||||||
TRANSPARENCY = .8 # how transparent the window looks. 0 = invisible, 1 = normal window
|
TRANSPARENCY = .8 # how transparent the window looks. 0 = invisible, 1 = normal window
|
||||||
NUM_COLS = 4
|
NUM_COLS = 4
|
||||||
POLL_FREQUENCY = 2000 # how often to update graphs in milliseconds
|
POLL_FREQUENCY = 1500 # how often to update graphs in milliseconds
|
||||||
|
|
||||||
colors = ('#23a0a0', '#56d856', '#be45be', '#5681d8', '#d34545', '#BE7C29')
|
colors = ('#23a0a0', '#56d856', '#be45be', '#5681d8', '#d34545', '#BE7C29')
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
import math
|
|
||||||
import random
|
|
||||||
|
|
||||||
|
|
||||||
class Gauge():
|
class Gauge():
|
||||||
def mapping(func, sequence, *argc):
|
def mapping(func, sequence, *argc):
|
||||||
"""
|
"""
|
||||||
|
@ -56,7 +54,7 @@ class Gauge():
|
||||||
: Return
|
: Return
|
||||||
Addition result for number1 and number2.
|
Addition result for number1 and number2.
|
||||||
"""
|
"""
|
||||||
return number1 + number1
|
return number1 + number2
|
||||||
|
|
||||||
def limit(number):
|
def limit(number):
|
||||||
"""
|
"""
|
||||||
|
@ -67,7 +65,6 @@ class Gauge():
|
||||||
angel degree in 0 ~ 360, return 0 if number < 0, 360 if number > 360.
|
angel degree in 0 ~ 360, return 0 if number < 0, 360 if number > 360.
|
||||||
"""
|
"""
|
||||||
return max(min(360, number), 0)
|
return max(min(360, number), 0)
|
||||||
|
|
||||||
class Clock():
|
class Clock():
|
||||||
"""
|
"""
|
||||||
Draw background circle or arc
|
Draw background circle or arc
|
||||||
|
@ -134,7 +131,7 @@ class Gauge():
|
||||||
self.figure = []
|
self.figure = []
|
||||||
self.stop_angle = angle
|
self.stop_angle = angle
|
||||||
self.graph_elem = graph_elem
|
self.graph_elem = graph_elem
|
||||||
self.new(degree=angle)
|
self.new(degree=angle, color=pointer_color)
|
||||||
|
|
||||||
def new(self, degree=0, color=None):
|
def new(self, degree=0, color=None):
|
||||||
"""
|
"""
|
||||||
|
@ -224,10 +221,10 @@ class Gauge():
|
||||||
All angles defined as count clockwise from negative x-axis.
|
All angles defined as count clockwise from negative x-axis.
|
||||||
Should create instance of clock, pointer, minor tick and major tick first.
|
Should create instance of clock, pointer, minor tick and major tick first.
|
||||||
"""
|
"""
|
||||||
def __init__(self, center=(0, 0), start_angle=0, stop_angle=180, major_tick_width=5, minor_tick_width=2,major_tick_start_radius=90, major_tick_stop_radius=100, major_tick_step=30, clock_radius=100, pointer_line_width=5, pointer_inner_radius=10, pointer_outer_radius=75, pointer_color='white', pointer_origin_color='black', pointer_outer_color='white', pointer_angle=0, degree=0, clock_color='white', major_tick_color='black', minor_tick_color='black', minor_tick_start_radius=90, minor_tick_stop_radius=100, graph_elem=None):
|
def __init__(self, center=(0, 0), start_angle=0, stop_angle=180, major_tick_width=5, minor_tick_width=2,major_tick_start_radius=90, major_tick_stop_radius=100, minor_tick_step=5, major_tick_step=30, clock_radius=100, pointer_line_width=5, pointer_inner_radius=10, pointer_outer_radius=75, pointer_color='white', pointer_origin_color='black', pointer_outer_color='white', pointer_angle=0, degree=0, clock_color='white', major_tick_color='black', minor_tick_color='black', minor_tick_start_radius=90, minor_tick_stop_radius=100, graph_elem=None):
|
||||||
|
|
||||||
self.clock = Gauge.Clock(start_angle=start_angle, stop_angle=stop_angle, fill_color=clock_color, radius=clock_radius, graph_elem=graph_elem)
|
self.clock = Gauge.Clock(start_angle=start_angle, stop_angle=stop_angle, fill_color=clock_color, radius=clock_radius, graph_elem=graph_elem)
|
||||||
self.minor_tick = Gauge.Tick(start_angle=start_angle, stop_angle=stop_angle, line_width=minor_tick_width, line_color=minor_tick_color, start_radius=minor_tick_start_radius, stop_radius=minor_tick_stop_radius, graph_elem=graph_elem)
|
self.minor_tick = Gauge.Tick(start_angle=start_angle, stop_angle=stop_angle, line_width=minor_tick_width, line_color=minor_tick_color, start_radius=minor_tick_start_radius, stop_radius=minor_tick_stop_radius, graph_elem=graph_elem, step=minor_tick_step)
|
||||||
self.major_tick = Gauge.Tick(start_angle=start_angle, stop_angle=stop_angle, line_width=major_tick_width, start_radius=major_tick_start_radius, stop_radius=major_tick_stop_radius, step=major_tick_step, line_color=major_tick_color, graph_elem=graph_elem)
|
self.major_tick = Gauge.Tick(start_angle=start_angle, stop_angle=stop_angle, line_width=major_tick_width, start_radius=major_tick_start_radius, stop_radius=major_tick_stop_radius, step=major_tick_step, line_color=major_tick_color, graph_elem=graph_elem)
|
||||||
self.pointer = Gauge.Pointer(angle=pointer_angle, inner_radius=pointer_inner_radius, outer_radius=pointer_outer_radius, pointer_color=pointer_color, outer_color=pointer_outer_color, origin_color=pointer_origin_color, line_width=pointer_line_width, graph_elem=graph_elem)
|
self.pointer = Gauge.Pointer(angle=pointer_angle, inner_radius=pointer_inner_radius, outer_radius=pointer_outer_radius, pointer_color=pointer_color, outer_color=pointer_outer_color, origin_color=pointer_origin_color, line_width=pointer_line_width, graph_elem=graph_elem)
|
||||||
|
|
||||||
|
@ -250,7 +247,7 @@ class Gauge():
|
||||||
if self.pointer:
|
if self.pointer:
|
||||||
self.pointer.move(delta_x, delta_y)
|
self.pointer.move(delta_x, delta_y)
|
||||||
|
|
||||||
def change(self, degree=None, step=1):
|
def change(self, degree=None, step=1, pointer_color=None):
|
||||||
"""
|
"""
|
||||||
Rotation of pointer
|
Rotation of pointer
|
||||||
call it with degree and step to set initial options for rotation.
|
call it with degree and step to set initial options for rotation.
|
||||||
|
@ -266,17 +263,17 @@ class Gauge():
|
||||||
new_degree = now + step
|
new_degree = now + step
|
||||||
if ((step > 0 and new_degree < self.pointer.stop_degree) or
|
if ((step > 0 and new_degree < self.pointer.stop_degree) or
|
||||||
(step < 0 and new_degree > self.pointer.stop_degree)):
|
(step < 0 and new_degree > self.pointer.stop_degree)):
|
||||||
self.pointer.new(degree=new_degree, color='red' if new_degree > 90 else None)
|
self.pointer.new(degree=new_degree, color=pointer_color)
|
||||||
return False
|
return False
|
||||||
else:
|
else:
|
||||||
self.pointer.new(degree=self.pointer.stop_degree, color='red' if self.pointer.stop_degree > 90 else None)
|
self.pointer.new(degree=self.pointer.stop_degree, color=pointer_color)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
# ------------------------------ BEGINNING OF CPU WIDGET GUI CODE ------------------------------
|
||||||
|
|
||||||
# DashGraph does the drawing of each graph
|
# DashGraph does the drawing of each graph
|
||||||
class DashGraph(object):
|
class DashGraph(object):
|
||||||
def __init__(self, graph_elem, text_elem, starting_count, color):
|
def __init__(self, gsize, graph_elem, text_elem, starting_count, color):
|
||||||
self.graph_current_item = 0
|
self.graph_current_item = 0
|
||||||
self.graph_elem = graph_elem # type: sg.Graph
|
self.graph_elem = graph_elem # type: sg.Graph
|
||||||
self.text_elem = text_elem
|
self.text_elem = text_elem
|
||||||
|
@ -285,26 +282,36 @@ class DashGraph(object):
|
||||||
self.color = color
|
self.color = color
|
||||||
self.line_list = [] # list of currently visible lines. Used to delete oild figures
|
self.line_list = [] # list of currently visible lines. Used to delete oild figures
|
||||||
|
|
||||||
self.gauge = Gauge(pointer_color=color, clock_color=color, major_tick_color=color,
|
self.gauge = Gauge(pointer_color=color,
|
||||||
minor_tick_color=color, pointer_outer_color=sg.theme_text_color(), major_tick_start_radius=45,
|
clock_color=color,
|
||||||
minor_tick_start_radius=45, minor_tick_stop_radius=50, major_tick_stop_radius=50, major_tick_step=30, clock_radius=50, pointer_line_width=3,
|
major_tick_color=color,
|
||||||
pointer_inner_radius=10, pointer_outer_radius=50, graph_elem=graph_elem)
|
minor_tick_color=color,
|
||||||
|
pointer_outer_color=color,
|
||||||
|
major_tick_start_radius=gsize[1] - 10,
|
||||||
|
minor_tick_start_radius=gsize[1] - 10,
|
||||||
|
minor_tick_stop_radius=gsize[1] - 5,
|
||||||
|
major_tick_stop_radius=gsize[1] - 5,
|
||||||
|
clock_radius=gsize[1] - 5,
|
||||||
|
pointer_outer_radius=gsize[1] - 5,
|
||||||
|
major_tick_step=30,
|
||||||
|
minor_tick_step=15,
|
||||||
|
pointer_line_width=3,
|
||||||
|
pointer_inner_radius=10,
|
||||||
|
graph_elem=graph_elem)
|
||||||
|
|
||||||
|
|
||||||
self.gauge.change(degree=0)
|
self.gauge.change(degree=0)
|
||||||
|
|
||||||
def graph_percentage_abs(self, value):
|
def graph_percentage_abs(self, value):
|
||||||
if self.gauge.change():
|
if self.gauge.change(pointer_color='red' if value > 50 else None):
|
||||||
new_angle = value*180/100
|
new_angle = value*180/100
|
||||||
self.gauge.change(degree=new_angle, step=new_angle)
|
self.gauge.change(degree=new_angle, step=100, pointer_color='red' if value > 50 else None)
|
||||||
self.gauge.change()
|
self.gauge.change(pointer_color='red' if value > 50 else None)
|
||||||
|
|
||||||
def text_display(self, text):
|
def text_display(self, text):
|
||||||
self.text_elem.update(text)
|
self.text_elem.update(text)
|
||||||
|
|
||||||
def main(location):
|
def main(location):
|
||||||
gsize = (100, 55)
|
|
||||||
|
|
||||||
|
|
||||||
# A couple of "User defined elements" that combine several elements and enable bulk edits
|
# A couple of "User defined elements" that combine several elements and enable bulk edits
|
||||||
def Txt(text, **kwargs):
|
def Txt(text, **kwargs):
|
||||||
return(sg.Text(text, font=('Helvetica 8'), **kwargs))
|
return(sg.Text(text, font=('Helvetica 8'), **kwargs))
|
||||||
|
@ -312,7 +319,7 @@ def main(location):
|
||||||
def GraphColumn(name, key):
|
def GraphColumn(name, key):
|
||||||
layout = [
|
layout = [
|
||||||
[sg.Graph(gsize, (-gsize[0] // 2, 0), (gsize[0] // 2, gsize[1]), key=key+'-GRAPH-')],
|
[sg.Graph(gsize, (-gsize[0] // 2, 0), (gsize[0] // 2, gsize[1]), key=key+'-GRAPH-')],
|
||||||
[sg.T(size=(5, 1), justification='c', font='Courier 14', k=key+'-GAUGE VALUE-')]]
|
[sg.T(size=(5, 1), justification='c', font='Courier 10', k=key+'-GAUGE VALUE-')]]
|
||||||
return sg.Column(layout, pad=(2, 2), element_justification='c')
|
return sg.Column(layout, pad=(2, 2), element_justification='c')
|
||||||
|
|
||||||
num_cores = len(psutil.cpu_percent(percpu=True)) # get the number of cores in the CPU
|
num_cores = len(psutil.cpu_percent(percpu=True)) # get the number of cores in the CPU
|
||||||
|
@ -320,9 +327,7 @@ def main(location):
|
||||||
sg.theme('black')
|
sg.theme('black')
|
||||||
sg.set_options(element_padding=(0,0), margins=(1,1), border_width=0)
|
sg.set_options(element_padding=(0,0), margins=(1,1), border_width=0)
|
||||||
|
|
||||||
# the clever Red X graphic
|
layout = [[ sg.Button(image_data=sg.red_x, button_color=('black', 'black'), key='Exit', tooltip='Closes window'),
|
||||||
red_x = "R0lGODlhEAAQAPeQAIsAAI0AAI4AAI8AAJIAAJUAAJQCApkAAJoAAJ4AAJkJCaAAAKYAAKcAAKcCAKcDA6cGAKgAAKsAAKsCAKwAAK0AAK8AAK4CAK8DAqUJAKULAKwLALAAALEAALIAALMAALMDALQAALUAALYAALcEALoAALsAALsCALwAAL8AALkJAL4NAL8NAKoTAKwbAbEQALMVAL0QAL0RAKsREaodHbkQELMsALg2ALk3ALs+ALE2FbgpKbA1Nbc1Nb44N8AAAMIWAMsvAMUgDMcxAKVABb9NBbVJErFYEq1iMrtoMr5kP8BKAMFLAMxKANBBANFCANJFANFEB9JKAMFcANFZANZcANpfAMJUEMZVEc5hAM5pAMluBdRsANR8AM9YOrdERMpIQs1UVMR5WNt8X8VgYMdlZcxtYtx4YNF/btp9eraNf9qXXNCCZsyLeNSLd8SSecySf82kd9qqc9uBgdyBgd+EhN6JgtSIiNuJieGHhOGLg+GKhOKamty1ste4sNO+ueenp+inp+HHrebGrefKuOPTzejWzera1O7b1vLb2/bl4vTu7fbw7ffx7vnz8f///wAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAEAAJAALAAAAAAQABAAAAjUACEJHEiwYEEABniQKfNFgQCDkATQwAMokEU+PQgUFDAjjR09e/LUmUNnh8aBCcCgUeRmzBkzie6EeQBAoAAMXuA8ciRGCaJHfXzUMCAQgYooWN48anTokR8dQk4sELggBhQrU9Q8evSHiJQgLCIIfMDCSZUjhbYuQkLFCRAMAiOQGGLE0CNBcZYmaRIDLqQFGF60eTRoSxc5jwjhACFWIAgMLtgUocJFy5orL0IQRHAiQgsbRZYswbEhBIiCCH6EiJAhAwQMKU5DjHCi9gnZEHMTDAgAOw=="
|
|
||||||
layout = [[ sg.Button(image_data=red_x, button_color=('black', 'black'), key='Exit', tooltip='Closes window'),
|
|
||||||
sg.Text(' CPU Core Usage')] ]
|
sg.Text(' CPU Core Usage')] ]
|
||||||
|
|
||||||
# add on the graphs
|
# add on the graphs
|
||||||
|
@ -342,10 +347,12 @@ def main(location):
|
||||||
use_default_focus=False,
|
use_default_focus=False,
|
||||||
finalize=True,
|
finalize=True,
|
||||||
location=location,
|
location=location,
|
||||||
right_click_menu=[[''], 'Exit'])
|
right_click_menu=[[''], 'Exit'],
|
||||||
|
# transparent_color='black',
|
||||||
|
)
|
||||||
|
|
||||||
# setup graphs & initial values
|
# setup graphs & initial values
|
||||||
graphs = [DashGraph(window['-CPU-'+str(i)+'-GRAPH-'],
|
graphs = [DashGraph(gsize, window['-CPU-'+str(i)+'-GRAPH-'],
|
||||||
window['-CPU-'+str(i) + '-GAUGE VALUE-'],
|
window['-CPU-'+str(i) + '-GAUGE VALUE-'],
|
||||||
0, colors[i%6]) for i in range(num_cores) ]
|
0, colors[i%6]) for i in range(num_cores) ]
|
||||||
|
|
||||||
|
|
|
@ -4,7 +4,7 @@ import sys
|
||||||
|
|
||||||
"""
|
"""
|
||||||
Another simple Desktop Widget using PySimpleGUI
|
Another simple Desktop Widget using PySimpleGUI
|
||||||
This time a RAM indicator. The Widget is square.
|
This time a CPU Usage indicator. The Widget is square.
|
||||||
The bottom section will be shaded to
|
The bottom section will be shaded to
|
||||||
represent the total amount CPU currently in use.
|
represent the total amount CPU currently in use.
|
||||||
Uses the theme's button color for colors.
|
Uses the theme's button color for colors.
|
||||||
|
@ -18,16 +18,12 @@ GSIZE = (160, 160)
|
||||||
UPDATE_FREQUENCY_MILLISECONDS = 2 * 1000
|
UPDATE_FREQUENCY_MILLISECONDS = 2 * 1000
|
||||||
|
|
||||||
|
|
||||||
def human_size(bytes, units=(' bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB')):
|
|
||||||
""" Returns a human readable string reprentation of bytes"""
|
|
||||||
return str(bytes) + ' ' + units[0] if bytes < 1024 else human_size(bytes >> 10, units[1:])
|
|
||||||
|
|
||||||
|
|
||||||
def main(location):
|
def main(location):
|
||||||
graph = sg.Graph(GSIZE, (0, 0), GSIZE, key='-GRAPH-', enable_events=True)
|
graph = sg.Graph(GSIZE, (0, 0), GSIZE, key='-GRAPH-')
|
||||||
|
|
||||||
layout = [[graph]]
|
layout = [[graph]]
|
||||||
|
|
||||||
window = sg.Window('CPU Usage Widget Square', layout, location=location, no_titlebar=True, grab_anywhere=True, margins=(0, 0), element_padding=(0, 0), alpha_channel=ALPHA, finalize=True, right_click_menu=sg.MENU_RIGHT_CLICK_EDITME_EXIT)
|
window = sg.Window('CPU Usage Widget Square', layout, location=location, no_titlebar=True, grab_anywhere=True, margins=(0, 0), element_padding=(0, 0), alpha_channel=ALPHA, finalize=True, right_click_menu=sg.MENU_RIGHT_CLICK_EDITME_EXIT, enable_close_attempted_event=True)
|
||||||
|
|
||||||
text_id2 = graph.draw_text(f'CPU', (GSIZE[0] // 2, GSIZE[1] // 4), font='Any 20', text_location=sg.TEXT_LOCATION_CENTER, color=sg.theme_button_color()[0])
|
text_id2 = graph.draw_text(f'CPU', (GSIZE[0] // 2, GSIZE[1] // 4), font='Any 20', text_location=sg.TEXT_LOCATION_CENTER, color=sg.theme_button_color()[0])
|
||||||
|
|
||||||
|
@ -45,7 +41,8 @@ def main(location):
|
||||||
|
|
||||||
# update the window, wait for a while, then check for exit
|
# update the window, wait for a while, then check for exit
|
||||||
event, values = window.read(timeout=UPDATE_FREQUENCY_MILLISECONDS)
|
event, values = window.read(timeout=UPDATE_FREQUENCY_MILLISECONDS)
|
||||||
if event == sg.WIN_CLOSED or event == 'Exit':
|
if event in (sg.WIN_CLOSE_ATTEMPTED_EVENT, 'Exit'):
|
||||||
|
sg.user_settings_set_entry('-location-', window.current_location()) # The line of code to save the position before exiting
|
||||||
break
|
break
|
||||||
if event == 'Edit Me':
|
if event == 'Edit Me':
|
||||||
sg.execute_editor(__file__)
|
sg.execute_editor(__file__)
|
||||||
|
@ -62,5 +59,5 @@ if __name__ == '__main__':
|
||||||
location = sys.argv[1].split(',')
|
location = sys.argv[1].split(',')
|
||||||
location = (int(location[0]), int(location[1]))
|
location = (int(location[0]), int(location[1]))
|
||||||
else:
|
else:
|
||||||
location = (None, None)
|
location = sg.user_settings_get_entry('-location-', (None, None))
|
||||||
main(location)
|
main(location)
|
||||||
|
|
|
@ -24,7 +24,6 @@ g_cpu_percent = 0
|
||||||
g_procs = None
|
g_procs = None
|
||||||
g_exit = False
|
g_exit = False
|
||||||
|
|
||||||
|
|
||||||
def CPU_thread(args):
|
def CPU_thread(args):
|
||||||
global g_interval, g_cpu_percent, g_procs, g_exit
|
global g_interval, g_cpu_percent, g_procs, g_exit
|
||||||
|
|
||||||
|
@ -53,7 +52,7 @@ def main(location):
|
||||||
]
|
]
|
||||||
|
|
||||||
window = sg.Window('Top CPU Processes', layout,
|
window = sg.Window('Top CPU Processes', layout,
|
||||||
no_titlebar=True, keep_on_top=True,location=location, use_default_focus=False, alpha_channel=.8, grab_anywhere=True)
|
no_titlebar=True, keep_on_top=True,location=location, use_default_focus=False, alpha_channel=.8, grab_anywhere=True, right_click_menu=sg.MENU_RIGHT_CLICK_EDITME_EXIT)
|
||||||
|
|
||||||
# start cpu measurement thread
|
# start cpu measurement thread
|
||||||
thread = Thread(target=CPU_thread, args=(None,))
|
thread = Thread(target=CPU_thread, args=(None,))
|
||||||
|
@ -67,9 +66,10 @@ def main(location):
|
||||||
# --------- Do Button Operations --------
|
# --------- Do Button Operations --------
|
||||||
if event in (sg.WIN_CLOSED, 'Exit'):
|
if event in (sg.WIN_CLOSED, 'Exit'):
|
||||||
break
|
break
|
||||||
|
elif event == 'Edit Me':
|
||||||
|
sg.execute_editor(__file__)
|
||||||
timeout_value = int(values['spin']) * 1000 # for now on, use spinner for timeout
|
timeout_value = int(values['spin']) * 1000 # for now on, use spinner for timeout
|
||||||
|
g_interval = int(values['spin'])
|
||||||
cpu_percent = g_cpu_percent
|
cpu_percent = g_cpu_percent
|
||||||
display_string = ''
|
display_string = ''
|
||||||
if g_procs:
|
if g_procs:
|
||||||
|
|
|
@ -50,7 +50,7 @@ class Gauge():
|
||||||
: Return
|
: Return
|
||||||
Addition result for number1 and number2.
|
Addition result for number1 and number2.
|
||||||
"""
|
"""
|
||||||
return number1 + number1
|
return number1 + number2
|
||||||
|
|
||||||
def limit(number):
|
def limit(number):
|
||||||
"""
|
"""
|
||||||
|
|
|
@ -0,0 +1,198 @@
|
||||||
|
import PySimpleGUI as sg
|
||||||
|
import sys
|
||||||
|
from datetime import datetime
|
||||||
|
from datetime import timedelta
|
||||||
|
"""
|
||||||
|
Desktop Widget - Display the date
|
||||||
|
Simple display of the date in the format of:
|
||||||
|
Day of week Day Month Year
|
||||||
|
|
||||||
|
You can change the format by modifying the function get_date_string
|
||||||
|
|
||||||
|
Copyright 2021 PySimpleGUI
|
||||||
|
"""
|
||||||
|
|
||||||
|
ALPHA = 0.9 # Initial alpha until user changes
|
||||||
|
THEME = 'Dark green 3' # Initial theme until user changes
|
||||||
|
refresh_font = title_font = 'Courier 8'
|
||||||
|
main_info_font = sg.user_settings_get_entry('-main info font-', 'Courier 60')
|
||||||
|
|
||||||
|
main_info_size = (12, 1)
|
||||||
|
UPDATE_FREQUENCY_MILLISECONDS = 1000 * 60 * 60 # update every hour by default until set by user
|
||||||
|
|
||||||
|
|
||||||
|
def choose_theme(location, size):
|
||||||
|
"""
|
||||||
|
A window to allow new themes to be tried out.
|
||||||
|
Changes the theme to the newly chosen one and returns theme's name
|
||||||
|
Automaticallyi switches to new theme and saves the setting in user settings file
|
||||||
|
|
||||||
|
:param location: (x,y) location of the Widget's window
|
||||||
|
:type location: Tuple[int, int]
|
||||||
|
:param size: Size in pixels of the Widget's window
|
||||||
|
:type size: Tuple[int, int]
|
||||||
|
:return: The name of the newly selected theme
|
||||||
|
:rtype: None | str
|
||||||
|
"""
|
||||||
|
layout = [[sg.Text('Try a theme')],
|
||||||
|
[sg.Listbox(values=sg.theme_list(), size=(20, 20), key='-LIST-', enable_events=True)],
|
||||||
|
[sg.OK(), sg.Cancel()]]
|
||||||
|
|
||||||
|
window = sg.Window('Look and Feel Browser', layout, location=location)
|
||||||
|
old_theme = sg.theme()
|
||||||
|
while True: # Event Loop
|
||||||
|
event, values = window.read()
|
||||||
|
if event in (sg.WIN_CLOSED, 'Exit', 'OK', 'Cancel'):
|
||||||
|
break
|
||||||
|
sg.theme(values['-LIST-'][0])
|
||||||
|
window.hide()
|
||||||
|
# make at test window to the left of the current one
|
||||||
|
test_window = make_window(location=((location[0] - size[0] * 1.2, location[1])), test_window=True)
|
||||||
|
test_window.read(close=True)
|
||||||
|
window.un_hide()
|
||||||
|
window.close()
|
||||||
|
|
||||||
|
# after choice made, save theme or restore the old one
|
||||||
|
if event == 'OK' and values['-LIST-']:
|
||||||
|
sg.theme(values['-LIST-'][0])
|
||||||
|
sg.user_settings_set_entry('-theme-', values['-LIST-'][0])
|
||||||
|
return values['-LIST-'][0]
|
||||||
|
else:
|
||||||
|
sg.theme(old_theme)
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
def make_window(location, test_window=False):
|
||||||
|
"""
|
||||||
|
Defines the layout and creates the window for the main window
|
||||||
|
If the parm test_window is True, then a simplified, and EASY to close version is shown
|
||||||
|
|
||||||
|
:param location: (x,y) location to create the window
|
||||||
|
:type location: Tuple[int, int]
|
||||||
|
:param test_window: If True, then this is a test window & will close by clicking on it
|
||||||
|
:type test_window: bool
|
||||||
|
:return: newly created window
|
||||||
|
:rtype: sg.Window
|
||||||
|
"""
|
||||||
|
title = sg.user_settings_get_entry('-title-', '')
|
||||||
|
if not test_window:
|
||||||
|
theme = sg.user_settings_get_entry('-theme-', THEME)
|
||||||
|
sg.theme(theme)
|
||||||
|
main_info_font = sg.user_settings_get_entry('-main info font-', 'Courier 60')
|
||||||
|
|
||||||
|
# ------------------- Window Layout -------------------
|
||||||
|
initial_text = get_date_string()
|
||||||
|
if test_window:
|
||||||
|
title_element = sg.Text('Click to close', font=title_font, enable_events=True)
|
||||||
|
right_click_menu = [[''], ['Exit', ]]
|
||||||
|
else:
|
||||||
|
title_element = sg.pin(sg.Text(title, size=(20, 1), font=title_font, justification='c', k='-TITLE-'))
|
||||||
|
right_click_menu = [[''],
|
||||||
|
['Choose Title', 'Edit Me', 'New Theme', 'Save Location', 'Font', 'Refresh', 'Set Refresh Rate', 'Show Refresh Info', 'Hide Refresh Info',
|
||||||
|
'Alpha', [str(x) for x in range(1, 11)], 'Exit', ]]
|
||||||
|
|
||||||
|
|
||||||
|
layout = [[title_element],
|
||||||
|
[sg.Text(initial_text, size=(len(initial_text)+2, 1), font=main_info_font, k='-MAIN INFO-', justification='c', enable_events=test_window)],
|
||||||
|
[sg.pin(
|
||||||
|
sg.Text(size=(15, 2), font=refresh_font, k='-REFRESHED-', justification='c', visible=sg.user_settings_get_entry('-show refresh-', True)))]]
|
||||||
|
|
||||||
|
# ------------------- Window Creation -------------------
|
||||||
|
try:
|
||||||
|
window = sg.Window('Desktop Widget Template', layout, location=location, no_titlebar=True, grab_anywhere=True, margins=(0, 0), element_justification='c', element_padding=(0, 0), alpha_channel=sg.user_settings_get_entry('-alpha-', ALPHA), finalize=True, right_click_menu=right_click_menu, right_click_menu_tearoff=False)
|
||||||
|
except Exception as e:
|
||||||
|
if sg.popup_yes_no('Error creating the window', e, 'Do you want to delete your settings file to fix?') == 'Yes':
|
||||||
|
sg.user_settings_delete_filename()
|
||||||
|
sg.popup('Settings file deleted. Please restart your program.')
|
||||||
|
exit()
|
||||||
|
return window
|
||||||
|
|
||||||
|
def get_date_string():
|
||||||
|
dtime_here = datetime.utcnow() + timedelta(hours=-5)
|
||||||
|
return dtime_here.strftime('%a %d %b %Y')
|
||||||
|
|
||||||
|
|
||||||
|
def main(location):
|
||||||
|
"""
|
||||||
|
Where execution begins
|
||||||
|
The Event Loop lives here, but the window creation is done in another function
|
||||||
|
This is an important design pattern
|
||||||
|
|
||||||
|
:param location: Location to create the main window if one is not found in the user settings
|
||||||
|
:type location: Tuple[int, int]
|
||||||
|
"""
|
||||||
|
|
||||||
|
window = make_window(sg.user_settings_get_entry('-location-', location))
|
||||||
|
|
||||||
|
refresh_frequency = sg.user_settings_get_entry('-fresh frequency-', UPDATE_FREQUENCY_MILLISECONDS)
|
||||||
|
|
||||||
|
while True: # Event Loop
|
||||||
|
# Normally a window.read goes here, but first we're updating the values in the window, then reading it
|
||||||
|
# First update the status information
|
||||||
|
window['-MAIN INFO-'].update(get_date_string())
|
||||||
|
# for debugging show the last update date time
|
||||||
|
if sg.user_settings_get_entry('-title-', 'None') in ('None', 'Hide'):
|
||||||
|
window['-TITLE-'].update(visible=False)
|
||||||
|
else:
|
||||||
|
window['-TITLE-'].update(sg.user_settings_get_entry('-title-', 'None'),visible=True)
|
||||||
|
window['-REFRESHED-'].update(datetime.now().strftime("%m/%d/%Y\n%I:%M:%S %p"))
|
||||||
|
|
||||||
|
# -------------- Start of normal event loop --------------
|
||||||
|
event, values = window.read(timeout=refresh_frequency)
|
||||||
|
print(event, values)
|
||||||
|
if event in (sg.WIN_CLOSED, 'Exit'): # standard exit test... ALWAYS do this
|
||||||
|
break
|
||||||
|
if event == 'Edit Me':
|
||||||
|
sg.execute_editor(__file__)
|
||||||
|
elif event == 'Choose Title':
|
||||||
|
new_title = sg.popup_get_text('Choose a title for your Widget\nEnter None if you do not want anything displayed', location=window.current_location())
|
||||||
|
if new_title is not None:
|
||||||
|
if new_title in ('None', 'Hide'):
|
||||||
|
window['-TITLE-'].update(visible=False)
|
||||||
|
else:
|
||||||
|
window['-TITLE-'].update(new_title, visible=True)
|
||||||
|
sg.user_settings_set_entry('-title-', new_title)
|
||||||
|
elif event == 'Show Refresh Info':
|
||||||
|
window['-REFRESHED-'].update(visible=True)
|
||||||
|
sg.user_settings_set_entry('-show refresh-', True)
|
||||||
|
elif event == 'Save Location':
|
||||||
|
sg.user_settings_set_entry('-location-', window.current_location())
|
||||||
|
sg.popup_notify(f'Saved your current window location:', window.current_location(), title='Saved Location')
|
||||||
|
elif event == 'Hide Refresh Info':
|
||||||
|
window['-REFRESHED-'].update(visible=False)
|
||||||
|
sg.user_settings_set_entry('-show refresh-', False)
|
||||||
|
elif event in [str(x) for x in range(1, 11)]: # if Alpha Channel was chosen
|
||||||
|
window.set_alpha(int(event) / 10)
|
||||||
|
sg.user_settings_set_entry('-alpha-', int(event) / 10)
|
||||||
|
elif event == 'Set Refresh Rate':
|
||||||
|
choice = sg.popup_get_text('How frequently to update window in seconds? (can be a float)',
|
||||||
|
default_text=sg.user_settings_get_entry('-fresh frequency-', UPDATE_FREQUENCY_MILLISECONDS) / 1000,
|
||||||
|
location=window.current_location())
|
||||||
|
if choice is not None:
|
||||||
|
try:
|
||||||
|
refresh_frequency = float(choice) * 1000 # convert to milliseconds
|
||||||
|
sg.user_settings_set_entry('-fresh frequency-', float(refresh_frequency))
|
||||||
|
except Exception as e:
|
||||||
|
sg.popup_error(f'You entered an incorrect number of seconds: {choice}', f'Error: {e}', location=window.current_location())
|
||||||
|
elif event == 'New Theme':
|
||||||
|
loc = window.current_location()
|
||||||
|
if choose_theme(window.current_location(), window.size) is not None:
|
||||||
|
window.close() # out with the old...
|
||||||
|
window = make_window(loc) # in with the new
|
||||||
|
elif event == 'Font':
|
||||||
|
font = sg.popup_get_text('Enter font string using PySimpleGUI font format (e.g. courier 70 or courier 70 bold)', default_text=sg.user_settings_get_entry('-main info font-'), keep_on_top=True)
|
||||||
|
if font:
|
||||||
|
sg.user_settings_set_entry('-main info font-', font)
|
||||||
|
loc = window.current_location()
|
||||||
|
_, window = window.close(), make_window(loc)
|
||||||
|
window.close()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
# To start the window at a specific location, get this location on the command line
|
||||||
|
# The location should be in form x,y with no spaces
|
||||||
|
location = (None, None) # assume no location provided
|
||||||
|
if len(sys.argv) > 1:
|
||||||
|
location = sys.argv[1].split(',')
|
||||||
|
location = (int(location[0]), int(location[1]))
|
||||||
|
main(location)
|
|
@ -19,6 +19,7 @@ import datetime
|
||||||
|
|
||||||
ALPHA = 0.9 # Initial alpha until user changes
|
ALPHA = 0.9 # Initial alpha until user changes
|
||||||
THEME = 'Dark green 3' # Initial theme until user changes
|
THEME = 'Dark green 3' # Initial theme until user changes
|
||||||
|
sg.user_settings_filename(filename='DaysUntil.json')
|
||||||
refresh_font = sg.user_settings_get_entry('-refresh font-', 'Courier 8')
|
refresh_font = sg.user_settings_get_entry('-refresh font-', 'Courier 8')
|
||||||
title_font = sg.user_settings_get_entry('-title font-', 'Courier 8')
|
title_font = sg.user_settings_get_entry('-title font-', 'Courier 8')
|
||||||
main_number_font = sg.user_settings_get_entry('-main number font-', 'Courier 70')
|
main_number_font = sg.user_settings_get_entry('-main number font-', 'Courier 70')
|
||||||
|
|
|
@ -0,0 +1,359 @@
|
||||||
|
#!/usr/bin/env python
|
||||||
|
import PySimpleGUI as sg
|
||||||
|
import psutil
|
||||||
|
import sys
|
||||||
|
import math
|
||||||
|
|
||||||
|
"""
|
||||||
|
Desktop "Rainmeter" style widget - Drive usage
|
||||||
|
Requires: psutil
|
||||||
|
Uses a "Gauge" to display drive usage
|
||||||
|
"""
|
||||||
|
|
||||||
|
ALPHA = 0.7
|
||||||
|
THEME = 'black'
|
||||||
|
UPDATE_FREQUENCY_MILLISECONDS = 20 * 1000
|
||||||
|
|
||||||
|
BAR_COLORS = ('#23a0a0', '#56d856', '#be45be', '#5681d8', '#d34545', '#BE7C29')
|
||||||
|
gsize = (50, 30)
|
||||||
|
|
||||||
|
class Gauge():
|
||||||
|
def mapping(func, sequence, *argc):
|
||||||
|
"""
|
||||||
|
Map function with extra argument, not for tuple.
|
||||||
|
: Parameters
|
||||||
|
func - function to call.
|
||||||
|
sequence - list for iteration.
|
||||||
|
argc - more arguments for func.
|
||||||
|
: Return
|
||||||
|
list of func(element of sequence, *argc)
|
||||||
|
"""
|
||||||
|
if isinstance(sequence, list):
|
||||||
|
return list(map(lambda i: func(i, *argc), sequence))
|
||||||
|
else:
|
||||||
|
return func(sequence, *argc)
|
||||||
|
|
||||||
|
def add(number1, number2):
|
||||||
|
"""
|
||||||
|
Add two number
|
||||||
|
: Parameter
|
||||||
|
number1 - number to add.
|
||||||
|
numeer2 - number to add.
|
||||||
|
: Return
|
||||||
|
Addition result for number1 and number2.
|
||||||
|
"""
|
||||||
|
return number1 + number1
|
||||||
|
|
||||||
|
def limit(number):
|
||||||
|
"""
|
||||||
|
Limit angle in range 0 ~ 360
|
||||||
|
: Parameter
|
||||||
|
number: angle degree.
|
||||||
|
: Return
|
||||||
|
angel degree in 0 ~ 360, return 0 if number < 0, 360 if number > 360.
|
||||||
|
"""
|
||||||
|
return max(min(360, number), 0)
|
||||||
|
class Clock():
|
||||||
|
"""
|
||||||
|
Draw background circle or arc
|
||||||
|
All angles defined as clockwise from negative x-axis.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, center_x=0, center_y=0, radius=100, start_angle=0,
|
||||||
|
stop_angle=360, fill_color='white', line_color='black', line_width=2, graph_elem=None):
|
||||||
|
|
||||||
|
instance = Gauge.mapping(isinstance, [center_x, center_y, radius, start_angle,
|
||||||
|
stop_angle, line_width], (int, float)) + Gauge.mapping(isinstance,
|
||||||
|
[fill_color, line_color], str)
|
||||||
|
if False in instance:
|
||||||
|
raise ValueError
|
||||||
|
start_angle, stop_angle = Gauge.limit(start_angle), Gauge.limit(stop_angle)
|
||||||
|
self.all = [center_x, center_y, radius, start_angle, stop_angle,
|
||||||
|
fill_color, line_color, line_width]
|
||||||
|
self.figure = []
|
||||||
|
self.graph_elem = graph_elem
|
||||||
|
self.new()
|
||||||
|
|
||||||
|
def new(self):
|
||||||
|
"""
|
||||||
|
Draw Arc or circle
|
||||||
|
"""
|
||||||
|
x, y, r, start, stop, fill, line, width = self.all
|
||||||
|
start, stop = (180 - start, 180 - stop) if stop < start else (180 - stop, 180 - start)
|
||||||
|
if start == stop % 360:
|
||||||
|
self.figure.append(self.graph_elem.DrawCircle((x, y), r, fill_color=fill,
|
||||||
|
line_color=line, line_width=width))
|
||||||
|
else:
|
||||||
|
self.figure.append(self.graph_elem.DrawArc((x - r, y + r), (x + r, y - r), stop - start,
|
||||||
|
start, style='arc', arc_color=fill))
|
||||||
|
|
||||||
|
def move(self, delta_x, delta_y):
|
||||||
|
"""
|
||||||
|
Move circle or arc in clock by delta x, delta y
|
||||||
|
"""
|
||||||
|
if False in Gauge.mapping(isinstance, [delta_x, delta_y], (int, float)):
|
||||||
|
raise ValueError
|
||||||
|
self.all[0] += delta_x
|
||||||
|
self.all[1] += delta_y
|
||||||
|
for figure in self.figure:
|
||||||
|
self.graph_elem.MoveFigure(figure, delta_x, delta_y)
|
||||||
|
|
||||||
|
class Pointer():
|
||||||
|
"""
|
||||||
|
Draw pointer of clock
|
||||||
|
All angles defined as clockwise from negative x-axis.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, center_x=0, center_y=0, angle=0, inner_radius=20,
|
||||||
|
outer_radius=80, outer_color='white', pointer_color='blue',
|
||||||
|
origin_color='black', line_width=2, graph_elem=None):
|
||||||
|
|
||||||
|
instance = Gauge.mapping(isinstance, [center_x, center_y, angle, inner_radius,
|
||||||
|
outer_radius, line_width], (int, float)) + Gauge.mapping(isinstance,
|
||||||
|
[outer_color, pointer_color, origin_color], str)
|
||||||
|
if False in instance:
|
||||||
|
raise ValueError
|
||||||
|
|
||||||
|
self.all = [center_x, center_y, angle, inner_radius, outer_radius,
|
||||||
|
outer_color, pointer_color, origin_color, line_width]
|
||||||
|
self.figure = []
|
||||||
|
self.stop_angle = angle
|
||||||
|
self.graph_elem = graph_elem
|
||||||
|
self.new(degree=angle, color=pointer_color)
|
||||||
|
|
||||||
|
def new(self, degree=0, color=None):
|
||||||
|
"""
|
||||||
|
Draw new pointer by angle, erase old pointer if exist
|
||||||
|
degree defined as clockwise from negative x-axis.
|
||||||
|
"""
|
||||||
|
(center_x, center_y, angle, inner_radius, outer_radius,
|
||||||
|
outer_color, pointer_color, origin_color, line_width) = self.all
|
||||||
|
pointer_color = color or pointer_color
|
||||||
|
if self.figure != []:
|
||||||
|
for figure in self.figure:
|
||||||
|
self.graph_elem.DeleteFigure(figure)
|
||||||
|
self.figure = []
|
||||||
|
d = degree - 90
|
||||||
|
self.all[2] = degree
|
||||||
|
dx1 = int(2 * inner_radius * math.sin(d / 180 * math.pi))
|
||||||
|
dy1 = int(2 * inner_radius * math.cos(d / 180 * math.pi))
|
||||||
|
dx2 = int(outer_radius * math.sin(d / 180 * math.pi))
|
||||||
|
dy2 = int(outer_radius * math.cos(d / 180 * math.pi))
|
||||||
|
self.figure.append(self.graph_elem.DrawLine((center_x - dx1, center_y - dy1),
|
||||||
|
(center_x + dx2, center_y + dy2),
|
||||||
|
color=pointer_color, width=line_width))
|
||||||
|
self.figure.append(self.graph_elem.DrawCircle((center_x, center_y), inner_radius,
|
||||||
|
fill_color=origin_color, line_color=outer_color, line_width=line_width))
|
||||||
|
|
||||||
|
def move(self, delta_x, delta_y):
|
||||||
|
"""
|
||||||
|
Move pointer with delta x and delta y
|
||||||
|
"""
|
||||||
|
if False in Gauge.mapping(isinstance, [delta_x, delta_y], (int, float)):
|
||||||
|
raise ValueError
|
||||||
|
self.all[:2] = [self.all[0] + delta_x, self.all[1] + delta_y]
|
||||||
|
for figure in self.figure:
|
||||||
|
self.graph_elem.MoveFigure(figure, delta_x, delta_y)
|
||||||
|
|
||||||
|
class Tick():
|
||||||
|
"""
|
||||||
|
Create tick on click for minor tick, also for major tick
|
||||||
|
All angles defined as clockwise from negative x-axis.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, center_x=0, center_y=0, start_radius=90, stop_radius=100,
|
||||||
|
start_angle=0, stop_angle=360, step=6, line_color='black', line_width=2, graph_elem=None):
|
||||||
|
|
||||||
|
instance = Gauge.mapping(isinstance, [center_x, center_y, start_radius,
|
||||||
|
stop_radius, start_angle, stop_angle, step, line_width],
|
||||||
|
(int, float)) + [Gauge.mapping(isinstance, line_color, (list, str))]
|
||||||
|
if False in instance:
|
||||||
|
raise ValueError
|
||||||
|
start_angle, stop_angle = Gauge.limit(start_angle), Gauge.limit(stop_angle)
|
||||||
|
self.all = [center_x, center_y, start_radius, stop_radius,
|
||||||
|
start_angle, stop_angle, step, line_color, line_width]
|
||||||
|
self.figure = []
|
||||||
|
self.graph_elem = graph_elem
|
||||||
|
|
||||||
|
self.new()
|
||||||
|
|
||||||
|
def new(self):
|
||||||
|
"""
|
||||||
|
Draw ticks on clock
|
||||||
|
"""
|
||||||
|
(x, y, start_radius, stop_radius, start_angle, stop_angle, step,
|
||||||
|
line_color, line_width) = self.all
|
||||||
|
start_angle, stop_angle = (180 - start_angle, 180 - stop_angle
|
||||||
|
) if stop_angle < start_angle else (180 - stop_angle, 180 - start_angle)
|
||||||
|
for i in range(start_angle, stop_angle + 1, step):
|
||||||
|
start_x = x + start_radius * math.cos(i / 180 * math.pi)
|
||||||
|
start_y = y + start_radius * math.sin(i / 180 * math.pi)
|
||||||
|
stop_x = x + stop_radius * math.cos(i / 180 * math.pi)
|
||||||
|
stop_y = y + stop_radius * math.sin(i / 180 * math.pi)
|
||||||
|
self.figure.append(self.graph_elem.DrawLine((start_x, start_y),
|
||||||
|
(stop_x, stop_y), color=line_color, width=line_width))
|
||||||
|
|
||||||
|
def move(self, delta_x, delta_y):
|
||||||
|
"""
|
||||||
|
Move ticks by delta x and delta y
|
||||||
|
"""
|
||||||
|
if False in Gauge.mapping(isinstance, [delta_x, delta_y], (int, float)):
|
||||||
|
raise ValueError
|
||||||
|
self.all[0] += delta_x
|
||||||
|
self.all[1] += delta_y
|
||||||
|
for figure in self.figure:
|
||||||
|
self.graph_elem.MoveFigure(figure, delta_x, delta_y)
|
||||||
|
|
||||||
|
"""
|
||||||
|
Create Gauge
|
||||||
|
All angles defined as count clockwise from negative x-axis.
|
||||||
|
Should create instance of clock, pointer, minor tick and major tick first.
|
||||||
|
"""
|
||||||
|
def __init__(self, center=(0, 0), start_angle=0, stop_angle=180, major_tick_width=5, minor_tick_width=2,major_tick_start_radius=90, major_tick_stop_radius=100, minor_tick_step=5, major_tick_step=30, clock_radius=100, pointer_line_width=5, pointer_inner_radius=10, pointer_outer_radius=75, pointer_color='white', pointer_origin_color='black', pointer_outer_color='white', pointer_angle=0, degree=0, clock_color='white', major_tick_color='black', minor_tick_color='black', minor_tick_start_radius=90, minor_tick_stop_radius=100, graph_elem=None):
|
||||||
|
|
||||||
|
self.clock = Gauge.Clock(start_angle=start_angle, stop_angle=stop_angle, fill_color=clock_color, radius=clock_radius, graph_elem=graph_elem)
|
||||||
|
self.minor_tick = Gauge.Tick(start_angle=start_angle, stop_angle=stop_angle, line_width=minor_tick_width, line_color=minor_tick_color, start_radius=minor_tick_start_radius, stop_radius=minor_tick_stop_radius, graph_elem=graph_elem, step=minor_tick_step)
|
||||||
|
self.major_tick = Gauge.Tick(start_angle=start_angle, stop_angle=stop_angle, line_width=major_tick_width, start_radius=major_tick_start_radius, stop_radius=major_tick_stop_radius, step=major_tick_step, line_color=major_tick_color, graph_elem=graph_elem)
|
||||||
|
self.pointer = Gauge.Pointer(angle=pointer_angle, inner_radius=pointer_inner_radius, outer_radius=pointer_outer_radius, pointer_color=pointer_color, outer_color=pointer_outer_color, origin_color=pointer_origin_color, line_width=pointer_line_width, graph_elem=graph_elem)
|
||||||
|
|
||||||
|
self.center_x, self.center_y = self.center = center
|
||||||
|
self.degree = degree
|
||||||
|
self.dx = self.dy = 1
|
||||||
|
|
||||||
|
def move(self, delta_x, delta_y):
|
||||||
|
"""
|
||||||
|
Move gauge to move all componenets in gauge.
|
||||||
|
"""
|
||||||
|
self.center_x, self.center_y =self.center = (
|
||||||
|
self.center_x+delta_x, self.center_y+delta_y)
|
||||||
|
if self.clock:
|
||||||
|
self.clock.move(delta_x, delta_y)
|
||||||
|
if self.minor_tick:
|
||||||
|
self.minor_tick.move(delta_x, delta_y)
|
||||||
|
if self.major_tick:
|
||||||
|
self.major_tick.move(delta_x, delta_y)
|
||||||
|
if self.pointer:
|
||||||
|
self.pointer.move(delta_x, delta_y)
|
||||||
|
|
||||||
|
def change(self, degree=None, step=1, pointer_color=None):
|
||||||
|
"""
|
||||||
|
Rotation of pointer
|
||||||
|
call it with degree and step to set initial options for rotation.
|
||||||
|
Without any option to start rotation.
|
||||||
|
"""
|
||||||
|
if self.pointer:
|
||||||
|
if degree != None:
|
||||||
|
self.pointer.stop_degree = degree
|
||||||
|
self.pointer.step = step if self.pointer.all[2] < degree else -step
|
||||||
|
return True
|
||||||
|
now = self.pointer.all[2]
|
||||||
|
step = self.pointer.step
|
||||||
|
new_degree = now + step
|
||||||
|
if ((step > 0 and new_degree < self.pointer.stop_degree) or
|
||||||
|
(step < 0 and new_degree > self.pointer.stop_degree)):
|
||||||
|
self.pointer.new(degree=new_degree, color=pointer_color)
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
self.pointer.new(degree=self.pointer.stop_degree, color=pointer_color)
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def human_size(bytes, units=(' bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB')):
|
||||||
|
""" Returns a human readable string reprentation of bytes"""
|
||||||
|
return str(bytes) + ' ' + units[0] if bytes < 1024 else human_size(bytes >> 10, units[1:])
|
||||||
|
|
||||||
|
|
||||||
|
def update_window(window):
|
||||||
|
particians = psutil.disk_partitions()
|
||||||
|
for count, part in enumerate(particians):
|
||||||
|
mount = part[0]
|
||||||
|
try:
|
||||||
|
usage = psutil.disk_usage(mount)
|
||||||
|
window[('-NAME-', mount)].update(mount)
|
||||||
|
# window[('-PROG-', mount)].update_bar(int(usage.percent))
|
||||||
|
window[('-%-', mount)].update(f'{usage.percent}%')
|
||||||
|
window[('-STATS-', mount)].update(f'{human_size(usage.used)} / {human_size(usage.total)} = {human_size(usage.free)} free')
|
||||||
|
gauge = Gauge(pointer_color=window[('-GRAPH-', mount)].metadata,
|
||||||
|
clock_color=window[('-GRAPH-', mount)].metadata,
|
||||||
|
major_tick_color=sg.theme_input_background_color(),
|
||||||
|
minor_tick_color=sg.theme_input_text_color(),
|
||||||
|
pointer_outer_color=sg.theme_input_background_color(),
|
||||||
|
major_tick_start_radius=gsize[1] - 10,
|
||||||
|
minor_tick_start_radius=gsize[1] - 10,
|
||||||
|
minor_tick_stop_radius=gsize[1] - 5,
|
||||||
|
major_tick_stop_radius=gsize[1] - 5,
|
||||||
|
clock_radius=gsize[1] - 5,
|
||||||
|
pointer_outer_radius=gsize[1] - 5,
|
||||||
|
major_tick_step=30,
|
||||||
|
minor_tick_step=15,
|
||||||
|
pointer_line_width=3,
|
||||||
|
pointer_inner_radius=10,
|
||||||
|
graph_elem=window[('-GRAPH-', mount)])
|
||||||
|
gauge.change(degree=0)
|
||||||
|
gauge.change(degree=180 * usage.percent / 100, step=180)
|
||||||
|
gauge.change()
|
||||||
|
except KeyError as e: # A key error means a new drive was added
|
||||||
|
print('Got a key error, so a new drive was added. Window will restart')
|
||||||
|
return False
|
||||||
|
except BaseException as e:
|
||||||
|
print(e)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
def create_window(location):
|
||||||
|
layout = [[sg.Text('Drive Status', font='Any 16')]]
|
||||||
|
|
||||||
|
# Add a row for every partician that has a bar graph and text stats
|
||||||
|
particians = psutil.disk_partitions()
|
||||||
|
for count, part in enumerate(particians):
|
||||||
|
mount = part[0]
|
||||||
|
try:
|
||||||
|
bar_color = sg.theme_progress_bar_color()
|
||||||
|
this_color = BAR_COLORS[count % len(BAR_COLORS)]
|
||||||
|
usage = psutil.disk_usage(mount)
|
||||||
|
stats_info = f'{human_size(usage.used)} / {human_size(usage.total)} = {human_size(usage.free)} free'
|
||||||
|
layout += [[sg.Text(mount, size=(3, 1), key=('-NAME-', mount)),
|
||||||
|
sg.Graph(gsize, (-gsize[0] // 2, 0), (gsize[0] // 2, gsize[1]), key=('-GRAPH-', mount), metadata=this_color),
|
||||||
|
# sg.ProgressBar(100, 'h', size=(10, 15), key=('-PROG-', mount), bar_color=(this_color, bar_color[1])),
|
||||||
|
sg.Text(f'{usage.percent}%', size=(6, 1), key=('-%-', mount)), sg.T(stats_info, size=(30, 1), key=('-STATS-', mount)),
|
||||||
|
]]
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
layout += [[sg.Text('Refresh', font='Any 8', key='-REFRESH-', enable_events=True), sg.Text('❎', enable_events=True, key='Exit Text')]]
|
||||||
|
|
||||||
|
# ---------------- Create Window ----------------
|
||||||
|
window = sg.Window('Drive Status Widget', layout, location=location, keep_on_top=True, grab_anywhere=True, no_titlebar=True, alpha_channel=ALPHA, use_default_focus=False,
|
||||||
|
finalize=True)
|
||||||
|
return window
|
||||||
|
|
||||||
|
def main(location):
|
||||||
|
# Turn off the popups because key errors are normal in this program.
|
||||||
|
# Will get a key error is a new drive is added. Want to get the key error as an exception.
|
||||||
|
sg.set_options(suppress_error_popups=True, suppress_raise_key_errors=False, suppress_key_guessing=True)
|
||||||
|
sg.theme(THEME)
|
||||||
|
|
||||||
|
window = create_window(location)
|
||||||
|
|
||||||
|
update_window(window) # sets the progress bars
|
||||||
|
|
||||||
|
# ---------------- Event Loop ----------------
|
||||||
|
while True:
|
||||||
|
event, values = window.read(timeout=UPDATE_FREQUENCY_MILLISECONDS)
|
||||||
|
if event == sg.WIN_CLOSED or event.startswith('Exit'):
|
||||||
|
break
|
||||||
|
if not update_window(window):
|
||||||
|
window.close()
|
||||||
|
window = create_window(location)
|
||||||
|
update_window(window)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
if len(sys.argv) > 1:
|
||||||
|
location = sys.argv[1].split(',')
|
||||||
|
location = (int(location[0]), int(location[1]))
|
||||||
|
else:
|
||||||
|
location = (None, None)
|
||||||
|
main(location)
|
||||||
|
|
|
@ -150,6 +150,8 @@ def main():
|
||||||
while True:
|
while True:
|
||||||
event, values = window.read(timeout=1000*60*60) # wake every hour
|
event, values = window.read(timeout=1000*60*60) # wake every hour
|
||||||
if event == sg.WIN_CLOSED or event == 'Exit':
|
if event == sg.WIN_CLOSED or event == 'Exit':
|
||||||
|
if event == 'Exit':
|
||||||
|
sg.user_settings_set_entry('-location-', window.current_location())
|
||||||
break
|
break
|
||||||
if event == 'Add Package':
|
if event == 'Add Package':
|
||||||
window.metadata += 1
|
window.metadata += 1
|
||||||
|
|
|
@ -121,7 +121,7 @@ def main():
|
||||||
if new_count is not None:
|
if new_count is not None:
|
||||||
counter = int(new_count)
|
counter = int(new_count)
|
||||||
elif event == 'Choose Title':
|
elif event == 'Choose Title':
|
||||||
new_title = sg.popup_get_text('Choose a title for your counter', default_text=sg.user_settings_get_entry('-title-', ''), location=window.current_location(), keep_on_top=True)
|
new_title = sg.popup_get_text('Choose a title for your counter', default_text=sg.user_settings_get_entry('-title-', ''), location=window.current_location(), )
|
||||||
if new_title is not None:
|
if new_title is not None:
|
||||||
window['-TITLE-'].update(new_title)
|
window['-TITLE-'].update(new_title)
|
||||||
sg.user_settings_set_entry('-title-', new_title)
|
sg.user_settings_set_entry('-title-', new_title)
|
||||||
|
@ -134,7 +134,7 @@ def main():
|
||||||
# this is result of hacking code down to 99 lines in total. Not tried it before. Interesting test.
|
# this is result of hacking code down to 99 lines in total. Not tried it before. Interesting test.
|
||||||
_, window = window.close(), make_window(loc)
|
_, window = window.close(), make_window(loc)
|
||||||
elif event == 'Set Main Font':
|
elif event == 'Set Main Font':
|
||||||
font = sg.popup_get_text('Main Information Font and Size (e.g. courier 70)', default_text=sg.user_settings_get_entry('-main number font-'), keep_on_top=True, location=window.current_location())
|
font = sg.popup_get_text('Main Information Font and Size (e.g. courier 70)', default_text=sg.user_settings_get_entry('-main number font-'), keep_on_top=False, location=window.current_location())
|
||||||
if font:
|
if font:
|
||||||
sg.user_settings_set_entry('-main number font-', font)
|
sg.user_settings_set_entry('-main number font-', font)
|
||||||
_, window = window.close(), make_window(loc)
|
_, window = window.close(), make_window(loc)
|
||||||
|
@ -164,7 +164,7 @@ def main():
|
||||||
if sound_file is not None:
|
if sound_file is not None:
|
||||||
sg.user_settings_set_entry('-sound file-', sound_file)
|
sg.user_settings_set_entry('-sound file-', sound_file)
|
||||||
elif event =='Show Settings':
|
elif event =='Show Settings':
|
||||||
sg.Print(sg.UserSettings._default_for_function_interface)
|
sg.popup_scrolled(sg.UserSettings._default_for_function_interface, location=window.current_location())
|
||||||
|
|
||||||
sg.user_settings_set_entry('-counter-', counter)
|
sg.user_settings_set_entry('-counter-', counter)
|
||||||
|
|
||||||
|
@ -172,4 +172,5 @@ def main():
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
sg.set_options(keep_on_top=True)
|
||||||
main()
|
main()
|
|
@ -51,7 +51,7 @@ class Gauge():
|
||||||
: Return
|
: Return
|
||||||
Addition result for number1 and number2.
|
Addition result for number1 and number2.
|
||||||
"""
|
"""
|
||||||
return number1 + number1
|
return number1 + number2
|
||||||
|
|
||||||
def limit(number):
|
def limit(number):
|
||||||
"""
|
"""
|
||||||
|
|
|
@ -30,7 +30,7 @@ def main(location):
|
||||||
graph = sg.Graph(GSIZE, (0, 0), GSIZE, key='-GRAPH-', enable_events=True)
|
graph = sg.Graph(GSIZE, (0, 0), GSIZE, key='-GRAPH-', enable_events=True)
|
||||||
layout = [[graph]]
|
layout = [[graph]]
|
||||||
|
|
||||||
window = sg.Window('RAM Usage Widget Square', layout, location=location, no_titlebar=True, grab_anywhere=True, margins=(0, 0), element_padding=(0, 0), alpha_channel=ALPHA, finalize=True, right_click_menu=sg.MENU_RIGHT_CLICK_EDITME_EXIT)
|
window = sg.Window('RAM Usage Widget Square', layout, location=location, no_titlebar=True, grab_anywhere=True, margins=(0, 0), element_padding=(0, 0), alpha_channel=ALPHA, finalize=True, right_click_menu=sg.MENU_RIGHT_CLICK_EDITME_EXIT, enable_close_attempted_event=True)
|
||||||
|
|
||||||
|
|
||||||
while True: # Event Loop
|
while True: # Event Loop
|
||||||
|
@ -43,7 +43,9 @@ def main(location):
|
||||||
text_id2 = graph.draw_text(f'{human_size(ram.used)} used', (GSIZE[0] // 2, GSIZE[1] // 4), font='Any 20', text_location=sg.TEXT_LOCATION_CENTER, color=sg.theme_button_color()[0])
|
text_id2 = graph.draw_text(f'{human_size(ram.used)} used', (GSIZE[0] // 2, GSIZE[1] // 4), font='Any 20', text_location=sg.TEXT_LOCATION_CENTER, color=sg.theme_button_color()[0])
|
||||||
|
|
||||||
event, values = window.read(timeout=UPDATE_FREQUENCY_MILLISECONDS)
|
event, values = window.read(timeout=UPDATE_FREQUENCY_MILLISECONDS)
|
||||||
if event == sg.WIN_CLOSED or event == 'Exit':
|
if event in (sg.WIN_CLOSED, 'Exit', sg.WIN_CLOSE_ATTEMPTED_EVENT):
|
||||||
|
if event != sg.WIN_CLOSED:
|
||||||
|
sg.user_settings_set_entry('-location-', window.current_location()) # The line of code to save the position before exiting
|
||||||
break
|
break
|
||||||
elif event == 'Edit Me':
|
elif event == 'Edit Me':
|
||||||
sg.execute_editor(__file__)
|
sg.execute_editor(__file__)
|
||||||
|
@ -59,5 +61,5 @@ if __name__ == '__main__':
|
||||||
location = sys.argv[1].split(',')
|
location = sys.argv[1].split(',')
|
||||||
location = (int(location[0]), int(location[1]))
|
location = (int(location[0]), int(location[1]))
|
||||||
else:
|
else:
|
||||||
location = (None, None)
|
location = sg.user_settings_get_entry('-location-', (None, None))
|
||||||
main(location)
|
main(location)
|
||||||
|
|
|
@ -208,11 +208,12 @@ def create_window(win_location):
|
||||||
|
|
||||||
layout = [[top_col],
|
layout = [[top_col],
|
||||||
[lf_col, rt_col],
|
[lf_col, rt_col],
|
||||||
[bot_col]]
|
[bot_col],
|
||||||
|
[sg.Text(f'{sg.ver} {sg.framework_version} {sys.version}', font=('Arial', 8), background_color=BG_COLOR, text_color=TXT_COLOR, pad=(0,0))]]
|
||||||
|
|
||||||
window = sg.Window(layout=layout, title='Weather Widget', margins=(0, 0), finalize=True, location=win_location,
|
window = sg.Window(layout=layout, title='Weather Widget', margins=(0, 0), finalize=True, location=win_location,
|
||||||
element_justification='center', keep_on_top=True, no_titlebar=True, grab_anywhere=True, alpha_channel=ALPHA,
|
element_justification='center', keep_on_top=True, no_titlebar=True, grab_anywhere=True, alpha_channel=ALPHA,
|
||||||
right_click_menu=[[''], 'Exit'])
|
right_click_menu=[[''], ['Edit Me', 'Versions', 'Exit',]], enable_close_attempted_event=True)
|
||||||
|
|
||||||
for col in ['COL1', 'COL2', 'TopCOL', 'BotCOL', '-QUIT-']:
|
for col in ['COL1', 'COL2', 'TopCOL', 'BotCOL', '-QUIT-']:
|
||||||
window[col].expand(expand_y=True, expand_x=True)
|
window[col].expand(expand_y=True, expand_x=True)
|
||||||
|
@ -261,14 +262,19 @@ def main(refresh_rate, win_location):
|
||||||
|
|
||||||
while True: # Event Loop
|
while True: # Event Loop
|
||||||
event, values = window.read(timeout=refresh_in_milliseconds)
|
event, values = window.read(timeout=refresh_in_milliseconds)
|
||||||
if event in (None, '-QUIT-', 'Exit'):
|
if event in (None, '-QUIT-', 'Exit', sg.WIN_CLOSE_ATTEMPTED_EVENT):
|
||||||
|
sg.user_settings_set_entry('-win location-', window.current_location()) # The line of code to save the position before exiting
|
||||||
break
|
break
|
||||||
if event == '-CHANGE-':
|
if event == '-CHANGE-':
|
||||||
x, y = window.current_location()
|
x, y = window.current_location()
|
||||||
settings = change_settings(settings, (x + 200, y+50))
|
settings = change_settings(settings, (x + 200, y+50))
|
||||||
elif event == '-REFRESH-':
|
elif event == '-REFRESH-':
|
||||||
sg.popup_quick_message('Refreshing...', keep_on_top=True, background_color='red', text_color='white',
|
sg.popup_quick_message('Refreshing...', keep_on_top=True, background_color='red', text_color='white',
|
||||||
auto_close_duration=3, non_blocking=False, location=(win_location[0]+170, win_location[1]+150))
|
auto_close_duration=3, non_blocking=False, location=(window.current_location()[0]+window.size[0]//2-30, window.current_location()[1]+window.size[1]//2-10))
|
||||||
|
elif event == 'Edit Me':
|
||||||
|
sg.execute_editor(__file__)
|
||||||
|
elif event == 'Versions':
|
||||||
|
sg.main_get_debug_data()
|
||||||
elif event != sg.TIMEOUT_KEY:
|
elif event != sg.TIMEOUT_KEY:
|
||||||
sg.Print('Unknown event received\nEvent & values:\n', event, values, location=win_location)
|
sg.Print('Unknown event received\nEvent & values:\n', event, values, location=win_location)
|
||||||
|
|
||||||
|
@ -282,5 +288,6 @@ if __name__ == '__main__':
|
||||||
win_location = sys.argv[1].split(',')
|
win_location = sys.argv[1].split(',')
|
||||||
win_location = (int(win_location[0]), int(win_location[1]))
|
win_location = (int(win_location[0]), int(win_location[1]))
|
||||||
else:
|
else:
|
||||||
win_location = (None, None)
|
win_location = sg.user_settings_get_entry('-win location-', (None, None))
|
||||||
|
|
||||||
main(refresh_rate=1, win_location=win_location)
|
main(refresh_rate=1, win_location=win_location)
|
|
@ -25,34 +25,38 @@ import PySimpleGUI as sg
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# ------------------- The Old Way -------------------
|
# ------------------- The Old Way -------------------
|
||||||
|
import PySimpleGUI as sg
|
||||||
|
|
||||||
layout = [[sg.Text('My Window')],
|
layout = [[sg.Text('My Window')],
|
||||||
[sg.InputText(size=(50, 1), key='-FILENAME-'), sg.FileBrowse()],
|
[sg.InputText(size=(50, 1), key='-FILENAME-'), sg.FileBrowse()],
|
||||||
[sg.Button('Go'), sg.Button('Exit')]]
|
[sg.Button('Go'), sg.Button('Exit')]]
|
||||||
|
|
||||||
event1, values1 = sg.Window('Window Title', layout).read(close=True)
|
event1, values1 = sg.Window('Normal Filename', layout).read(close=True)
|
||||||
|
|
||||||
# ------------------- The New Way with history -------------------
|
# ------------------- The New Way with history -------------------
|
||||||
|
import PySimpleGUI as sg
|
||||||
|
|
||||||
layout = [[sg.Text('My Window')],
|
layout = [[sg.Text('My Window')],
|
||||||
[sg.Combo(sg.user_settings_get_entry('-filenames-', []), default_value=sg.user_settings_get_entry('-last filename-', ''), size=(50, 1), key='-FILENAME-'),
|
[sg.Combo(sg.user_settings_get_entry('-filenames-', []), default_value=sg.user_settings_get_entry('-last filename-', ''), size=(50, 1), key='-FILENAME-'),
|
||||||
sg.FileBrowse()],
|
sg.FileBrowse()],
|
||||||
[sg.Button('Go'), sg.Button('Exit')]]
|
[sg.Button('Go'), sg.Button('Exit')]]
|
||||||
|
|
||||||
event, values = sg.Window('Window Title', layout).read(close=True)
|
event, values = sg.Window('Filename with History', layout).read(close=True)
|
||||||
|
|
||||||
if event == 'Go':
|
if event == 'Go':
|
||||||
sg.user_settings_set_entry('-filenames-', list(set(sg.user_settings_get_entry('-filenames-', []) + [values['-FILENAME-'], ])))
|
sg.user_settings_set_entry('-filenames-', list(set(sg.user_settings_get_entry('-filenames-', []) + [values['-FILENAME-'], ])))
|
||||||
sg.user_settings_set_entry('-last filename-', values['-FILENAME-'])
|
sg.user_settings_set_entry('-last filename-', values['-FILENAME-'])
|
||||||
|
|
||||||
|
|
||||||
# ------------------- The New Way with history and clear -------------------
|
# ------------------- The New Way with history and clear -------------------
|
||||||
|
import PySimpleGUI as sg
|
||||||
|
|
||||||
layout = [[sg.Text('My Window')],
|
layout = [[sg.Text('My Window')],
|
||||||
[sg.Combo(sg.user_settings_get_entry('-filenames-', []), default_value=sg.user_settings_get_entry('-last filename-', ''), size=(50, 1), key='-FILENAME-'),
|
[sg.Combo(sg.user_settings_get_entry('-filenames-', []), default_value=sg.user_settings_get_entry('-last filename-', ''), size=(50, 1), key='-FILENAME-'),
|
||||||
sg.FileBrowse()],
|
sg.FileBrowse()],
|
||||||
[sg.Button('Go'), sg.B('Clear'), sg.Button('Exit')]]
|
[sg.Button('Go'), sg.B('Clear'), sg.Button('Exit')]]
|
||||||
|
|
||||||
window = sg.Window('Window Title', layout)
|
window = sg.Window('Filename History Clearable', layout)
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
event, values = window.read()
|
event, values = window.read()
|
||||||
|
|
Loading…
Reference in New Issue