MoveAction.py 6.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. from synergine.synergy.event.Action import Action
  2. from intelligine.synergy.event.move.MoveEvent import MoveEvent
  3. from random import randint, choice, randrange
  4. from xyzworld.cst import POSITION
  5. from intelligine.cst import PREVIOUS_DIRECTION, BLOCKED_SINCE, MOVE_MODE, MOVE_MODE_EXPLO
  6. from intelligine.cst import COL_TRANSPORTER_NOT_CARRYING, COL_TRANSPORTER_CARRYING
  7. from intelligine.synergy.event.move.direction import directions_same_level, directions_modifiers, directions_slighty
  8. from intelligine.synergy.event.move.direction import get_position_with_direction_decal
  9. from intelligine.core.exceptions import NoPheromone, BestPheromoneHere
  10. from intelligine.simulation.pheromone.DirectionPheromone import DirectionPheromone
  11. class MoveAction(Action):
  12. _listen = MoveEvent
  13. def __init__(self, object_id, parameters):
  14. super().__init__(object_id, parameters)
  15. self._move_to_point = None
  16. self._move_to_direction = None
  17. def prepare(self, context):
  18. object_point = context.metas.value.get(POSITION, self._object_id)
  19. try:
  20. direction = self._get_direction_with_pheromones(context, object_point)
  21. except NoPheromone:
  22. direction = self._get_random_direction(context)
  23. move_to_point = get_position_with_direction_decal(direction, object_point)
  24. if self._direction_point_is_possible(context, move_to_point):
  25. self._move_to_point = move_to_point
  26. self._move_to_direction = direction
  27. else:
  28. # TODO: mettre self._dont_move = True ?
  29. pass
  30. def _get_direction_with_pheromones(self, context, object_point):
  31. object_movement_mode = context.metas.value.get(MOVE_MODE, self._object_id)
  32. pheromone_type = DirectionPheromone.get_pheromone_type_for_move_mode(object_movement_mode)
  33. try:
  34. direction = self._get_pheromone_direction_for_point(context, object_point, pheromone_type)
  35. except NoPheromone:
  36. try:
  37. direction = self._get_direction_of_pheromone(context, object_point, pheromone_type)
  38. except NoPheromone:
  39. raise
  40. return direction
  41. @staticmethod
  42. def _get_pheromone_direction_for_point(context, point, pheromone_type):
  43. return DirectionPheromone.get_direction_for_point(context, point, pheromone_type)
  44. @staticmethod
  45. def _get_direction_of_pheromone(context, point, pheromone_type):
  46. search_pheromone_in_points = context.get_arround_points_of(point, distance=1)
  47. try:
  48. best_pheromone_direction = DirectionPheromone.get_best_pheromone_direction_in(context,
  49. point,
  50. search_pheromone_in_points,
  51. pheromone_type)
  52. return best_pheromone_direction
  53. except NoPheromone as err:
  54. raise err
  55. def _get_random_direction(self, context):
  56. try:
  57. blocked_since = context.metas.value.get(BLOCKED_SINCE, self._object_id)
  58. except KeyError:
  59. blocked_since = 0
  60. direction_name = None
  61. if blocked_since <= 3: #TODO: config
  62. try:
  63. previous_direction = context.metas.value.get(PREVIOUS_DIRECTION, self._object_id)
  64. # TODO: Faut mettre ca en plus propre (proba d'aller tou droit, config, etc)
  65. if randrange(100) < 75: # 75% de change d'aller tout droit
  66. # Dans le futur: les fourmis vont moins tout droit quand elle se croient et se touche
  67. return previous_direction
  68. directions_list = directions_slighty[previous_direction]
  69. # TODO: TMP tant que 1 niveau (z)
  70. directions_list = [direction for direction in directions_list if direction > 9 and direction < 19]
  71. direction_name = choice(directions_list)
  72. except KeyError:
  73. pass
  74. if not direction_name:
  75. direction_name = randint(directions_same_level[0], directions_same_level[1])
  76. return direction_name
  77. @staticmethod
  78. def _direction_point_is_possible(context, direction_point):
  79. return context.position_is_penetrable(direction_point)
  80. def run(self, obj, context, synergy_manager):
  81. if self._move_to_point is not None and self._move_to_direction != 14: # TODO: il ne faut pas choisir une direction 14.
  82. obj.set_position(self._move_to_point)
  83. context.metas.value.set(PREVIOUS_DIRECTION, self._object_id, self._move_to_direction)
  84. context.metas.value.set(BLOCKED_SINCE, self._object_id, 0)
  85. self._appose_pheromone(obj, context)
  86. # TEST: le temps de tout tester
  87. if self._move_to_point == obj.get_colony().get_start_position() and obj.is_carrying():
  88. obj_transported = obj.get_carried()
  89. obj_transported.set_carried_by(None)
  90. obj.put_carry(obj_transported, (-1, 0, 0))
  91. obj.get_brain().switch_to_mode(MOVE_MODE_EXPLO)
  92. context.metas.collections.add_remove(obj.get_id(),
  93. COL_TRANSPORTER_NOT_CARRYING,
  94. COL_TRANSPORTER_CARRYING)
  95. else:
  96. try:
  97. blocked_since = context.metas.value.get(BLOCKED_SINCE, self._object_id)
  98. except:
  99. blocked_since = 0
  100. context.metas.value.set(BLOCKED_SINCE, self._object_id, blocked_since+1)
  101. @staticmethod
  102. def _appose_pheromone(obj, context):
  103. # TODO: Cette action de pheromone doit etre une surcharge de Move afin d'avoir une Action Move generique.
  104. obj.get_brain().host_moved()
  105. try:
  106. DirectionPheromone.appose(context,
  107. obj.get_position(),
  108. obj.get_movement_pheromone_gland().get_movement_molecules())
  109. except BestPheromoneHere as best_pheromone_here:
  110. obj.get_brain().set_distance_from_objective(best_pheromone_here.get_best_distance())