diff --git a/DemoPrograms/Demo_Desktop_Widget_Count_To_A_Goal.py b/DemoPrograms/Demo_Desktop_Widget_Count_To_A_Goal.py new file mode 100644 index 00000000..43fc4d74 --- /dev/null +++ b/DemoPrograms/Demo_Desktop_Widget_Count_To_A_Goal.py @@ -0,0 +1,428 @@ +import PySimpleGUI as sg +import math +""" + Another simple Desktop Widget using PySimpleGUI + + A Counter Widget... X out of Y + + Maybe you're counting the number of classes left in a course you're + working your way through, or the number of pokemons left to capture. + + Whatever it is, sometimes knowing your progress helps. This widget shows + you the current count and the total along with your % complete via a gauge. + (Again, thank you to Jason for the gauge!) + + Copyright 2021 PySimpleGUI +""" + +ALPHA = 0.9 # Initial alpha until user changes +THEME = 'Dark green 3' # Initial theme until user changes +refresh_font = sg.user_settings_get_entry('-refresh 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_info_size = (3,1) + + + + +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) + + def new(self, degree=0): + """ + 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 + 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, 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) + 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): + """ + 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) + return False + else: + self.pointer.new(degree=self.pointer.stop_degree) + return True + +GSIZE = (160, 160) + +def choose_theme(location): + layout = [[sg.Text(f'Current theme {sg.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, keep_on_top=True) + 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]) + test_window=make_window(location=(location[0]-200, location[1]), test_window=True) + test_window.read(close=True) + window.close() + + 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): + title_font = sg.user_settings_get_entry('-title font-', 'Courier 8') + title = sg.user_settings_get_entry('-title-', '') + main_number_font = sg.user_settings_get_entry('-main number font-', 'Courier 70') + + if not test_window: + theme = sg.user_settings_get_entry('-theme-', THEME) + sg.theme(theme) + + alpha = sg.user_settings_get_entry('-alpha-', ALPHA) + + # ------------------- Window Layout ------------------- + # If this is a test window (for choosing theme), then uses some extra Text Elements to display theme info + # and also enables events for the elements to make the window easy to close + if test_window: + top_elements = [[sg.Text(title, size=(20, 1), font=title_font, justification='c', k='-TITLE-', enable_events=True)], + [sg.Text('Click to close', font=title_font, enable_events=True)], + [sg.Text('This is theme', font=title_font, enable_events=True)], + [sg.Text(sg.theme(), font=title_font, enable_events=True)]] + right_click_menu = [[''], ['Exit',]] + else: + top_elements = [[sg.Text(title, size=(20, 1), font=title_font, justification='c', k='-TITLE-')]] + right_click_menu = [[''], ['Set Count','Set Goal','Choose Title', 'Edit Me', 'Change Theme', 'Save Location', 'Refresh', 'Set Title Font', 'Set Main Font','Alpha', [str(x) for x in range(1, 11)], 'Exit', ]] + + gsize = (100, 55) + + + layout = top_elements + \ + [[sg.Text('0', size=main_info_size, font=main_number_font, k='-MAIN INFO-', justification='c', enable_events=test_window)], + sg.vbottom([sg.Text(0, size=(3, 1), justification='r', font='courier 20'), + sg.Graph(gsize, (-gsize[0] // 2, 0), (gsize[0] // 2, gsize[1]), key='-Graph-'), + sg.Text(0, size=(3, 1), font='courier 20', k='-GOAL-')]), + ] + + + + try: + window = sg.Window('Counter Widget', layout, location=location, no_titlebar=True, grab_anywhere=True, margins=(0, 0), element_justification='c', element_padding=(0, 0), alpha_channel=alpha, finalize=True, right_click_menu=right_click_menu, right_click_menu_tearoff=False, keep_on_top=True) + except Exception as e: + if sg.popup_yes_no('Error creating your window', e, 'These are your current settings:', sg.user_settings(), 'Do you want to delete your settings file?') == 'Yes': + sg.user_settings_delete_filename() + sg.popup('Settings deleted.','Please restart your program') + exit() + window = None + + window.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_start_radius=45, minor_tick_stop_radius=50, major_tick_stop_radius=50, major_tick_step=30, clock_radius=50, pointer_line_width=3, pointer_inner_radius=10, pointer_outer_radius=50, graph_elem=window['-Graph-']) + + window.gauge.change(degree=0) + + return window + +def main(): + loc = sg.user_settings_get_entry('-location-', (None, None)) + window = make_window(loc) + try: + current_count = int(sg.user_settings_get_entry('-current count-', 0)) + current_goal = int(sg.user_settings_get_entry('-goal-', 100)) + current_goal = current_goal if current_goal != 0 else 100 + except: + if sg.popup_yes_no('Your count or goal number is not good. Do you want to delete your settings file?', location=window.current_location()) == 'Yes': + sg.user_settings_delete_filename() + sg.popup('Settings deleted.','Please restart your program', location=window.current_location()) + exit() + + window['-MAIN INFO-'].update(current_count) + window['-GOAL-'].update(current_goal) + + while True: # Event Loop + if window.gauge.change(): + new_angle = current_count / current_goal * 180 + window.gauge.change(degree=new_angle, step=180) + window.gauge.change() + window['-GOAL-'].update(current_goal) + window['-MAIN INFO-'].update(current_count) + + # -------------- Start of normal event loop -------------- + event, values = window.read() + print(event, values) + if event == sg.WIN_CLOSED or event == 'Exit': + break + if event == 'Edit Me': + sg.execute_editor(__file__) + elif event == 'Set Count': + new_count = sg.popup_get_text('Enter current count', default_text=current_count, location=window.current_location(), keep_on_top=True) + if new_count is not None: + try: + current_count = int(new_count) + except: + sg.popup_error('Your count is not good. Ignoring input.', location=window.current_location()) + continue + sg.user_settings_set_entry('-current count-', current_count) + elif event == 'Set Goal': + new_goal = sg.popup_get_text('Enter Goal', default_text=current_goal, location=window.current_location(), keep_on_top=True) + if new_goal is not None: + try: + current_goal = int(new_goal) + except: + sg.popup_error('Your goal number is not good. Ignoring input.', location=window.current_location()) + continue + current_goal = current_goal if current_goal != 0 else 100 + sg.user_settings_set_entry('-goal-', current_goal) + elif event == 'Choose Title': + new_title = sg.popup_get_text('Choose a title for your date', location=window.current_location(), keep_on_top=True) + if new_title is not None: + window['-TITLE-'].update(new_title) + sg.user_settings_set_entry('-title-', new_title) + elif event == 'Save Location': + sg.user_settings_set_entry('-location-', window.current_location()) + elif event in [str(x) for x in range(1,11)]: + window.set_alpha(int(event)/10) + sg.user_settings_set_entry('-alpha-', int(event)/10) + elif event == 'Change Theme': + loc = window.current_location() + if choose_theme(loc) is not None: + # this is result of hacking code down to 99 lines in total. Not tried it before. Interesting test. + _, window = window.close(), make_window(loc) + 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) + if font: + sg.user_settings_set_entry('-main number font-', font) + _, window = window.close(), make_window(loc) + elif event == 'Set Title Font': + font = sg.popup_get_text('Title Font and Size (e.g. courier 8)', default_text=sg.user_settings_get_entry('-title font-'), keep_on_top=True) + if font: + sg.user_settings_set_entry('-title font-', font) + _, window = window.close(), make_window(loc) + + + + window.close() + + +if __name__ == '__main__': + main() \ No newline at end of file diff --git a/DemoPrograms/Demo_Sudoku_1_Line.py b/DemoPrograms/Demo_Sudoku_1_Line.py index 093765df..a66bd4f3 100644 --- a/DemoPrograms/Demo_Sudoku_1_Line.py +++ b/DemoPrograms/Demo_Sudoku_1_Line.py @@ -10,4 +10,9 @@ import PySimpleGUI as sg Copyright 2021 PySimpleGUI """ -sg.Window('Sudoku', [[sg.Frame('', [[sg.Input(justification='r', size=(3,1)) for col in range(3)] for row in range(3)]) for frame_col in range(3)] for frame_row in range(3)], use_custom_titlebar=True).read() +sg.Window('Sudoku', [[sg.Frame('', [[sg.Input(justification='r', size=(3,1)) + for col in range(3)] + for row in range(3)]) + for frame_col in range(3)] + for frame_row in range(3)], + use_custom_titlebar=True).read()