Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fixed hex-space draw function to avoid overlaps #2609

Merged
merged 21 commits into from
Jan 26, 2025
Merged
Changes from 7 commits
Commits
Show all changes
21 commits
Select commit Hold shift + click to select a range
f716287
fixed hex-space draw to avoid overlaps
Sahil-Chhoker Jan 10, 2025
75f06be
removed unused imports
Sahil-Chhoker Jan 10, 2025
f88551f
Merge remote-tracking branch 'origin/main' into fix-hex-tile-overlapping
Sahil-Chhoker Jan 10, 2025
cfd8a67
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Jan 10, 2025
d7c570a
Merge branch 'main' into fix-hex-tile-overlapping
Sahil-Chhoker Jan 10, 2025
f17e99d
Add Polygon and PatchCollection imports for visualization
Sahil-Chhoker Jan 10, 2025
70e37f1
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Jan 10, 2025
b0849cf
added reviewed changes
Sahil-Chhoker Jan 13, 2025
d09ef05
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Jan 13, 2025
c248700
reformatting
Sahil-Chhoker Jan 13, 2025
933fa54
Merge branch 'fix-hex-tile-overlapping' of https://github.com/Sahil-C…
Sahil-Chhoker Jan 13, 2025
9db46ca
applied pre-commit changes
Sahil-Chhoker Jan 13, 2025
379f32f
Merge branch 'main' into fix-hex-tile-overlapping
Sahil-Chhoker Jan 13, 2025
612bd64
added reviewed changes
Sahil-Chhoker Jan 13, 2025
2be551b
Merge branch 'main' into fix-hex-tile-overlapping
Sahil-Chhoker Jan 14, 2025
0a34e2f
added empty loc check
Sahil-Chhoker Jan 20, 2025
b2f57c5
Merge branch 'fix-hex-tile-overlapping' of https://github.com/Sahil-C…
Sahil-Chhoker Jan 20, 2025
45c3e76
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Jan 20, 2025
dea512b
Merge branch 'main' into fix-hex-tile-overlapping
Sahil-Chhoker Jan 20, 2025
8c6d801
Merge branch 'main' into fix-hex-tile-overlapping
quaquel Jan 20, 2025
69e15dc
Merge branch 'main' into fix-hex-tile-overlapping
Sahil-Chhoker Jan 23, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
126 changes: 70 additions & 56 deletions mesa/visualization/mpl_space_drawing.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,6 @@

import contextlib
import itertools
import math
import warnings
from collections.abc import Callable
from typing import Any
Expand All @@ -18,9 +17,9 @@
from matplotlib import pyplot as plt
from matplotlib.axes import Axes
from matplotlib.cm import ScalarMappable
from matplotlib.collections import PatchCollection
from matplotlib.collections import LineCollection, PatchCollection
from matplotlib.colors import LinearSegmentedColormap, Normalize, to_rgba
from matplotlib.patches import Polygon, RegularPolygon
from matplotlib.patches import Polygon

import mesa
from mesa.experimental.cell_space import (
Expand Down Expand Up @@ -308,13 +307,6 @@ def draw_hex_grid(
ax: a Matplotlib Axes instance. If none is provided a new figure and ax will be created using plt.subplots
draw_grid: whether to draw the grid
kwargs: additional keyword arguments passed to ax.scatter

Returns:
Returns the Axes object with the plot drawn onto it.

``agent_portrayal`` is called with an agent and should return a dict. Valid fields in this dict are "color",
"size", "marker", and "zorder". Other field are ignored and will result in a user warning.

"""
if ax is None:
fig, ax = plt.subplots()
Expand All @@ -323,62 +315,84 @@ def draw_hex_grid(
s_default = (180 / max(space.width, space.height)) ** 2
arguments = collect_agent_data(space, agent_portrayal, size=s_default)

# for hexgrids we have to go from logical coordinates to visual coordinates
# this is a bit messy.

# give all even rows an offset in the x direction
# give all rows an offset in the y direction

# numbers here are based on a distance of 1 between centers of hexes
offset = math.sqrt(0.75)
# Parameters for hexagon grid
size = 1.0
x_spacing = np.sqrt(3) * size
y_spacing = 1.5 * size

loc = arguments["loc"].astype(float)

logical = np.mod(loc[:, 1], 2) == 0
loc[:, 0][logical] += 0.5
loc[:, 1] *= offset
# Calculate hexagon centers
loc[:, 0] = loc[:, 0] * x_spacing + (loc[:, 1] % 2) * (x_spacing / 2)
Sahil-Chhoker marked this conversation as resolved.
Show resolved Hide resolved
loc[:, 1] = loc[:, 1] * y_spacing
arguments["loc"] = loc

# plot the agents
_scatter(ax, arguments, **kwargs)

# further styling and adding of grid
ax.set_xlim(-1, space.width + 0.5)
ax.set_ylim(-offset, space.height * offset)

def setup_hexmesh(
width,
height,
):
"""Helper function for creating the hexmaesh."""
# fixme: this should be done once, rather than in each update
# fixme check coordinate system in hexgrid (see https://www.redblobgames.com/grids/hexagons/#coordinates-offset)

patches = []
for x, y in itertools.product(range(width), range(height)):
if y % 2 == 0:
x += 0.5 # noqa: PLW2901
y *= offset # noqa: PLW2901
hex = RegularPolygon(
(x, y),
numVertices=6,
radius=math.sqrt(1 / 3),
orientation=np.radians(120),
)
patches.append(hex)
mesh = PatchCollection(
patches, edgecolor="k", facecolor=(1, 1, 1, 0), linestyle="dotted", lw=1
# Calculate proper bounds that account for the full hexagon width and height
x_max = space.width * x_spacing + (space.height % 2) * (x_spacing / 2)
y_max = space.height * y_spacing

# Add padding that accounts for the hexagon points
x_padding = (
size * np.sqrt(3) / 2
) # Distance from center to rightmost point of hexagon
y_padding = size # Distance from center to topmost point of hexagon

# Plot limits to perfectly contain the hexagonal grid
# Determined through physical testing.
ax.set_xlim(-2 * x_padding, x_max + x_padding)
ax.set_ylim(-2 * y_padding, y_max + y_padding)

def setup_hexmesh(width, height):
"""Helper function for creating the hexmesh with unique edges."""
edges = set()
size = 1.0
x_spacing = np.sqrt(3) * size
y_spacing = 1.5 * size

def get_hex_vertices(
center_x: float, center_y: float
) -> list[tuple[float, float]]:
"""Get vertices for a hexagon centered at (center_x, center_y)."""
vertices = [
(center_x, center_y + size), # top
(center_x + size * np.sqrt(3) / 2, center_y + size / 2), # top right
(center_x + size * np.sqrt(3) / 2, center_y - size / 2), # bottom right
(center_x, center_y - size), # bottom
(center_x - size * np.sqrt(3) / 2, center_y - size / 2), # bottom left
(center_x - size * np.sqrt(3) / 2, center_y + size / 2), # top left
]
return vertices

# Generate edges for each hexagon
for row, col in itertools.product(range(height), range(width)):
# Calculate center position for each hexagon with offset for even rows
x = col * x_spacing + (row % 2) * (x_spacing / 2)
y = row * y_spacing

vertices = get_hex_vertices(x, y)

# Edge logic, connecting each vertex to the next
for i in range(len(vertices)):
v1 = vertices[i]
v2 = vertices[(i + 1) % len(vertices)]
Sahil-Chhoker marked this conversation as resolved.
Show resolved Hide resolved

# Sort vertices to ensure consistent edge representation
edge = tuple(sorted([tuple(np.round(v1, 6)), tuple(np.round(v2, 6))]))
Sahil-Chhoker marked this conversation as resolved.
Show resolved Hide resolved
edges.add(edge)

# Convert to LineCollection format
edges_list = [np.array(edge) for edge in edges]
Sahil-Chhoker marked this conversation as resolved.
Show resolved Hide resolved
return LineCollection(
edges_list, linestyle=":", color="black", linewidth=1, alpha=1
)
return mesh

if draw_grid:
# add grid
ax.add_collection(
setup_hexmesh(
space.width,
space.height,
)
)
ax.add_collection(setup_hexmesh(space.width, space.height))

# Set aspect ratio to 'equal' to ensure hexagons appear regular
ax.set_aspect("equal")
Sahil-Chhoker marked this conversation as resolved.
Show resolved Hide resolved
return ax


Expand Down
Loading