Shuvit game master repo. http://shuvit.org
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

walkers.py 5.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. import bge
  2. import random
  3. import FSM
  4. import astar
  5. import observer
  6. car_colors = [[.2,.01,.01,1], [.5,.5,.4,1], [.005,.01,.015,1], [.005, .1, 0.003, 1], [.1, .1, .1, 1]]
  7. npcs = ['carlos', 'maria', 'denise', 'charlie']
  8. def idle_run_check(self):
  9. output = False
  10. for x in bge.logic.getCurrentScene().objects:
  11. if 'walker_idle_spot' in x.name:
  12. self.idle_spots.append(x)
  13. x['in_use'] = False
  14. output = True
  15. print(output, 'walker idle spots')
  16. return output
  17. def get_idle_spots():
  18. op = []
  19. for obj in bge.logic.getCurrentScene().objects:
  20. if 'idle_spot' in obj.name:
  21. ps = IdleSpot(obj, 'available')
  22. if 'corner' in obj:
  23. ps.type = 'corner'
  24. op.append(ps)
  25. print('&&&&&& idle spots')
  26. print(op)
  27. return op
  28. def get_intersections():
  29. op = []
  30. for obj in bge.logic.getCurrentScene().objects:
  31. if 'intersection' in obj:
  32. op.append(obj)
  33. return op
  34. def add_walker(self, x):
  35. print(x.obj.worldPosition, '--walker here')
  36. #walker = bge.logic.getCurrentScene().addObject('larryCube', x.obj, 0)
  37. #walker = bge.logic.getCurrentScene().addObject('npcCube', x.obj, 0)
  38. choice = random.choice(npcs)
  39. walker = bge.logic.getCurrentScene().addObject(choice, x.obj, 0)
  40. print(walker.worldPosition)
  41. #walker.worldPosition = x.obj.worldPosition
  42. walker.worldOrientation = x.obj.worldOrientation
  43. walker.worldPosition.z += 2.8
  44. walker.name = 'lwalker' + str(len(self.manager.walkers))
  45. walker.childrenRecursive['shirt'].color = [random.random(),random.random(),random.random(),1]
  46. walker['npc'] = True
  47. x.status = 'in_use'
  48. return walker
  49. #====================================
  50. class IdleSpot:
  51. def __init__(self, obj, status):
  52. self.obj = obj
  53. self.status = status
  54. self.type = None
  55. class Walker:
  56. def __init__(self, own, start_empty):
  57. self.manager = own
  58. self.life = 0
  59. self.start_empty = start_empty
  60. self.obj = add_walker(self, self.start_empty)
  61. self.speed_targ = self.manager.default_speed
  62. self.speed_inc = 100
  63. self.FSM = FSM.WalkerFSM(self)
  64. self.active = False
  65. self.target = start_empty
  66. self.lane_point = self.obj.worldPosition
  67. self.point = self.obj.worldPosition
  68. self.last_lane_point = self.obj.worldPosition
  69. self.path = None
  70. self.path_index = 0
  71. self.path_display = []
  72. self.cla = None
  73. self.manager.pub.register("path found", self)
  74. #self.obj.worldPosition = self.start_empty.obj.worldPosition
  75. def Execute(self):
  76. self.FSM.Execute()
  77. self.life += 1
  78. def ReceiveMessage(self, message):
  79. #print(message[0])
  80. if message[0] == 'path':
  81. if self == message[1]:
  82. self.path = message[2]
  83. self.FSM.FSM.ToTransition('toExitParallelPark')
  84. #print('path got')
  85. elif message[0] == 'hit':
  86. if self.obj == message[2]:
  87. #print('something was hit')
  88. active_ = False
  89. if self in self.manager.walkers_active:
  90. active_ = True
  91. #print(self.FSM.FSM.curState.__class__.__name__, 'name')
  92. if self.FSM.FSM.curState.__class__.__name__ == 'NavigateToTarget':
  93. self.FSM.FSM.ToTransition('toWalkingHitBySkater')
  94. else:
  95. self.manager.walkers_active.append(self)
  96. self.FSM.FSM.ToTransition('toHitBySkater')
  97. #print('i was hit', message[2], active_)
  98. class WalkerManager:
  99. def __init__(self, own):
  100. self.parent = own
  101. self.navmesh = None
  102. self.pub = observer.Publisher(['path found', 'working'])
  103. self.navmesh2 = astar.Astar('walker_nav_points', self.pub)
  104. self.walkers = []
  105. self.max_walkers = 6
  106. self.max_active = 3
  107. self.targets = []
  108. self.target_loc = None
  109. self.idle_spots = get_idle_spots()
  110. self.active = True
  111. self.walkers_active = []
  112. self.walkers_loaded = False
  113. self.life = 0
  114. self.default_speed = 1.0#5.0
  115. self.lane_position = 0.5#1.75
  116. def load_walkers(self):
  117. iter_ = 0
  118. if npcs[0] in bge.logic.getCurrentScene().objectsInactive:
  119. start_choices = self.idle_spots.copy()
  120. while len(self.walkers) < self.max_walkers:
  121. #get starting position
  122. start_choice = random.choice(start_choices)
  123. start_choices.remove(start_choice)
  124. walker_ = Walker(self, start_choice)
  125. #walker_.cla = self
  126. self.walkers.append(walker_)
  127. for x in self.idle_spots:
  128. iter_ += 1
  129. self.walkers_loaded = True
  130. else:
  131. for obj in npcs:
  132. mainDir = bge.logic.expandPath("//npc_walkers/")
  133. npc = obj
  134. fileName = mainDir + str(npc) + '.blend'
  135. path = bge.logic.expandPath(fileName)
  136. print('loading npc')
  137. try:
  138. bge.logic.LibLoad(fileName, 'Scene', load_actions=True)
  139. print(' loaded', obj)
  140. except Exception as e:
  141. print('loading', fileName, 'failed', e)
  142. def activator_check(self):
  143. if len(self.walkers_active) < self.max_active:
  144. l = []
  145. for c in self.walkers:
  146. if not c.active:
  147. l.append(c)
  148. walker = random.choice(l)
  149. self.walkers_active.append(walker)
  150. walker.active = True
  151. walker.obj.worldPosition = walker.start_empty.obj.worldPosition
  152. walker.FSM.FSM.ToTransition('toRequestPath')
  153. #print('activating walker')
  154. def update(self):
  155. self.life += 1
  156. if self.walkers_loaded:
  157. if self.life % 180 == 0:
  158. self.activator_check()
  159. for walker in self.walkers_active:
  160. walker.Execute()
  161. self.navmesh2.update()
  162. else:
  163. self.load_walkers()
  164. def Execute(cont):
  165. own = cont.owner
  166. if 'walker_manager' not in own:
  167. own['walker_manager'] = WalkerManager(own)
  168. walker_man = own['walker_manager']
  169. if walker_man.active:
  170. walker_man.update()