Legend of the Gold Box... A game written for the LOWREZJAM 2018 game jam
No puede seleccionar más de 25 temas Los temas deben comenzar con una letra o número, pueden incluir guiones ('-') y pueden tener hasta 35 caracteres de largo.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765
  1. from . import gbe
  2. import pygame
  3. class NodeOptions(gbe.nodes.Node2D):
  4. def __init__(self, name="Options", parent=None):
  5. try:
  6. gbe.nodes.Node2D.__init__(self, name, parent)
  7. except gbe.nodes.NodeError as e:
  8. raise e
  9. self._options = []
  10. self._oindex = 0
  11. self._color_select = (255,255,0)
  12. self._color_idle = (255,255,255)
  13. def add_option(self, font_src, size, text, event, params={}):
  14. nodeName = "OpText{}".format(len(self._options))
  15. nodeOption = gbe.nodes.NodeText(nodeName, self)
  16. nodeOption.font_src = font_src
  17. nodeOption.size = size
  18. nodeOption.text = text
  19. nodeOption.antialias = False
  20. if len(self._options) == 0:
  21. nodeOption.set_color(*self._color_select)
  22. else:
  23. nodeOption.set_color(*self._color_idle)
  24. li = len(self._options)-1
  25. lp = self._options[li][0].position
  26. ls = self._options[li][0].size
  27. nodeOption.position = (lp[0], lp[1] + ls + 1)
  28. self._options.append([nodeOption, event, params])
  29. def on_start(self):
  30. self.listen("KEYPRESSED", self.on_keypressed)
  31. def on_pause(self):
  32. self.unlisten("KEYPRESSED", self.on_keypressed)
  33. def on_keypressed(self, event, params):
  34. if params["key_name"] == "w":
  35. if self._oindex > 0:
  36. self._options[self._oindex][0].set_color(*self._color_idle)
  37. self._oindex -= 1
  38. self._options[self._oindex][0].set_color(*self._color_select)
  39. elif params["key_name"] == "s":
  40. if self._oindex < len(self._options) - 1:
  41. self._options[self._oindex][0].set_color(*self._color_idle)
  42. self._oindex += 1
  43. self._options[self._oindex][0].set_color(*self._color_select)
  44. elif params["key_name"] in ["enter", "return"]:
  45. if len(self._options) > 0:
  46. op = self._options[self._oindex]
  47. self.emit(op[1], op[2])
  48. elif params["key_name"] == "escape":
  49. self.emit("QUIT")
  50. class NodeGameMap(gbe.nodes.Node2D):
  51. def __init__(self, name="GameMap", parent=None):
  52. try:
  53. gbe.nodes.Node2D.__init__(self, name, parent)
  54. except gbe.nodes.NodeError as e:
  55. raise e
  56. self._renderMode = 0 # 0 = Top-down | 1 = Perspective
  57. self._layer = {}
  58. self._currentLayer = ""
  59. self._res = {
  60. "env_src":"",
  61. "env":None,
  62. "h_index":0,
  63. "g_index":0,
  64. "wall_src":"",
  65. "walls":None,
  66. "wall_index":-1,
  67. "door_index":-1
  68. }
  69. self._topdown = {
  70. "size":8, # Pixels square
  71. "wall_color":pygame.Color(255, 255, 255),
  72. "blocked_color":pygame.Color(255, 0, 0)
  73. }
  74. self._cellpos = [0,0]
  75. self._orientation = "n"
  76. @property
  77. def environment_source(self):
  78. return self._res["env_src"]
  79. @property
  80. def wall_source(self):
  81. return self._res["wall_src"]
  82. @property
  83. def layer_count(self):
  84. return len(self._layer)
  85. @property
  86. def current_layer(self):
  87. return self._currentLayer
  88. @property
  89. def current_layer_width(self):
  90. if self._currentLayer != "":
  91. return self._layer[self._currentLayer]["w"]
  92. return 0
  93. @property
  94. def current_layer_height(self):
  95. if self._currentLayer != "":
  96. return self._layer[self._currentLayer]["h"]
  97. return 0
  98. @property
  99. def layer_names(self):
  100. names = []
  101. for key in self._layer:
  102. names.append(names)
  103. return names
  104. @property
  105. def orientation(self):
  106. return self._orientation
  107. @property
  108. def cell_position(self):
  109. return (self._cellpos[0], self._cellpos[1])
  110. def set_resources(self, env_src, wall_src):
  111. res = self.resource
  112. if env_src != "" and env_src != self._res["env_src"]:
  113. if res.is_valid("json", env_src):
  114. if not res.has("json", env_src):
  115. res.store("json", env_src)
  116. e = res.get("json", env_src)
  117. if e is not None and e() is not None:
  118. self._res["env_src"] = env_src
  119. self._res["env"] = e
  120. # NOTE: Making a lot of assumptions to the structural validity of the data file.
  121. isrc1 = e().data["horizon"]["src"]
  122. isrc2 = e().data["ground"]["src"]
  123. if res.is_valid("graphic", isrc1) and res.is_valid("graphic", isrc2):
  124. if not res.has("graphic", isrc1):
  125. res.store("graphic", isrc1)
  126. if not res.has("graphic", isrc2):
  127. res.store("graphic", isrc2)
  128. if wall_src != "" and wall_src != self._res["wall_src"]:
  129. if res.is_valid("json", wall_src):
  130. if not res.has("json", wall_src):
  131. print("Storing resource {}".format(wall_src))
  132. res.store("json", wall_src)
  133. w = res.get("json", wall_src)
  134. if w is not None and w() is not None:
  135. self._res["wall_src"] = wall_src
  136. self._res["walls"] = w
  137. # NOTE: I'm making a lot of assumptions to the structural validity of the data file, but...
  138. imgsrc = w().data["src"]
  139. if res.is_valid("graphic", imgsrc):
  140. if not res.has("graphic", imgsrc):
  141. res.store("graphic", imgsrc)
  142. else:
  143. print("Failed to get JSON instance {}".format(wall_src))
  144. else:
  145. print("Invalid JSON {}".format(wall_src))
  146. def add_layer(self, name, w, h):
  147. if name == "" or name in self._layer:
  148. return
  149. self._layer[name] = {
  150. "w":w,
  151. "h":h,
  152. "cells":[]
  153. }
  154. for c in range(0, w*h):
  155. self._layer[name]["cells"].append({
  156. "h":0, # Horizon
  157. "g":0, # Ground
  158. "n":[-1, False, -1, None], # North Wall | [<graphic index, -1 = None>, <blocking>, <door index, -1 = None, 0 = closed, 1=open>, <door target>]
  159. "s":[-1, False, -1, None], # South Wall
  160. "e":[-1, False, -1, None], # East Wall
  161. "w":[-1, False, -1, None], # West Wall
  162. })
  163. if self._currentLayer == "":
  164. self._currentLayer = name
  165. def set_active_layer(self, name, x=0, y=0):
  166. if name == "" or not (name in self._layers):
  167. return
  168. layer = self._layers[name]
  169. if x >= 0 and x < layer["w"] and y >= 0 and y < layer["h"]:
  170. self._currentLayer = name
  171. self._cellpos = [x,y]
  172. def set_cell_env(self, x, y, ceiling=-1, ground=-1):
  173. if self._currentLayer == "":
  174. return
  175. layer = self._layer[self._currentLayer]
  176. if x >= 0 and x < layer["w"] and y >= 0 and y < layer["h"]:
  177. index = (y * layer["w"]) + x
  178. cell = layer["cells"]
  179. if ceiling >= 0:
  180. cell[index]["c"] = ceiling
  181. if ground >= 0:
  182. cell[index]["g"] = ground
  183. def fill_cell_env(self, x1, y1, x2, y2, ceiling=-1, ground=-1):
  184. if self._currentLayer == "":
  185. return
  186. for y in range(y1, y2+1):
  187. for x in range(x1, x2+1):
  188. self.set_cell_env(x, y, ceiling, ground)
  189. def set_cell_face(self, x, y, face, gi=-2, blocking=None):
  190. if self._currentLayer == "" or not (face == "n" or face == "s" or face == "w" or face == "e"):
  191. return
  192. layer = self._layer[self._currentLayer]
  193. if x >= 0 and x < layer["w"] and y >= 0 and y < layer["h"]:
  194. index = (y * layer["w"]) + x
  195. cell = layer["cells"]
  196. if gi <= -2:
  197. gi = self._res["wall_index"]
  198. if gi >= -1:
  199. cell[index][face][0] = gi
  200. if blocking is not None:
  201. cell[index][face][1] = (blocking == True)
  202. else:
  203. if gi == -1: # If gi = -1, there is no wall, so, by default, there is no blocking.
  204. cell[index][face][1] = False
  205. else: # Otherwise, blocking is assumed :)
  206. cell[index][face][1] = True
  207. elif blocking is not None:
  208. blocking = (blocking == True) # Forcing a boolean
  209. cell[index][face][1] = blocking
  210. def next_wall(self):
  211. if self._res["walls"] is not None:
  212. w = self._res["walls"]()
  213. if w is not None:
  214. windex = self._res["wall_index"] + 1
  215. if windex >= len(w.data["walls"]):
  216. windex = -1
  217. self._res["wall_index"] = windex
  218. print("Next Wall Index: {}".format(windex))
  219. def prev_wall(self):
  220. if self._res["walls"] is not None:
  221. w = self._res["walls"]()
  222. if w is not None:
  223. windex = self._res["wall_index"] - 1
  224. if windex < -1:
  225. windex = len(w.data["walls"]) - 1
  226. self._res["wall_index"] = windex
  227. print("Prev Wall Index: {}".format(windex))
  228. def turn_left(self):
  229. onum = self._d_s2n(self._orientation)
  230. onum -= 1
  231. if onum < 0:
  232. onum = 3
  233. self._orientation = self._d_n2s(onum)
  234. def turn_right(self):
  235. onum = self._d_s2n(self._orientation)
  236. onum += 1
  237. if onum > 3:
  238. onum = 0
  239. self._orientation = self._d_n2s(onum)
  240. def move_to(self, x, y):
  241. if x >= 0 and x < self.current_layer_width and y >= 0 and y < self.current_layer_height:
  242. self._cellpos = [x, y]
  243. def move_forward(self, ignore_passible=False):
  244. if ignore_passible or self.is_passible(self._cellpos[0], self._cellpos[1], self._orientation):
  245. x = self._cellpos[0]
  246. y = self._cellpos[1]
  247. if self._orientation == "n":
  248. y -= 1
  249. elif self._orientation == "e":
  250. x += 1
  251. elif self._orientation == "s":
  252. y += 1
  253. elif self._orientation == "w":
  254. x -= 1
  255. self.move_to(x, y)
  256. def move_backward(self, ignore_passible=False):
  257. orient = self._orientation
  258. if self._orientation == "n":
  259. self._orientation = "s"
  260. elif self._orientation == "s":
  261. self._orientation = "n"
  262. elif self._orientation == "e":
  263. self._orientation = "w"
  264. elif self._orientation == "w":
  265. self._orientation = "e"
  266. self.move_forward(ignore_passible)
  267. self._orientation = orient
  268. def is_passible(self, x, y, d):
  269. """
  270. Returns true if it's possible to move forward from the x, y map position in the direction given.
  271. d - 0 = North, 1 = East, 2 = South, 3 = West
  272. """
  273. if self._currentLayer == "" or d < 0 or d >= 4:
  274. return False
  275. layer = self._layer[self._currentLayer]
  276. if x >= 0 and x < layer["w"] and y >= 0 and y < layer["h"]:
  277. index = (y * layer["w"]) + x
  278. d = self._d_n2s(d)
  279. return not layer["cells"][index][d][1]
  280. return False
  281. def toggle_render_mode(self):
  282. if self._renderMode == 0:
  283. self._renderMode = 1
  284. else:
  285. self._renderMode = 0
  286. def set_render_mode(self, mode):
  287. if mode <= 0:
  288. self._renderMode = 0
  289. else:
  290. self._renderMode = 1
  291. def get_render_mode(self):
  292. return self._renderMode
  293. def _d_n2s(self, d): # _(d)irection_(n)umber_to_(s)tring
  294. if d == 0:
  295. return "n"
  296. elif d == 1:
  297. return "e"
  298. elif d == 2:
  299. return "s"
  300. elif d == 3:
  301. return "w"
  302. return ""
  303. def _d_s2n(self, d):
  304. if d == "n":
  305. return 0
  306. elif d == "e":
  307. return 1
  308. elif d == "s":
  309. return 2
  310. elif d == "w":
  311. return 3
  312. return -1
  313. def _indexFromPos(self, x, y):
  314. if x >= 0 and x < self.current_layer_width and y >= 0 and y < self.current_layer_height:
  315. return (y * self.current_layer_width) + x
  316. return -1
  317. def _getCell(self, x, y):
  318. if x >=- 0 and x < self.current_layer_width and y >= 0 and y < self.current_layer_height:
  319. index = self._indexFromPos(x, y)
  320. return self._layer[self._currentLayer]["cells"][index]
  321. return None
  322. def _getOrientVec(self):
  323. if self._orientation == "n":
  324. return (0, -1)
  325. elif self._orientation == "s":
  326. return (0, 1)
  327. elif self._orientation == "e":
  328. return (1, 0)
  329. elif self._orientation == "w":
  330. return (-1, 0)
  331. return (0,0)
  332. def _RenderTopDown(self):
  333. cell_size = self._topdown["size"]
  334. size = self.resolution
  335. pos = self._cellpos
  336. lsize = (self.current_layer_width, self.current_layer_height)
  337. hcells = int(size[0] / cell_size)
  338. vcells = int(size[1] / cell_size)
  339. cx = pos[0] - int(hcells * 0.5)
  340. cy = pos[1] - int(vcells * 0.5)
  341. ry = -int(cell_size*0.5)
  342. for j in range(0, vcells+1):
  343. y = cy + j
  344. if y >= 0 and y < lsize[1]:
  345. rx = -int(cell_size*0.5)
  346. for i in range(0, hcells+1):
  347. x = cx + i
  348. if x >= 0 and x < lsize[0]:
  349. cell = self._getCell(x, y)
  350. if cell["n"][0] >= 0:
  351. self.draw_rect((rx, ry, cell_size, 2), self._topdown["wall_color"], 0, self._topdown["wall_color"])
  352. if cell["e"][0] >= 0:
  353. self.draw_rect((rx+(cell_size-2), ry, 2, cell_size), self._topdown["wall_color"], 0, self._topdown["wall_color"])
  354. if cell["s"][0] >= 0:
  355. self.draw_rect((rx, ry+(cell_size-2), cell_size, 2), self._topdown["wall_color"], 0, self._topdown["wall_color"])
  356. if cell["w"][0] >= 0:
  357. self.draw_rect((rx, ry, 2, cell_size), self._topdown["wall_color"], 0, self._topdown["wall_color"])
  358. if cell["n"][1] == True:
  359. self.draw_lines([(rx+1, ry+1), (rx+(cell_size-2), ry+1)], self._topdown["blocked_color"], 1)
  360. if cell["e"][1] == True:
  361. self.draw_lines([(rx+(cell_size-2), ry+1), (rx+(cell_size-2), ry+(cell_size-2))], self._topdown["blocked_color"], 1)
  362. if cell["s"][1] == True:
  363. self.draw_lines([(rx+1, ry+(cell_size-2)), (rx+(cell_size-2), ry+(cell_size-2))], self._topdown["blocked_color"], 1)
  364. if cell["w"][1] == True:
  365. self.draw_lines([(rx+1, ry+1), (rx+1, ry+(cell_size-2))], self._topdown["blocked_color"], 1)
  366. rx += cell_size
  367. ry += cell_size
  368. def _RenderPersFar(self, pos, size, o, orr, orl, wdat, wsurf):
  369. ovec = self._getOrientVec()
  370. fx = pos[0] + (ovec[0]*2)
  371. fy = pos[1] + (ovec[1]*2)
  372. fcell = self._getCell(fx, fy)
  373. if fcell == None:
  374. return # If we can't see the cell directly ahead, the other's won't be visible either!
  375. lcell = llcell = rcell = rrcell = None
  376. if ovec[0] == 0: # Facing North/South
  377. lcell = self._getCell(fx + ovec[1], fy)
  378. llcell = self._getCell(fx + (ovec[1]*2), fy)
  379. rcell = self._getCell(fx - ovec[1], fy)
  380. rrcell = self._getCell(fx - (ovec[1]*2), fy)
  381. else: # Facing East/West
  382. lcell = self._getCell(fx, fy - ovec[0])
  383. llcell = self._getCell(fx, fy - (ovec[0]*2))
  384. rcell = self._getCell(fx, fy + ovec[0])
  385. rrcell = self._getCell(fx, fy + (ovec[0]*2))
  386. hsw = int(size[0]*0.5)
  387. hsh = int(size[1]*0.5)
  388. # Rendering from edges to center
  389. if llcell is not None:
  390. if llcell[o][0] >= 0:
  391. rect = wdat["walls"][llcell[o][0]]["f_far"]
  392. hw = int(rect[2]*0.5)
  393. hh = int(rect[3]*0.5)
  394. self.draw_image(wsurf, (0, hsh-hh), (rect[0], rect[1], hw, rect[3]))
  395. if rrcell is not None:
  396. if rrcell[o][0] >= 0:
  397. rect = wdat["walls"][rrcell[o][0]]["f_far"]
  398. hw = int(rect[2]*0.5)
  399. hh = int(rect[3]*0.5)
  400. self.draw_image(wsurf, (size[0]-hw, hsh-hh), (rect[0]+hw, rect[1], hw, rect[3]))
  401. if lcell is not None:
  402. if lcell[o][0] >= 0:
  403. rect = wdat["walls"][lcell[o][0]]["f_far"]
  404. hw = int(rect[2]*0.5)
  405. hh = int(rect[3]*0.5)
  406. self.draw_image(wsurf, (hw, hsh-hh), (rect[0], rect[1], rect[2], rect[3]))
  407. if lcell[orl][0] >= 0:
  408. rect = wdat["walls"][lcell[orl][0]]["s_far"]
  409. lsurf = pygame.transform.flip(wsurf.subsurface(rect), True, False)
  410. hh = int(rect[3]*0.5)
  411. self.draw_image(lsurf, (0, hsh-hh))
  412. if rcell is not None:
  413. if rcell[o][0] >= 0:
  414. rect = wdat["walls"][rcell[o][0]]["f_far"]
  415. hw = int(rect[2]*0.5)
  416. hh = int(rect[3]*0.5)
  417. self.draw_image(wsurf, (size[0]-(rect[2]+hw), hsh-hh), (rect[0], rect[1], rect[2], rect[3]))
  418. if rcell[orr][0] >= 0:
  419. rect = wdat["walls"][rcell[orr][0]]["s_far"]
  420. hh = int(rect[3]*0.5)
  421. self.draw_image(wsurf, (size[0]-rect[2], hsh-hh), (rect[0], rect[1], rect[2], rect[3]))
  422. # Rendering the main cell!!
  423. frect = None # This will be used to place walls
  424. if fcell[o][0] >= 0:
  425. frect = wdat["walls"][fcell[o][0]]["f_far"]
  426. hw = int(frect[2]*0.5)
  427. hh = int(frect[3]*0.5)
  428. self.draw_image(wsurf, (hsw-hw, hsh-hh), (frect[0], frect[1], frect[2], frect[3]))
  429. if fcell[orl][0] >= 0:
  430. rect = wdat["walls"][fcell[orl][0]]["s_far"]
  431. if frect is None:
  432. # Kinda cheating in that it's known that all walls are the same size.
  433. frect = wdat["walls"][fcell[orl][0]]["f_far"]
  434. hw = int(frect[2]*0.5)
  435. lsurf = pygame.transform.flip(wsurf.subsurface(rect), True, False)
  436. self.draw_image(lsurf, (hsw-(hw+rect[2]), hsh-int(rect[3]*0.5)))
  437. if fcell[orr][0] >= 0:
  438. rect = wdat["walls"][fcell[orr][0]]["s_far"]
  439. if frect is None:
  440. frect = wdat["walls"][fcell[orr][0]]["f_far"]
  441. hw = int(frect[2]*0.5)
  442. self.draw_image(wsurf, (hsw+hw, hsh-int(rect[3]*0.5)), (rect[0], rect[1], rect[2], rect[3]))
  443. def _RenderPersMid(self, pos, size, o, orr, orl, wdat, wsurf):
  444. ovec = self._getOrientVec()
  445. fx = pos[0] + ovec[0]
  446. fy = pos[1] + ovec[1]
  447. fcell = self._getCell(fx, fy)
  448. if fcell == None:
  449. return # If we can't see the cell directly ahead, the other's won't be visible either!
  450. lcell = rcell = None
  451. if ovec[0] == 0: # Facing North/South
  452. lcell = self._getCell(fx + ovec[1], fy)
  453. rcell = self._getCell(fx - ovec[1], fy)
  454. else: # Facing East/West
  455. lcell = self._getCell(fx, fy - ovec[0])
  456. rcell = self._getCell(fx, fy + ovec[0])
  457. hsw = int(size[0]*0.5)
  458. hsh = int(size[1]*0.5)
  459. # Render from outside inwards!
  460. if lcell is not None:
  461. if lcell[o][0] >= 0:
  462. rect = wdat["walls"][lcell[o][0]]["f_mid"]
  463. hw = int(rect[2]*0.5)
  464. hh = int(rect[3]*0.5)
  465. self.draw_image(wsurf, (0, hsh-hh), (rect[0]+hw, rect[1], int(rect[2]*0.5), rect[3]))
  466. if rcell is not None:
  467. if rcell[o][0] >= 0:
  468. rect = wdat["walls"][rcell[o][0]]["f_mid"]
  469. hw = int(rect[2]*0.5)
  470. hh = int(rect[3]*0.5)
  471. self.draw_image(wsurf, (size[0]-hw, hsh-hh), (rect[0], rect[1], hw, rect[3]))
  472. # Rendering the main cell!!
  473. frect = None # This will be used to place walls
  474. if fcell[o][0] >= 0:
  475. frect = wdat["walls"][fcell[o][0]]["f_mid"]
  476. hw = int(frect[2]*0.5)
  477. hh = int(frect[3]*0.5)
  478. self.draw_image(wsurf, (hsw-hw, hsh-hh), (frect[0], frect[1], frect[2], frect[3]))
  479. if fcell[orl][0] >= 0:
  480. rect = wdat["walls"][fcell[orl][0]]["s_mid"]
  481. if frect is None:
  482. # Kinda cheating in that it's known that all walls are the same size.
  483. frect = wdat["walls"][fcell[orl][0]]["f_mid"]
  484. hw = int(frect[2]*0.5)
  485. lsurf = pygame.transform.flip(wsurf.subsurface(rect), True, False)
  486. self.draw_image(lsurf, (hsw-(hw+rect[2]), hsh-int(rect[3]*0.5)))
  487. if fcell[orr][0] >= 0:
  488. rect = wdat["walls"][fcell[orr][0]]["s_mid"]
  489. if frect is None:
  490. frect = wdat["walls"][fcell[orr][0]]["f_mid"]
  491. hw = int(frect[2]*0.5)
  492. self.draw_image(wsurf, (hsw+hw, hsh-int(rect[3]*0.5)), (rect[0], rect[1], rect[2], rect[3]))
  493. def _RenderPersClose(self, pos, size, o, orr, orl, cell, wdat, wsurf):
  494. fcell = self._getCell(pos[0], pos[1])
  495. hsw = int(size[0]*0.5)
  496. hsh = int(size[1]*0.5)
  497. ovec = self._getOrientVec()
  498. lcell = rcell = None
  499. if ovec[0] == 0: # Facing North/South
  500. lcell = self._getCell(pos[0] + ovec[1], pos[1])
  501. rcell = self._getCell(pos[0] - ovec[1], pos[1])
  502. else: # Facing East/West
  503. lcell = self._getCell(pos[0], pos[1] - ovec[0])
  504. rcell = self._getCell(pos[0], pos[1] + ovec[0])
  505. hsw = int(size[0]*0.5)
  506. hsh = int(size[1]*0.5)
  507. # Render from outside inwards!
  508. if lcell is not None:
  509. if lcell[o][0] >= 0:
  510. rect = wdat["walls"][lcell[o][0]]["f_close"]
  511. hw = int(rect[2]*0.5)
  512. hh = int(rect[3]*0.5)
  513. rw = hsw - hw
  514. self.draw_image(wsurf, (0, hsh-hh), (rect[0]+(rect[2]-rw), rect[1], rw, rect[3]))
  515. if rcell is not None:
  516. if rcell[o][0] >= 0:
  517. rect = wdat["walls"][rcell[o][0]]["f_close"]
  518. hw = int(rect[2]*0.5)
  519. hh = int(rect[3]*0.5)
  520. rw = hsw - hw
  521. self.draw_image(wsurf, (size[0]-rw, hsh-hh), (rect[0], rect[1], rw, rect[3]))
  522. # Rendering the main cell!!
  523. frect = None # This will be used to place walls
  524. if fcell[o][0] >= 0:
  525. idx = fcell[o][0]
  526. frect = wdat["walls"][idx]["f_close"]
  527. hw = int(frect[2]*0.5)
  528. hh = int(frect[3]*0.5)
  529. self.draw_image(wsurf, (hsw-hw, hsh-hh), (frect[0], frect[1], frect[2], frect[3]))
  530. if fcell[orl][0] >= 0:
  531. idx = fcell[orl][0]
  532. rect = wdat["walls"][idx]["s_close"]
  533. if frect is None:
  534. # Kinda cheating in that it's known that all walls are the same size.
  535. frect = wdat["walls"][idx]["f_close"]
  536. hw = int(frect[2]*0.5)
  537. lsurf = pygame.transform.flip(wsurf.subsurface(rect), True, False)
  538. self.draw_image(lsurf, (hsw-(hw+rect[2]), hsh-int(rect[3]*0.5)))
  539. if fcell[orr][0] >= 0:
  540. idx = fcell[orr][0]
  541. rect = wdat["walls"][idx]["s_close"]
  542. if frect is None:
  543. frect = wdat["walls"][idx]["f_close"]
  544. hw = int(frect[2]*0.5)
  545. self.draw_image(wsurf, (hsw+hw, hsh-int(rect[3]*0.5)), (rect[0], rect[1], rect[2], rect[3]))
  546. def _RenderPerspective(self):
  547. # Getting ALL of the core resources
  548. rm = self.resource
  549. edat = self._res["env"]
  550. wdat = self._res["walls"]
  551. if edat is None or wdat is None or edat() is None or wdat() is None:
  552. return
  553. edat = edat().data
  554. wdat = wdat().data
  555. ehsurf = rm.get("graphic", edat["horizon"]["src"])
  556. egsurf = rm.get("graphic", edat["ground"]["src"])
  557. wsurf = rm.get("graphic", wdat["src"])
  558. if ehsurf is None or egsurf is None or wsurf is None:
  559. return
  560. if ehsurf() is None or egsurf() is None or wsurf() is None:
  561. return
  562. oshift = lambda d: d if d >= 0 and d < 4 else (0 if d > 3 else 3)
  563. px = self._cellpos[0]
  564. py = self._cellpos[1]
  565. orl = self._d_n2s(oshift(self._d_s2n(self._orientation) - 1))
  566. orr = self._d_n2s(oshift(self._d_s2n(self._orientation) + 1))
  567. cell = self._getCell(px, py)
  568. # First, output the ground and horizon
  569. # TODO Later, perhaps cut the horizon and ground to represent each possible cell instead of just the current one?
  570. self.draw_image(ehsurf(), (0,0), edat["horizon"]["defs"][cell["h"]]["rect"])
  571. self.draw_image(egsurf(), (0,32), edat["ground"]["defs"][cell["g"]]["rect"])
  572. # Rendering the rest
  573. size = self.resolution
  574. self._RenderPersFar((px, py), size, self._orientation, orr, orl, wdat, wsurf())
  575. self._RenderPersMid((px, py), size, self._orientation, orr, orl, wdat, wsurf())
  576. self._RenderPersClose((px, py), size, self._orientation, orr, orl, cell, wdat, wsurf())
  577. def on_render(self):
  578. if self._renderMode == 0:
  579. self._RenderTopDown()
  580. else:
  581. self._RenderPerspective()
  582. class NodeMapEditor(gbe.nodes.Node2D):
  583. def __init__(self, name="MapEditor", parent=None):
  584. try:
  585. gbe.nodes.Node2D.__init__(self, name, parent)
  586. except gbe.nodes.Node2D as e:
  587. raise e
  588. self._last_orientation = ""
  589. self._size = 0
  590. self._thickness = 1
  591. self._color = pygame.Color(255,255,255)
  592. self._points = None
  593. self.pointer_size = 4
  594. def _getPoints(self, size):
  595. p = self.parent
  596. o = p.orientation
  597. points = self._points[o]
  598. hw = int(size[0]*0.5)
  599. hh = int(size[1]*0.5)
  600. return (
  601. (points[0][0] + hw, points[0][1] + hh),
  602. (points[1][0] + hw, points[1][1] + hh),
  603. (points[2][0] + hw, points[2][1] + hh)
  604. )
  605. @property
  606. def pointer_size(self):
  607. return self._size
  608. @pointer_size.setter
  609. def pointer_size(self, size):
  610. if not isinstance(size, int):
  611. raise TypeError("Size expected to be an integer.")
  612. if size <= 0:
  613. raise ValueError("Size must be greater than zero.")
  614. if size != self._size:
  615. self._size = size
  616. # Now updating all of the pointer... ummm... points
  617. hs = max(1, int(size * 0.5))
  618. self._points = {
  619. "n": ((0,-hs),(hs,hs),(-hs,hs)),
  620. "s": ((0,hs),(hs,-hs),(-hs,-hs)),
  621. "e": ((-hs,hs),(hs,0),(-hs,-hs)),
  622. "w": ((hs,hs),(-hs,0),(hs,-hs))
  623. }
  624. def set_color(self, color):
  625. if isinstance(color, (tuple, list)):
  626. clen = len(color)
  627. if clen == 3 or clen == 4:
  628. iscolor = lambda v : isinstance(v, int) and v >= 0 and v < 256
  629. if iscolor(color[0]) and iscolor(color[1]) and iscolor(color[2]):
  630. if clen == 3:
  631. self._color = pygame.Color(color[0], color[1], color[2])
  632. elif clen == 4 and iscolor(color[3]):
  633. self._color = pygame.Color(color[0], color[1], color[2], color[3])
  634. def get_color(self):
  635. return (self._color.r, self._color.g, self._color.b, self._color.a)
  636. def on_start(self):
  637. self.listen("KEYPRESSED", self.on_keypressed)
  638. def on_pause(self):
  639. self.unlisten("KEYPRESSED", self.on_keypressed)
  640. def on_keypressed(self, event, data):
  641. p = self.parent
  642. if p is None or not isinstance(p, NodeGameMap):
  643. return
  644. if data["key_name"] == "escape":
  645. self.emit("SCENECHANGE", {"scene":"MAIN_MENU", "hold":False})
  646. #self.emit("QUIT")
  647. if data["key_name"] == "tab":
  648. p.toggle_render_mode()
  649. elif data["key_name"] == "w":
  650. p.move_forward(True)
  651. elif data["key_name"] == "s":
  652. p.move_backward(True)
  653. elif data["key_name"] == "a":
  654. p.turn_left()
  655. elif data["key_name"] == "d":
  656. p.turn_right()
  657. elif data["key_name"] == "space":
  658. o = p.orientation
  659. cpos = p.cell_position
  660. p.set_cell_face(cpos[0], cpos[1], o)
  661. elif data["key_name"] == "e":
  662. p.next_wall()
  663. elif data["key_name"] == "q":
  664. p.prev_wall()
  665. def on_render(self):
  666. size = self.resolution
  667. self.draw_lines(self._getPoints(size), self._color, self._thickness, True)