Merge pull request 'improved code and folder structure' (#22) from improved_code into main

Reviewed-on: Fabelous/MPENN#22
This commit is contained in:
Falko Victor Habel 2024-04-03 16:04:20 +00:00
commit e48ae11d70
3 changed files with 274 additions and 270 deletions

View File

@ -1,271 +1,275 @@
# this is a modified version of the CTkColorPicker from GitHub """Fabelous CTK Color Picker for customtkinter
----------------------------------------
import customtkinter as Ctk Based on the original work by Akash Bora (Akascape)
from PIL import Image, ImageTk Contributions from Victor Vimbert-Guerlais (helloHackYnow)
import os Original: https://github.com/Akascape/CTkColorPicker/tree/main
import math """
PATH = os.path.dirname(os.path.realpath(__file__))
import customtkinter as Ctk
class AskColor(Ctk.CTkToplevel): from PIL import Image, ImageTk
import os
def __init__(self, import math
master,
font = None,
width: int = 250, class AskColor(Ctk.CTkToplevel):
title: str = "Choose Color",
initial_color: str = None, def __init__(self,
text: str = "apply", master,
corner_radius: int = 16, font = None,
slider_border: int = 1, width: int = 250,
**button_kwargs): title: str = "Choose Color",
super().__init__(master, **button_kwargs) initial_color: str = None,
self.selected_color = None text: str = "apply",
self.title(title) corner_radius: int = 16,
WIDTH = width if width >= 250 else 250 slider_border: int = 1,
HEIGHT = WIDTH + 110 **button_kwargs):
self.font = font if font is not None else Ctk.CTkFont(family="kDefaultFont", size=16) super().__init__(master, **button_kwargs)
self.image_dimension = self._apply_window_scaling(WIDTH - 100) self.selected_color = None
self.target_dimension = self._apply_window_scaling(20) self.title(title)
self.initial_color = initial_color WIDTH = width if width >= 250 else 250
self.maxsize(WIDTH, HEIGHT) HEIGHT = WIDTH + 110
self.minsize(WIDTH, HEIGHT) self.font = font if font is not None else Ctk.CTkFont(family="kDefaultFont", size=16)
self.resizable(width=False, height=False) self.image_dimension = self._apply_window_scaling(WIDTH - 100)
self.transient(self.master) self.target_dimension = self._apply_window_scaling(20)
self.lift() self.initial_color = initial_color
self.after(10) self.maxsize(WIDTH, HEIGHT)
self.protocol("WM_DELETE_WINDOW", self._on_closing) self.minsize(WIDTH, HEIGHT)
self.resizable(width=False, height=False)
self.default_hex_color = "#ffffff" self.transient(self.master)
self.default_rgb = [255, 255, 255] self.lift()
self.rgb_color = self.default_rgb[:] self.after(10)
self.button_text = text self.protocol("WM_DELETE_WINDOW", self._on_closing)
self.corner_radius = corner_radius
self.slider_border = 10 if slider_border >= 10 else slider_border self.default_hex_color = "#ffffff"
self.default_rgb = [255, 255, 255]
self.frame = Ctk.CTkFrame(master=self) self.rgb_color = self.default_rgb[:]
self.frame.grid(sticky="nswe", padx=5, pady=5) self.button_text = text
self.fg_color = self.fg_color = self._apply_appearance_mode(Ctk.ThemeManager.theme["CTkFrame"]["fg_color"]) self.corner_radius = corner_radius
self.canvas = Ctk.CTkCanvas(self.frame, height=self.image_dimension, width=self.image_dimension, highlightthickness=0, bg=self.fg_color) self.slider_border = 10 if slider_border >= 10 else slider_border
self.canvas.grid(row=0, column=0, columnspan=2, pady=20)
self.frame = Ctk.CTkFrame(master=self)
self.frame.grid(sticky="nswe", padx=5, pady=5)
self.canvas.bind("<B1-Motion>", self.on_mouse_drag) self.fg_color = self.fg_color = self._apply_appearance_mode(Ctk.ThemeManager.theme["CTkFrame"]["fg_color"])
self.canvas = Ctk.CTkCanvas(self.frame, height=self.image_dimension, width=self.image_dimension, highlightthickness=0, bg=self.fg_color)
self.img1 = Image.open(os.path.join(PATH, 'color_wheel.png')).resize((self.image_dimension, self.image_dimension), Image.Resampling.LANCZOS) self.canvas.grid(row=0, column=0, columnspan=2, pady=20)
self.img2 = Image.open(os.path.join(PATH, 'target.png')).resize((self.target_dimension, self.target_dimension), Image.Resampling.LANCZOS)
self.wheel = ImageTk.PhotoImage(self.img1) self.canvas.bind("<B1-Motion>", self.on_mouse_drag)
self.target = ImageTk.PhotoImage(self.img2)
self.img1 = Image.open('icons/color_wheel.png').resize((self.image_dimension, self.image_dimension), Image.Resampling.LANCZOS)
self.canvas.create_image(self.image_dimension/2, self.image_dimension/2, image=self.wheel) self.img2 = Image.open('icons/target.png').resize((self.target_dimension, self.target_dimension), Image.Resampling.LANCZOS)
self.set_initial_color(initial_color) self.wheel = ImageTk.PhotoImage(self.img1)
self.target = ImageTk.PhotoImage(self.img2)
self.brightness_slider_value = Ctk.IntVar()
self.brightness_slider_value.set(255) self.canvas.create_image(self.image_dimension/2, self.image_dimension/2, image=self.wheel)
self.set_initial_color(initial_color)
self.slider = Ctk.CTkSlider(master=self.frame, height=20, border_width=self.slider_border,
button_length=15, progress_color=self.default_hex_color, from_=0, to=255, self.brightness_slider_value = Ctk.IntVar()
variable=self.brightness_slider_value, number_of_steps=256, self.brightness_slider_value.set(255)
button_corner_radius=self.corner_radius, corner_radius=self.corner_radius,
command=lambda x:self.update_colors()) self.slider = Ctk.CTkSlider(master=self.frame, height=20, border_width=self.slider_border,
button_length=15, progress_color=self.default_hex_color, from_=0, to=255,
self.slider.grid(row=1, column=0, columnspan=2, pady=(0, 15), padx=20-self.slider_border) variable=self.brightness_slider_value, number_of_steps=256,
button_corner_radius=self.corner_radius, corner_radius=self.corner_radius,
self.label = Ctk.CTkLabel(master=self.frame, text_color="#000000", height=50, width=75, fg_color=self.default_hex_color, command=lambda x:self.update_colors())
corner_radius=self.corner_radius, text="")
self.slider.grid(row=1, column=0, columnspan=2, pady=(0, 15), padx=20-self.slider_border)
self.color_entry = Ctk.CTkEntry(master=self.frame, height=50, corner_radius=self.corner_radius, font=self.font, width=100)
self.color_entry.configure(placeholder_text=self.default_hex_color) # Insert the new text self.label = Ctk.CTkLabel(master=self.frame, text_color="#000000", height=50, width=75, fg_color=self.default_hex_color,
corner_radius=self.corner_radius, text="")
self.color_entry = Ctk.CTkEntry(master=self.frame, height=50, corner_radius=self.corner_radius, font=self.font, width=100)
self.button = Ctk.CTkButton(master=self.frame, text=self.button_text, height=50, corner_radius=self.corner_radius,width = 200, self.color_entry.configure(placeholder_text=self.default_hex_color) # Insert the new text
command=self._ok_event, **button_kwargs)
self.label.grid(row=2, column=0, padx=(5, 20), pady=10, sticky="e") self.button = Ctk.CTkButton(master=self.frame, text=self.button_text, height=50, corner_radius=self.corner_radius,width = 200,
self.color_entry.grid(row=2, column=1, padx=(5, 5), pady=10, sticky="w") command=self._ok_event, **button_kwargs)
self.button.grid(row=3, column=0, columnspan=3, padx=5, pady=10)
self.after(150, lambda: self.label.focus())
self.label.grid(row=2, column=0, padx=(5, 20), pady=10, sticky="e")
self.grab_set() self.color_entry.grid(row=2, column=1, padx=(5, 5), pady=10, sticky="w")
self.button.grid(row=3, column=0, columnspan=3, padx=5, pady=10)
self.after(150, lambda: self.label.focus())
def get(self):
# Use the stored selected_color instead of accessing the widget self.grab_set()
return self.selected_color
def _ok_event(self, event=None): def get(self):
input_string = self.color_entry.get() # Use the stored selected_color instead of accessing the widget
self.selected_color = self.check_rgb_hex_color(input_string) # Store the selected color return self.selected_color
self.grab_release()
self.destroy() def _ok_event(self, event=None):
del self.img1 input_string = self.color_entry.get()
del self.img2 self.selected_color = self.check_rgb_hex_color(input_string) # Store the selected color
del self.wheel self.grab_release()
del self.target self.destroy()
del self.img1
def check_rgb_hex_color(self, input_string): del self.img2
if not input_string: del self.wheel
return self.default_hex_color del self.target
if not input_string.startswith("#"):
input_string = "#" + input_string def check_rgb_hex_color(self, input_string):
hex_color = input_string.lstrip("#") if not input_string:
if len(hex_color) != 6: return self.default_hex_color
return None if not input_string.startswith("#"):
try: input_string = "#" + input_string
int(hex_color, 16) hex_color = input_string.lstrip("#")
return input_string if len(hex_color) != 6:
except ValueError: return None
return None try:
int(hex_color, 16)
return input_string
def _on_closing(self): except ValueError:
self._color = None return None
self.grab_release()
self.destroy()
del self.img1 def _on_closing(self):
del self.img2 self._color = None
del self.wheel self.grab_release()
del self.target self.destroy()
del self.img1
def on_mouse_drag(self, event): del self.img2
x = event.x del self.wheel
y = event.y del self.target
self.canvas.delete("all")
self.canvas.create_image(self.image_dimension/2, self.image_dimension/2, image=self.wheel) def on_mouse_drag(self, event):
x = event.x
d_from_center = math.sqrt(((self.image_dimension/2)-x)**2 + ((self.image_dimension/2)-y)**2) y = event.y
self.canvas.delete("all")
if d_from_center < self.image_dimension/2: self.canvas.create_image(self.image_dimension/2, self.image_dimension/2, image=self.wheel)
self.target_x, self.target_y = x, y
else: d_from_center = math.sqrt(((self.image_dimension/2)-x)**2 + ((self.image_dimension/2)-y)**2)
self.target_x, self.target_y = self.projection_on_circle(x, y, self.image_dimension/2, self.image_dimension/2, self.image_dimension/2 -1)
if d_from_center < self.image_dimension/2:
self.canvas.create_image(self.target_x, self.target_y, image=self.target) self.target_x, self.target_y = x, y
else:
self.get_target_color() self.target_x, self.target_y = self.projection_on_circle(x, y, self.image_dimension/2, self.image_dimension/2, self.image_dimension/2 -1)
self.update_colors()
self.canvas.create_image(self.target_x, self.target_y, image=self.target)
def get_target_color(self):
try: self.get_target_color()
self.rgb_color = self.img1.getpixel((self.target_x, self.target_y)) self.update_colors()
r = self.rgb_color[0] def get_target_color(self):
g = self.rgb_color[1] try:
b = self.rgb_color[2] self.rgb_color = self.img1.getpixel((self.target_x, self.target_y))
self.rgb_color = [r, g, b]
r = self.rgb_color[0]
except AttributeError: g = self.rgb_color[1]
self.rgb_color = self.default_rgb b = self.rgb_color[2]
self.rgb_color = [r, g, b]
def update_colors(self):
brightness = self.brightness_slider_value.get() except AttributeError:
self.rgb_color = self.default_rgb
self.get_target_color()
def update_colors(self):
r = int(self.rgb_color[0] * (brightness/255)) brightness = self.brightness_slider_value.get()
g = int(self.rgb_color[1] * (brightness/255))
b = int(self.rgb_color[2] * (brightness/255)) self.get_target_color()
self.rgb_color = [r, g, b] r = int(self.rgb_color[0] * (brightness/255))
g = int(self.rgb_color[1] * (brightness/255))
self.default_hex_color = "#{:02x}{:02x}{:02x}".format(*self.rgb_color) b = int(self.rgb_color[2] * (brightness/255))
self.slider.configure(progress_color=self.default_hex_color) self.rgb_color = [r, g, b]
self.label.configure(fg_color=self.default_hex_color)
self.default_hex_color = "#{:02x}{:02x}{:02x}".format(*self.rgb_color)
self.color_entry.delete(0, Ctk.END) # Delete any existing text in the color_entry widget
self.color_entry.configure(placeholder_text=self.default_hex_color) # Insert the new text self.slider.configure(progress_color=self.default_hex_color)
self.label.configure(fg_color=self.default_hex_color)
def projection_on_circle(self, point_x, point_y, circle_x, circle_y, radius): self.color_entry.delete(0, Ctk.END) # Delete any existing text in the color_entry widget
angle = math.atan2(point_y - circle_y, point_x - circle_x) self.color_entry.configure(placeholder_text=self.default_hex_color) # Insert the new text
projection_x = circle_x + radius * math.cos(angle)
projection_y = circle_y + radius * math.sin(angle)
def projection_on_circle(self, point_x, point_y, circle_x, circle_y, radius):
return projection_x, projection_y angle = math.atan2(point_y - circle_y, point_x - circle_x)
projection_x = circle_x + radius * math.cos(angle)
def set_initial_color(self, initial_color): projection_y = circle_y + radius * math.sin(angle)
"""
Sets the initial color of the target if it matches the specified `initial_color`. return projection_x, projection_y
Falls back to the center of the image if no matching color is found or `initial_color` is invalid.
def set_initial_color(self, initial_color):
Parameters: """
- initial_color (str): The hexadecimal color value (e.g., '#RRGGBB') to match in the image. Sets the initial color of the target if it matches the specified `initial_color`.
Falls back to the center of the image if no matching color is found or `initial_color` is invalid.
Note: This method is in beta stage and may not accurately handle all colors.
""" Parameters:
if not initial_color or not initial_color.startswith("#"): - initial_color (str): The hexadecimal color value (e.g., '#RRGGBB') to match in the image.
self._place_image_at_center()
return Note: This method is in beta stage and may not accurately handle all colors.
"""
r, g, b = self._hex_to_rgb(initial_color) if not initial_color or not initial_color.startswith("#"):
self._place_image_at_center()
for i in range(self.image_dimension): return
for j in range(self.image_dimension):
if self._pixel_matches_color(i, j, (r, g, b)): r, g, b = self._hex_to_rgb(initial_color)
self._place_image_at(i, j)
return for i in range(self.image_dimension):
for j in range(self.image_dimension):
self._place_image_at_center() if self._pixel_matches_color(i, j, (r, g, b)):
self._place_image_at(i, j)
def _hex_to_rgb(self, hex_color): return
"""
Converts a hexadecimal color to an RGB tuple. self._place_image_at_center()
Parameters: def _hex_to_rgb(self, hex_color):
- hex_color (str): The hexadecimal color string (e.g., '#RRGGBB'). """
Converts a hexadecimal color to an RGB tuple.
Returns:
(tuple): A tuple containing the RGB values (r, g, b). Parameters:
""" - hex_color (str): The hexadecimal color string (e.g., '#RRGGBB').
try:
return tuple(int(hex_color.lstrip('#')[i:i+2], 16) for i in (0, 2, 4)) Returns:
except ValueError: (tuple): A tuple containing the RGB values (r, g, b).
return None """
try:
def _pixel_matches_color(self, x, y, color): return tuple(int(hex_color.lstrip('#')[i:i+2], 16) for i in (0, 2, 4))
""" except ValueError:
Checks if the pixel at (x, y) matches the specified `color`. return None
Parameters: def _pixel_matches_color(self, x, y, color):
- x (int): The x-coordinate of the pixel. """
- y (int): The y-coordinate of the pixel. Checks if the pixel at (x, y) matches the specified `color`.
- color (tuple): The RGB tuple to match.
Parameters:
Returns: - x (int): The x-coordinate of the pixel.
(bool): True if the pixel matches the `color`; False otherwise. - y (int): The y-coordinate of the pixel.
""" - color (tuple): The RGB tuple to match.
try:
return self.img1.getpixel((x, y))[:3] == color Returns:
except IndexError: (bool): True if the pixel matches the `color`; False otherwise.
# Outside the image bounds """
return False try:
return self.img1.getpixel((x, y))[:3] == color
def _place_image_at(self, x, y): except IndexError:
""" # Outside the image bounds
Places the image at the specified (x, y) coordinates. return False
Parameters: def _place_image_at(self, x, y):
- x (int): The x-coordinate where the image should be placed. """
- y (int): The y-coordinate where the image should be placed. Places the image at the specified (x, y) coordinates.
"""
self.default_hex_color = self.initial_color Parameters:
self.canvas.create_image(x, y, image=self.target) - x (int): The x-coordinate where the image should be placed.
self.target_x = x - y (int): The y-coordinate where the image should be placed.
self.target_y = y """
self.default_hex_color = self.initial_color
def _place_image_at_center(self): self.canvas.create_image(x, y, image=self.target)
""" self.target_x = x
Places the image at the center of the canvas. self.target_y = y
"""
center = self.image_dimension // 2 def _place_image_at_center(self):
self.canvas.create_image(center, center, image=self.target) """
Places the image at the center of the canvas.
if __name__ == "__main__": """
app = AskColor(master=None, font=None) center = self.image_dimension // 2
app.wait_window(app) # This waits until the window is closed self.canvas.create_image(center, center, image=self.target)
color = app.get() # Access the color stored before the window was destroyed
if __name__ == "__main__":
app = AskColor(master=None, font=None)
app.wait_window(app) # This waits until the window is closed
color = app.get() # Access the color stored before the window was destroyed
print("Selected color:", color) print("Selected color:", color)

View File

Before

Width:  |  Height:  |  Size: 109 KiB

After

Width:  |  Height:  |  Size: 109 KiB

View File

Before

Width:  |  Height:  |  Size: 933 B

After

Width:  |  Height:  |  Size: 933 B