doric-engine/hexmap.py

130 lines
4.6 KiB
Python
Raw Normal View History

import collections
from math import ceil
from kivy.logger import Logger
from kivy.uix.label import Label
from kivy.graphics import Color, Line, Ellipse
2016-07-17 00:06:06 -04:00
import kivy.utils
from kivy.vector import Vector
from terrain import choose_random_terrain, Terrains
MapCoords = collections.namedtuple('MapCoords', ['row', 'col'])
class Tile(Label):
def __init__(self, row=0, col=0, **kwargs):
super(Tile, self).__init__(**kwargs)
self.coords = MapCoords(row, col)
2016-07-17 00:06:06 -04:00
# set the cube coordinates of the hexagon
# as [x, y, z]
self.cube_coords = self.even_r_to_cube(self.coords.row, self.coords.col)
self.selected = False
# Pick a random terrain for each hex.
self.terrain = choose_random_terrain()
2016-11-15 09:19:41 -05:00
self.terrain_colour = kivy.utils.get_color_from_hex(Terrains[self.terrain]['color'])
radius = self.height / 2
2016-11-15 09:19:41 -05:00
self.bind(pos=self.update_pos, size=self.update_pos)
with self.canvas.after:
# Create the solid background of the hexagon, from the bottom left coordinate of the hex.
Color(*self.terrain_colour)
self.solid = Ellipse(pos=(self.x, self.y), size=(self.height, self.height), segments=6)
# Create the outline of hexagon, based off the centre of the hex.
Color(*kivy.utils.get_color_from_hex('#000000'))
self.ell = Line(circle=(self.center_x, self.center_y, radius, 0, 360, 6), width=2)
Color(0, 0, 0, 1)
self.coord_label = Label(
2016-11-15 09:19:41 -05:00
text=self.even_r_coordinate_text(),
center_x=self.center_x,
center_y=self.center_y)
2016-07-17 00:06:06 -04:00
@staticmethod
def even_r_to_cube(row, col):
"""compute cube coordinates from even-r hex coordinates"""
x = int(col - ceil(float(row)/2))
z = row
y = - x - z
2016-07-17 00:06:06 -04:00
return [x, y, z]
2016-07-17 00:06:06 -04:00
@staticmethod
def cube_to_even_r(x, y, z):
row = int(x + ceil(z / 2))
col = z
2016-11-15 09:19:41 -05:00
return [row, col]
@property
def even_r_coords(self):
2016-07-17 00:06:06 -04:00
"""return even-r coordinates of the hexagon."""
return self.cube_to_even_r(*self.cube_coords)
@even_r_coords.setter
def even_r_coords(self, value):
self.cube_coords = self.even_r_to_cube(*value)
def even_r_coordinate_text(self):
return '{}'.format(self.even_r_coords)
def cube_coordinate_text(self):
2016-07-17 00:06:06 -04:00
return '{!r}'.format(self.cube_coords)
def map_display_text(self):
return "{}\n{} \n {}".format(self.even_r_coordinate_text(), self.cube_coordinate_text(), self.terrain)
def update_pos(self, instance, value):
self.canvas.clear()
2016-11-15 09:19:41 -05:00
radius = self.height / 2
# Resize the outline of the cell.
self.ell.circle = (self.center_x, self.center_y, radius, 0, 360, 6)
# Resize the actual cell.
2016-11-15 09:19:41 -05:00
self.solid.pos = (self.x, self.y)
self.solid.size = (self.height, self.height)
2016-11-15 09:19:41 -05:00
self.coord_label.center_x = self.center_x
self.coord_label.center_y = self.center_y
def on_touch_down(self, touch):
if super(Tile, self).on_touch_down(touch):
return False
coord_x, coord_y = self.even_r_coords
with self.canvas.after:
Color(*kivy.utils.get_color_from_hex('#000000'))
2016-11-15 09:19:41 -05:00
radius = self.height / 2
self.ell = Line(circle=(self.center_x, self.center_y, radius, 0, 360, 6), width=2)
if not self.collide_with_bounding_circle(touch.x, touch.y):
return False
Logger.debug('Selected: ({}, {})'.format(coord_x, coord_y))
with self.canvas.after:
if 'button' in touch.profile and touch.button == 'left':
Color(*kivy.utils.get_color_from_hex('#00FF00'))
if 'button' in touch.profile and touch.button == 'right':
# TODO Will refactor to have separate on_touch_up for selected target hex instead.
Color(*kivy.utils.get_color_from_hex('#FF0000'))
2016-11-15 09:19:41 -05:00
radius = self.height / 2
self.ell = Line(circle=(self.center_x, self.center_y, radius, 0, 360, 6), width=2)
2016-11-15 09:19:41 -05:00
self.parent.parent.game.update_selected_cell(self)
return True
def collide_with_bounding_circle(self, coord_x, coord_y):
# Register if within bounds of circle that the hex is inscribed in.
Logger.debug('Detected: ({}, {})'.format(coord_x, coord_y))
2016-11-15 09:19:41 -05:00
radius = self.height / 2
dist = Vector(self.x, self.y).distance((coord_x, coord_y))
Logger.debug('({}, {}) -> ({}, {})'.format(self.x, self.y, coord_x, coord_y))
Logger.debug('Dist: {} Diff: {}'.format(dist, dist - radius))
return dist - radius <= 0