Multi-window support (experimental release), BringToFront method
This commit is contained in:
		
							parent
							
								
									70f6002ab9
								
							
						
					
					
						commit
						f3776ff5d4
					
				
					 1 changed files with 73 additions and 18 deletions
				
			
		|  | @ -178,6 +178,9 @@ ThisRow = 555666777  # magic number | |||
| # DEFAULT_WINDOW_ICON = '' | ||||
| MESSAGE_BOX_LINE_WIDTH = 60 | ||||
| 
 | ||||
| # Key representing a Read timeout | ||||
| TIMEOUT_KEY = '__timeout__' | ||||
| 
 | ||||
| 
 | ||||
| # a shameful global variable. This represents the top-level window information. Needed because opening a second window is different than opening the first. | ||||
| class MyWindows(): | ||||
|  | @ -380,6 +383,7 @@ class Element(): | |||
|         else: | ||||
|             self.ParentForm.LastButtonClicked = self.DisplayText | ||||
|         self.ParentForm.FormRemainedOpen = True | ||||
|         if self.ParentForm.CurrentlyRunningMainloop: | ||||
|             self.ParentForm.TKroot.quit()  # kick the users out of the mainloop | ||||
| 
 | ||||
|     def ReturnKeyHandler(self, event): | ||||
|  | @ -397,6 +401,7 @@ class Element(): | |||
|         else: | ||||
|             self.ParentForm.LastButtonClicked = '' | ||||
|         self.ParentForm.FormRemainedOpen = True | ||||
|         if self.ParentForm.CurrentlyRunningMainloop: | ||||
|             self.ParentForm.TKroot.quit()  # kick the users out of the mainloop | ||||
| 
 | ||||
|     def ComboboxSelectHandler(self, event): | ||||
|  | @ -408,6 +413,7 @@ class Element(): | |||
|         else: | ||||
|             self.ParentForm.LastButtonClicked = '' | ||||
|         self.ParentForm.FormRemainedOpen = True | ||||
|         if self.ParentForm.CurrentlyRunningMainloop: | ||||
|             self.ParentForm.TKroot.quit()  # kick the users out of the mainloop | ||||
| 
 | ||||
|     def CheckboxHandler(self): | ||||
|  | @ -417,6 +423,7 @@ class Element(): | |||
|         else: | ||||
|             self.ParentForm.LastButtonClicked = '' | ||||
|         self.ParentForm.FormRemainedOpen = True | ||||
|         if self.ParentForm.CurrentlyRunningMainloop: | ||||
|             self.ParentForm.TKroot.quit() | ||||
| 
 | ||||
|     def TabGroupSelectHandler(self, event): | ||||
|  | @ -426,6 +433,7 @@ class Element(): | |||
|         else: | ||||
|             self.ParentForm.LastButtonClicked = '' | ||||
|         self.ParentForm.FormRemainedOpen = True | ||||
|         if self.ParentForm.CurrentlyRunningMainloop: | ||||
|             self.ParentForm.TKroot.quit() | ||||
| 
 | ||||
|     def __del__(self): | ||||
|  | @ -883,6 +891,7 @@ class Spin(Element): | |||
|         else: | ||||
|             self.ParentForm.LastButtonClicked = '' | ||||
|         self.ParentForm.FormRemainedOpen = True | ||||
|         if self.ParentForm.CurrentlyRunningMainloop: | ||||
|             self.ParentForm.TKroot.quit()  # kick the users out of the mainloop | ||||
| 
 | ||||
|     def __del__(self): | ||||
|  | @ -1225,6 +1234,7 @@ class Button(Element): | |||
|             self.ParentForm.LastButtonClicked = self.Key | ||||
|         else: | ||||
|             self.ParentForm.LastButtonClicked = self.ButtonText | ||||
|         if self.ParentForm.CurrentlyRunningMainloop: | ||||
|             self.ParentForm.TKroot.quit()           # kick out of loop if read was called | ||||
| 
 | ||||
|     # -------  Button Callback  ------- # | ||||
|  | @ -1300,6 +1310,7 @@ class Button(Element): | |||
|                 self.ParentForm.LastButtonClicked = self.ButtonText | ||||
|             self.ParentForm.FormRemainedOpen = False | ||||
|             self.ParentForm._Close() | ||||
|             if self.ParentForm.CurrentlyRunningMainloop: | ||||
|                 self.ParentForm.TKroot.quit() | ||||
|             if self.ParentForm.NonBlocking: | ||||
|                 self.ParentForm.TKroot.destroy() | ||||
|  | @ -1312,6 +1323,7 @@ class Button(Element): | |||
|             else: | ||||
|                 self.ParentForm.LastButtonClicked = self.ButtonText | ||||
|             self.ParentForm.FormRemainedOpen = True | ||||
|             if self.ParentForm.CurrentlyRunningMainloop:            # if this window is running the mainloop, kick out | ||||
|                 self.ParentForm.TKroot.quit()  # kick the users out of the mainloop | ||||
|         elif self.BType == BUTTON_TYPE_CLOSES_WIN_ONLY:  # special kind of button that does not exit main loop | ||||
|             self.ParentForm._Close() | ||||
|  | @ -1329,6 +1341,7 @@ class Button(Element): | |||
|         if should_submit_window: | ||||
|             self.ParentForm.LastButtonClicked = target_element.Key | ||||
|             self.ParentForm.FormRemainedOpen = True | ||||
|             if self.ParentForm.CurrentlyRunningMainloop: | ||||
|                 self.ParentForm.TKroot.quit()  # kick the users out of the mainloop | ||||
| 
 | ||||
|         return | ||||
|  | @ -1984,6 +1997,7 @@ class Slider(Element): | |||
|         else: | ||||
|             self.ParentForm.LastButtonClicked = '' | ||||
|         self.ParentForm.FormRemainedOpen = True | ||||
|         if self.ParentForm.CurrentlyRunningMainloop: | ||||
|             self.ParentForm.TKroot.quit()  # kick the users out of the mainloop | ||||
| 
 | ||||
|     def __del__(self): | ||||
|  | @ -2367,6 +2381,7 @@ class Menu(Element): | |||
|         # print('IN MENU ITEM CALLBACK', item_chosen) | ||||
|         self.ParentForm.LastButtonClicked = item_chosen | ||||
|         self.ParentForm.FormRemainedOpen = True | ||||
|         if self.ParentForm.CurrentlyRunningMainloop: | ||||
|             self.ParentForm.TKroot.quit()  # kick the users out of the mainloop | ||||
| 
 | ||||
|     def __del__(self): | ||||
|  | @ -2622,6 +2637,7 @@ class Window: | |||
|         self.NonBlocking = False | ||||
|         self.TKroot = None | ||||
|         self.TKrootDestroyed = False | ||||
|         self.CurrentlyRunningMainloop = False | ||||
|         self.FormRemainedOpen = False | ||||
|         self.TKAfterID = None | ||||
|         self.ProgressBarColor = progress_bar_color | ||||
|  | @ -2753,20 +2769,23 @@ class Window: | |||
|     def _TimeoutAlarmCallback(self): | ||||
|         # first, get the results table built | ||||
|         # modify the Results table in the parent FlexForm object | ||||
|         # print('TIMEOUT CALLBACK') | ||||
|         if self.TimerCancelled: | ||||
|             # print('** timer was cancelled **') | ||||
|             return | ||||
|         self.LastButtonClicked = self.TimeoutKey | ||||
|         self.FormRemainedOpen = True | ||||
|         self.TKroot.quit()  # kick the users out of the mainloop | ||||
| 
 | ||||
|     def Read(self, timeout=None, timeout_key='_timeout_'): | ||||
|         if timeout == 0: | ||||
|     def Read(self, timeout=None, timeout_key=TIMEOUT_KEY): | ||||
|         if timeout == 0:                            # timeout of zero runs the old readnonblocking | ||||
|             event, values =  self.ReadNonBlocking() | ||||
|             if event is None: | ||||
|                 event = timeout_key | ||||
|             if values is None: | ||||
|                 event = None | ||||
|             return event, values | ||||
|             return event, values                    # make event None if values was None and return | ||||
|         # Read with a timeout | ||||
|         self.Timeout = timeout | ||||
|         self.TimeoutKey = timeout_key | ||||
|         self.NonBlocking = False | ||||
|  | @ -2775,27 +2794,48 @@ class Window: | |||
|         if not self.Shown: | ||||
|             self.Show() | ||||
|         else: | ||||
|             # if already have a button waiting, the return previously built results | ||||
|             if self.LastButtonClicked is not None and not self.LastButtonClickedWasRealtime: | ||||
|                 # print(f'*** Found previous clicked saved {self.LastButtonClicked}') | ||||
|                 results = BuildResults(self, False, self) | ||||
|                 self.LastButtonClicked = None | ||||
|                 return results | ||||
|             InitializeResults(self) | ||||
|             # if the last button clicked was realtime, emulate a read non-blocking | ||||
|             # the idea is to quickly return realtime buttons without any blocks until released | ||||
|             if self.LastButtonClickedWasRealtime: | ||||
|                 # print(f'RTime down {self.LastButtonClicked}' ) | ||||
|                 try: | ||||
|                     rc = self.TKroot.update() | ||||
|                 except: | ||||
|                     self.TKrootDestroyed = True | ||||
|                     _my_windows.Decrement() | ||||
|                     print('ROOT Destroyed') | ||||
|                 results = BuildResults(self, False, self) | ||||
|                 if results[0] != None and results[0] != timeout_key: | ||||
|                    return results | ||||
|                 else: | ||||
|                     pass | ||||
| 
 | ||||
|                 # else: | ||||
|                 #     print("** REALTIME PROBLEM FOUND **", results) | ||||
| 
 | ||||
|             # normal read blocking code.... | ||||
|             if timeout != None: | ||||
|                 self.TimerCancelled = False | ||||
|                 self.TKAfterID = self.TKroot.after(timeout, self._TimeoutAlarmCallback) | ||||
|             self.CurrentlyRunningMainloop = True | ||||
|             # print(f'In main {self.Title}') | ||||
|             self.TKroot.mainloop() | ||||
|             # print('Out main') | ||||
|             self.CurrentlyRunningMainloop = False | ||||
|             # if self.LastButtonClicked != TIMEOUT_KEY: | ||||
|             #     print(f'Window {self.Title} Last button clicked = {self.LastButtonClicked}') | ||||
|             try: | ||||
|                 self.TKroot.after_cancel(self.TKAfterID) | ||||
|             except: | ||||
|                 pass | ||||
|                 # print('** tkafter cancel failed **') | ||||
|             self.TimerCancelled = True | ||||
|             if self.RootNeedsDestroying: | ||||
|                 self.TKroot.destroy() | ||||
|  | @ -2803,8 +2843,12 @@ class Window: | |||
|             # if form was closed with X | ||||
|             if self.LastButtonClicked is None and self.LastKeyboardEvent is None and self.ReturnValues[0] is None: | ||||
|                 _my_windows.Decrement() | ||||
|         # Determine return values | ||||
|         if self.LastKeyboardEvent is not None or self.LastButtonClicked is not None: | ||||
|             return BuildResults(self, False, self) | ||||
|             results =  BuildResults(self, False, self) | ||||
|             if not self.LastButtonClickedWasRealtime: | ||||
|                 self.LastButtonClicked = None | ||||
|             return results | ||||
|         else: | ||||
|             return self.ReturnValues | ||||
| 
 | ||||
|  | @ -2820,6 +2864,7 @@ class Window: | |||
|         except: | ||||
|             self.TKrootDestroyed = True | ||||
|             _my_windows.Decrement() | ||||
|             # print("read failed") | ||||
|             # return None, None | ||||
|         return BuildResults(self, False, self) | ||||
| 
 | ||||
|  | @ -2928,6 +2973,7 @@ class Window: | |||
|             self.LastKeyboardEvent = str(event.keysym) + ':' + str(event.keycode) | ||||
|         if not self.NonBlocking: | ||||
|             BuildResults(self, False, self) | ||||
|         if self.CurrentlyRunningMainloop:       # quit if this is the current mainloop, otherwise don't quit! | ||||
|             self.TKroot.quit() | ||||
| 
 | ||||
|     def _MouseWheelCallback(self, event): | ||||
|  | @ -2936,6 +2982,7 @@ class Window: | |||
|         self.LastKeyboardEvent = 'MouseWheel:Down' if event.delta < 0 else 'MouseWheel:Up' | ||||
|         if not self.NonBlocking: | ||||
|             BuildResults(self, False, self) | ||||
|         if self.CurrentlyRunningMainloop:       # quit if this is the current mainloop, otherwise don't quit! | ||||
|             self.TKroot.quit() | ||||
| 
 | ||||
|     def _Close(self): | ||||
|  | @ -3004,6 +3051,12 @@ class Window: | |||
|         self._AlphaChannel = alpha | ||||
|         self.TKroot.attributes('-alpha', alpha) | ||||
| 
 | ||||
|     def BringToFront(self): | ||||
|         try: | ||||
|             self.TKroot.lift() | ||||
|         except: | ||||
|             pass | ||||
| 
 | ||||
|     def __enter__(self): | ||||
|         return self | ||||
| 
 | ||||
|  | @ -4531,7 +4584,9 @@ def StartupTK(my_flex_form): | |||
|         # my_flex_form.TKroot.protocol("WM_DELETE_WINDOW", my_flex_form.OnClosingCallback()) | ||||
|     else:  # it's a blocking form | ||||
|         # print('..... CALLING MainLoop') | ||||
|         my_flex_form.CurrentlyRunningMainloop = True | ||||
|         my_flex_form.TKroot.mainloop() | ||||
|         my_flex_form.CurrentlyRunningMainloop = False | ||||
|         my_flex_form.TimerCancelled = True | ||||
|         # print('..... BACK from MainLoop') | ||||
|         if not my_flex_form.FormRemainedOpen: | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue