mirror of
https://github.com/TomSchimansky/CustomTkinter.git
synced 2023-08-10 21:13:13 +03:00
Compare commits
9 Commits
Author | SHA1 | Date | |
---|---|---|---|
f587109618 | |||
8bfd763786 | |||
2a0ae06426 | |||
16b9ce3c5f | |||
e15bc5933d | |||
11c7363d28 | |||
d4d0cf1188 | |||
22b4dfb2d3 | |||
9146e02718 |
18
CHANGELOG.md
18
CHANGELOG.md
@ -4,6 +4,24 @@ All notable changes to this project will be documented in this file.
|
|||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## [4.5.0] - 2022-06-23
|
||||||
|
### Added
|
||||||
|
- CTkScrollbar (vertical, horizontal)
|
||||||
|
|
||||||
|
## [4.4.0] - 2022-06-14
|
||||||
|
### Changed
|
||||||
|
- Changed custom dropdown menu to normal tkinter.Menu because of multiple platform specific bugs
|
||||||
|
|
||||||
|
## [4.3.0] - 2022-06-1
|
||||||
|
### Added
|
||||||
|
- Added CTkComboBox
|
||||||
|
- Small fixes for new dropdown menu
|
||||||
|
|
||||||
|
## [4.2.0] - 2022-05-30
|
||||||
|
### Added
|
||||||
|
- CTkOptionMenu with custom dropdown menu
|
||||||
|
- Support for clicking on labels of CTkCheckBox, CTkRadioButton, CTkSwitch
|
||||||
|
|
||||||
## [4.1.0] - 2022-05-24
|
## [4.1.0] - 2022-05-24
|
||||||
### Added
|
### Added
|
||||||
- Configure width and height for frame, button, label, progressbar, slider, entry
|
- Configure width and height for frame, button, label, progressbar, slider, entry
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
__version__ = "4.5.0"
|
__version__ = "4.5.1"
|
||||||
|
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
@ -49,6 +49,7 @@ if FontManager.load_font(os.path.join(script_directory, "assets", "fonts", "Cust
|
|||||||
DrawEngine.preferred_drawing_method = "circle_shapes"
|
DrawEngine.preferred_drawing_method = "circle_shapes"
|
||||||
|
|
||||||
# import widgets
|
# import widgets
|
||||||
|
from .widgets.widget_base_class import CTkBaseClass
|
||||||
from .widgets.ctk_button import CTkButton
|
from .widgets.ctk_button import CTkButton
|
||||||
from .widgets.ctk_checkbox import CTkCheckBox
|
from .widgets.ctk_checkbox import CTkCheckBox
|
||||||
from .widgets.ctk_entry import CTkEntry
|
from .widgets.ctk_entry import CTkEntry
|
||||||
@ -62,6 +63,7 @@ from .widgets.ctk_switch import CTkSwitch
|
|||||||
from .widgets.ctk_optionmenu import CTkOptionMenu
|
from .widgets.ctk_optionmenu import CTkOptionMenu
|
||||||
from .widgets.ctk_combobox import CTkComboBox
|
from .widgets.ctk_combobox import CTkComboBox
|
||||||
from .widgets.ctk_scrollbar import CTkScrollbar
|
from .widgets.ctk_scrollbar import CTkScrollbar
|
||||||
|
from .widgets.ctk_textbox import CTkTextbox
|
||||||
|
|
||||||
# import windows
|
# import windows
|
||||||
from .windows.ctk_tk import CTk
|
from .windows.ctk_tk import CTk
|
||||||
|
@ -50,7 +50,10 @@ class AppearanceModeTracker:
|
|||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def remove(cls, callback: Callable):
|
def remove(cls, callback: Callable):
|
||||||
|
try:
|
||||||
cls.callback_list.remove(callback)
|
cls.callback_list.remove(callback)
|
||||||
|
except ValueError:
|
||||||
|
return
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def detect_appearance_mode() -> int:
|
def detect_appearance_mode() -> int:
|
||||||
|
@ -120,6 +120,13 @@ class CTkComboBox(CTkBaseClass):
|
|||||||
def set_scaling(self, *args, **kwargs):
|
def set_scaling(self, *args, **kwargs):
|
||||||
super().set_scaling(*args, **kwargs)
|
super().set_scaling(*args, **kwargs)
|
||||||
|
|
||||||
|
# change entry font size and grid padding
|
||||||
|
left_section_width = self._current_width - self._current_height
|
||||||
|
self.entry.configure(font=self.apply_font_scaling(self.text_font))
|
||||||
|
self.entry.grid(row=0, column=0, rowspan=1, columnspan=1, sticky="ew",
|
||||||
|
padx=(max(self.apply_widget_scaling(self.corner_radius), self.apply_widget_scaling(3)),
|
||||||
|
max(self.apply_widget_scaling(self._current_width - left_section_width + 3), self.apply_widget_scaling(3))))
|
||||||
|
|
||||||
self.canvas.configure(width=self.apply_widget_scaling(self._desired_width),
|
self.canvas.configure(width=self.apply_widget_scaling(self._desired_width),
|
||||||
height=self.apply_widget_scaling(self._desired_height))
|
height=self.apply_widget_scaling(self._desired_height))
|
||||||
self.draw()
|
self.draw()
|
||||||
|
@ -218,3 +218,9 @@ class CTkEntry(CTkBaseClass):
|
|||||||
return ""
|
return ""
|
||||||
else:
|
else:
|
||||||
return self.entry.get()
|
return self.entry.get()
|
||||||
|
|
||||||
|
def focus(self):
|
||||||
|
self.entry.focus()
|
||||||
|
|
||||||
|
def focus_force(self):
|
||||||
|
self.entry.focus_force()
|
||||||
|
@ -126,9 +126,12 @@ class CTkOptionMenu(CTkBaseClass):
|
|||||||
def set_scaling(self, *args, **kwargs):
|
def set_scaling(self, *args, **kwargs):
|
||||||
super().set_scaling(*args, **kwargs)
|
super().set_scaling(*args, **kwargs)
|
||||||
|
|
||||||
if self.text_label is not None:
|
# change label text size and grid padding
|
||||||
self.text_label.destroy()
|
left_section_width = self._current_width - self._current_height
|
||||||
self.text_label = None
|
self.text_label.configure(font=self.apply_font_scaling(self.text_font))
|
||||||
|
self.text_label.grid(row=0, column=0, sticky="w",
|
||||||
|
padx=(max(self.apply_widget_scaling(self.corner_radius), self.apply_widget_scaling(3)),
|
||||||
|
max(self.apply_widget_scaling(self._current_width - left_section_width + 3), self.apply_widget_scaling(3))))
|
||||||
|
|
||||||
self.canvas.configure(width=self.apply_widget_scaling(self._desired_width),
|
self.canvas.configure(width=self.apply_widget_scaling(self._desired_width),
|
||||||
height=self.apply_widget_scaling(self._desired_height))
|
height=self.apply_widget_scaling(self._desired_height))
|
||||||
@ -153,9 +156,6 @@ class CTkOptionMenu(CTkBaseClass):
|
|||||||
self.apply_widget_scaling(self._current_height / 2),
|
self.apply_widget_scaling(self._current_height / 2),
|
||||||
self.apply_widget_scaling(self._current_height / 3))
|
self.apply_widget_scaling(self._current_height / 3))
|
||||||
|
|
||||||
if self.current_value is not None:
|
|
||||||
self.text_label.configure(text=self.current_value)
|
|
||||||
|
|
||||||
if no_color_updates is False or requires_recoloring or requires_recoloring_2:
|
if no_color_updates is False or requires_recoloring or requires_recoloring_2:
|
||||||
|
|
||||||
self.canvas.configure(bg=ThemeManager.single_color(self.bg_color, self._appearance_mode))
|
self.canvas.configure(bg=ThemeManager.single_color(self.bg_color, self._appearance_mode))
|
||||||
@ -287,10 +287,7 @@ class CTkOptionMenu(CTkBaseClass):
|
|||||||
def set(self, value: str, from_variable_callback: bool = False):
|
def set(self, value: str, from_variable_callback: bool = False):
|
||||||
self.current_value = value
|
self.current_value = value
|
||||||
|
|
||||||
if self.text_label is not None:
|
|
||||||
self.text_label.configure(text=self.current_value)
|
self.text_label.configure(text=self.current_value)
|
||||||
else:
|
|
||||||
self.draw()
|
|
||||||
|
|
||||||
if self.variable is not None and not from_variable_callback:
|
if self.variable is not None and not from_variable_callback:
|
||||||
self.variable_callback_blocked = True
|
self.variable_callback_blocked = True
|
||||||
|
159
customtkinter/widgets/ctk_textbox.py
Normal file
159
customtkinter/widgets/ctk_textbox.py
Normal file
@ -0,0 +1,159 @@
|
|||||||
|
import tkinter
|
||||||
|
|
||||||
|
from .ctk_canvas import CTkCanvas
|
||||||
|
from ..theme_manager import ThemeManager
|
||||||
|
from ..draw_engine import DrawEngine
|
||||||
|
from .widget_base_class import CTkBaseClass
|
||||||
|
|
||||||
|
|
||||||
|
class CTkTextbox(CTkBaseClass):
|
||||||
|
def __init__(self, *args,
|
||||||
|
bg_color=None,
|
||||||
|
fg_color="default_theme",
|
||||||
|
border_color="default_theme",
|
||||||
|
border_width="default_theme",
|
||||||
|
corner_radius="default_theme",
|
||||||
|
text_font="default_theme",
|
||||||
|
text_color="default_theme",
|
||||||
|
width=200,
|
||||||
|
height=200,
|
||||||
|
**kwargs):
|
||||||
|
|
||||||
|
# transfer basic functionality (bg_color, size, _appearance_mode, scaling) to CTkBaseClass
|
||||||
|
super().__init__(*args, bg_color=bg_color, width=width, height=height, **kwargs)
|
||||||
|
|
||||||
|
# color
|
||||||
|
self.fg_color = ThemeManager.theme["color"]["entry"] if fg_color == "default_theme" else fg_color
|
||||||
|
self.border_color = ThemeManager.theme["color"]["frame_border"] if border_color == "default_theme" else border_color
|
||||||
|
|
||||||
|
# shape
|
||||||
|
self.corner_radius = ThemeManager.theme["shape"]["frame_corner_radius"] if corner_radius == "default_theme" else corner_radius
|
||||||
|
self.border_width = ThemeManager.theme["shape"]["frame_border_width"] if border_width == "default_theme" else border_width
|
||||||
|
|
||||||
|
self.text_color = ThemeManager.theme["color"]["text"] if text_color == "default_theme" else text_color
|
||||||
|
self.text_font = (ThemeManager.theme["text"]["font"], ThemeManager.theme["text"]["size"]) if text_font == "default_theme" else text_font
|
||||||
|
|
||||||
|
# configure 1x1 grid
|
||||||
|
self.grid_rowconfigure(0, weight=1)
|
||||||
|
self.grid_columnconfigure(0, weight=1)
|
||||||
|
|
||||||
|
self.canvas = CTkCanvas(master=self,
|
||||||
|
highlightthickness=0,
|
||||||
|
width=self.apply_widget_scaling(self._current_width),
|
||||||
|
height=self.apply_widget_scaling(self._current_height))
|
||||||
|
self.canvas.grid(row=0, column=0, padx=0, pady=0, rowspan=1, columnspan=1, sticky="nsew")
|
||||||
|
self.canvas.configure(bg=ThemeManager.single_color(self.bg_color, self._appearance_mode))
|
||||||
|
self.draw_engine = DrawEngine(self.canvas)
|
||||||
|
|
||||||
|
self.textbox = tkinter.Text(self,
|
||||||
|
fg=ThemeManager.single_color(self.text_color, self._appearance_mode),
|
||||||
|
width=0,
|
||||||
|
height=0,
|
||||||
|
font=self.text_font,
|
||||||
|
highlightthickness=0,
|
||||||
|
bg=ThemeManager.single_color(self.fg_color, self._appearance_mode),
|
||||||
|
**kwargs)
|
||||||
|
self.textbox.grid(row=0, column=0, padx=self.corner_radius, pady=self.corner_radius, rowspan=1, columnspan=1, sticky="nsew")
|
||||||
|
|
||||||
|
self.bind('<Configure>', self.update_dimensions_event)
|
||||||
|
|
||||||
|
self.draw()
|
||||||
|
|
||||||
|
def winfo_children(self):
|
||||||
|
""" winfo_children of CTkFrame without self.canvas widget,
|
||||||
|
because it's not a child but part of the CTkFrame itself """
|
||||||
|
|
||||||
|
child_widgets = super().winfo_children()
|
||||||
|
try:
|
||||||
|
child_widgets.remove(self.canvas)
|
||||||
|
return child_widgets
|
||||||
|
except ValueError:
|
||||||
|
return child_widgets
|
||||||
|
|
||||||
|
def set_scaling(self, *args, **kwargs):
|
||||||
|
super().set_scaling(*args, **kwargs)
|
||||||
|
|
||||||
|
self.canvas.configure(width=self.apply_widget_scaling(self._desired_width), height=self.apply_widget_scaling(self._desired_height))
|
||||||
|
self.draw()
|
||||||
|
|
||||||
|
def set_dimensions(self, width=None, height=None):
|
||||||
|
super().set_dimensions(width, height)
|
||||||
|
|
||||||
|
self.canvas.configure(width=self.apply_widget_scaling(self._desired_width),
|
||||||
|
height=self.apply_widget_scaling(self._desired_height))
|
||||||
|
self.draw()
|
||||||
|
|
||||||
|
def draw(self, no_color_updates=False):
|
||||||
|
|
||||||
|
requires_recoloring = self.draw_engine.draw_rounded_rect_with_border(self.apply_widget_scaling(self._current_width),
|
||||||
|
self.apply_widget_scaling(self._current_height),
|
||||||
|
self.apply_widget_scaling(self.corner_radius),
|
||||||
|
self.apply_widget_scaling(self.border_width))
|
||||||
|
|
||||||
|
if no_color_updates is False or requires_recoloring:
|
||||||
|
if self.fg_color is None:
|
||||||
|
self.canvas.itemconfig("inner_parts",
|
||||||
|
fill=ThemeManager.single_color(self.bg_color, self._appearance_mode),
|
||||||
|
outline=ThemeManager.single_color(self.bg_color, self._appearance_mode))
|
||||||
|
else:
|
||||||
|
self.canvas.itemconfig("inner_parts",
|
||||||
|
fill=ThemeManager.single_color(self.fg_color, self._appearance_mode),
|
||||||
|
outline=ThemeManager.single_color(self.fg_color, self._appearance_mode))
|
||||||
|
|
||||||
|
self.canvas.itemconfig("border_parts",
|
||||||
|
fill=ThemeManager.single_color(self.border_color, self._appearance_mode),
|
||||||
|
outline=ThemeManager.single_color(self.border_color, self._appearance_mode))
|
||||||
|
self.canvas.configure(bg=ThemeManager.single_color(self.bg_color, self._appearance_mode))
|
||||||
|
|
||||||
|
self.canvas.tag_lower("inner_parts")
|
||||||
|
self.canvas.tag_lower("border_parts")
|
||||||
|
|
||||||
|
def configure(self, *args, **kwargs):
|
||||||
|
require_redraw = False # some attribute changes require a call of self.draw() at the end
|
||||||
|
|
||||||
|
if "fg_color" in kwargs:
|
||||||
|
self.fg_color = kwargs["fg_color"]
|
||||||
|
require_redraw = True
|
||||||
|
del kwargs["fg_color"]
|
||||||
|
|
||||||
|
# check if CTk widgets are children of the frame and change their bg_color to new frame fg_color
|
||||||
|
for child in self.winfo_children():
|
||||||
|
if isinstance(child, CTkBaseClass):
|
||||||
|
child.configure(bg_color=self.fg_color)
|
||||||
|
|
||||||
|
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"]
|
||||||
|
|
||||||
|
if "border_color" in kwargs:
|
||||||
|
self.border_color = kwargs["border_color"]
|
||||||
|
require_redraw = True
|
||||||
|
del kwargs["border_color"]
|
||||||
|
|
||||||
|
if "corner_radius" in kwargs:
|
||||||
|
self.corner_radius = kwargs["corner_radius"]
|
||||||
|
require_redraw = True
|
||||||
|
del kwargs["corner_radius"]
|
||||||
|
|
||||||
|
if "border_width" in kwargs:
|
||||||
|
self.border_width = kwargs["border_width"]
|
||||||
|
require_redraw = True
|
||||||
|
del kwargs["border_width"]
|
||||||
|
|
||||||
|
if "width" in kwargs:
|
||||||
|
self.set_dimensions(width=kwargs["width"])
|
||||||
|
del kwargs["width"]
|
||||||
|
|
||||||
|
if "height" in kwargs:
|
||||||
|
self.set_dimensions(height=kwargs["height"])
|
||||||
|
del kwargs["height"]
|
||||||
|
|
||||||
|
self.textbox.configure(*args, **kwargs)
|
||||||
|
|
||||||
|
if require_redraw:
|
||||||
|
self.draw()
|
@ -138,7 +138,7 @@ class CTkBaseClass(tkinter.Frame):
|
|||||||
self.draw()
|
self.draw()
|
||||||
|
|
||||||
def update_dimensions_event(self, event):
|
def update_dimensions_event(self, event):
|
||||||
# only redraw if dimensions changed (for performance)
|
# only redraw if dimensions changed (for performance), independent of scaling
|
||||||
if round(self._current_width) != round(event.width / self._widget_scaling) or round(self._current_height) != round(event.height / self._widget_scaling):
|
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_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
|
self._current_height = (event.height / self._widget_scaling) # _current_width and _current_height are independent of the scale
|
||||||
|
@ -7,7 +7,7 @@ build-backend = "setuptools.build_meta"
|
|||||||
github_url = "https://github.com/TomSchimansky/CustomTkinter"
|
github_url = "https://github.com/TomSchimansky/CustomTkinter"
|
||||||
|
|
||||||
[tool.tbump.version]
|
[tool.tbump.version]
|
||||||
current = "4.5.0"
|
current = "4.5.1"
|
||||||
|
|
||||||
# Example of a semver regexp.
|
# Example of a semver regexp.
|
||||||
# Make sure this matches current_version before
|
# Make sure this matches current_version before
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
[metadata]
|
[metadata]
|
||||||
name = customtkinter
|
name = customtkinter
|
||||||
version = 4.5.0
|
version = 4.5.1
|
||||||
description = Create modern looking GUIs with Python
|
description = Create modern looking GUIs with Python
|
||||||
long_description = CustomTkinter UI-Library\n\n[](https://github.com/TomSchimansky/CustomTkinter/blob/master/documentation_images/Windows_dark.png)\n\nMore Information: https://github.com/TomSchimansky/CustomTkinter
|
long_description = CustomTkinter UI-Library\n\n[](https://github.com/TomSchimansky/CustomTkinter/blob/master/documentation_images/Windows_dark.png)\n\nMore Information: https://github.com/TomSchimansky/CustomTkinter
|
||||||
long_description_content_type = text/markdown
|
long_description_content_type = text/markdown
|
||||||
|
@ -4,7 +4,7 @@ import customtkinter
|
|||||||
|
|
||||||
app = customtkinter.CTk()
|
app = customtkinter.CTk()
|
||||||
app.title('Test OptionMenu ComboBox.py')
|
app.title('Test OptionMenu ComboBox.py')
|
||||||
app.geometry('400x300')
|
app.geometry('400x500')
|
||||||
|
|
||||||
|
|
||||||
def select_callback(choice):
|
def select_callback(choice):
|
||||||
@ -33,4 +33,12 @@ combobox_tk.pack(pady=10, padx=10)
|
|||||||
combobox_1 = customtkinter.CTkComboBox(app, variable=None, values=countries, command=select_callback, width=300)
|
combobox_1 = customtkinter.CTkComboBox(app, variable=None, values=countries, command=select_callback, width=300)
|
||||||
combobox_1.pack(pady=20, padx=10)
|
combobox_1.pack(pady=20, padx=10)
|
||||||
|
|
||||||
|
def set_new_scaling(scaling):
|
||||||
|
customtkinter.set_spacing_scaling(scaling)
|
||||||
|
customtkinter.set_window_scaling(scaling)
|
||||||
|
customtkinter.set_widget_scaling(scaling)
|
||||||
|
|
||||||
|
scaling_slider = customtkinter.CTkSlider(app, command=set_new_scaling, from_=0, to=2)
|
||||||
|
scaling_slider.pack(pady=20, padx=10)
|
||||||
|
|
||||||
app.mainloop()
|
app.mainloop()
|
||||||
|
11
test/manual_integration_tests/test_textbox.py
Normal file
11
test/manual_integration_tests/test_textbox.py
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import customtkinter
|
||||||
|
|
||||||
|
app = customtkinter.CTk()
|
||||||
|
app.grid_rowconfigure(0, weight=1)
|
||||||
|
app.grid_columnconfigure(0, weight=1)
|
||||||
|
|
||||||
|
textbox = customtkinter.CTkTextbox(app)
|
||||||
|
textbox.grid(row=0, column=0, padx=20, pady=20, sticky="nsew")
|
||||||
|
|
||||||
|
|
||||||
|
app.mainloop()
|
Reference in New Issue
Block a user