Surn commited on
Commit
4a4069a
·
1 Parent(s): 2abe227

Update Hex generation to allow Triangles, Squares and Hexagons, the only 3 shapes that allow tiling

Browse files
Files changed (3) hide show
  1. app.py +6 -4
  2. utils/color_utils.py +73 -1
  3. utils/hex_grid.py +76 -28
app.py CHANGED
@@ -67,6 +67,7 @@ from utils.image_utils import (
67
  from utils.hex_grid import (
68
  generate_hexagon_grid,
69
  generate_hexagon_grid_interface,
 
70
  )
71
 
72
  from utils.excluded_colors import (
@@ -1360,10 +1361,11 @@ with gr.Blocks(css_paths="style_20250314.css", title=title, theme='Surn/beeuty',
1360
  end_x = gr.Number(label="End X", value=-20, minimum=-512, maximum= 512, precision=0)
1361
  end_y = gr.Number(label="End Y", value=-20, minimum=-512, maximum= 512, precision=0)
1362
  with gr.Row():
1363
- x_spacing = gr.Number(label="Adjust Horizontal spacing", value=-8, minimum=-200, maximum=200, precision=1)
1364
  y_spacing = gr.Number(label="Adjust Vertical spacing", value=3, minimum=-200, maximum=200, precision=1)
1365
  with gr.Row():
1366
  rotation = gr.Slider(-90, 180, 0.0, 0.1, label="Hexagon Rotation (degree)")
 
1367
  add_hex_text = gr.Dropdown(label="Add Text to Hexagons", choices=[None, "Column-Row Coordinates", "Column(Letter)-Row Coordinates", "Column-Row(Letter) Coordinates", "Sequential Numbers", "Playing Cards Sequential", "Playing Cards Alternate Red and Black", "Custom List"], value=None)
1368
  with gr.Row():
1369
  custom_text_list = gr.TextArea(label="Custom Text List", value=constants.cards_alternating, visible=False,)
@@ -1486,9 +1488,9 @@ with gr.Blocks(css_paths="style_20250314.css", title=title, theme='Surn/beeuty',
1486
  delete_button.click(fn=delete_color, inputs=[selected_row, color_display], outputs=[color_display])
1487
  exclude_color_button.click(fn=add_color, inputs=[color_picker, gr.State(excluded_color_list)], outputs=[color_display, gr.State(excluded_color_list)])
1488
  hex_button.click(
1489
- fn=lambda hex_size, border_size, input_image, start_x, start_y, end_x, end_y, rotation, background_color, background_opacity, border_color, border_opacity, fill_hex, color_display, filter_color, x_spacing, y_spacing, add_hex_text, custom_text_list, custom_text_color_list:
1490
- gr.Warning("Please upload an Input Image to get started.") if input_image is None else hex_create(hex_size, border_size, input_image, start_x, start_y, end_x, end_y, rotation, background_color, background_opacity, border_color, border_opacity, fill_hex, color_display, filter_color, x_spacing, y_spacing, add_hex_text, custom_text_list, custom_text_color_list),
1491
- inputs=[hex_size, border_size, input_image, start_x, start_y, end_x, end_y, rotation, background_color, background_opacity, border_color, border_opacity, fill_hex, color_display, filter_color, x_spacing, y_spacing, add_hex_text, custom_text_list, custom_text_color_list],
1492
  outputs=[output_image, overlay_image],
1493
  scroll_to_output=True
1494
  )
 
67
  from utils.hex_grid import (
68
  generate_hexagon_grid,
69
  generate_hexagon_grid_interface,
70
+ map_sides
71
  )
72
 
73
  from utils.excluded_colors import (
 
1361
  end_x = gr.Number(label="End X", value=-20, minimum=-512, maximum= 512, precision=0)
1362
  end_y = gr.Number(label="End Y", value=-20, minimum=-512, maximum= 512, precision=0)
1363
  with gr.Row():
1364
+ x_spacing = gr.Number(label="Adjust Horizontal spacing", value=-10, minimum=-200, maximum=200, precision=1)
1365
  y_spacing = gr.Number(label="Adjust Vertical spacing", value=3, minimum=-200, maximum=200, precision=1)
1366
  with gr.Row():
1367
  rotation = gr.Slider(-90, 180, 0.0, 0.1, label="Hexagon Rotation (degree)")
1368
+ sides = gr.Dropdown(label="Grid Shapes", info="other choices do not form grids",choices=["triangle", "square", "hexagon"], value="hexagon")
1369
  add_hex_text = gr.Dropdown(label="Add Text to Hexagons", choices=[None, "Column-Row Coordinates", "Column(Letter)-Row Coordinates", "Column-Row(Letter) Coordinates", "Sequential Numbers", "Playing Cards Sequential", "Playing Cards Alternate Red and Black", "Custom List"], value=None)
1370
  with gr.Row():
1371
  custom_text_list = gr.TextArea(label="Custom Text List", value=constants.cards_alternating, visible=False,)
 
1488
  delete_button.click(fn=delete_color, inputs=[selected_row, color_display], outputs=[color_display])
1489
  exclude_color_button.click(fn=add_color, inputs=[color_picker, gr.State(excluded_color_list)], outputs=[color_display, gr.State(excluded_color_list)])
1490
  hex_button.click(
1491
+ fn=lambda hex_size, border_size, input_image, start_x, start_y, end_x, end_y, rotation, sides, background_color, background_opacity, border_color, border_opacity, fill_hex, color_display, filter_color, x_spacing, y_spacing, add_hex_text, custom_text_list, custom_text_color_list:
1492
+ gr.Warning("Please upload an Input Image to get started.") if input_image is None else hex_create(hex_size, border_size, input_image, start_x, start_y, end_x, end_y, rotation, background_color, background_opacity, border_color, border_opacity, fill_hex, color_display, filter_color, x_spacing, y_spacing, add_hex_text, custom_text_list, custom_text_color_list, map_sides(sides)),
1493
+ inputs=[hex_size, border_size, input_image, start_x, start_y, end_x, end_y, rotation, sides, background_color, background_opacity, border_color, border_opacity, fill_hex, color_display, filter_color, x_spacing, y_spacing, add_hex_text, custom_text_list, custom_text_color_list],
1494
  outputs=[output_image, overlay_image],
1495
  scroll_to_output=True
1496
  )
utils/color_utils.py CHANGED
@@ -5,6 +5,7 @@ import re
5
  import cairocffi as cairo
6
  import pangocffi
7
  import pangocairocffi
 
8
 
9
 
10
  def multiply_and_clamp(value, scale, min_value=0, max_value=255):
@@ -196,7 +197,7 @@ def draw_text_with_emojis(image, text, font_color, offset_x, offset_y, font_name
196
  # Set text color
197
  r, g, b, a = parse_hex_color(font_color)
198
  context.set_source_rgba(r , g , b , a )
199
- # Move to the position (top-left corner adjusted to center the text)
200
  context.move_to(offset_x, offset_y)
201
  # Render the text
202
  pangocairocffi.show_layout(context, layout)
@@ -211,4 +212,75 @@ def draw_text_with_emojis(image, text, font_color, offset_x, offset_y, font_name
211
  "BGRA", # Cairo stores data in BGRA order
212
  surface.get_stride(),
213
  ).convert("RGBA")
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
214
  return modified_image
 
5
  import cairocffi as cairo
6
  import pangocffi
7
  import pangocairocffi
8
+ import math
9
 
10
 
11
  def multiply_and_clamp(value, scale, min_value=0, max_value=255):
 
197
  # Set text color
198
  r, g, b, a = parse_hex_color(font_color)
199
  context.set_source_rgba(r , g , b , a )
200
+ # Move to the position (top-left corner adjusted to position the text)
201
  context.move_to(offset_x, offset_y)
202
  # Render the text
203
  pangocairocffi.show_layout(context, layout)
 
212
  "BGRA", # Cairo stores data in BGRA order
213
  surface.get_stride(),
214
  ).convert("RGBA")
215
+ return modified_image
216
+
217
+ def draw_rotated_text_with_emojis(image, text, font_color, offset_x, offset_y, font_name, font_size, angle=0):
218
+ """
219
+ Draws text with emojis directly onto the given PIL image at specified coordinates with the specified color and rotation.
220
+ Parameters:
221
+ image (PIL.Image.Image): The RGBA image to draw on.
222
+ text (str): The text to draw, including emojis.
223
+ font_color (tuple): RGBA color tuple for the text (e.g., (255, 0, 0, 255)).
224
+ offset_x (int): The x-coordinate for the text center position.
225
+ offset_y (int): The y-coordinate for the text center position.
226
+ font_name (str): The name of the font family.
227
+ font_size (int): Size of the font.
228
+ angle (float): Rotation angle in degrees (default is 0, positive values rotate counter-clockwise).
229
+ Returns:
230
+ PIL.Image.Image: The modified image with the text drawn.
231
+ """
232
+ if image.mode != 'RGBA':
233
+ raise ValueError("Image must be in RGBA mode.")
234
+ # Convert PIL image to a mutable bytearray
235
+ img_data = bytearray(image.tobytes("raw", "BGRA"))
236
+
237
+ # Create a Cairo ImageSurface that wraps the image's data
238
+ surface = cairo.ImageSurface.create_for_data(
239
+ img_data,
240
+ cairo.FORMAT_ARGB32,
241
+ image.width,
242
+ image.height,
243
+ image.width * 4
244
+ )
245
+ context = cairo.Context(surface)
246
+ # Create Pango layout
247
+ layout = pangocairocffi.create_layout(context)
248
+ layout._set_text(text)
249
+ # Set font description
250
+ desc = pangocffi.FontDescription()
251
+ desc._set_family(font_name)
252
+ desc._set_size(pangocffi.units_from_double(font_size))
253
+ layout._set_font_description(desc)
254
+ # Get the ink and logical rectangles of the layout
255
+ #ink_rect, logical_rect = layout.get_extents()
256
+ # Extract width and height from the logical rectangle, convert from pango units to pixels
257
+ #width = logical_rect.width / 1024.0
258
+ #height = logical_rect.height /1024.0
259
+ # Set text color
260
+ r, g, b, a = parse_hex_color(font_color)
261
+ context.set_source_rgba(r, g, b, a)
262
+ # Save the current context state
263
+ context.save()
264
+ # Move to the position (top-left corner adjusted to position the text)
265
+ context.translate(offset_x, offset_y)
266
+ # Rotate by the specified angle (convert degrees to radians)
267
+ context.rotate(math.radians(angle))
268
+ # Move to the position to center the text at (0, 0) in the transformed context
269
+ #context.move_to(-width / 2,-height / 2)
270
+ # Render the text
271
+ pangocairocffi.show_layout(context, layout)
272
+ # Restore the original context state
273
+ context.restore()
274
+ # Flush the surface to ensure all drawing operations are complete
275
+ surface.flush()
276
+ # Convert the modified bytearray back to a PIL Image
277
+ modified_image = Image.frombuffer(
278
+ "RGBA",
279
+ (image.width, image.height),
280
+ bytes(img_data),
281
+ "raw",
282
+ "BGRA",
283
+ surface.get_stride(),
284
+ ).convert("RGBA")
285
+
286
  return modified_image
utils/hex_grid.py CHANGED
@@ -11,7 +11,7 @@ from utils.excluded_colors import (
11
  excluded_color_list,
12
  )
13
  from utils.image_utils import multiply_and_blend_images
14
- from utils.color_utils import update_color_opacity, parse_hex_color, draw_text_with_emojis, hex_to_rgb
15
  import random # For random text options
16
  import utils.constants as constants # Import constants
17
  import ast
@@ -33,6 +33,10 @@ def calculate_font_size(hex_size, padding=0.6, size_ceil=20, min_font_size=8):
33
  return None # Hex is too small for text
34
  return min(font_size, size_ceil)
35
 
 
 
 
 
36
  def generate_hexagon_grid(hex_size, border_size, input_image=None, image_width=0, image_height=0, start_x=0, start_y=0, end_x=0, end_y=0, rotation=0, background_color="#ede9ac44", border_color="#12165380", fill_hex=True, excluded_color_list=excluded_color_list, filter_color=False, x_spacing=0, y_spacing=0, sides=6):
37
  if input_image:
38
  image_width, image_height = input_image.size
@@ -67,13 +71,13 @@ def generate_hexagon_grid(hex_size, border_size, input_image=None, image_width=0
67
  draw = ImageDraw.Draw(image, mode="RGBA")
68
  hex_width = hex_size * 2
69
  hex_height = hex_size * 2
70
- hex_horizontal_spacing = (hex_width ) + (hex_border_size if hex_border_size < 0 else 0) + x_spacing #* 0.8660254
71
  hex_vertical_spacing = hex_height + (hex_border_size if hex_border_size < 0 else 0) + y_spacing
72
  col = 0
73
  row = 0
74
  def draw_hexagon(x, y, color="#FFFFFFFF", rotation=0, outline_color="#12165380", outline_width=0, sides=6):
75
- #side_length = (hex_size * 2) / math.sqrt(3)
76
- side_length = 2 * hex_size * math.tan(math.pi / sides)
77
  points = [(x + side_length * math.cos(math.radians(angle + rotation)), y + side_length * math.sin(math.radians(angle + rotation))) for angle in range(0, 360, (360 // sides))]
78
  draw.polygon(points, fill=color, outline=outline_color, width=max(-5, outline_width))
79
  # Function to range a floating number
@@ -88,11 +92,32 @@ def generate_hexagon_grid(hex_size, border_size, input_image=None, image_width=0
88
  col = 0
89
  for x in frange(start_x, max(image_width + additional_width, image_width, rotated_image_width) + (end_x - start_x), hex_horizontal_spacing):
90
  col += 1
91
- x_offset = hex_width // 2
92
- y_offset = (hex_height // 2) #* 1.15470054342517
93
- # Adjust y_offset for columns 1 and 3 to overlap
94
- if col % 2 == 1:
95
- y_offset -= (hex_height // 2) #* 0.8660254
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
96
  if rotated_input_image:
97
  # Sample the colors of the pixels in the hexagon, if fill_hex is True
98
  if fill_hex:
@@ -121,12 +146,12 @@ def generate_hexagon_grid(hex_size, border_size, input_image=None, image_width=0
121
  else:
122
  print(f"color found: {avg_color}")
123
  #draw_hexagon(x + x_offset, y + y_offset, color="#{:02x}{:02x}{:02x}{:02x}".format(*avg_color if fill_hex else (0,0,0,0)), outline_color=border_color, outline_width=hex_border_size)
124
- draw_hexagon(x + x_offset, y + y_offset, color="#{:02x}{:02x}{:02x}{:02x}".format(*avg_color), outline_color=border_color, outline_width=hex_border_size, sides=sides)
125
  else:
126
- draw_hexagon(x + x_offset, y + y_offset, color="#000000", outline_color=border_color, outline_width=hex_border_size, sides=sides)
127
  else:
128
  color = "#%02x%02x%02x%02x" % (128, math.ceil(y) % 255, math.ceil(x) % 255, 255) if fill_hex else (0,0,0,0)
129
- draw_hexagon(x + x_offset, y + y_offset, color=color, outline_color=border_color, outline_width=hex_border_size, sides=sides)
130
  if rotation != 0:
131
  #image.show()
132
  # Rotate the final image
@@ -178,8 +203,8 @@ def generate_hexagon_grid_with_text(hex_size, border_size, input_image=None, ima
178
  draw = ImageDraw.Draw(image, mode="RGBA")
179
  hex_width = hex_size * 2
180
  hex_height = hex_size * 2
181
- hex_horizontal_spacing = (hex_width ) + (hex_border_size if hex_border_size < 0 else 0) + x_spacing #* 0.8660254
182
- hex_vertical_spacing = hex_height + (hex_border_size if hex_border_size < 0 else 0) + y_spacing
183
  col = 0
184
  row = 0
185
  ## Function to draw optional text
@@ -213,8 +238,8 @@ def generate_hexagon_grid_with_text(hex_size, border_size, input_image=None, ima
213
  pass
214
  hex_index = -1 # Initialize hex index
215
  def draw_hexagon(x, y, color="#FFFFFFFF", rotation=0, outline_color="#12165380", outline_width=0, sides=6):
216
- #side_length = (hex_size * 2) / math.sqrt(3)
217
- side_length = 2 * hex_size * math.tan(math.pi / sides)
218
  points = [(x + side_length * math.cos(math.radians(angle + rotation)), y + side_length * math.sin(math.radians(angle + rotation))) for angle in range(0, 360, (360 // sides))]
219
  draw.polygon(points, fill=color, outline=outline_color, width=max(-5, outline_width))
220
  # Function to range a floating number
@@ -230,11 +255,33 @@ def generate_hexagon_grid_with_text(hex_size, border_size, input_image=None, ima
230
  for x in frange(start_x, max(image_width + additional_width, image_width, rotated_image_width) + (end_x - start_x), hex_horizontal_spacing):
231
  col += 1
232
  hex_index += 1 # Increment hex index
233
- x_offset = hex_width // 2
234
- y_offset = (hex_height // 2) #* 1.15470054342517
235
- # Adjust y_offset for columns 1 and 3 to overlap
236
- if col % 2 == 1:
237
- y_offset -= (hex_height // 2) #* 0.8660254
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
238
  if rotated_input_image:
239
  # Sample the colors of the pixels in the hexagon, if fill_hex is True
240
  if fill_hex:
@@ -263,12 +310,12 @@ def generate_hexagon_grid_with_text(hex_size, border_size, input_image=None, ima
263
  else:
264
  print(f"color found: {avg_color}")
265
  #draw_hexagon(x + x_offset, y + y_offset, color="#{:02x}{:02x}{:02x}{:02x}".format(*avg_color if fill_hex else (0,0,0,0)), outline_color=border_color, outline_width=hex_border_size, sides=sides)
266
- draw_hexagon(x + x_offset, y + y_offset, color="#{:02x}{:02x}{:02x}{:02x}".format(*avg_color), outline_color=border_color, outline_width=hex_border_size, sides=sides)
267
  else:
268
- draw_hexagon(x + x_offset, y + y_offset, color="#00000000", outline_color=border_color, outline_width=hex_border_size, sides=sides)
269
  else:
270
  color = "#%02x%02x%02x%02x" % (128, math.ceil(y) % 255, math.ceil(x) % 255, 255) if fill_hex else (0,0,0,0)
271
- draw_hexagon(x + x_offset, y + y_offset, color=color, outline_color=border_color, outline_width=hex_border_size, sides=sides)
272
  # Draw text in hexagon
273
  if add_hex_text_option != None:
274
  font_size = calculate_font_size(hex_size, 0.333, 20, 7)
@@ -322,18 +369,19 @@ def generate_hexagon_grid_with_text(hex_size, border_size, input_image=None, ima
322
  # Calculate position to center text in hexagon
323
  # text_x = x + x_offset - (text_width / 2)
324
  # text_y = y + y_offset - (text_height / 2)
325
- # Calculate position to center text in hexagon
326
  text_x = x + x_offset - (hex_size / 1.75)
327
- text_y = y + y_offset - (hex_size / 1.25)
328
  # Draw the text directly onto the image
329
- font_image = draw_text_with_emojis(
330
  image=font_image,
331
  text=text,
332
  font_color=update_color_opacity(text_color,255),
333
  offset_x=text_x,
334
  offset_y=text_y,
335
  font_name=font_name,
336
- font_size=font_size
 
337
  )
338
  # # Use Pilmoji to draw text with emojis
339
  # with Pilmoji(image) as pilmoji:
 
11
  excluded_color_list,
12
  )
13
  from utils.image_utils import multiply_and_blend_images
14
+ from utils.color_utils import update_color_opacity, parse_hex_color, draw_rotated_text_with_emojis, hex_to_rgb
15
  import random # For random text options
16
  import utils.constants as constants # Import constants
17
  import ast
 
33
  return None # Hex is too small for text
34
  return min(font_size, size_ceil)
35
 
36
+ def map_sides(selected_side):
37
+ mapping = {"triangle": 3, "square": 4, "hexagon": 6}
38
+ return mapping[selected_side]
39
+
40
  def generate_hexagon_grid(hex_size, border_size, input_image=None, image_width=0, image_height=0, start_x=0, start_y=0, end_x=0, end_y=0, rotation=0, background_color="#ede9ac44", border_color="#12165380", fill_hex=True, excluded_color_list=excluded_color_list, filter_color=False, x_spacing=0, y_spacing=0, sides=6):
41
  if input_image:
42
  image_width, image_height = input_image.size
 
71
  draw = ImageDraw.Draw(image, mode="RGBA")
72
  hex_width = hex_size * 2
73
  hex_height = hex_size * 2
74
+ hex_horizontal_spacing = hex_width + (hex_border_size if hex_border_size < 0 else 0) + x_spacing #* 0.8660254
75
  hex_vertical_spacing = hex_height + (hex_border_size if hex_border_size < 0 else 0) + y_spacing
76
  col = 0
77
  row = 0
78
  def draw_hexagon(x, y, color="#FFFFFFFF", rotation=0, outline_color="#12165380", outline_width=0, sides=6):
79
+ #side_length = (hex_size * 2) / math.sqrt(3) #hexagons only
80
+ side_length = 2 * hex_size * math.tan(math.pi / sides) #hexagons, squares, triangle can tile
81
  points = [(x + side_length * math.cos(math.radians(angle + rotation)), y + side_length * math.sin(math.radians(angle + rotation))) for angle in range(0, 360, (360 // sides))]
82
  draw.polygon(points, fill=color, outline=outline_color, width=max(-5, outline_width))
83
  # Function to range a floating number
 
92
  col = 0
93
  for x in frange(start_x, max(image_width + additional_width, image_width, rotated_image_width) + (end_x - start_x), hex_horizontal_spacing):
94
  col += 1
95
+
96
+ # Calculate offsets based on the number of sides
97
+ if sides == 4:
98
+ # Squares line up perfectly; no vertical offset is needed.
99
+ x_offset = int(hex_width + hex_horizontal_spacing * 2)
100
+ y_offset = 0
101
+ rotation_offset = -45
102
+ elif sides == 3:
103
+ # For equilateral triangles, you might offset rows by about one-third
104
+ # of the triangle�s height. Adjust as needed.
105
+ x_offset = hex_width + hex_horizontal_spacing
106
+ y_offset = int((hex_height * 2) + hex_vertical_spacing)
107
+ rotation_offset = -60
108
+ # Adjust y_offset for columns 1 and 3 to overlap
109
+ if col % 2 == 1:
110
+ y_offset -= (hex_height // 2) #* 0.8660254
111
+ rotation_offset = 0
112
+ else:
113
+ # Default behavior (6 sides)
114
+ x_offset = hex_width // 2
115
+ y_offset = (hex_height // 2) #* 1.15470054342517
116
+ rotation_offset = 0
117
+ # Adjust y_offset for columns 1 and 3 to overlap
118
+ if col % 2 == 1:
119
+ y_offset -= (hex_height // 2) #* 0.8660254
120
+
121
  if rotated_input_image:
122
  # Sample the colors of the pixels in the hexagon, if fill_hex is True
123
  if fill_hex:
 
146
  else:
147
  print(f"color found: {avg_color}")
148
  #draw_hexagon(x + x_offset, y + y_offset, color="#{:02x}{:02x}{:02x}{:02x}".format(*avg_color if fill_hex else (0,0,0,0)), outline_color=border_color, outline_width=hex_border_size)
149
+ draw_hexagon(x + x_offset, y + y_offset, color="#{:02x}{:02x}{:02x}{:02x}".format(*avg_color), rotation=rotation_offset, outline_color=border_color, outline_width=hex_border_size, sides=sides)
150
  else:
151
+ draw_hexagon(x + x_offset, y + y_offset, color="#000000", rotation=rotation_offset, outline_color=border_color, outline_width=hex_border_size, sides=sides)
152
  else:
153
  color = "#%02x%02x%02x%02x" % (128, math.ceil(y) % 255, math.ceil(x) % 255, 255) if fill_hex else (0,0,0,0)
154
+ draw_hexagon(x + x_offset, y + y_offset, color=color, rotation=rotation_offset, outline_color=border_color, outline_width=hex_border_size, sides=sides)
155
  if rotation != 0:
156
  #image.show()
157
  # Rotate the final image
 
203
  draw = ImageDraw.Draw(image, mode="RGBA")
204
  hex_width = hex_size * 2
205
  hex_height = hex_size * 2
206
+ hex_horizontal_spacing = (hex_width + (hex_border_size if hex_border_size < 0 else 0) + x_spacing) * ((6 / sides) if sides > 3 else 1.3333) #* 0.8660254
207
+ hex_vertical_spacing = (hex_height + (hex_border_size if hex_border_size < 0 else 0) + y_spacing) * ((6 / sides) if sides > 3 else 3.0)
208
  col = 0
209
  row = 0
210
  ## Function to draw optional text
 
238
  pass
239
  hex_index = -1 # Initialize hex index
240
  def draw_hexagon(x, y, color="#FFFFFFFF", rotation=0, outline_color="#12165380", outline_width=0, sides=6):
241
+ #side_length = (hex_size * 2) / math.sqrt(3) #hexagons only
242
+ side_length = 2 * hex_size * math.tan(math.pi / sides) #hexagons, squares, triangle can tile
243
  points = [(x + side_length * math.cos(math.radians(angle + rotation)), y + side_length * math.sin(math.radians(angle + rotation))) for angle in range(0, 360, (360 // sides))]
244
  draw.polygon(points, fill=color, outline=outline_color, width=max(-5, outline_width))
245
  # Function to range a floating number
 
255
  for x in frange(start_x, max(image_width + additional_width, image_width, rotated_image_width) + (end_x - start_x), hex_horizontal_spacing):
256
  col += 1
257
  hex_index += 1 # Increment hex index
258
+
259
+ # Calculate offsets based on the number of sides
260
+ if sides == 4:
261
+ # Squares line up perfectly; no vertical offset is needed.
262
+ x_offset = hex_size
263
+ y_offset = 0
264
+ rotation_offset = -45
265
+ elif sides == 3:
266
+ # For equilateral triangles, you might offset rows by about one-third
267
+ # of the triangle�s height. Adjust as needed.
268
+ x_offset = -hex_border_size * 2 #hex_width * math.tan(math.pi / sides) - hex_border_size * 2
269
+ y_offset = -hex_border_size * 2
270
+ rotation_offset = -60
271
+ # Adjust y_offset for columns 1 and 2 to overlap
272
+ if col % 2 == 1:
273
+ x_offset += int(hex_size * 0.8660254)
274
+ y_offset -= int(hex_height * 1.5)
275
+ rotation_offset = 0
276
+ else:
277
+ # Default behavior (6 sides)
278
+ x_offset = hex_width // 2
279
+ y_offset = (hex_height // 2) #* 1.15470054342517
280
+ rotation_offset = 0
281
+ # Adjust y_offset for columns 1 and 3 to overlap
282
+ if col % 2 == 1:
283
+ y_offset -= (hex_height // 2) #* 0.8660254
284
+
285
  if rotated_input_image:
286
  # Sample the colors of the pixels in the hexagon, if fill_hex is True
287
  if fill_hex:
 
310
  else:
311
  print(f"color found: {avg_color}")
312
  #draw_hexagon(x + x_offset, y + y_offset, color="#{:02x}{:02x}{:02x}{:02x}".format(*avg_color if fill_hex else (0,0,0,0)), outline_color=border_color, outline_width=hex_border_size, sides=sides)
313
+ draw_hexagon(x + x_offset, y + y_offset, color="#{:02x}{:02x}{:02x}{:02x}".format(*avg_color), rotation=rotation_offset, outline_color=border_color, outline_width=hex_border_size, sides=sides)
314
  else:
315
+ draw_hexagon(x + x_offset, y + y_offset, color="#00000000", rotation=rotation_offset, outline_color=border_color, outline_width=hex_border_size, sides=sides)
316
  else:
317
  color = "#%02x%02x%02x%02x" % (128, math.ceil(y) % 255, math.ceil(x) % 255, 255) if fill_hex else (0,0,0,0)
318
+ draw_hexagon(x + x_offset, y + y_offset, color=color, rotation=rotation_offset, outline_color=border_color, outline_width=hex_border_size, sides=sides)
319
  # Draw text in hexagon
320
  if add_hex_text_option != None:
321
  font_size = calculate_font_size(hex_size, 0.333, 20, 7)
 
369
  # Calculate position to center text in hexagon
370
  # text_x = x + x_offset - (text_width / 2)
371
  # text_y = y + y_offset - (text_height / 2)
372
+ # Calculate position to top left text in hexagon
373
  text_x = x + x_offset - (hex_size / 1.75)
374
+ text_y = y + y_offset - (hex_size / 1.75)
375
  # Draw the text directly onto the image
376
+ font_image = draw_rotated_text_with_emojis(
377
  image=font_image,
378
  text=text,
379
  font_color=update_color_opacity(text_color,255),
380
  offset_x=text_x,
381
  offset_y=text_y,
382
  font_name=font_name,
383
+ font_size=font_size,
384
+ angle = -1.0 * rotation
385
  )
386
  # # Use Pilmoji to draw text with emojis
387
  # with Pilmoji(image) as pilmoji: