From 4716219c8986e90f0d435167a3fe60ac89cae14e Mon Sep 17 00:00:00 2001 From: alanwatsonforster <68709385+alanwatsonforster@users.noreply.github.com> Date: Thu, 21 Sep 2023 23:05:06 -0600 Subject: [PATCH] Implement and use (x,y) to hex number. --- airpower/aircraft.py | 18 +++-- airpower/maps.py | 162 +++++++++++++++++++++++++++++++++++++------ 2 files changed, 153 insertions(+), 27 deletions(-) diff --git a/airpower/aircraft.py b/airpower/aircraft.py index 92b0c7b2..415d5dec 100644 --- a/airpower/aircraft.py +++ b/airpower/aircraft.py @@ -1,20 +1,20 @@ import airpower.draw as apdraw import airpower.altitude as apaltitude import airpower.azimuth as apazimuth +import airpower.maps as apmaps import math class Aircraft: - def __init__(self, name, x, y, azimuth, altitude): + def __init__(self, name, hex, azimuth, altitude): apaltitude._checkaltitude(altitude) self._turn = 0 self._name = name - self._x = x - self._y = y + self._x, self._y = apmaps.numbertohex(hex) self._facing = apazimuth.tofacing(azimuth) self._altitude = altitude self._altitudecarry = 0 @@ -26,7 +26,13 @@ def __init__(self, name, x, y, azimuth, altitude): self._drawaircraft("end") def __str__(self): - return "%s: (%.2f,%.2f) %s %02d (%+.02f)" % (self._name, self._x, self._y, apazimuth.toazimuth(self._facing), self._altitude, self._altitudecarry) + return "%s: %s %02d %s (%+.03f)" % ( + self._name, + apmaps.hextonumber(self._x, self._y), + self._altitude, + apazimuth.toazimuth(self._facing), + self._altitudecarry + ) def _restore(self, i): self._x, self._y, self._facing, self._altitude, self._altitudecarry, self._destroyed = self._saved[i] @@ -46,8 +52,8 @@ def _drawaircraft(self, when): apdraw.drawaircraft(self._x, self._y, self._facing, self._name, self._altitude, when) def _position(self): - return "(%.2f,%.2f) %d %s" % ( - self._x, self._y, + return "%s %d %s" % ( + apmaps.hextonumber(self._x, self._y), self._altitude, apazimuth.toazimuth(self._facing) ) diff --git a/airpower/maps.py b/airpower/maps.py index 7c75399c..422a3418 100644 --- a/airpower/maps.py +++ b/airpower/maps.py @@ -8,27 +8,35 @@ matplotlib.rcParams['figure.figsize'] = [7.5, 10] plt.rcParams.update({'font.size': 10}) -_maps = [] +_mapsgrid = [] +_mapslist = [] _nxmaps = 0 _nymaps = 0 _compassrose = None -def setmaps(maps, compassrose, verbose=True): +def setmaps(mapsgrid, compassrose, verbose=True): - global _maps + global _mapsgrid + global _mapslist global _nymaps global _nxmaps global _compass # The maps argument follows visual layout, so we need to flip it vertically # so that the lower-left map has indices (0,0). - _maps = list(reversed(maps)) - _nymaps = len(maps) - _nxmaps = len(maps[0]) + _mapsgrid = list(reversed(mapsgrid)) + _nymaps = len(_mapsgrid) + _nxmaps = len(_mapsgrid[0]) if verbose: for iy in range (0, _nymaps): - print("%s" % " ".join(maps[iy])) + print("%s" % " ".join(mapsgrid[iy])) + + _mapslist = [] + for iy in range (0, _nymaps): + for ix in range (0, _nxmaps): + if _mapsgrid[iy][ix] != "--": + _mapslist.append(_mapsgrid[iy][ix]) _compassrose = compassrose @@ -37,14 +45,14 @@ def setmaps(maps, compassrose, verbose=True): def _mapx0(map): for iy in range (0, _nymaps): for ix in range (0, _nxmaps): - if map == _maps[iy][ix]: + if map == _mapsgrid[iy][ix]: return ix * 20 raise ValueError("map %s is not in use." % map) def _mapy0(map): for iy in range (0, _nymaps): for ix in range (0, _nxmaps): - if map == _maps[iy][ix]: + if map == _mapsgrid[iy][ix]: return iy * 15 raise ValueError("map %s is not in use." % map) @@ -97,22 +105,134 @@ def numbertomap(n): return mapletter + mapnumber +def checknumber(n): + + if isinstance(n, int): + return + + if isinstance(n, float) and n % 1.0 == 0.0: + return + + if not isinstance(n, str): + raise ValueError("invalid hex number %s." % n) + + m = n.split("/") + if len(m) == 1: + try: + n = int(n) + return + except: + raise ValueError("invalid hex number %s." % n) + elif len(m) == 2: + try: + int(m[0]) + int(m[1]) + except: + raise ValueError("invalid hex number %s." % n) + else: + raise ValueError("invalid hex number %s." % n) + def numbertohex(n): - map = numbertomap(n) + checknumber(n) - xx = n // 100 - yy = n % 100 + try: + + n = int(n) + + map = numbertomap(n) + + xx = n // 100 + yy = n % 100 + + dx = xx - _mapxx0(map) + dy = _mapyy0(map) - yy + if xx % 2 == 0: + dy += 0.5 + + x0 = _mapx0(map) + y0 = _mapy0(map) + + return x0 + dx, y0 + dy - dx = xx - _mapxx0(map) - dy = _mapyy0(map) - yy - if xx % 2 == 0: - dy += 0.5 + except: - x0 = _mapx0(map) - y0 = _mapy0(map) + m = n.split("/") + x0, y0 = numbertohex(m[0]) + x1, y1 = numbertohex(m[1]) + + return 0.5 * (x0 + x1), 0.5 * (y0 + y1) + +def hexiscentered(x, y): + if x % 2 == 0.0 and y % 1.0 == 0.00: + return True + elif x % 2 == 1.0 and y % 1.0 == 0.50: + return True + else: + return False + +def hexisonedge(x, y): + if x % 2 == 0.0 and y % 1.0 == 0.50: + return True + elif x % 2 == 0.5 and y % 0.5 == 0.25: + return True + elif x % 2 == 1.0 and y % 1.0 == 0.00: + return True + elif x % 2 == 1.5 and y % 0.5 == 0.25: + return True + else: + return False + +def checkhex(x, y): + if not hexiscentered(x, y) and not hexisonedge(x, y): + raise ValueError("invalid hex coordinates (%s,%s)." % (x,y)) + +def hextomap(x, y): + checkhex(x, y) + for map in _mapslist: + x0 = _mapx0(map) + y0 = _mapy0(map) + if x0 - 0.5 <= x and x < x0 + 19.5 and y0 - 0.5 <= y and y < y0 + 14.5: + return map + return None + +def hextonumber(x, y): + + checkhex(x, y) + + if hexiscentered(x, y): + + map = hextomap(x, y) + x0 = _mapx0(map) + y0 = _mapy0(map) + xx0 = _mapxx0(map) + yy0 = _mapyy0(map) + xx = xx0 + (x - x0) + yy = yy0 - (y - y0) + if xx % 2 == 0: + yy += 0.5 + + return "%04d" % int(xx * 100 + yy) + + else: - return x0 + dx, y0 + dy + if x % 1 == 0: + n0 = hextonumber(x, y + 0.5) + n1 = hextonumber(x, y - 0.5) + elif x % 2 == 0.5 and y % 1 == 0.25: + n0 = hextonumber(x - 0.5, y - 0.25) + n1 = hextonumber(x + 0.5, y + 0.25) + elif x % 2 == 0.5 and y % 1 == 0.75: + n0 = hextonumber(x - 0.5, y + 0.25) + n1 = hextonumber(x + 0.5, y - 0.25) + elif x % 2 == 1.5 and y % 1 == 0.25: + n0 = hextonumber(x - 0.5, y + 0.25) + n1 = hextonumber(x + 0.5, y - 0.25) + elif x % 2 == 1.5 and y % 1 == 0.75: + n0 = hextonumber(x - 0.5, y - 0.25) + n1 = hextonumber(x + 0.5, y + 0.25) + + return "%s/%s" % (n0, n1) def _drawmap(map): xx0 = _mapxx0(map) @@ -146,8 +266,8 @@ def drawmaps(): for iy in range (0, _nymaps): for ix in range (0, _nxmaps): - if _maps[iy][ix] != "--": - _drawmap(_maps[iy][ix]) + if _mapsgrid[iy][ix] != "--": + _drawmap(_mapsgrid[iy][ix]) if _compassrose != None: apdraw.drawcompass(*numbertohex(_compassrose), apazimuth.tofacing("N"))