MoveAction.py 6.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  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, PHEROMONE_SEARCHING
  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. pheromone_type = context.metas.value.get(PHEROMONE_SEARCHING, self._object_id)
  32. try:
  33. direction = self._get_pheromone_direction_for_point(context, object_point, pheromone_type)
  34. except NoPheromone:
  35. try:
  36. direction = self._get_direction_of_pheromone(context, object_point, pheromone_type)
  37. except NoPheromone:
  38. raise
  39. return direction
  40. @staticmethod
  41. def _get_pheromone_direction_for_point(context, point, pheromone_type):
  42. return DirectionPheromone.get_direction_for_point(context, point, pheromone_type)
  43. @staticmethod
  44. def _get_direction_of_pheromone(context, point, pheromone_type):
  45. search_pheromone_in_points = context.get_arround_points_of(point, distance=1)
  46. try:
  47. best_pheromone_direction = DirectionPheromone.get_best_pheromone_direction_in(context,
  48. point,
  49. search_pheromone_in_points,
  50. pheromone_type)
  51. return best_pheromone_direction
  52. except NoPheromone as err:
  53. raise err
  54. def _get_random_direction(self, context):
  55. try:
  56. blocked_since = context.metas.value.get(BLOCKED_SINCE, self._object_id)
  57. except KeyError:
  58. blocked_since = 0
  59. direction_name = None
  60. if blocked_since <= 3: #TODO: config
  61. try:
  62. previous_direction = context.metas.value.get(PREVIOUS_DIRECTION, self._object_id)
  63. # TODO: Faut mettre ca en plus propre (proba d'aller tou droit, config, etc)
  64. if randrange(100) < 75: # 75% de change d'aller tout droit
  65. # Dans le futur: les fourmis vont moins tout droit quand elle se croient et se touche
  66. return previous_direction
  67. directions_list = directions_slighty[previous_direction]
  68. # TODO: TMP tant que 1 niveau (z)
  69. directions_list = [direction for direction in directions_list if direction > 9 and direction < 19]
  70. direction_name = choice(directions_list)
  71. except KeyError:
  72. pass
  73. if not direction_name:
  74. direction_name = randint(directions_same_level[0], directions_same_level[1])
  75. return direction_name
  76. @staticmethod
  77. def _direction_point_is_possible(context, direction_point):
  78. return context.position_is_penetrable(direction_point)
  79. def run(self, obj, context, synergy_manager):
  80. if self._move_to_point is not None and self._move_to_direction != 14: # TODO: il ne faut pas choisir une direction 14.
  81. obj.set_position(self._move_to_point)
  82. context.metas.value.set(PREVIOUS_DIRECTION, self._object_id, self._move_to_direction)
  83. context.metas.value.set(BLOCKED_SINCE, self._object_id, 0)
  84. self._appose_pheromone(obj, context)
  85. # TEST: le temps de tout tester
  86. if self._move_to_point == obj.get_colony().get_start_position() and obj.is_carrying():
  87. obj_transported = obj.get_carried()
  88. obj_transported.set_carried_by(None)
  89. obj.put_carry(obj_transported, (-1, 0, 0))
  90. obj.get_brain().switch_to_mode(MOVE_MODE_EXPLO)
  91. context.metas.collections.add_remove(obj.get_id(),
  92. COL_TRANSPORTER_NOT_CARRYING,
  93. COL_TRANSPORTER_CARRYING)
  94. else:
  95. try:
  96. blocked_since = context.metas.value.get(BLOCKED_SINCE, self._object_id)
  97. except:
  98. blocked_since = 0
  99. context.metas.value.set(BLOCKED_SINCE, self._object_id, blocked_since+1)
  100. @staticmethod
  101. def _appose_pheromone(obj, context):
  102. # TODO: Cette action de pheromone doit etre une surcharge de Move afin d'avoir une Action Move generique.
  103. try:
  104. DirectionPheromone.appose(context,
  105. obj.get_position(),
  106. obj.get_movement_pheromone_gland().get_movement_molecules())
  107. except BestPheromoneHere as best_pheromone_here:
  108. obj.get_brain().set_distance_from_objective(best_pheromone_here.get_best_distance())