PySimpleGUI/ThemeCreator/Theme_Maker.py

263 lines
14 KiB
Python

import PySimpleGUI as sg
import color_themes
"""
Program that is used to create new themes for use with PySimpleGUI's "Look and Feel" settings.
You are presented with a grid of mini-windows that is created for color palettes downloaded from https://colorhunt.co/palettes
The file color_themes.py contains a large dictionary of approx 1780 palettes.
For each palette you are shown 4 candidate themes, 2 "light" and 2 "dark". The window shows 5 palettes so you'll have a
total of 20 candidate themes displayed in total.
Each candidate theme has a 3 options - The button color (text and background), the text color for Input/Multiline elements,
and the name of the theme when you save it. These you choose using the radio buttons and one input field.
To "save" one of these candidate themes, check the checkbox to the left of the layout, choose the radio buttons for button
& text settings and optionally change the theme name that is shown above the grid of OK buttons. By default the name starts
with either "Dark" or "Light" and is followed by the first 2 "tags" that were posted with the palette on the colorhunt site.
After you've selected the themes you want to save from the window of 20 click any "OK" button in the window or close the
window with the "X". You will see the dictionary text in the Debug Window and the values will also be written to a file.
You'll then be shown the next group of 20 candidate themes.
If you want to exit the program entirely, click any "Cancel" button the page. Note - cliicking "Cancel" will not save any
theme you have checked with the checkbox. You should only exit from a window you have not selected any themes for saving
If a Theme is selected for saving, then the values for the LOOK_AND_FEEL dictionary are displayed in a debug window and are
also appended to the file new_theme_dict.py. You will need to edit the new_theme_dict.py file to get the syntax correct.
A "," or "}" may need to be added in order to make the table be correct.
If you're using this program it's assumed you know what you're doing and understand the LOOK_AND_FEEL dictionary and can
figure out how to get the syntax correct for adding it to the main dictionary of themes.
"""
#----------------- Window to get layout for window ---------#
CANDIDATES_PER_ROW = 4
PALETTES_PER_WINDOW = 5
STARTING_PALETTE = 0
sg.change_look_and_feel('Dark Blue 3')
layout = [
[sg.T('Choose your window layout (default is a huge window)')],
[sg.In(default_text=CANDIDATES_PER_ROW),sg.T('Candidates Per Row')],
[sg.In(default_text=PALETTES_PER_WINDOW),sg.T('Palettes Per Window (4 candidates for each)')],
[sg.In(default_text=0),sg.T(f'Starting palette number of {len(color_themes.themes)} palettes')],
[sg.OK()]]
window = sg.Window('Choose Theme Layout', layout,default_element_size=(4,1))
event, values = window.read()
window.close()
if event is None:
sg.popup_no_buttons('Aborting....', no_titlebar=True, auto_close=True, keep_on_top=True)
exit()
try:
CANDIDATES_PER_ROW = int(values[0])
PALETTES_PER_WINDOW = int(values[1])
STARTING_PALETTE = int(values[2])
except:
sg.popup_no_buttons('Bad input... Aborting....', no_titlebar=True, auto_close=True, keep_on_top=True)
exit()
TOTAL_CANDIDATES_PER_PAGE = PALETTES_PER_WINDOW * 4
#-----------------------------------------------------------#
def rgb_to_hsl(r, g, b):
r = float(r)
g = float(g)
b = float(b)
high = max(r, g, b)
low = min(r, g, b)
h, s, v = ((high + low) / 2,)*3
if high == low:
h = s = 0.0
else:
d = high - low
l = (high + low) / 2
s = d / (2 - high - low) if l > 0.5 else d / (high + low)
h = {
r: (g - b) / d + (6 if g < b else 0),
g: (b - r) / d + 2,
b: (r - g) / d + 4,
}[high]
h /= 6
return h, s, v
def hex_to_rgb(hex):
hex = hex.lstrip('#')
hlen = len(hex)
return tuple(int(hex[i:i + hlen // 3], 16) for i in range(0, hlen, hlen // 3))
def sorted_tuple(tup, position):
tup.sort(key=lambda x: x[position])
return tup
sg.popup_quick_message('Hang on this could me a few moments....', background_color='red', text_color='white', keep_on_top=True)
sg.LOOK_AND_FEEL_TABLE = {} # Entirely replace the look and feel table in PySimpleGUI
palette_counter = 0
for key, colors in color_themes.themes.items():
if palette_counter < STARTING_PALETTE:
palette_counter += 1
continue
# Sort the colors from darkest to lightest
color_lightness_pairs = []
for color in colors:
if type(color) in (tuple, list):
continue
r,g,b = hex_to_rgb(color)
lightness = (rgb_to_hsl(r=r, g=g, b=b))[2]
color_lightness_pairs.append((lightness, color))
sorted_colors_tuples = sorted_tuple(color_lightness_pairs, 0) # sort the pairs by the first item (lightness)
scolors = [c for l, c in sorted_colors_tuples] # Colors sorted from darkest to lightest
# Create a "Dark" and a "Light" theme based on the sorted colors
sg.LOOK_AND_FEEL_TABLE['Dark'+key] = {'BACKGROUND': scolors[0],
'TEXT': scolors[3],
'INPUT': scolors[2],
'TEXT_INPUT': '#000000',
'SCROLL': scolors[2],
'BUTTON': ('#FFFFFF', scolors[1]),
'PROGRESS': sg.DEFAULT_PROGRESS_BAR_COLOR,
'BORDER': 1,
'SLIDER_DEPTH': 0,
'PROGRESS_DEPTH': 0,
'COLOR_LIST':scolors,
'DESCRIPTION':colors[4]}
sg.LOOK_AND_FEEL_TABLE['Light'+key] = {'BACKGROUND': scolors[3],
'TEXT': scolors[0],
'INPUT': scolors[1],
'TEXT_INPUT': '#FFFFFF',
'SCROLL': scolors[0],
'BUTTON': ('#FFFFFF', scolors[2]),
'PROGRESS': sg.DEFAULT_PROGRESS_BAR_COLOR,
'BORDER': 1,
'SLIDER_DEPTH': 0,
'PROGRESS_DEPTH': 0,
'COLOR_LIST': scolors,
'DESCRIPTION':colors[4]}
sg.LOOK_AND_FEEL_TABLE['Dark2'+key] = {'BACKGROUND': scolors[1],
'TEXT': scolors[3],
'INPUT': scolors[2],
'TEXT_INPUT': '#000000',
'SCROLL': scolors[2],
'BUTTON': ('#FFFFFF', scolors[1]),
'PROGRESS': sg.DEFAULT_PROGRESS_BAR_COLOR,
'BORDER': 1,
'SLIDER_DEPTH': 0,
'PROGRESS_DEPTH': 0,
'COLOR_LIST':scolors,
'DESCRIPTION':colors[4]}
sg.LOOK_AND_FEEL_TABLE['Light2'+key] = {'BACKGROUND': scolors[2],
'TEXT': scolors[0],
'INPUT': scolors[1],
'TEXT_INPUT': '#FFFFFF',
'SCROLL': scolors[0],
'BUTTON': ('#FFFFFF', scolors[2]),
'PROGRESS': sg.DEFAULT_PROGRESS_BAR_COLOR,
'BORDER': 1,
'SLIDER_DEPTH': 0,
'PROGRESS_DEPTH': 0,
'COLOR_LIST': scolors,
'DESCRIPTION':colors[4]}
WINDOW_BACKGROUND = 'lightblue'
def sample_layout(theme_name, colors, description):
name = 'Dark' if theme_name.startswith('D') else 'Light'
name += "".join(description[:2])
layout = [[sg.Text('Text element', size=(12,1)), sg.InputText(' '.join(colors),text_color='#000000' ),sg.Radio('',theme+'1', key='-INPUT_RAD0-'+theme, default=True, metadata='#000000'),
sg.Slider((0,10),size=(10,20), orientation='h')],
[sg.T(size=(12,1)), sg.InputText(colors[0], text_color='#FFFFFF'),sg.Radio('',theme+'1', key='-INPUT_RAD1-'+theme, metadata='#FFFFFF')],
[sg.T(size=(12,1)), sg.InputText(colors[0], text_color=colors[0]),sg.Radio('',theme+'1', key='-INPUT_RAD2-'+theme, metadata=colors[0])],
[sg.T(size=(12,1)),sg.InputText(colors[3], text_color=colors[3]),sg.Radio('',theme+'1', key='-INPUT_RAD3-'+theme, metadata=colors[3])],
[sg.T(', '.join(description)), sg.In(name, key='-NEW_THEME_NAME-'+theme)],
[sg.Button('OK'), sg.Radio('',theme+'2',key='-BTN_RAD1-'+theme, default=True, metadata=sg.DEFAULT_BUTTON_COLOR),
sg.Button('OK', button_color=('white', colors[0])),sg.Radio('',theme+'2',key='-BTN_RAD2-'+theme, metadata=('white', colors[0])),
sg.Button('OK', button_color=('black', colors[0])),sg.Radio('',theme+'2',key='-BTN_RAD9-'+theme, metadata=('black', colors[0])),
sg.Button('OK', button_color=('white', colors[3])),sg.Radio('', theme+'2', key='-BTN_RAD10-' + theme, metadata=('white', colors[3])),
sg.Button('OK', button_color=('black', colors[3])),sg.Radio('', theme+'2', key='-BTN_RAD11-' + theme, metadata=('black', colors[3]))],
[sg.Button('OK', button_color=(colors[0],colors[1])),sg.Radio('',theme+'2',key='-BTN_RAD3-'+theme, metadata=(colors[0], colors[1])),
sg.Button('OK', button_color=(colors[2],colors[1])),sg.Radio('',theme+'2',key='-BTN_RAD4-'+theme, metadata=(colors[2], colors[1])),
sg.Button('OK', button_color=(colors[3],colors[1])),sg.Radio('',theme+'2',key='-BTN_RAD5-'+theme, metadata=(colors[3], colors[1])),
sg.Button('OK', button_color=(colors[3],colors[0])),sg.Radio('',theme+'2',key='-BTN_RAD7-'+theme, metadata=(colors[3], colors[0])),
sg.Button('OK', button_color=(colors[0],colors[3])),sg.Radio('',theme+'2',key='-BTN_RAD8-'+theme, metadata=(colors[0], colors[3])),
sg.Button('Cancel', button_color=(colors[3], colors[2])),sg.Radio('',theme+'2',key='-BTN_RAD6-'+theme, metadata=(colors[3], colors[2])),
] ]
return layout
# layout = [[sg.Text('Here is list of some themes', font='Default 18', background_color=WINDOW_BACKGROUND)]]
layout = []
row = []
layouts = []
for count, theme in enumerate(sg.LOOK_AND_FEEL_TABLE.keys()):
sg.change_look_and_feel(theme)
if count and not(count % CANDIDATES_PER_ROW):
layout += [row]
row = []
row += [sg.CB('',text_color='black', background_color=WINDOW_BACKGROUND, key='-CB-'+theme)]
row += [sg.Frame(theme, sample_layout(theme, sg.LOOK_AND_FEEL_TABLE[theme]['COLOR_LIST'], sg.LOOK_AND_FEEL_TABLE[theme]['DESCRIPTION']))]
if count and not (count % TOTAL_CANDIDATES_PER_PAGE):
if layout:
layouts.append(layout)
layout = []
if row:
layout += [row]
if layout:
layouts.append(layout)
for layout in layouts:
window = sg.Window('PySimpleGUI Theme Maker', layout, background_color=WINDOW_BACKGROUND, default_element_size=(30,1))
event, values = window.read()
if event is not None and event.startswith('Cancel'):
break
for key, value in values.items():
if type(key) is str and key.startswith('-CB-') and value:
theme = key[4:]
theme_entry = sg.LOOK_AND_FEEL_TABLE[theme]
if values['-INPUT_RAD1-'+theme]:
input_text_color = window['-INPUT_RAD1-'+theme].metadata
elif values['-INPUT_RAD2-'+theme]:
input_text_color = window['-INPUT_RAD2-'+theme].metadata
elif values['-INPUT_RAD3-'+theme]:
input_text_color = window['-INPUT_RAD3-'+theme].metadata
elif values['-INPUT_RAD0-'+theme]:
input_text_color = window['-INPUT_RAD0-'+theme].metadata
else:
print('** ERROR none of the radio buttons are true for input text **')
continue
if values['-BTN_RAD1-'+theme]:
b_color = window['-BTN_RAD1-'+theme].metadata
elif values['-BTN_RAD2-'+theme]:
b_color = window['-BTN_RAD2-'+theme].metadata
elif values['-BTN_RAD3-'+theme]:
b_color = window['-BTN_RAD3-'+theme].metadata
elif values['-BTN_RAD4-'+theme]:
b_color = window['-BTN_RAD4-'+theme].metadata
elif values['-BTN_RAD5-'+theme]:
b_color = window['-BTN_RAD5-'+theme].metadata
elif values['-BTN_RAD6-'+theme]:
b_color = window['-BTN_RAD6-'+theme].metadata
elif values['-BTN_RAD7-'+theme]:
b_color = window['-BTN_RAD7-'+theme].metadata
elif values['-BTN_RAD8-'+theme]:
b_color = window['-BTN_RAD8-'+theme].metadata
elif values['-BTN_RAD9-'+theme]:
b_color = window['-BTN_RAD9-'+theme].metadata
elif values['-BTN_RAD10-'+theme]:
b_color = window['-BTN_RAD10-'+theme].metadata
elif values['-BTN_RAD11-'+theme]:
b_color = window['-BTN_RAD11-'+theme].metadata
else:
print('** ERROR none of the radio buttons are true for button color **')
continue
sg.LOOK_AND_FEEL_TABLE[theme]['TEXT_INPUT'] = input_text_color
sg.LOOK_AND_FEEL_TABLE[theme]['BUTTON'] = b_color
with open('new_theme_dict.py', 'a') as outfile:
outfile.write(f"'{values['-NEW_THEME_NAME-'+theme]}' : {sg.LOOK_AND_FEEL_TABLE[theme]},\n")
sg.Print(f"'{values['-NEW_THEME_NAME-'+theme]}' : {sg.LOOK_AND_FEEL_TABLE[theme]}\n")
window.close()
del window