Merge pull request #3655 from PySimpleGUI/Dev-latest
Initial Checkin - A desktop widget that is the same as grid of bar gr…
This commit is contained in:
		
						commit
						be550079b1
					
				
					 1 changed files with 375 additions and 0 deletions
				
			
		
							
								
								
									
										375
									
								
								DemoPrograms/Demo_Desktop_Widget_CPU_Grid_Of_Gauges.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										375
									
								
								DemoPrograms/Demo_Desktop_Widget_CPU_Grid_Of_Gauges.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,375 @@ | |||
| #!/usr/bin/env python | ||||
| import PySimpleGUI as sg | ||||
| import sys | ||||
| import psutil | ||||
| 
 | ||||
| """ | ||||
|     Desktop floating widget - CPU Cores as Gauges | ||||
|     Uses psutil to display: | ||||
|         CPU usage of each individual core | ||||
|     CPU utilization is updated every 2000 ms by default | ||||
|     Utiliziation is shown as a gauge | ||||
|     Pointer turns red when usage > 50% | ||||
|     To achieve a "rainmeter-style" of window, these featurees were used: | ||||
|       An alpha-channel setting of 0.8 to give a little transparency | ||||
|       No titlebar | ||||
|       Grab anywhere, making window easy to move around | ||||
|     Copyright 2020 PySimpleGUI | ||||
| """ | ||||
| 
 | ||||
| GRAPH_WIDTH = 120       # each individual graph size in pixels | ||||
| GRAPH_HEIGHT = 40 | ||||
| TRANSPARENCY = .8       # how transparent the window looks. 0 = invisible, 1 = normal window | ||||
| NUM_COLS = 4 | ||||
| POLL_FREQUENCY = 2000    # how often to update graphs in milliseconds | ||||
| 
 | ||||
| colors = ('#23a0a0', '#56d856', '#be45be', '#5681d8', '#d34545', '#BE7C29') | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| import math | ||||
| import random | ||||
| 
 | ||||
| 
 | ||||
| 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, 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, 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, color='red' if new_degree > 90 else None) | ||||
|                     return False | ||||
|             else: | ||||
|                 self.pointer.new(degree=self.pointer.stop_degree, color='red' if self.pointer.stop_degree > 90 else None) | ||||
|                 return True | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| # DashGraph does the drawing of each graph | ||||
| class DashGraph(object): | ||||
|     def __init__(self, graph_elem, text_elem, starting_count, color): | ||||
|         self.graph_current_item = 0 | ||||
|         self.graph_elem = graph_elem        # type: sg.Graph | ||||
|         self.text_elem = text_elem | ||||
|         self.prev_value = starting_count | ||||
|         self.max_sent = 1 | ||||
|         self.color = color | ||||
|         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, | ||||
|                   minor_tick_color=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=graph_elem) | ||||
| 
 | ||||
|         self.gauge.change(degree=0) | ||||
| 
 | ||||
|     def graph_percentage_abs(self, value): | ||||
|         if self.gauge.change(): | ||||
|             new_angle = value*180/100 | ||||
|             self.gauge.change(degree=new_angle, step=new_angle) | ||||
|             self.gauge.change() | ||||
| 
 | ||||
|     def text_display(self, text): | ||||
|         self.text_elem.update(text) | ||||
| 
 | ||||
| def main(location): | ||||
|     gsize = (100, 55) | ||||
| 
 | ||||
| 
 | ||||
|     # A couple of "User defined elements" that combine several elements and enable bulk edits | ||||
|     def Txt(text, **kwargs): | ||||
|         return(sg.Text(text, font=('Helvetica 8'), **kwargs)) | ||||
| 
 | ||||
|     def GraphColumn(name, key): | ||||
|         layout = [ | ||||
|             [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-')]] | ||||
|         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 | ||||
| 
 | ||||
|     sg.theme('black') | ||||
|     sg.set_options(element_padding=(0,0), margins=(1,1), border_width=0) | ||||
| 
 | ||||
|     # the clever Red X graphic | ||||
|     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')] ] | ||||
| 
 | ||||
|     # add on the graphs | ||||
|     for rows in range(num_cores//NUM_COLS+1): | ||||
|         # for cols in range(min(num_cores-rows*NUM_COLS, NUM_COLS)): | ||||
|         layout += [[GraphColumn('CPU '+str(rows*NUM_COLS+cols), '-CPU-'+str(rows*NUM_COLS+cols)) for cols in range(min(num_cores-rows*NUM_COLS, NUM_COLS))]] | ||||
| 
 | ||||
|     # ----------------  Create Window  ---------------- | ||||
|     window = sg.Window('CPU Cores Usage Widget', layout, | ||||
|                        keep_on_top=True, | ||||
|                        auto_size_buttons=False, | ||||
|                        grab_anywhere=True, | ||||
|                        no_titlebar=True, | ||||
|                        default_button_element_size=(12, 1), | ||||
|                        return_keyboard_events=True, | ||||
|                        alpha_channel=TRANSPARENCY, | ||||
|                        use_default_focus=False, | ||||
|                        finalize=True, | ||||
|                        location=location, | ||||
|                        right_click_menu=[[''], 'Exit']) | ||||
| 
 | ||||
|     # setup graphs & initial values | ||||
|     graphs = [DashGraph(window['-CPU-'+str(i)+'-GRAPH-'], | ||||
|                                 window['-CPU-'+str(i) + '-GAUGE VALUE-'], | ||||
|                                 0, colors[i%6]) for i in range(num_cores) ] | ||||
| 
 | ||||
|     # ----------------  main loop  ---------------- | ||||
|     while True : | ||||
|         # --------- Read and update window once every Polling Frequency -------- | ||||
|         event, values = window.read(timeout=POLL_FREQUENCY/2) | ||||
|         if event in (sg.WIN_CLOSED, 'Exit'):         # Be nice and give an exit | ||||
|             break | ||||
|         # read CPU for each core | ||||
|         stats = psutil.cpu_percent(interval=POLL_FREQUENCY/2/1000, percpu=True) | ||||
| 
 | ||||
|         # update each graph | ||||
|         for i, util in enumerate(stats): | ||||
|             graphs[i].graph_percentage_abs(util) | ||||
|             graphs[i].text_display('{:2.0f}'.format(util)) | ||||
| 
 | ||||
|     window.close() | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     # when invoking this program, if a location is set on the command line, then the window will be created there. Use x,y with no ( ) | ||||
|     if len(sys.argv) > 1: | ||||
|         location = sys.argv[1].split(',') | ||||
|         location = (int(location[0]), int(location[1])) | ||||
|     else: | ||||
|         location = (None, None) | ||||
|     main(location) | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue