Spaces:
Running
on
Zero
Running
on
Zero
| #---------------------------------------------------------------------------------------------------------------------# | |
| # Comfyroll Studio custom nodes by RockOfFire and Akatsuzi https://github.com/Suzie1/ComfyUI_Comfyroll_CustomNodes | |
| # for ComfyUI https://github.com/comfyanonymous/ComfyUI | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| import torch | |
| import numpy as np | |
| import os | |
| import sys | |
| import io | |
| import folder_paths | |
| from PIL import Image | |
| from ..categories import icons | |
| try: | |
| import matplotlib.pyplot as plt | |
| except ImportError: | |
| import pip | |
| pip.main(['install', 'matplotlib']) | |
| import matplotlib.pyplot as plt | |
| from matplotlib.patches import RegularPolygon | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| #icons = { | |
| # "Comfyroll/Graphics/Pattern": "🧩 Comfyroll/👾 Graphics/🌈 Pattern", | |
| #} | |
| # Dictionary to map color names to RGB values | |
| color_mapping = { | |
| "white": (255, 255, 255), | |
| "black": (0, 0, 0), | |
| "red": (255, 0, 0), | |
| "green": (0, 255, 0), | |
| "blue": (0, 0, 255), | |
| "yellow": (255, 255, 0), | |
| "cyan": (0, 255, 255), | |
| "magenta": (255, 0, 255), | |
| "orange": (255, 165, 0), | |
| "purple": (128, 0, 128), | |
| "pink": (255, 192, 203), | |
| "brown": (165, 42, 42), | |
| "gray": (128, 128, 128), | |
| "lightgray": (211, 211, 211), | |
| "darkgray": (169, 169, 169), | |
| "olive": (128, 128, 0), | |
| "lime": (0, 128, 0), | |
| "teal": (0, 128, 128), | |
| "navy": (0, 0, 128), | |
| "maroon": (128, 0, 0), | |
| "fuchsia": (255, 0, 128), | |
| "aqua": (0, 255, 128), | |
| "silver": (192, 192, 192), | |
| "gold": (255, 215, 0), | |
| "turquoise": (64, 224, 208), | |
| "lavender": (230, 230, 250), | |
| "violet": (238, 130, 238), | |
| "coral": (255, 127, 80), | |
| "indigo": (75, 0, 130), | |
| } | |
| COLORS = ["custom", "white", "black", "red", "green", "blue", "yellow", | |
| "cyan", "magenta", "orange", "purple", "pink", "brown", "gray", | |
| "lightgray", "darkgray", "olive", "lime", "teal", "navy", "maroon", | |
| "fuchsia", "aqua", "silver", "gold", "turquoise", "lavender", | |
| "violet", "coral", "indigo"] | |
| STYLES = ["Accent","afmhot","autumn","binary","Blues","bone","BrBG","brg", | |
| "BuGn","BuPu","bwr","cividis","CMRmap","cool","coolwarm","copper","cubehelix","Dark2","flag", | |
| "gist_earth","gist_gray","gist_heat","gist_rainbow","gist_stern","gist_yarg","GnBu","gnuplot","gnuplot2","gray","Greens", | |
| "Greys","hot","hsv","inferno","jet","magma","nipy_spectral","ocean","Oranges","OrRd", | |
| "Paired","Pastel1","Pastel2","pink","PiYG","plasma","PRGn","prism","PuBu","PuBuGn", | |
| "PuOr","PuRd","Purples","rainbow","RdBu","RdGy","RdPu","RdYlBu","RdYlGn","Reds","seismic", | |
| "Set1","Set2","Set3","Spectral","spring","summer","tab10","tab20","tab20b","tab20c","terrain", | |
| "turbo","twilight","twilight_shifted","viridis","winter","Wistia","YlGn","YlGnBu","YlOrBr","YlOrRd"] | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| def rgb_to_hex(rgb): | |
| r, g, b = rgb | |
| return "#{:02X}{:02X}{:02X}".format(r, g, b) | |
| def hex_to_rgb(hex_color): | |
| hex_color = hex_color.lstrip('#') # Remove the '#' character, if present | |
| r = int(hex_color[0:2], 16) | |
| g = int(hex_color[2:4], 16) | |
| b = int(hex_color[4:6], 16) | |
| return (r, g, b) | |
| def tensor2pil(image): | |
| return Image.fromarray(np.clip(255. * image.cpu().numpy().squeeze(), 0, 255).astype(np.uint8)) | |
| def pil2tensor(image): | |
| return torch.from_numpy(np.array(image).astype(np.float32) / 255.0).unsqueeze(0) | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| class CR_HalftoneGrid: | |
| def INPUT_TYPES(s): | |
| return {"required": { | |
| "width": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "height": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "dot_style": (STYLES,), | |
| "reverse_dot_style": (["No", "Yes"],), | |
| "dot_frequency": ("INT", {"default": 50, "min": 1, "max":200, "step": 1}), | |
| "background_color": (COLORS,), | |
| "x_pos": ("FLOAT", {"default": 0.5, "min": 0, "max": 1, "step": .01}), | |
| "y_pos": ("FLOAT", {"default": 0.5, "min": 0, "max": 1, "step": .01}), | |
| }, | |
| "optional": { | |
| "bg_color_hex": ("STRING", {"multiline": False, "default": "#000000"}) | |
| } | |
| } | |
| RETURN_TYPES = ("IMAGE", "STRING", ) | |
| RETURN_NAMES = ("IMAGE", "show_help", ) | |
| FUNCTION = "halftone" | |
| CATEGORY = icons.get("Comfyroll/Graphics/Pattern") | |
| def halftone(self, width, height, dot_style, reverse_dot_style, dot_frequency, background_color, x_pos, y_pos, bg_color_hex='#000000'): | |
| if background_color == "custom": | |
| bgc = bg_color_hex | |
| else: | |
| bgc = background_color | |
| reverse = "" | |
| if reverse_dot_style == "Yes": | |
| reverse = "_r" | |
| fig, ax = plt.subplots(figsize=(width/100,height/100)) | |
| dotsx = np.linspace(0, 1, dot_frequency) | |
| dotsy = np.linspace(0, 1, dot_frequency) | |
| X, Y = np.meshgrid(dotsx, dotsy) | |
| dist = np.sqrt((X - x_pos)**2 + (Y - y_pos)**2) | |
| fig.patch.set_facecolor(bgc) | |
| ax.scatter(X, Y, c=dist, cmap=dot_style+reverse) | |
| plt.axis('off') | |
| plt.tight_layout(pad=0, w_pad=0, h_pad=0) | |
| plt.autoscale(tight=True) | |
| img_buf = io.BytesIO() | |
| plt.savefig(img_buf, format='png') | |
| img = Image.open(img_buf) | |
| image_out = pil2tensor(img.convert("RGB")) | |
| show_help = "https://github.com/Suzie1/ComfyUI_Comfyroll_CustomNodes/wiki/Pattern-Nodes#cr-halftone-grid" | |
| return(image_out, show_help, ) | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| class CR_ColorBars: | |
| def INPUT_TYPES(s): | |
| modes = ["2-color"] | |
| return {"required": { | |
| "mode": (modes,), | |
| "width": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "height": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "color_1": (COLORS,), | |
| "color_2": (COLORS,), | |
| "orientation": (["vertical", "horizontal", "diagonal", "alt_diagonal"],), #added 135 angle for diagonals | |
| "bar_frequency": ("INT", {"default": 5, "min": 1, "max":200, "step": 1}), | |
| "offset": ("FLOAT", {"default": 0, "min": 0, "max":20, "step": 0.05}), | |
| }, | |
| "optional": { | |
| "color1_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| "color2_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| } | |
| } | |
| RETURN_TYPES = ("IMAGE", "STRING", ) | |
| RETURN_NAMES = ("IMAGE", "show_help", ) | |
| FUNCTION = "draw" | |
| CATEGORY = icons.get("Comfyroll/Graphics/Pattern") | |
| def draw(self, mode, width, height, color_1, color_2, | |
| orientation, bar_frequency, offset=0, | |
| color1_hex='#000000', color2_hex='#000000'): | |
| # Get RGB values | |
| if color_1 == "custom": | |
| color1_rgb = hex_to_rgb(color1_hex) | |
| else: | |
| color1_rgb = color_mapping.get(color_1, (255, 255, 255)) # Default to white if the color is not found | |
| if color_2 == "custom": | |
| color2_rgb = hex_to_rgb(color2_hex) | |
| else: | |
| color2_rgb = color_mapping.get(color_2, (0, 0, 0)) # Default to black if the color is not found | |
| canvas = np.zeros((height, width, 3), dtype=np.uint8) | |
| bar_width = width / bar_frequency | |
| bar_height = height / bar_frequency | |
| offset_pixels = int(offset * max(width, height)) | |
| if orientation == "vertical": | |
| for j in range(height): | |
| for i in range(width): | |
| if ((i + offset_pixels) // bar_width) % 2 == 0: # Check for even index | |
| canvas[j, i] = color1_rgb | |
| else: | |
| canvas[j, i] = color2_rgb | |
| elif orientation == "horizontal": | |
| for j in range(height): | |
| for i in range(width): | |
| if ((j + offset_pixels) // bar_height) % 2 == 0: # Check for even index | |
| canvas[j, i] = color1_rgb | |
| else: | |
| canvas[j, i] = color2_rgb | |
| elif orientation == "diagonal": | |
| # Calculate the bar width based on a 45 degree angle | |
| bar_width = int(bar_height / np.tan(np.pi / 4)) * 2 | |
| for j in range(height): | |
| for i in range(width): | |
| # Calculate which diagonal bar the pixel belongs to with the offset | |
| bar_number = (i + j + offset_pixels) // bar_width | |
| if bar_number % 2 == 0: # Check for even bar number | |
| canvas[j, i] = color1_rgb | |
| else: | |
| canvas[j, i] = color2_rgb | |
| elif orientation == "alt_diagonal": | |
| bar_width = int(bar_height / np.tan(np.pi / 4)) * 2 | |
| for j in range(height): | |
| for i in range(width): | |
| # Calculate which diagonal bar the pixel belongs to with the offset | |
| bar_number = (i - j + width + offset_pixels) // bar_width | |
| if bar_number % 2 == 0: # Check for even bar number | |
| canvas[j, i] = color1_rgb | |
| else: | |
| canvas[j, i] = color2_rgb | |
| fig, ax = plt.subplots(figsize=(width/100, height/100)) | |
| ax.imshow(canvas) | |
| plt.axis('off') | |
| plt.tight_layout(pad=0, w_pad=0, h_pad=0) | |
| plt.autoscale(tight=True) | |
| img_buf = io.BytesIO() | |
| plt.savefig(img_buf, format='png') | |
| img = Image.open(img_buf) | |
| image_out = pil2tensor(img.convert("RGB")) | |
| show_help = "https://github.com/Suzie1/ComfyUI_Comfyroll_CustomNodes/wiki/Pattern-Nodes#cr-color-bars" | |
| return (image_out, show_help, ) | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| class CR_StyleBars: | |
| def INPUT_TYPES(s): | |
| modes = ["color bars", "sin wave", "gradient bars"] | |
| return {"required": { | |
| "mode": (modes,), | |
| "width": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "height": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "bar_style": (STYLES,), | |
| "orientation": (["vertical", "horizontal", ],), | |
| "bar_frequency": ("INT", {"default": 5, "min": 1, "max":200, "step": 1}), | |
| } | |
| } | |
| RETURN_TYPES = ("IMAGE", "STRING", ) | |
| RETURN_NAMES = ("IMAGE", "show_help", ) | |
| FUNCTION = "draw" | |
| CATEGORY = icons.get("Comfyroll/Graphics/Pattern") | |
| def draw(self, mode, width, height, bar_style, orientation, bar_frequency): | |
| # Create a horizontal or vertical bar depending on the orientation | |
| if orientation == "vertical": | |
| x = np.linspace(0, 1, width) | |
| y = np.zeros((height, width)) | |
| elif orientation == "horizontal": | |
| x = np.zeros((height, width)) | |
| y = np.linspace(0, 1, height) | |
| # Create a grid of colors for the bar | |
| X, Y = np.meshgrid(x, y) | |
| if mode == "color bars": | |
| bar_width = 1 / bar_frequency | |
| if orientation == "vertical": | |
| colors = (X // bar_width) % 2 | |
| elif orientation == "horizontal": | |
| colors = (Y // bar_width) % 2 | |
| elif mode == "sin wave": | |
| if orientation == "vertical": | |
| colors = np.sin(2 * np.pi * bar_frequency * X) | |
| elif orientation == "horizontal": | |
| colors = np.sin(2 * np.pi * bar_frequency * Y) | |
| elif mode == "gradient bars": | |
| if orientation == "vertical": | |
| colors = (X * bar_frequency * 2) % 2 | |
| elif orientation == "horizontal": | |
| colors = (Y * bar_frequency * 2) % 2 | |
| fig, ax = plt.subplots(figsize=(width/100, height/100)) | |
| ax.imshow(colors, cmap=bar_style, aspect='auto') | |
| plt.axis('off') | |
| plt.tight_layout(pad=0, w_pad=0, h_pad=0) | |
| plt.autoscale(tight=True) | |
| img_buf = io.BytesIO() | |
| plt.savefig(img_buf, format='png') | |
| img = Image.open(img_buf) | |
| image_out = pil2tensor(img.convert("RGB")) | |
| show_help = "https://github.com/Suzie1/ComfyUI_Comfyroll_CustomNodes/wiki/Pattern-Nodes#cr-style-bars" | |
| return (image_out, show_help, ) | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| class CR_ColorGradient: | |
| def INPUT_TYPES(s): | |
| return {"required": { | |
| "width": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "height": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "start_color": (COLORS,), | |
| "end_color": (COLORS,), | |
| "gradient_distance": ("FLOAT", {"default": 1, "min": 0, "max": 2, "step": 0.05}), | |
| "linear_transition": ("FLOAT", {"default": 0.5, "min": 0, "max": 1, "step": 0.05}), | |
| "orientation": (["vertical", "horizontal", ],), | |
| }, | |
| "optional": { | |
| "start_color_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| "end_color_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| } | |
| } | |
| RETURN_TYPES = ("IMAGE", "STRING", ) | |
| RETURN_NAMES = ("IMAGE", "show_help", ) | |
| FUNCTION = "draw" | |
| CATEGORY = icons.get("Comfyroll/Graphics/Pattern") | |
| def draw(self, width, height, start_color, end_color, orientation, | |
| linear_transition=0.5, gradient_distance=1, | |
| start_color_hex='#000000', end_color_hex='#000000'): # Default to .5 if the value is not found | |
| # Get RGB values | |
| if start_color == "custom": | |
| color1_rgb = hex_to_rgb(start_color_hex) | |
| else: | |
| color1_rgb = color_mapping.get(start_color, (255, 255, 255)) # Default to white if the color is not found | |
| if end_color == "custom": | |
| color2_rgb = hex_to_rgb(end_color_hex) | |
| else: | |
| color2_rgb = color_mapping.get(end_color, (0, 0, 0)) # Default to black if the color is not found | |
| # Create a blank canvas | |
| canvas = np.zeros((height, width, 3), dtype=np.uint8) | |
| transition_pixel = int(linear_transition * (width if orientation == 'horizontal' else height)) #getting center point for gradient | |
| def get_gradient_value(pos, length, linear_transition, gradient_distance): #getting the distance we use to apply gradient | |
| # Calculate the start and end of the transition | |
| transition_length = length * gradient_distance | |
| transition_start = linear_transition * length - transition_length / 2 | |
| transition_end = linear_transition * length + transition_length / 2 | |
| # Return the gradient value based on position | |
| if pos < transition_start: | |
| return 0 | |
| elif pos > transition_end: | |
| return 1 | |
| else: | |
| return (pos - transition_start) / transition_length | |
| if orientation == 'horizontal': | |
| # Define the x-values for interpolation | |
| x = [0, width * linear_transition - 0.5 * width * gradient_distance, width * linear_transition + 0.5 * width * gradient_distance, width] | |
| # Define the y-values for interpolation (t-values) | |
| y = [0, 0, 1, 1] | |
| # Interpolate | |
| t_values = np.interp(np.arange(width), x, y) | |
| for i, t in enumerate(t_values): | |
| interpolated_color = [int(c1 * (1 - t) + c2 * t) for c1, c2 in zip(color1_rgb, color2_rgb)] | |
| canvas[:, i] = interpolated_color | |
| elif orientation == 'vertical': | |
| # Define the x-values for interpolation | |
| x = [0, height * linear_transition - 0.5 * height * gradient_distance, height * linear_transition + 0.5 * height * gradient_distance, height] | |
| # Define the y-values for interpolation (t-values) | |
| y = [0, 0, 1, 1] | |
| # Interpolate | |
| t_values = np.interp(np.arange(height), x, y) | |
| for j, t in enumerate(t_values): | |
| interpolated_color = [int(c1 * (1 - t) + c2 * t) for c1, c2 in zip(color1_rgb, color2_rgb)] | |
| canvas[j, :] = interpolated_color | |
| fig, ax = plt.subplots(figsize=(width / 100, height / 100)) | |
| ax.imshow(canvas) | |
| plt.axis('off') | |
| plt.tight_layout(pad=0, w_pad=0, h_pad=0) | |
| plt.autoscale(tight=True) | |
| img_buf = io.BytesIO() | |
| plt.savefig(img_buf, format='png') | |
| img = Image.open(img_buf) | |
| image_out = pil2tensor(img.convert("RGB")) | |
| show_help = "https://github.com/Suzie1/ComfyUI_Comfyroll_CustomNodes/wiki/Pattern-Nodes#cr-color-gradient" | |
| return (image_out, show_help, ) | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| class CR_RadialGradient: | |
| def INPUT_TYPES(s): | |
| return {"required": { | |
| "width": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "height": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "start_color": (COLORS,), | |
| "end_color": (COLORS,), | |
| "gradient_distance": ("FLOAT", {"default": 1, "min": 0, "max": 2, "step": 0.05}), | |
| "radial_center_x": ("FLOAT", {"default": 0.5, "min": 0, "max": 1, "step": 0.05}), | |
| "radial_center_y": ("FLOAT", {"default": 0.5, "min": 0, "max": 1, "step": 0.05}), | |
| }, | |
| "optional": { | |
| "start_color_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| "end_color_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| } | |
| } | |
| RETURN_TYPES = ("IMAGE", "STRING", ) | |
| RETURN_NAMES = ("IMAGE", "show_Help", ) | |
| FUNCTION = "draw" | |
| CATEGORY = icons.get("Comfyroll/Graphics/Pattern") | |
| def draw(self, width, height, start_color, end_color, | |
| radial_center_x=0.5, radial_center_y=0.5, gradient_distance=1, | |
| start_color_hex='#000000', end_color_hex='#000000'): # Default to .5 if the value is not found | |
| # Get RGB values | |
| if start_color == "custom": | |
| color1_rgb = hex_to_rgb(start_color_hex) | |
| else: | |
| color1_rgb = color_mapping.get(start_color, (255, 255, 255)) # Default to white if the color is not found | |
| if end_color == "custom": | |
| color2_rgb = hex_to_rgb(end_color_hex) | |
| else: | |
| color2_rgb = color_mapping.get(end_color, (0, 0, 0)) # Default to black if the color is not found | |
| # Create a blank canvas | |
| canvas = np.zeros((height, width, 3), dtype=np.uint8) | |
| center_x = int(radial_center_x * width) | |
| center_y = int(radial_center_y * height) | |
| # Computation for max_distance | |
| max_distance = (np.sqrt(max(center_x, width - center_x)**2 + max(center_y, height - center_y)**2))*gradient_distance | |
| for i in range(width): | |
| for j in range(height): | |
| distance_to_center = np.sqrt((i - center_x) ** 2 + (j - center_y) ** 2) | |
| t = distance_to_center / max_distance | |
| # Ensure t is between 0 and 1 | |
| t = max(0, min(t, 1)) | |
| interpolated_color = [int(c1 * (1 - t) + c2 * t) for c1, c2 in zip(color1_rgb, color2_rgb)] | |
| canvas[j, i] = interpolated_color | |
| fig, ax = plt.subplots(figsize=(width / 100, height / 100)) | |
| ax.imshow(canvas) | |
| plt.axis('off') | |
| plt.tight_layout(pad=0, w_pad=0, h_pad=0) | |
| plt.autoscale(tight=True) | |
| img_buf = io.BytesIO() | |
| plt.savefig(img_buf, format='png') | |
| img = Image.open(img_buf) | |
| image_out = pil2tensor(img.convert("RGB")) | |
| show_help = "https://github.com/Suzie1/ComfyUI_Comfyroll_CustomNodes/wiki/Pattern-Nodes#cr-radial-gradiant" | |
| return (image_out, show_help, ) | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| class CR_CheckerPattern: | |
| def INPUT_TYPES(s): | |
| modes = ["regular", "stepped"] | |
| return {"required": { | |
| "mode": (modes,), | |
| "width": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "height": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "color_1": (COLORS,), | |
| "color_2": (COLORS,), | |
| "grid_frequency": ("INT", {"default": 8, "min": 1, "max": 200, "step": 1}), | |
| "step": ("INT", {"default": 2, "min": 2, "max": 200, "step": 1}), | |
| }, | |
| "optional": { | |
| "color1_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| "color2_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| } | |
| } | |
| RETURN_TYPES = ("IMAGE", "STRING", ) | |
| RETURN_NAMES = ("IMAGE", "show_help", ) | |
| FUNCTION = "draw" | |
| CATEGORY = icons.get("Comfyroll/Graphics/Pattern") | |
| def draw(self, mode, width, height, color_1, color_2, | |
| grid_frequency, step, | |
| color1_hex='#000000', color2_hex='#000000'): | |
| # Get RGB values | |
| if color_1 == "custom": | |
| color1_rgb = hex_to_rgb(color1_hex) | |
| else: | |
| color1_rgb = color_mapping.get(color_1, (255, 255, 255)) # Default to white if the color is not found | |
| if color_2 == "custom": | |
| color2_rgb = hex_to_rgb(color2_hex) | |
| else: | |
| color2_rgb = color_mapping.get(color_2, (0, 0, 0)) # Default to black if the color is not found | |
| # Create a blank canvas | |
| canvas = np.zeros((height, width, 3), dtype=np.uint8) | |
| grid_size = width / grid_frequency | |
| for i in range(width): | |
| for j in range(height): | |
| if mode == "regular": | |
| if (i // grid_size) % 2 == (j // grid_size) % 2: | |
| canvas[j, i] = color1_rgb | |
| else: | |
| canvas[j, i] = color2_rgb | |
| elif mode == "stepped": | |
| if (i // grid_size) % step != (j // grid_size) % step: | |
| canvas[j, i] = color1_rgb | |
| else: | |
| canvas[j, i] = color2_rgb | |
| fig, ax = plt.subplots(figsize=(width/100, height/100)) | |
| ax.imshow(canvas) | |
| plt.axis('off') | |
| plt.tight_layout(pad=0, w_pad=0, h_pad=0) | |
| plt.autoscale(tight=True) | |
| img_buf = io.BytesIO() | |
| plt.savefig(img_buf, format='png') | |
| img = Image.open(img_buf) | |
| image_out = pil2tensor(img.convert("RGB")) | |
| show_help = "https://github.com/Suzie1/ComfyUI_Comfyroll_CustomNodes/wiki/Pattern-Nodes#cr-checker-pattern" | |
| return (image_out, show_help, ) | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| class CR_Polygons: | |
| def INPUT_TYPES(s): | |
| modes = ["hexagons", "triangles"] | |
| return {"required": { | |
| "mode": (modes,), | |
| "width": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "height": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "rows": ("INT", {"default": 5, "min": 1, "max": 512}), | |
| "columns": ("INT", {"default": 5, "min": 1, "max": 512}), | |
| "face_color": (COLORS,), | |
| "background_color": (COLORS,), | |
| "line_color": (COLORS,), | |
| "line_width": ("INT", {"default": 2, "min": 0, "max": 512}), | |
| }, | |
| "optional": { | |
| "face_color_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| "bg_color_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| "line_color_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| } | |
| } | |
| RETURN_TYPES = ("IMAGE", "STRING", ) | |
| RETURN_NAMES = ("IMAGE", "show_help", ) | |
| FUNCTION = "draw" | |
| CATEGORY = icons.get("Comfyroll/Graphics/Pattern") | |
| def draw(self, mode, width, height, rows, columns, | |
| face_color, background_color, line_color, line_width, | |
| face_color_hex='#000000', bg_color_hex='#000000', line_color_hex='#000000'): | |
| # Get RGB values | |
| if face_color == "custom": | |
| face_color = face_color_hex | |
| if line_color == "custom": | |
| line_color = line_color_hex | |
| if background_color == "custom": | |
| background_color = bg_color_hex | |
| fig, ax = plt.subplots(figsize=(width/100, height/100)) | |
| fig.set_facecolor(background_color) | |
| plt.xlim(0, width/100) | |
| plt.ylim(0, height/100) | |
| plt.axis('off') | |
| plt.tight_layout(pad=0, w_pad=0, h_pad=0) | |
| plt.autoscale(False) | |
| # Get polygon shape | |
| if mode == "hexagons": | |
| vertices = 6 | |
| elif mode == "triangles": | |
| vertices = 3 | |
| # Define the height and width of a hexagon | |
| cell_width = (width/100) / columns | |
| cell_height = (width/height) * np.sqrt(3) * (height/100) / (2 * columns) | |
| for row in range(rows + 2): | |
| for col in range(columns + 2): | |
| x = col * cell_width | |
| y = row * cell_height | |
| # Shift every other row | |
| if row % 2 == 1: | |
| x += cell_width / 2 | |
| # Create a hexagon as a polygon patch | |
| hexagon = RegularPolygon((x, y), numVertices=vertices, radius=cell_width/1.732, edgecolor=line_color, linewidth=line_width, facecolor=face_color) | |
| ax.add_patch(hexagon) | |
| img_buf = io.BytesIO() | |
| plt.savefig(img_buf, format='png') | |
| img = Image.open(img_buf) | |
| image_out = pil2tensor(img.convert("RGB")) | |
| show_help = "https://github.com/Suzie1/ComfyUI_Comfyroll_CustomNodes/wiki/Pattern-Nodes#cr-polygons" | |
| return (image_out, show_help, ) | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| class CR_StarburstLines: | |
| def INPUT_TYPES(s): | |
| return {"required": { | |
| "width": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "height": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "num_lines": ("INT", {"default": 6, "min": 1, "max": 500}), | |
| "line_length": ("FLOAT", {"default": 5, "min": 0, "max": 100, "step": 0.1}), | |
| "line_width": ("INT", {"default": 5, "min": 1, "max": 512}), | |
| "line_color": (COLORS,), | |
| "background_color": (COLORS,), | |
| "center_x": ("INT", {"default": 0, "min": 0, "max": 1024}), | |
| "center_y": ("INT", {"default": 0, "min": 0, "max": 1024}), | |
| "rotation": ("FLOAT", {"default": 0, "min": 0, "max": 720}), | |
| }, | |
| "optional": { | |
| "line_color_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| "bg_color_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| } | |
| } | |
| RETURN_TYPES = ("IMAGE", "STRING", ) | |
| RETURN_NAMES = ("IMAGE", "show_help", ) | |
| FUNCTION = "draw" | |
| CATEGORY = icons.get("Comfyroll/Graphics/Pattern") | |
| def draw(self, width, height, num_lines, line_length, line_width, line_color, background_color, | |
| center_x, center_y, rotation=0, | |
| line_color_hex='#000000', bg_color_hex='#000000'): | |
| if line_color == "custom": | |
| line_color = line_color_hex | |
| else: | |
| line_color = line_color | |
| if background_color == "custom": | |
| bgc = bg_color_hex | |
| else: | |
| bgc = background_color | |
| # Define the angle for the spokes in the starburst | |
| angle = 360 / num_lines | |
| # Set up the plot | |
| fig, ax = plt.subplots(figsize=(width/100,height/100)) | |
| plt.xlim(-width/100, width/100) | |
| plt.ylim(-height/100, height/100) | |
| plt.axis('off') | |
| plt.tight_layout(pad=0, w_pad=0, h_pad=0) | |
| plt.autoscale(False) | |
| # Coordinates of the central point | |
| center_x = center_x/100 | |
| center_y = center_y/100 | |
| # Draw the starburst lines | |
| for i in range(num_lines): | |
| # Calculate the endpoint of each line | |
| x_unrotated = center_x + line_length * np.cos(np.radians(i * angle)) | |
| y_unrotated = center_y + line_length * np.sin(np.radians(i * angle)) | |
| # Apply rotation transformation | |
| x = center_x + x_unrotated * np.cos(np.radians(rotation)) - y_unrotated * np.sin(np.radians(rotation)) | |
| y = center_y + x_unrotated * np.sin(np.radians(rotation)) + y_unrotated * np.cos(np.radians(rotation)) | |
| # Plot the line | |
| fig.patch.set_facecolor(bgc) | |
| ax.plot([center_x, x], [center_y, y], color=line_color, linewidth=line_width) | |
| img_buf = io.BytesIO() | |
| plt.savefig(img_buf, format='png') | |
| img = Image.open(img_buf) | |
| image_out = pil2tensor(img.convert("RGB")) | |
| show_help = "https://github.com/Suzie1/ComfyUI_Comfyroll_CustomNodes/wiki/Pattern-Nodes#cr-starburst-lines" | |
| return (image_out, show_help, ) | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| class CR_StarburstColors: | |
| def INPUT_TYPES(s): | |
| return {"required": { | |
| "width": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "height": ("INT", {"default": 512, "min": 64, "max": 4096}), | |
| "num_triangles": ("INT", {"default": 6, "min": 1, "max": 512}), | |
| "color_1": (COLORS,), | |
| "color_2": (COLORS,), | |
| "center_x": ("INT", {"default": 0, "min": 0, "max": 512}), | |
| "center_y": ("INT", {"default": 0, "min": 0, "max": 512}), | |
| "rotation": ("FLOAT", {"default": 0, "min": 0, "max": 720}), | |
| "bbox_factor": ("FLOAT", {"default": 2, "min": 0, "max": 2, "step": .01}), | |
| }, | |
| "optional": { | |
| "color1_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| "color2_hex": ("STRING", {"multiline": False, "default": "#000000"}), | |
| } | |
| } | |
| RETURN_TYPES = ("IMAGE", "STRING", ) | |
| RETURN_NAMES = ("IMAGE", "show_help", ) | |
| FUNCTION = "draw" | |
| CATEGORY = icons.get("Comfyroll/Graphics/Pattern") | |
| def draw(self, width, height, num_triangles, color_1, color_2, | |
| center_x, center_y, bbox_factor, rotation=0, | |
| color1_hex='#000000', color2_hex='#000000'): | |
| # Get RGB values for the text color | |
| if color_1 == "custom": | |
| color_1 = color1_hex | |
| else: | |
| color_1 = color_1 | |
| if color_2 == "custom": | |
| color_2 = color2_hex | |
| else: | |
| color_2 = color_2 | |
| # Set up the plot | |
| fig, ax = plt.subplots() | |
| x = width/100 | |
| y = height/100 | |
| fig, ax = plt.subplots(figsize=(x,y)) | |
| plt.xlim(-x/2, x/2) | |
| plt.ylim(-y/2, y/2) | |
| plt.axis('off') | |
| plt.tight_layout(pad=0, w_pad=0, h_pad=0) | |
| plt.autoscale(False) | |
| # Set the size of the starburst bounding box in x and y dimensions | |
| box_width = bbox_factor * x | |
| box_height = bbox_factor * y | |
| # Initialize a color list for alternating colors | |
| colors = [color_1, color_2] | |
| tri = num_triangles | |
| # Draw the starburst triangles with alternating colors and square pattern | |
| for i in range(tri): | |
| # Calculate the endpoints of the triangle with varying length | |
| x1 = center_x/100 | |
| y1 = center_y/100 | |
| x2_unrotated = (box_width / 2) * np.cos(np.radians(i * 360 / tri)) | |
| y2_unrotated = (box_height / 2) * np.sin(np.radians(i * 360 / tri)) | |
| x3_unrotated = (box_width / 2) * np.cos(np.radians((i + 1) * 360 / tri)) | |
| y3_unrotated = (box_height / 2) * np.sin(np.radians((i + 1) * 360 / tri)) | |
| #apply rotation transform | |
| x2 = x2_unrotated * np.cos(np.radians(rotation)) - y2_unrotated * np.sin(np.radians(rotation)) | |
| y2 = x2_unrotated * np.sin(np.radians(rotation)) + y2_unrotated * np.cos(np.radians(rotation)) | |
| x3 = x3_unrotated * np.cos(np.radians(rotation)) - y3_unrotated * np.sin(np.radians(rotation)) | |
| y3 = x3_unrotated * np.sin(np.radians(rotation)) + y3_unrotated * np.cos(np.radians(rotation)) | |
| # Plot the triangle with alternating colors | |
| ax.fill([x1, x2, x3, x1], [y1, y2, y3, y1], color=colors[i % 2]) | |
| img_buf = io.BytesIO() | |
| plt.savefig(img_buf, format='png') | |
| img = Image.open(img_buf) | |
| image_out = pil2tensor(img.convert("RGB")) | |
| show_help = "https://github.com/Suzie1/ComfyUI_Comfyroll_CustomNodes/wiki/Pattern-Nodes#cr-starburst-colors" | |
| return (image_out, show_help, ) | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| # MAPPINGS | |
| #---------------------------------------------------------------------------------------------------------------------# | |
| # For reference only, actual mappings are in __init__.py | |
| ''' | |
| NODE_CLASS_MAPPINGS = { | |
| "CR Color Bars": CR_ColorBars, | |
| "CR Style Bars": CR_StyleBars, | |
| "CR Checker Pattern": CR_CheckerPattern, | |
| "CR Polygons":CR_Polygons, | |
| "CR Halftone Grid": CR_HalftoneGrid, | |
| "CR Color Gradient": CR_ColorGradient, | |
| "CR Radial Gradient": CR_RadialGradient, | |
| "CR Overlay Text": CR_OverlayText, | |
| "CR Starburst Lines": CR_StarburstLines, | |
| "CR Starburst Colors": CR_StarburstColors, | |
| } | |
| ''' | |