From b8f131077e38dc7f1e18cd885710daf2bd69d1fc Mon Sep 17 00:00:00 2001 From: Alex Goodwin Date: Sun, 19 Jan 2025 01:22:56 +1000 Subject: [PATCH] Add __eq__ to Hex, & type-hint it --- PyRoute/Position/Hex.py | 77 +++++++++++++++++++++++++---------------- 1 file changed, 48 insertions(+), 29 deletions(-) diff --git a/PyRoute/Position/Hex.py b/PyRoute/Position/Hex.py index 71fb8016f..fb20e8271 100644 --- a/PyRoute/Position/Hex.py +++ b/PyRoute/Position/Hex.py @@ -4,24 +4,28 @@ @author: CyberiaResurrection """ import functools +from typing import Tuple +from typing_extensions import TypeAlias + +HexPos: TypeAlias = Tuple[int, int] class Hex(object): - position = None # Hex location - row = 0 # Location in the sector. - col = 0 - dx = 0 # location in the whole space, row/column coordinate - dy = 0 - q = 0 # location in the whole of space, axial coordinates - r = 0 + position: str = None # Hex location + row: int = 0 # Location in the sector. + col: int = 0 + dx: int = 0 # location in the whole space, row/column coordinate + dy: int = 0 + q: int = 0 # location in the whole of space, axial coordinates + r: int = 0 # Hex-side alias constants BOTTOM = 1 BOTTOMRIGHT = 2 BOTTOMLEFT = 4 - def __init__(self, sector, position): + def __init__(self, sector, position: str): """ # The zero point of the co-ordinate system used is Reference (Core 0140). # As a result, hex position 01,40 becomes q=0, r=0, x=0, y=0, z=0. @@ -33,21 +37,39 @@ def __init__(self, sector, position): self.dx = sector.x * 32 + self.col - 1 self.dy = sector.y * 40 + self.row - 1 self.q, self.r = Hex.hex_to_axial(self.dx, Hex.dy_offset(self.row, sector.y)) - - def distance(self, other): - return Hex.axial_distance((self.q, self.r), (other.q, other.r)) + self._hash = hash((self.position, self.dx, self.dy)) def __str__(self): return f"{self.col:02d}{self.row:02d}" + def __hash__(self): + return self._hash + + def __eq__(self, other): + if self.__hash__() != other.__hash__(): + return False + if not isinstance(other, Hex): + return False + + if self.position != other.position: + return False + if self.dx != other.dx: + return False + if self.dy != other.dy: + return False + return True + + def distance(self, other): + return Hex.axial_distance((self.q, self.r), (other.q, other.r)) + # Used to calculate distances for the AllyGen, which keeps only the q/r (axial) coordinates. @staticmethod - def axial_distance(Hex1, Hex2): + def axial_distance(Hex1: HexPos, Hex2: HexPos): return Hex._axial_core(Hex1[0] - Hex2[0], Hex1[1] - Hex2[1]) @staticmethod @functools.cache - def _axial_core(dq, dr): + def _axial_core(dq: int, dr: int): return (abs(dq) + abs(dr) + abs(dq + dr)) // 2 # Used to calculate distances for the TradeCalculation via the Network Graph, which requires a function @@ -60,15 +82,18 @@ def hex_distance(self, star): @staticmethod @functools.cache - def _hex_core(dx, dy, dz): + def _hex_core(dx: int, dy: int, dz: int): return max(abs(dx), abs(dy), abs(dz)) @functools.cache - def hex_position(self): - return (self.q, self.r) + def hex_position(self) -> HexPos: + return self.q, self.r + + def get_neighbour(self, direction: int, distance: int = 1): + return Hex.get_neighbor(self.hex_position(), direction, distance) @staticmethod - def get_neighbor(hex_pos, direction, distance=1): + def get_neighbor(hex_pos: HexPos, direction: int, distance: int = 1) -> HexPos: """ determine neighboring hex from the q,r position and direction. Direction index is: @@ -79,30 +104,24 @@ def get_neighbor(hex_pos, direction, distance=1): 4 => Down / left 5 => Down """ - d = Hex._get_neighbour_core(direction, distance) - return int(hex_pos[0] + d[0]), int(hex_pos[1] + d[1]) - - @staticmethod - @functools.cache - def _get_neighbour_core(direction, distance): neighbors = [ [+1, -1], [+1, 0], [0, +1], [-1, +1], [-1, 0], [0, -1] ] d = neighbors[direction] - d[0] *= distance - d[1] *= distance - return d + qn = hex_pos[0] + (d[0] * distance) + rn = hex_pos[1] + (d[1] * distance) + return int(qn), int(rn) @staticmethod - def hex_to_axial(row, col): + def hex_to_axial(row: int, col: int) -> HexPos: q = row q_offset = (q + (q & 1)) // 2 r = col - q_offset return q, r @staticmethod - def axial_to_hex(q, r): + def axial_to_hex(q: int, r: int): row = q q_offset = (q + (q & 1)) // 2 col = r + q_offset @@ -110,7 +129,7 @@ def axial_to_hex(q, r): return row, col @staticmethod - def axial_to_sector(q, r, flip=False): + def axial_to_sector(q: int, r: int, flip: bool = False): (raw_row, raw_col) = Hex.axial_to_hex(q, r) col, _ = Hex.dy_reverse(raw_col)