Legend of the Gold Box... A game written for the LOWREZJAM 2018 game jam
Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.

nodes.py 32KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862
  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 load_map(self, src, user=True):
  111. rtype = "maps"
  112. if user == True:
  113. rtype = "user_maps"
  114. rm = self.resource
  115. m = None
  116. try:
  117. m = rm.load(rtype, src)
  118. m = m.data
  119. except Exception as e:
  120. print ("Failed to load '{}': {}".format(src, e))
  121. if "version" not in m:
  122. print("Invalid Map Data!")
  123. return
  124. if m["version"] != "0.0.1":
  125. print ("Invalid map version.")
  126. return
  127. if "count" not in m:
  128. print("Invalid Map Data!")
  129. return
  130. if not isinstance(m["count"], int):
  131. print("Invalid Map Data!")
  132. return
  133. if "layers" not in m:
  134. print ("Invalid Map Data!")
  135. return
  136. if len(m["layers"]) != m["count"]:
  137. print ("Map Layer count mismatch!")
  138. return
  139. if "player" not in m:
  140. print ("Invalid Map Data!")
  141. return
  142. # OK... these are weak tests, but we'll just accept it from here!
  143. self._layer = m["layers"]
  144. self._currentLayer = m["player"]["layer_name"]
  145. self._cellpos = m["player"]["pos"]
  146. self._orientation = m["player"]["orientation"]
  147. print("Map '{}' loaded!".format(src))
  148. def save_map(self, dst):
  149. rm = self.resource
  150. m = {
  151. "version":"0.0.1",
  152. "count":self.layer_count,
  153. "layers":self._layer,
  154. "player":{
  155. "layer_name":self._currentLayer,
  156. "pos":(self._cellpos[0], self._cellpos[1]),
  157. "orientation":self._orientation
  158. }
  159. }
  160. try:
  161. rm.save("user_maps", dst, m)
  162. except Exception as e:
  163. print("Failed to save '{}': {}".format(dst, e))
  164. return
  165. print ("Map '{}' saved.".format(dst))
  166. def set_resources(self, env_src, wall_src):
  167. res = self.resource
  168. if env_src != "" and env_src != self._res["env_src"]:
  169. if res.is_valid("json", env_src):
  170. if not res.has("json", env_src):
  171. res.store("json", env_src)
  172. e = res.get("json", env_src)
  173. if e is not None and e() is not None:
  174. self._res["env_src"] = env_src
  175. self._res["env"] = e
  176. # NOTE: Making a lot of assumptions to the structural validity of the data file.
  177. isrc1 = e().data["horizon"]["src"]
  178. isrc2 = e().data["ground"]["src"]
  179. if res.is_valid("graphic", isrc1) and res.is_valid("graphic", isrc2):
  180. if not res.has("graphic", isrc1):
  181. res.store("graphic", isrc1)
  182. if not res.has("graphic", isrc2):
  183. res.store("graphic", isrc2)
  184. if wall_src != "" and wall_src != self._res["wall_src"]:
  185. if res.is_valid("json", wall_src):
  186. if not res.has("json", wall_src):
  187. print("Storing resource {}".format(wall_src))
  188. res.store("json", wall_src)
  189. w = res.get("json", wall_src)
  190. if w is not None and w() is not None:
  191. self._res["wall_src"] = wall_src
  192. self._res["walls"] = w
  193. # NOTE: I'm making a lot of assumptions to the structural validity of the data file, but...
  194. imgsrc = w().data["src"]
  195. if res.is_valid("graphic", imgsrc):
  196. if not res.has("graphic", imgsrc):
  197. res.store("graphic", imgsrc)
  198. else:
  199. print("Failed to get JSON instance {}".format(wall_src))
  200. else:
  201. print("Invalid JSON {}".format(wall_src))
  202. def add_layer(self, name, w, h):
  203. if name == "" or name in self._layer:
  204. return
  205. self._layer[name] = {
  206. "w":w,
  207. "h":h,
  208. "cells":[]
  209. }
  210. for c in range(0, w*h):
  211. self._layer[name]["cells"].append({
  212. "h":0, # Horizon
  213. "g":0, # Ground
  214. "n":[-1, False, -1, None], # North Wall | [<graphic index, -1 = None>, <blocking>, <door index, -1 = None, 0 = closed, 1=open>, <door target>]
  215. "s":[-1, False, -1, None], # South Wall
  216. "e":[-1, False, -1, None], # East Wall
  217. "w":[-1, False, -1, None], # West Wall
  218. })
  219. if self._currentLayer == "":
  220. self._currentLayer = name
  221. def set_active_layer(self, name, x=0, y=0):
  222. if name == "" or not (name in self._layers):
  223. return
  224. layer = self._layers[name]
  225. if x >= 0 and x < layer["w"] and y >= 0 and y < layer["h"]:
  226. self._currentLayer = name
  227. self._cellpos = [x,y]
  228. def set_cell_env(self, x, y, ceiling=-1, ground=-1):
  229. if self._currentLayer == "":
  230. return
  231. layer = self._layer[self._currentLayer]
  232. if x >= 0 and x < layer["w"] and y >= 0 and y < layer["h"]:
  233. index = (y * layer["w"]) + x
  234. cell = layer["cells"]
  235. if ceiling >= 0:
  236. cell[index]["c"] = ceiling
  237. if ground >= 0:
  238. cell[index]["g"] = ground
  239. def fill_cell_env(self, x1, y1, x2, y2, ceiling=-1, ground=-1):
  240. if self._currentLayer == "":
  241. return
  242. for y in range(y1, y2+1):
  243. for x in range(x1, x2+1):
  244. self.set_cell_env(x, y, ceiling, ground)
  245. def set_cell_face(self, x, y, face, gi=-2, blocking=None):
  246. if self._currentLayer == "" or not (face == "n" or face == "s" or face == "w" or face == "e"):
  247. return
  248. layer = self._layer[self._currentLayer]
  249. if x >= 0 and x < layer["w"] and y >= 0 and y < layer["h"]:
  250. index = (y * layer["w"]) + x
  251. cell = layer["cells"]
  252. if gi <= -2:
  253. gi = self._res["wall_index"]
  254. if gi >= -1:
  255. cell[index][face][0] = gi
  256. if blocking is not None:
  257. cell[index][face][1] = (blocking == True)
  258. else:
  259. if gi == -1: # If gi = -1, there is no wall, so, by default, there is no blocking.
  260. cell[index][face][1] = False
  261. else: # Otherwise, blocking is assumed :)
  262. cell[index][face][1] = True
  263. elif blocking is not None:
  264. blocking = (blocking == True) # Forcing a boolean
  265. cell[index][face][1] = blocking
  266. def next_wall(self):
  267. if self._res["walls"] is not None:
  268. w = self._res["walls"]()
  269. if w is not None:
  270. windex = self._res["wall_index"] + 1
  271. if windex >= len(w.data["walls"]):
  272. windex = -1
  273. self._res["wall_index"] = windex
  274. print("Next Wall Index: {}".format(windex))
  275. def prev_wall(self):
  276. if self._res["walls"] is not None:
  277. w = self._res["walls"]()
  278. if w is not None:
  279. windex = self._res["wall_index"] - 1
  280. if windex < -1:
  281. windex = len(w.data["walls"]) - 1
  282. self._res["wall_index"] = windex
  283. print("Prev Wall Index: {}".format(windex))
  284. def turn_left(self):
  285. onum = self._d_s2n(self._orientation)
  286. onum -= 1
  287. if onum < 0:
  288. onum = 3
  289. self._orientation = self._d_n2s(onum)
  290. def turn_right(self):
  291. onum = self._d_s2n(self._orientation)
  292. onum += 1
  293. if onum > 3:
  294. onum = 0
  295. self._orientation = self._d_n2s(onum)
  296. def move_to(self, x, y):
  297. if x >= 0 and x < self.current_layer_width and y >= 0 and y < self.current_layer_height:
  298. self._cellpos = [x, y]
  299. def move_forward(self, ignore_passible=False):
  300. if ignore_passible or self.is_passible(self._cellpos[0], self._cellpos[1], self._orientation):
  301. x = self._cellpos[0]
  302. y = self._cellpos[1]
  303. if self._orientation == "n":
  304. y -= 1
  305. elif self._orientation == "e":
  306. x += 1
  307. elif self._orientation == "s":
  308. y += 1
  309. elif self._orientation == "w":
  310. x -= 1
  311. self.move_to(x, y)
  312. def move_backward(self, ignore_passible=False):
  313. orient = self._orientation
  314. if self._orientation == "n":
  315. self._orientation = "s"
  316. elif self._orientation == "s":
  317. self._orientation = "n"
  318. elif self._orientation == "e":
  319. self._orientation = "w"
  320. elif self._orientation == "w":
  321. self._orientation = "e"
  322. self.move_forward(ignore_passible)
  323. self._orientation = orient
  324. def is_passible(self, x, y, d):
  325. """
  326. Returns true if it's possible to move forward from the x, y map position in the direction given.
  327. d - 0 = North, 1 = East, 2 = South, 3 = West
  328. """
  329. if self._currentLayer == "" or d < 0 or d >= 4:
  330. return False
  331. layer = self._layer[self._currentLayer]
  332. if x >= 0 and x < layer["w"] and y >= 0 and y < layer["h"]:
  333. index = (y * layer["w"]) + x
  334. d = self._d_n2s(d)
  335. return not layer["cells"][index][d][1]
  336. return False
  337. def toggle_render_mode(self):
  338. if self._renderMode == 0:
  339. self._renderMode = 1
  340. else:
  341. self._renderMode = 0
  342. def set_render_mode(self, mode):
  343. if mode <= 0:
  344. self._renderMode = 0
  345. else:
  346. self._renderMode = 1
  347. def get_render_mode(self):
  348. return self._renderMode
  349. def _d_n2s(self, d): # _(d)irection_(n)umber_to_(s)tring
  350. if d == 0:
  351. return "n"
  352. elif d == 1:
  353. return "e"
  354. elif d == 2:
  355. return "s"
  356. elif d == 3:
  357. return "w"
  358. return ""
  359. def _d_s2n(self, d):
  360. if d == "n":
  361. return 0
  362. elif d == "e":
  363. return 1
  364. elif d == "s":
  365. return 2
  366. elif d == "w":
  367. return 3
  368. return -1
  369. def _indexFromPos(self, x, y):
  370. if x >= 0 and x < self.current_layer_width and y >= 0 and y < self.current_layer_height:
  371. return (y * self.current_layer_width) + x
  372. return -1
  373. def _getCell(self, x, y):
  374. if x >=- 0 and x < self.current_layer_width and y >= 0 and y < self.current_layer_height:
  375. index = self._indexFromPos(x, y)
  376. return self._layer[self._currentLayer]["cells"][index]
  377. return None
  378. def _getOrientVec(self):
  379. if self._orientation == "n":
  380. return (0, -1)
  381. elif self._orientation == "s":
  382. return (0, 1)
  383. elif self._orientation == "e":
  384. return (1, 0)
  385. elif self._orientation == "w":
  386. return (-1, 0)
  387. return (0,0)
  388. def _RenderTopDown(self):
  389. cell_size = self._topdown["size"]
  390. size = self.resolution
  391. pos = self._cellpos
  392. lsize = (self.current_layer_width, self.current_layer_height)
  393. hcells = int(size[0] / cell_size)
  394. vcells = int(size[1] / cell_size)
  395. cx = pos[0] - int(hcells * 0.5)
  396. cy = pos[1] - int(vcells * 0.5)
  397. ry = -int(cell_size*0.5)
  398. for j in range(0, vcells+1):
  399. y = cy + j
  400. if y >= 0 and y < lsize[1]:
  401. rx = -int(cell_size*0.5)
  402. for i in range(0, hcells+1):
  403. x = cx + i
  404. if x >= 0 and x < lsize[0]:
  405. cell = self._getCell(x, y)
  406. if cell["n"][0] >= 0:
  407. self.draw_rect((rx, ry, cell_size, 2), self._topdown["wall_color"], 0, self._topdown["wall_color"])
  408. if cell["e"][0] >= 0:
  409. self.draw_rect((rx+(cell_size-2), ry, 2, cell_size), self._topdown["wall_color"], 0, self._topdown["wall_color"])
  410. if cell["s"][0] >= 0:
  411. self.draw_rect((rx, ry+(cell_size-2), cell_size, 2), self._topdown["wall_color"], 0, self._topdown["wall_color"])
  412. if cell["w"][0] >= 0:
  413. self.draw_rect((rx, ry, 2, cell_size), self._topdown["wall_color"], 0, self._topdown["wall_color"])
  414. if cell["n"][1] == True:
  415. self.draw_lines([(rx+1, ry+1), (rx+(cell_size-2), ry+1)], self._topdown["blocked_color"], 1)
  416. if cell["e"][1] == True:
  417. self.draw_lines([(rx+(cell_size-2), ry+1), (rx+(cell_size-2), ry+(cell_size-2))], self._topdown["blocked_color"], 1)
  418. if cell["s"][1] == True:
  419. self.draw_lines([(rx+1, ry+(cell_size-2)), (rx+(cell_size-2), ry+(cell_size-2))], self._topdown["blocked_color"], 1)
  420. if cell["w"][1] == True:
  421. self.draw_lines([(rx+1, ry+1), (rx+1, ry+(cell_size-2))], self._topdown["blocked_color"], 1)
  422. rx += cell_size
  423. ry += cell_size
  424. def _RenderPersFar(self, pos, size, o, orr, orl, wdat, wsurf):
  425. ovec = self._getOrientVec()
  426. fx = pos[0] + (ovec[0]*2)
  427. fy = pos[1] + (ovec[1]*2)
  428. fcell = self._getCell(fx, fy)
  429. if fcell == None:
  430. return # If we can't see the cell directly ahead, the other's won't be visible either!
  431. lcell = llcell = rcell = rrcell = None
  432. if ovec[0] == 0: # Facing North/South
  433. lcell = self._getCell(fx + ovec[1], fy)
  434. llcell = self._getCell(fx + (ovec[1]*2), fy)
  435. rcell = self._getCell(fx - ovec[1], fy)
  436. rrcell = self._getCell(fx - (ovec[1]*2), fy)
  437. else: # Facing East/West
  438. lcell = self._getCell(fx, fy - ovec[0])
  439. llcell = self._getCell(fx, fy - (ovec[0]*2))
  440. rcell = self._getCell(fx, fy + ovec[0])
  441. rrcell = self._getCell(fx, fy + (ovec[0]*2))
  442. hsw = int(size[0]*0.5)
  443. hsh = int(size[1]*0.5)
  444. # Rendering from edges to center
  445. if llcell is not None:
  446. if llcell[o][0] >= 0:
  447. rect = wdat["walls"][llcell[o][0]]["f_far"]
  448. hw = int(rect[2]*0.5)
  449. hh = int(rect[3]*0.5)
  450. self.draw_image(wsurf, (0, hsh-hh), (rect[0], rect[1], hw, rect[3]))
  451. if rrcell is not None:
  452. if rrcell[o][0] >= 0:
  453. rect = wdat["walls"][rrcell[o][0]]["f_far"]
  454. hw = int(rect[2]*0.5)
  455. hh = int(rect[3]*0.5)
  456. self.draw_image(wsurf, (size[0]-hw, hsh-hh), (rect[0]+hw, rect[1], hw, rect[3]))
  457. if lcell is not None:
  458. if lcell[o][0] >= 0:
  459. rect = wdat["walls"][lcell[o][0]]["f_far"]
  460. hw = int(rect[2]*0.5)
  461. hh = int(rect[3]*0.5)
  462. self.draw_image(wsurf, (hw, hsh-hh), (rect[0], rect[1], rect[2], rect[3]))
  463. if lcell[orl][0] >= 0:
  464. rect = wdat["walls"][lcell[orl][0]]["s_far"]
  465. lsurf = pygame.transform.flip(wsurf.subsurface(rect), True, False)
  466. hh = int(rect[3]*0.5)
  467. self.draw_image(lsurf, (0, hsh-hh))
  468. if rcell is not None:
  469. if rcell[o][0] >= 0:
  470. rect = wdat["walls"][rcell[o][0]]["f_far"]
  471. hw = int(rect[2]*0.5)
  472. hh = int(rect[3]*0.5)
  473. self.draw_image(wsurf, (size[0]-(rect[2]+hw), hsh-hh), (rect[0], rect[1], rect[2], rect[3]))
  474. if rcell[orr][0] >= 0:
  475. rect = wdat["walls"][rcell[orr][0]]["s_far"]
  476. hh = int(rect[3]*0.5)
  477. self.draw_image(wsurf, (size[0]-rect[2], hsh-hh), (rect[0], rect[1], rect[2], rect[3]))
  478. # Rendering the main cell!!
  479. frect = None # This will be used to place walls
  480. if fcell[o][0] >= 0:
  481. frect = wdat["walls"][fcell[o][0]]["f_far"]
  482. hw = int(frect[2]*0.5)
  483. hh = int(frect[3]*0.5)
  484. self.draw_image(wsurf, (hsw-hw, hsh-hh), (frect[0], frect[1], frect[2], frect[3]))
  485. if fcell[orl][0] >= 0:
  486. rect = wdat["walls"][fcell[orl][0]]["s_far"]
  487. if frect is None:
  488. # Kinda cheating in that it's known that all walls are the same size.
  489. frect = wdat["walls"][fcell[orl][0]]["f_far"]
  490. hw = int(frect[2]*0.5)
  491. lsurf = pygame.transform.flip(wsurf.subsurface(rect), True, False)
  492. self.draw_image(lsurf, (hsw-(hw+rect[2]), hsh-int(rect[3]*0.5)))
  493. if fcell[orr][0] >= 0:
  494. rect = wdat["walls"][fcell[orr][0]]["s_far"]
  495. if frect is None:
  496. frect = wdat["walls"][fcell[orr][0]]["f_far"]
  497. hw = int(frect[2]*0.5)
  498. self.draw_image(wsurf, (hsw+hw, hsh-int(rect[3]*0.5)), (rect[0], rect[1], rect[2], rect[3]))
  499. def _RenderPersMid(self, pos, size, o, orr, orl, wdat, wsurf):
  500. ovec = self._getOrientVec()
  501. fx = pos[0] + ovec[0]
  502. fy = pos[1] + ovec[1]
  503. fcell = self._getCell(fx, fy)
  504. if fcell == None:
  505. return # If we can't see the cell directly ahead, the other's won't be visible either!
  506. lcell = rcell = None
  507. if ovec[0] == 0: # Facing North/South
  508. lcell = self._getCell(fx + ovec[1], fy)
  509. rcell = self._getCell(fx - ovec[1], fy)
  510. else: # Facing East/West
  511. lcell = self._getCell(fx, fy - ovec[0])
  512. rcell = self._getCell(fx, fy + ovec[0])
  513. hsw = int(size[0]*0.5)
  514. hsh = int(size[1]*0.5)
  515. # Render from outside inwards!
  516. if lcell is not None:
  517. if lcell[o][0] >= 0:
  518. rect = wdat["walls"][lcell[o][0]]["f_mid"]
  519. hw = int(rect[2]*0.5)
  520. hh = int(rect[3]*0.5)
  521. self.draw_image(wsurf, (0, hsh-hh), (rect[0]+hw, rect[1], int(rect[2]*0.5), rect[3]))
  522. if rcell is not None:
  523. if rcell[o][0] >= 0:
  524. rect = wdat["walls"][rcell[o][0]]["f_mid"]
  525. hw = int(rect[2]*0.5)
  526. hh = int(rect[3]*0.5)
  527. self.draw_image(wsurf, (size[0]-hw, hsh-hh), (rect[0], rect[1], hw, rect[3]))
  528. # Rendering the main cell!!
  529. frect = None # This will be used to place walls
  530. if fcell[o][0] >= 0:
  531. frect = wdat["walls"][fcell[o][0]]["f_mid"]
  532. hw = int(frect[2]*0.5)
  533. hh = int(frect[3]*0.5)
  534. self.draw_image(wsurf, (hsw-hw, hsh-hh), (frect[0], frect[1], frect[2], frect[3]))
  535. if fcell[orl][0] >= 0:
  536. rect = wdat["walls"][fcell[orl][0]]["s_mid"]
  537. if frect is None:
  538. # Kinda cheating in that it's known that all walls are the same size.
  539. frect = wdat["walls"][fcell[orl][0]]["f_mid"]
  540. hw = int(frect[2]*0.5)
  541. lsurf = pygame.transform.flip(wsurf.subsurface(rect), True, False)
  542. self.draw_image(lsurf, (hsw-(hw+rect[2]), hsh-int(rect[3]*0.5)))
  543. if fcell[orr][0] >= 0:
  544. rect = wdat["walls"][fcell[orr][0]]["s_mid"]
  545. if frect is None:
  546. frect = wdat["walls"][fcell[orr][0]]["f_mid"]
  547. hw = int(frect[2]*0.5)
  548. self.draw_image(wsurf, (hsw+hw, hsh-int(rect[3]*0.5)), (rect[0], rect[1], rect[2], rect[3]))
  549. def _RenderPersClose(self, pos, size, o, orr, orl, cell, wdat, wsurf):
  550. fcell = self._getCell(pos[0], pos[1])
  551. hsw = int(size[0]*0.5)
  552. hsh = int(size[1]*0.5)
  553. ovec = self._getOrientVec()
  554. lcell = rcell = None
  555. if ovec[0] == 0: # Facing North/South
  556. lcell = self._getCell(pos[0] + ovec[1], pos[1])
  557. rcell = self._getCell(pos[0] - ovec[1], pos[1])
  558. else: # Facing East/West
  559. lcell = self._getCell(pos[0], pos[1] - ovec[0])
  560. rcell = self._getCell(pos[0], pos[1] + ovec[0])
  561. hsw = int(size[0]*0.5)
  562. hsh = int(size[1]*0.5)
  563. # Render from outside inwards!
  564. if lcell is not None:
  565. if lcell[o][0] >= 0:
  566. rect = wdat["walls"][lcell[o][0]]["f_close"]
  567. hw = int(rect[2]*0.5)
  568. hh = int(rect[3]*0.5)
  569. rw = hsw - hw
  570. self.draw_image(wsurf, (0, hsh-hh), (rect[0]+(rect[2]-rw), rect[1], rw, rect[3]))
  571. if rcell is not None:
  572. if rcell[o][0] >= 0:
  573. rect = wdat["walls"][rcell[o][0]]["f_close"]
  574. hw = int(rect[2]*0.5)
  575. hh = int(rect[3]*0.5)
  576. rw = hsw - hw
  577. self.draw_image(wsurf, (size[0]-rw, hsh-hh), (rect[0], rect[1], rw, rect[3]))
  578. # Rendering the main cell!!
  579. frect = None # This will be used to place walls
  580. if fcell[o][0] >= 0:
  581. idx = fcell[o][0]
  582. frect = wdat["walls"][idx]["f_close"]
  583. hw = int(frect[2]*0.5)
  584. hh = int(frect[3]*0.5)
  585. self.draw_image(wsurf, (hsw-hw, hsh-hh), (frect[0], frect[1], frect[2], frect[3]))
  586. if fcell[orl][0] >= 0:
  587. idx = fcell[orl][0]
  588. rect = wdat["walls"][idx]["s_close"]
  589. if frect is None:
  590. # Kinda cheating in that it's known that all walls are the same size.
  591. frect = wdat["walls"][idx]["f_close"]
  592. hw = int(frect[2]*0.5)
  593. lsurf = pygame.transform.flip(wsurf.subsurface(rect), True, False)
  594. self.draw_image(lsurf, (hsw-(hw+rect[2]), hsh-int(rect[3]*0.5)))
  595. if fcell[orr][0] >= 0:
  596. idx = fcell[orr][0]
  597. rect = wdat["walls"][idx]["s_close"]
  598. if frect is None:
  599. frect = wdat["walls"][idx]["f_close"]
  600. hw = int(frect[2]*0.5)
  601. self.draw_image(wsurf, (hsw+hw, hsh-int(rect[3]*0.5)), (rect[0], rect[1], rect[2], rect[3]))
  602. def _RenderPerspective(self):
  603. # Getting ALL of the core resources
  604. rm = self.resource
  605. edat = self._res["env"]
  606. wdat = self._res["walls"]
  607. if edat is None or wdat is None or edat() is None or wdat() is None:
  608. return
  609. edat = edat().data
  610. wdat = wdat().data
  611. ehsurf = rm.get("graphic", edat["horizon"]["src"])
  612. egsurf = rm.get("graphic", edat["ground"]["src"])
  613. wsurf = rm.get("graphic", wdat["src"])
  614. if ehsurf is None or egsurf is None or wsurf is None:
  615. return
  616. if ehsurf() is None or egsurf() is None or wsurf() is None:
  617. return
  618. oshift = lambda d: d if d >= 0 and d < 4 else (0 if d > 3 else 3)
  619. px = self._cellpos[0]
  620. py = self._cellpos[1]
  621. orl = self._d_n2s(oshift(self._d_s2n(self._orientation) - 1))
  622. orr = self._d_n2s(oshift(self._d_s2n(self._orientation) + 1))
  623. cell = self._getCell(px, py)
  624. # First, output the ground and horizon
  625. # TODO Later, perhaps cut the horizon and ground to represent each possible cell instead of just the current one?
  626. self.draw_image(ehsurf(), (0,0), edat["horizon"]["defs"][cell["h"]]["rect"])
  627. self.draw_image(egsurf(), (0,32), edat["ground"]["defs"][cell["g"]]["rect"])
  628. # Rendering the rest
  629. size = self.resolution
  630. self._RenderPersFar((px, py), size, self._orientation, orr, orl, wdat, wsurf())
  631. self._RenderPersMid((px, py), size, self._orientation, orr, orl, wdat, wsurf())
  632. self._RenderPersClose((px, py), size, self._orientation, orr, orl, cell, wdat, wsurf())
  633. def on_render(self):
  634. if self._renderMode == 0:
  635. self._RenderTopDown()
  636. else:
  637. self._RenderPerspective()
  638. class NodeMapEditor(gbe.nodes.Node2D):
  639. def __init__(self, name="MapEditor", parent=None):
  640. try:
  641. gbe.nodes.Node2D.__init__(self, name, parent)
  642. except gbe.nodes.Node2D as e:
  643. raise e
  644. self._last_orientation = ""
  645. self._size = 0
  646. self._thickness = 1
  647. self._color = pygame.Color(255,255,255)
  648. self._points = None
  649. self._filename = None
  650. self._fileiomode = 0 # 0 = Save | 1 = Load
  651. self.pointer_size = 4
  652. self._fnnode = gbe.nodes.NodeText("FileText", self)
  653. self._fnnode.size = 4
  654. self._fnnode.font_src = "IttyBitty.ttf"
  655. self._fnnode.antialias = False
  656. self._fnnode.set_color(255,255,255)
  657. self._fnnode.set_background(0,0,0,128)
  658. self._fnnode.visible = False
  659. def _getPoints(self, size):
  660. p = self.parent
  661. o = p.orientation
  662. points = self._points[o]
  663. hw = int(size[0]*0.5)
  664. hh = int(size[1]*0.5)
  665. return (
  666. (points[0][0] + hw, points[0][1] + hh),
  667. (points[1][0] + hw, points[1][1] + hh),
  668. (points[2][0] + hw, points[2][1] + hh)
  669. )
  670. @property
  671. def pointer_size(self):
  672. return self._size
  673. @pointer_size.setter
  674. def pointer_size(self, size):
  675. if not isinstance(size, int):
  676. raise TypeError("Size expected to be an integer.")
  677. if size <= 0:
  678. raise ValueError("Size must be greater than zero.")
  679. if size != self._size:
  680. self._size = size
  681. # Now updating all of the pointer... ummm... points
  682. hs = max(1, int(size * 0.5))
  683. self._points = {
  684. "n": ((0,-hs),(hs,hs),(-hs,hs)),
  685. "s": ((0,hs),(hs,-hs),(-hs,-hs)),
  686. "e": ((-hs,hs),(hs,0),(-hs,-hs)),
  687. "w": ((hs,hs),(-hs,0),(hs,-hs))
  688. }
  689. def set_color(self, color):
  690. if isinstance(color, (tuple, list)):
  691. clen = len(color)
  692. if clen == 3 or clen == 4:
  693. iscolor = lambda v : isinstance(v, int) and v >= 0 and v < 256
  694. if iscolor(color[0]) and iscolor(color[1]) and iscolor(color[2]):
  695. if clen == 3:
  696. self._color = pygame.Color(color[0], color[1], color[2])
  697. elif clen == 4 and iscolor(color[3]):
  698. self._color = pygame.Color(color[0], color[1], color[2], color[3])
  699. def get_color(self):
  700. return (self._color.r, self._color.g, self._color.b, self._color.a)
  701. def on_start(self):
  702. self.listen("KEYPRESSED", self.on_keypressed)
  703. def on_pause(self):
  704. self.unlisten("KEYPRESSED", self.on_keypressed)
  705. def on_keypressed(self, event, data):
  706. p = self.parent
  707. if p is None or not isinstance(p, NodeGameMap):
  708. return
  709. if data["key_name"] == "escape":
  710. if self._filename is None:
  711. self.emit("SCENECHANGE", {"scene":"MAIN_MENU", "hold":False})
  712. else:
  713. self._filename = None
  714. self._fnnode.visible = False
  715. elif data["key_name"] in ["enter", "return"] and self._filename is not None:
  716. if len(self._filename) > 0:
  717. if self._fileiomode == 0:
  718. p.save_map(self._filename)
  719. else:
  720. p.load_map(self._filename)
  721. # Call SAVE on the game map.
  722. self._filename = None
  723. self._fnnode.visible = False
  724. if self._filename is None:
  725. if data["key_name"] == "tab":
  726. p.toggle_render_mode()
  727. elif data["key_name"] == "w":
  728. p.move_forward(True)
  729. elif data["key_name"] == "s":
  730. p.move_backward(True)
  731. elif data["key_name"] == "a":
  732. p.turn_left()
  733. elif data["key_name"] == "d":
  734. p.turn_right()
  735. elif data["key_name"] == "space":
  736. o = p.orientation
  737. cpos = p.cell_position
  738. p.set_cell_face(cpos[0], cpos[1], o)
  739. elif data["key_name"] == "e":
  740. p.next_wall()
  741. elif data["key_name"] == "q":
  742. p.prev_wall()
  743. elif data["key_name"] in ["o", "l"]:
  744. self._fileiomode = 0
  745. if data["key_name"] == "l":
  746. self._fileiomode = 1
  747. self._filename = ""
  748. self._fnnode.text = ""
  749. self._fnnode.visible = True
  750. elif len(data["key_name"]) == 1:
  751. self._filename = "{}{}".format(self._filename, data["key_name"])
  752. self._fnnode.text = self._filename
  753. def on_render(self):
  754. size = self.resolution
  755. self.draw_lines(self._getPoints(size), self._color, self._thickness, True)