2022-04-20 23:50:57 +03:00
|
|
|
import tkinter
|
|
|
|
import tkinter.ttk as ttk
|
2022-04-21 19:34:58 +03:00
|
|
|
import copy
|
|
|
|
import re
|
2022-05-04 16:38:13 +03:00
|
|
|
import math
|
2022-05-02 00:29:14 +03:00
|
|
|
from typing import Callable, Union, TypedDict
|
2022-04-20 23:50:57 +03:00
|
|
|
|
2022-05-02 00:29:14 +03:00
|
|
|
from ..windows.ctk_tk import CTk
|
|
|
|
from ..windows.ctk_toplevel import CTkToplevel
|
2022-04-20 23:50:57 +03:00
|
|
|
from ..appearance_mode_tracker import AppearanceModeTracker
|
2022-04-21 19:34:58 +03:00
|
|
|
from ..scaling_tracker import ScalingTracker
|
|
|
|
from ..theme_manager import CTkThemeManager
|
2022-04-20 23:50:57 +03:00
|
|
|
|
|
|
|
|
|
|
|
class CTkBaseClass(tkinter.Frame):
|
|
|
|
def __init__(self, *args, bg_color=None, width, height, **kwargs):
|
|
|
|
super().__init__(*args, width=width, height=height, **kwargs) # set desired size of underlying tkinter.Frame
|
|
|
|
|
|
|
|
self.bg_color = self.detect_color_of_master() if bg_color is None else bg_color
|
2022-05-02 00:29:14 +03:00
|
|
|
|
|
|
|
self.current_width = width # current_width and current_height in pixel, represent current size of the widget (not the desired size by init)
|
|
|
|
self.current_height = height # current_width and current_height are independent of the scale
|
|
|
|
self.desired_width = width
|
|
|
|
self.desired_height = height
|
2022-04-21 19:34:58 +03:00
|
|
|
|
|
|
|
# add set_scaling method to callback list of ScalingTracker for automatic scaling changes
|
|
|
|
ScalingTracker.add_widget(self.set_scaling, self)
|
2022-05-02 00:29:14 +03:00
|
|
|
self.widget_scaling = ScalingTracker.get_widget_scaling(self)
|
|
|
|
self.spacing_scaling = ScalingTracker.get_spacing_scaling(self)
|
|
|
|
|
2022-05-04 16:38:13 +03:00
|
|
|
super().configure(width=self.round_size(self.apply_widget_scaling(self.desired_width)),
|
|
|
|
height=self.round_size(self.apply_widget_scaling(self.desired_height)))
|
2022-05-02 00:38:02 +03:00
|
|
|
|
2022-05-02 00:29:14 +03:00
|
|
|
# save latest geometry function and kwargs
|
|
|
|
class GeometryCallDict(TypedDict):
|
|
|
|
function: Callable
|
|
|
|
kwargs: dict
|
|
|
|
|
|
|
|
self.last_geometry_manager_call: Union[GeometryCallDict, None] = None
|
2022-04-20 23:50:57 +03:00
|
|
|
|
|
|
|
# add set_appearance_mode method to callback list of AppearanceModeTracker for appearance mode changes
|
|
|
|
AppearanceModeTracker.add(self.set_appearance_mode, self)
|
|
|
|
self.appearance_mode = AppearanceModeTracker.get_mode() # 0: "Light" 1: "Dark"
|
|
|
|
|
2022-04-21 19:34:58 +03:00
|
|
|
super().configure(bg=CTkThemeManager.single_color(self.bg_color, self.appearance_mode))
|
|
|
|
|
2022-04-20 23:50:57 +03:00
|
|
|
# overwrite configure methods of master when master is tkinter widget, so that bg changes get applied on child CTk widget too
|
|
|
|
if isinstance(self.master, (tkinter.Tk, tkinter.Toplevel, tkinter.Frame)) and not isinstance(self.master, (CTkBaseClass, CTk, CTkToplevel)):
|
|
|
|
master_old_configure = self.master.config
|
|
|
|
|
|
|
|
def new_configure(*args, **kwargs):
|
|
|
|
if "bg" in kwargs:
|
|
|
|
self.configure(bg_color=kwargs["bg"])
|
|
|
|
elif "background" in kwargs:
|
|
|
|
self.configure(bg_color=kwargs["background"])
|
|
|
|
|
|
|
|
# args[0] is dict when attribute gets changed by widget[<attribute>] syntax
|
|
|
|
elif len(args) > 0 and type(args[0]) == dict:
|
|
|
|
if "bg" in args[0]:
|
|
|
|
self.configure(bg_color=args[0]["bg"])
|
|
|
|
elif "background" in args[0]:
|
|
|
|
self.configure(bg_color=args[0]["background"])
|
|
|
|
master_old_configure(*args, **kwargs)
|
|
|
|
|
|
|
|
self.master.config = new_configure
|
|
|
|
self.master.configure = new_configure
|
|
|
|
|
|
|
|
def destroy(self):
|
|
|
|
AppearanceModeTracker.remove(self.set_appearance_mode)
|
|
|
|
super().destroy()
|
|
|
|
|
2022-05-02 00:29:14 +03:00
|
|
|
def place(self, **kwargs):
|
|
|
|
self.last_geometry_manager_call = {"function": super().place, "kwargs": kwargs}
|
|
|
|
super().place(**self.apply_argument_scaling(kwargs))
|
|
|
|
|
|
|
|
def pack(self, **kwargs):
|
|
|
|
self.last_geometry_manager_call = {"function": super().pack, "kwargs": kwargs}
|
|
|
|
super().pack(**self.apply_argument_scaling(kwargs))
|
|
|
|
|
|
|
|
def grid(self, **kwargs):
|
|
|
|
self.last_geometry_manager_call = {"function": super().grid, "kwargs": kwargs}
|
|
|
|
super().grid(**self.apply_argument_scaling(kwargs))
|
|
|
|
|
|
|
|
def apply_argument_scaling(self, kwargs: dict) -> dict:
|
|
|
|
scaled_kwargs = copy.copy(kwargs)
|
|
|
|
|
|
|
|
if "pady" in scaled_kwargs:
|
|
|
|
if isinstance(scaled_kwargs["pady"], (int, float, str)):
|
|
|
|
scaled_kwargs["pady"] = self.apply_spacing_scaling(scaled_kwargs["pady"])
|
|
|
|
elif isinstance(scaled_kwargs["pady"], tuple):
|
|
|
|
scaled_kwargs["pady"] = tuple([self.apply_spacing_scaling(v) for v in scaled_kwargs["pady"]])
|
|
|
|
if "padx" in kwargs:
|
|
|
|
if isinstance(scaled_kwargs["padx"], (int, float, str)):
|
|
|
|
scaled_kwargs["padx"] = self.apply_spacing_scaling(scaled_kwargs["padx"])
|
|
|
|
elif isinstance(scaled_kwargs["padx"], tuple):
|
|
|
|
scaled_kwargs["padx"] = tuple([self.apply_spacing_scaling(v) for v in scaled_kwargs["padx"]])
|
|
|
|
|
|
|
|
if "x" in scaled_kwargs:
|
|
|
|
scaled_kwargs["x"] = self.apply_spacing_scaling(scaled_kwargs["x"])
|
|
|
|
if "y" in scaled_kwargs:
|
|
|
|
scaled_kwargs["y"] = self.apply_spacing_scaling(scaled_kwargs["y"])
|
|
|
|
|
|
|
|
return scaled_kwargs
|
|
|
|
|
2022-04-20 23:50:57 +03:00
|
|
|
def config(self, *args, **kwargs):
|
|
|
|
self.configure(*args, **kwargs)
|
|
|
|
|
|
|
|
def configure(self, *args, **kwargs):
|
|
|
|
""" basic configure with bg_color support, to be overridden """
|
|
|
|
|
|
|
|
require_redraw = False
|
|
|
|
|
|
|
|
if "bg_color" in kwargs:
|
|
|
|
if kwargs["bg_color"] is None:
|
|
|
|
self.bg_color = self.detect_color_of_master()
|
|
|
|
else:
|
|
|
|
self.bg_color = kwargs["bg_color"]
|
|
|
|
require_redraw = True
|
|
|
|
del kwargs["bg_color"]
|
|
|
|
|
|
|
|
super().configure(*args, **kwargs)
|
|
|
|
|
|
|
|
if require_redraw:
|
|
|
|
self.draw()
|
|
|
|
|
2022-04-21 19:34:58 +03:00
|
|
|
def update_dimensions_event(self, event):
|
2022-04-20 23:50:57 +03:00
|
|
|
# only redraw if dimensions changed (for performance)
|
2022-05-04 16:38:13 +03:00
|
|
|
if round(self.current_width) != round(event.width / self.widget_scaling) or round(self.current_height) != round(event.height / self.widget_scaling):
|
|
|
|
self.current_width = (event.width / self.widget_scaling) # adjust current size according to new size given by event
|
|
|
|
self.current_height = (event.height / self.widget_scaling) # current_width and current_height are independent of the scale
|
2022-04-20 23:50:57 +03:00
|
|
|
|
|
|
|
self.draw(no_color_updates=True) # faster drawing without color changes
|
|
|
|
|
|
|
|
def detect_color_of_master(self):
|
|
|
|
""" detect color of self.master widget to set correct bg_color """
|
|
|
|
|
|
|
|
if isinstance(self.master, CTkBaseClass) and hasattr(self.master, "fg_color"): # master is CTkFrame
|
|
|
|
return self.master.fg_color
|
|
|
|
|
|
|
|
elif isinstance(self.master, (ttk.Frame, ttk.LabelFrame, ttk.Notebook)): # master is ttk widget
|
|
|
|
try:
|
|
|
|
ttk_style = ttk.Style()
|
|
|
|
return ttk_style.lookup(self.master.winfo_class(), 'background')
|
|
|
|
except Exception:
|
|
|
|
return "#FFFFFF", "#000000"
|
|
|
|
|
|
|
|
else: # master is normal tkinter widget
|
|
|
|
try:
|
|
|
|
return self.master.cget("bg") # try to get bg color by .cget() method
|
|
|
|
except Exception:
|
|
|
|
return "#FFFFFF", "#000000"
|
|
|
|
|
|
|
|
def set_appearance_mode(self, mode_string):
|
|
|
|
if mode_string.lower() == "dark":
|
|
|
|
self.appearance_mode = 1
|
|
|
|
elif mode_string.lower() == "light":
|
|
|
|
self.appearance_mode = 0
|
|
|
|
|
|
|
|
if isinstance(self.master, (CTkBaseClass, CTk)) and hasattr(self.master, "fg_color"):
|
|
|
|
self.bg_color = self.master.fg_color
|
|
|
|
else:
|
|
|
|
self.bg_color = self.master.cget("bg")
|
|
|
|
|
|
|
|
self.draw()
|
|
|
|
|
2022-05-02 00:29:14 +03:00
|
|
|
def set_scaling(self, new_widget_scaling, new_spacing_scaling, new_window_scaling):
|
|
|
|
self.widget_scaling = new_widget_scaling
|
|
|
|
self.spacing_scaling = new_spacing_scaling
|
|
|
|
|
2022-05-04 16:38:13 +03:00
|
|
|
super().configure(width=self.round_size(self.apply_widget_scaling(self.desired_width)),
|
|
|
|
height=self.round_size(self.apply_widget_scaling(self.desired_height)))
|
2022-05-02 00:29:14 +03:00
|
|
|
|
|
|
|
if self.last_geometry_manager_call is not None:
|
|
|
|
self.last_geometry_manager_call["function"](**self.apply_argument_scaling(self.last_geometry_manager_call["kwargs"]))
|
2022-04-21 19:34:58 +03:00
|
|
|
|
2022-05-02 00:29:14 +03:00
|
|
|
def apply_widget_scaling(self, value):
|
|
|
|
if isinstance(value, (int, float)):
|
|
|
|
return value * self.widget_scaling
|
|
|
|
else:
|
|
|
|
return value
|
|
|
|
|
|
|
|
def apply_spacing_scaling(self, value):
|
|
|
|
if isinstance(value, (int, float)):
|
|
|
|
return value * self.spacing_scaling
|
|
|
|
else:
|
|
|
|
return value
|
2022-04-21 19:34:58 +03:00
|
|
|
|
|
|
|
def apply_font_scaling(self, font):
|
|
|
|
if type(font) == tuple or type(font) == list:
|
|
|
|
font_list = list(font)
|
|
|
|
for i in range(len(font_list)):
|
|
|
|
if (type(font_list[i]) == int or type(font_list[i]) == float) and font_list[i] < 0:
|
2022-05-02 00:29:14 +03:00
|
|
|
font_list[i] = int(font_list[i] * self.widget_scaling)
|
2022-04-21 19:34:58 +03:00
|
|
|
return tuple(font_list)
|
|
|
|
|
|
|
|
elif type(font) == str:
|
|
|
|
for negative_number in re.findall(r" -\d* ", font):
|
2022-05-02 00:29:14 +03:00
|
|
|
font = font.replace(negative_number, f" {int(int(negative_number) * self.widget_scaling)} ")
|
2022-04-21 19:34:58 +03:00
|
|
|
return font
|
|
|
|
|
|
|
|
elif isinstance(font, tkinter.font.Font):
|
|
|
|
new_font_object = copy.copy(font)
|
|
|
|
if font.cget("size") < 0:
|
2022-05-02 00:29:14 +03:00
|
|
|
new_font_object.config(size=int(font.cget("size") * self.widget_scaling))
|
2022-04-21 19:34:58 +03:00
|
|
|
return new_font_object
|
|
|
|
|
|
|
|
else:
|
|
|
|
return font
|
|
|
|
|
2022-05-04 16:38:13 +03:00
|
|
|
@staticmethod
|
|
|
|
def round_size(size: Union[int, float, str]):
|
|
|
|
if isinstance(size, (int, float)):
|
|
|
|
return math.floor(size / 2) * 2
|
|
|
|
else:
|
|
|
|
return size
|
|
|
|
|
2022-04-20 23:50:57 +03:00
|
|
|
def draw(self, no_color_updates=False):
|
|
|
|
""" abstract of draw method to be overridden """
|
|
|
|
pass
|
|
|
|
|
|
|
|
|