interior.py 4.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. # coding: utf-8
  2. import typing
  3. from PIL.PngImagePlugin import PngImageFile
  4. from synergine2_xyz.map import TMXMap
  5. from synergine2_xyz.utils import get_direct_around_positions_of_position
  6. class InteriorMapConfiguration(object):
  7. def __init__(
  8. self,
  9. layer_name: str='interiors',
  10. exterior_id: str='ext',
  11. interior_id: str='int',
  12. separator_id: str='sep',
  13. ) -> None:
  14. self.layer_name = layer_name
  15. self.exterior_id = exterior_id
  16. self.interior_id = interior_id
  17. self.separator_id = separator_id
  18. class InteriorManager(object):
  19. def __init__(
  20. self,
  21. map_: TMXMap,
  22. configuration: InteriorMapConfiguration=None,
  23. ) -> None:
  24. self.interiors = []
  25. self.map = map_
  26. self.configuration = configuration or InteriorMapConfiguration()
  27. self.interiors = self._compute_interiors()
  28. def _compute_interiors(self) -> typing.List[typing.List[typing.Tuple[int, int]]]:
  29. interiors = []
  30. layer_tiles = self.map.layer_tiles(self.configuration.layer_name)
  31. for tile_xy, tile in layer_tiles.items():
  32. # FIXME: on se retrouve avec des tuiles Grass la ou l'on a pas mis de tuile interior/exterior.
  33. # Faut pouvoir dire c'est tel tile la tile par defaut de tel layer
  34. if tile.property('id') == self.configuration.interior_id:
  35. x, y = map(int, tile_xy.split('.'))
  36. if not any([(x, y) in i for i in interiors]):
  37. new_interior = [(x, y)]
  38. positions_to_parse = []
  39. possible_positions_xyz = get_direct_around_positions_of_position((x, y, 0))
  40. possible_positions_xy = [(p[0], p[1]) for p in possible_positions_xyz]
  41. positions_to_parse.extend(possible_positions_xy)
  42. for possible_position_xyz in positions_to_parse:
  43. test_tile = None
  44. new_tile_x = possible_position_xyz[0]
  45. new_tile_y = possible_position_xyz[1]
  46. possible_position_key = '{}.{}'.format(new_tile_x, new_tile_y)
  47. if (new_tile_x, new_tile_y) in new_interior:
  48. continue
  49. try:
  50. test_tile = layer_tiles[possible_position_key]
  51. except KeyError:
  52. continue
  53. if test_tile.property('id') not in [
  54. self.configuration.interior_id,
  55. self.configuration.separator_id,
  56. ]:
  57. continue
  58. new_interior.append((new_tile_x, new_tile_y))
  59. if not test_tile.property('id') == self.configuration.separator_id:
  60. new_position_neighbour = get_direct_around_positions_of_position((new_tile_x, new_tile_y, 0))
  61. positions_to_parse.extend(new_position_neighbour)
  62. interiors.append(new_interior)
  63. return interiors
  64. def get_interiors(
  65. self,
  66. where_positions: typing.Iterable[typing.Tuple[int, int]]=None,
  67. ) -> typing.List[typing.List[typing.Tuple[int, int]]]:
  68. if where_positions is None:
  69. return self.interiors
  70. interiors = []
  71. for interior in self.interiors:
  72. for where_position in where_positions:
  73. if where_position in interior and interior not in interiors:
  74. interiors.append(interior)
  75. return interiors
  76. def update_image_for_interiors(
  77. self,
  78. image: PngImageFile,
  79. interiors: typing.List[typing.List[typing.Tuple[int, int]]],
  80. tile_width: int,
  81. tile_height: int,
  82. invert_y: bool=True,
  83. ) -> None:
  84. # TODO BS 20171213: Optimization can be done: keep in cache modifications on image instead change it entirely
  85. pixels = image.load()
  86. for interior in interiors:
  87. for tile_x, tile_y in interior:
  88. start_x = tile_x * tile_width
  89. start_y = tile_y * tile_height
  90. for x in range(start_x, start_x+tile_width):
  91. for y in range(start_y, start_y+tile_height):
  92. real_y = y
  93. if invert_y:
  94. real_y = image.height - 1 - y
  95. pixels[x, real_y] = (0, 0, 0, 0)