PySimpleGUI/DemoPrograms/Demo_Titlebar_Custom_Async.py

118 lines
4.9 KiB
Python
Raw Normal View History

import PySimpleGUI as sg
"""
Custom Titlebar - Async Version
Demo showing how to remove the titlebar and replace with your own
Unlike previous demos that lacked a titlebar, this one provides a way for you
to "minimize" your window that does not have a titlebar. This is done by faking
the window using a hidden window that is minimized.
While this demo uses the button colors for the titlebar color, you can use anything you want.
If possible it could be good to use combinations that are known to match like the input element colors
or the button colors.
This version of the demo allows for async execution of your code. In another demo of this
custom titlebar idea, the user window would stop when the window is minimized. This is OK
for most applications, but if you're running a window with a timeout value (an async window)
then stopping execution when the window is minimized is not good.
The way to achieve both async window and the custom titlebar is to use the "read_all_windows"
function call. Using this function with a timeout has the same effect as running your
window.read with a timeout.
Additionally, if you right click and choose "close" on the minimized window on your
taskbar, now the program will exist rather than restoring the window like the other demo does.
Copyright 2020 PySimpleGUI.org
"""
def minimize_main_window(main_window):
"""
Creates an icon on the taskbar that represents your custom titlebar window.
The FocusIn event is set so that if the user restores the window from the taskbar.
If this window is closed by right clicking on the icon and choosing close, then the
program will exit just as if the "X" was clicked on the main window.
"""
main_window.hide()
layout = [[sg.T('This is your window with a customized titlebar... you just cannot see it')]]
window = sg.Window(main_window.Title, layout, finalize=True, alpha_channel=0)
window.minimize()
window.bind('<FocusIn>', '-RESTORE-')
# store the dummy window as a function property
minimize_main_window.dummy_window = window
def restore_main_window(main_window):
"""
Call this function when you want to restore your main window
:param main_window:
:return:
"""
if hasattr(minimize_main_window, 'dummy_window'):
minimize_main_window.dummy_window.close()
minimize_main_window.dummy_window = None
main_window.un_hide()
def title_bar(title, text_color, background_color):
"""
Creates a "row" that can be added to a layout. This row looks like a titlebar
:param title: The "title" to show in the titlebar
:type title: str
:return: A list of elements (i.e. a "row" for a layout)
:type: List[sg.Element]
"""
bc = background_color
tc = text_color
return [sg.Col([[sg.T(title, text_color=tc, background_color=bc)]], pad=(0, 0), background_color=bc),
sg.Col([[sg.T('_', text_color=tc, background_color=bc, enable_events=True, key='-MINIMIZE-'),
sg.Text('', text_color=tc, background_color=bc, enable_events=True, key='Exit')]], element_justification='r', key='-TITLEBAR-',
pad=(0, 0), background_color=bc)]
def main():
# sg.theme('light green 3')
# sg.theme('dark red')
# sg.theme('dark green 3')
sg.theme('light brown 10')
title = 'Customized Titlebar Window'
# Here the titlebar colors are based on the theme. A few suggestions are shown. Try each of them
layout = [title_bar(title, sg.theme_button_color()[0], sg.theme_button_color()[1]),
# title_bar(title, sg.theme_button_color()[1], sg.theme_slider_color()),
# title_bar(title, sg.theme_slider_color(), sg.theme_button_color()[0]),
[sg.T('This is normal window text. The above is the fake "titlebar"')],
[sg.T('Input something:')],
[sg.Input(key='-IN-'), sg.Text(size=(12, 1), key='-OUT-')],
[sg.Button('Go')]]
window_main = sg.Window(title, layout, resizable=True, no_titlebar=True, grab_anywhere=True, keep_on_top=True, margins=(0, 0), finalize=True)
window_main['-TITLEBAR-'].expand(True, False, False) # expand the titlebar's rightmost column so that it resizes correctly
counter = 0
while True: # Event Loop
window, event, values = sg.read_all_windows(timeout=100)
if event == sg.WIN_CLOSED or event == 'Exit':
break
# ------ events to handle minimize and restore of window ------
if event == '-MINIMIZE-':
minimize_main_window(window_main)
continue
elif event == '-RESTORE-' or (event == sg.WINDOW_CLOSED and window != window_main):
restore_main_window(window_main)
continue
# ------ remainder of your "normal" events and window code ------
window_main['-OUT-'].update(counter)
counter += 1
window.close()
if __name__ == '__main__':
main()