init.lua 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835
  1. if not minetest.global_exists("obsidian_gateway") then obsidian_gateway = {} end
  2. obsidian_gateway.modpath = minetest.get_modpath("obsidian_gateway")
  3. -- Localize for performance.
  4. local vector_distance = vector.distance
  5. local vector_round = vector.round
  6. local math_random = math.random
  7. -- Gateway schematic.
  8. local o = {"default:obsidian", "griefer:grieferstone", "cavestuff:dark_obsidian", "cavestuff:glow_obsidian"}
  9. local a = function(name)
  10. -- Bones check needed here to prevent players from blocking a gate by dying in
  11. -- it. E.g., bones never decay while a hacker is online.
  12. if name == "air" or name == "bones:bones" then
  13. return true
  14. end
  15. if name == "nether:portal_liquid" or name == "nether:portal_hidden" then
  16. return true
  17. end
  18. end
  19. local gate_northsouth = {
  20. data = {
  21. {p={x=0, y=0, z=0}, n=o},
  22. {p={x=1, y=0, z=0}, n=o},
  23. {p={x=2, y=0, z=0}, n=o},
  24. {p={x=3, y=0, z=0}, n=o},
  25. {p={x=0, y=1, z=0}, n=o},
  26. {p={x=0, y=2, z=0}, n=o},
  27. {p={x=0, y=3, z=0}, n=o},
  28. {p={x=0, y=4, z=0}, n=o},
  29. {p={x=1, y=4, z=0}, n=o},
  30. {p={x=2, y=4, z=0}, n=o},
  31. {p={x=3, y=4, z=0}, n=o},
  32. {p={x=3, y=1, z=0}, n=o},
  33. {p={x=3, y=2, z=0}, n=o},
  34. {p={x=3, y=3, z=0}, n=o},
  35. {p={x=1, y=1, z=0}, n=a},
  36. {p={x=2, y=1, z=0}, n=a},
  37. {p={x=1, y=2, z=0}, n=a},
  38. {p={x=2, y=2, z=0}, n=a},
  39. {p={x=1, y=3, z=0}, n=a},
  40. {p={x=2, y=3, z=0}, n=a},
  41. },
  42. minp = {x=0, y=0, z=0},
  43. maxp = {x=3, y=4, z=0},
  44. }
  45. local gate_eastwest = {
  46. data = {
  47. {p={x=0, y=0, z=0}, n=o},
  48. {p={x=0, y=0, z=1}, n=o},
  49. {p={x=0, y=0, z=2}, n=o},
  50. {p={x=0, y=0, z=3}, n=o},
  51. {p={x=0, y=1, z=0}, n=o},
  52. {p={x=0, y=2, z=0}, n=o},
  53. {p={x=0, y=3, z=0}, n=o},
  54. {p={x=0, y=4, z=0}, n=o},
  55. {p={x=0, y=4, z=1}, n=o},
  56. {p={x=0, y=4, z=2}, n=o},
  57. {p={x=0, y=4, z=3}, n=o},
  58. {p={x=0, y=1, z=3}, n=o},
  59. {p={x=0, y=2, z=3}, n=o},
  60. {p={x=0, y=3, z=3}, n=o},
  61. {p={x=0, y=1, z=1}, n=a},
  62. {p={x=0, y=1, z=2}, n=a},
  63. {p={x=0, y=2, z=1}, n=a},
  64. {p={x=0, y=2, z=2}, n=a},
  65. {p={x=0, y=3, z=1}, n=a},
  66. {p={x=0, y=3, z=2}, n=a},
  67. },
  68. minp = {x=0, y=0, z=0},
  69. maxp = {x=0, y=4, z=3},
  70. }
  71. obsidian_gateway.gate_ns_data = gate_northsouth
  72. obsidian_gateway.gate_ew_data = gate_eastwest
  73. -- Quickly check for protection in an area.
  74. local function check_protection(pos, radius)
  75. -- How much beyond the radius to check for protections.
  76. local e = 3
  77. local minp = vector.new(pos.x-(radius+e), pos.y-(radius+e), pos.z-(radius+e))
  78. local maxp = vector.new(pos.x+(radius+e), pos.y+(radius+e), pos.z+(radius+e))
  79. -- Step size, to avoid checking every single node.
  80. -- This assumes protections cannot be smaller than this size.
  81. local ss = 3
  82. local check = minetest.test_protection
  83. for x=minp.x, maxp.x, ss do
  84. for y=minp.y, maxp.y, ss do
  85. for z=minp.z, maxp.z, ss do
  86. if check({x=x, y=y, z=z}, "") then
  87. -- Protections are present.
  88. return true
  89. end
  90. end
  91. end
  92. end
  93. -- Nothing in the area is protected.
  94. return false
  95. end
  96. -- Get a list of node positions inside the gate's frame.
  97. function obsidian_gateway.door_positions(origin, northsouth)
  98. local airpoints
  99. if northsouth then
  100. local o = origin
  101. airpoints = {
  102. {x=o.x+1, y=o.y+1, z=o.z+0},
  103. {x=o.x+2, y=o.y+1, z=o.z+0},
  104. {x=o.x+1, y=o.y+2, z=o.z+0},
  105. {x=o.x+2, y=o.y+2, z=o.z+0},
  106. {x=o.x+1, y=o.y+3, z=o.z+0},
  107. {x=o.x+2, y=o.y+3, z=o.z+0},
  108. }
  109. else
  110. local o = origin
  111. airpoints = {
  112. {x=o.x+0, y=o.y+1, z=o.z+1},
  113. {x=o.x+0, y=o.y+1, z=o.z+2},
  114. {x=o.x+0, y=o.y+2, z=o.z+1},
  115. {x=o.x+0, y=o.y+2, z=o.z+2},
  116. {x=o.x+0, y=o.y+3, z=o.z+1},
  117. {x=o.x+0, y=o.y+3, z=o.z+2},
  118. }
  119. end
  120. return airpoints
  121. end
  122. function obsidian_gateway.spawn_liquid(origin, northsouth, returngate, force)
  123. local color = 6
  124. local rotation = 1
  125. local str_origin = minetest.pos_to_string(origin)
  126. if northsouth then
  127. rotation = 0
  128. end
  129. if returngate then
  130. color = 5
  131. end
  132. -- Node's drawtype is "colorfacedir".
  133. local node = {
  134. name = "nether:portal_liquid",
  135. param2 = (color * 32 + rotation),
  136. }
  137. local vadd = vector.add
  138. local airpoints = obsidian_gateway.door_positions(origin, northsouth)
  139. local spawned = false
  140. local count = #airpoints
  141. for k = 1, count, 1 do
  142. local p = airpoints[k]
  143. local oldnode = minetest.get_node(p)
  144. -- Make sure node to be replaced is air. If we were to overwrite any
  145. -- existing portal liquid, that would cause callbacks to run, which would
  146. -- interfere with what we're doing here.
  147. if oldnode.name == "air" then
  148. -- Run 'on_construct' callbacks, etc.
  149. minetest.set_node(p, node)
  150. -- This tells the particle code (inside node's on_timer) what color to use.
  151. local meta = minetest.get_meta(p)
  152. if returngate then
  153. meta:set_string("color", "red")
  154. else
  155. meta:set_string("color", "gold")
  156. end
  157. meta:set_string("gate_origin", str_origin)
  158. meta:set_string("gate_northsouth", tostring(northsouth))
  159. spawned = true
  160. elseif force and oldnode.name == "nether:portal_hidden" then
  161. minetest.swap_node(p, node)
  162. -- Manually execute callback.
  163. local ndef = minetest.registered_nodes[node.name]
  164. ndef.on_construct(p)
  165. -- Do not need to set "color" meta here, the hidden node already has it.
  166. -- Same with 'gate_origin'.
  167. spawned = true
  168. end
  169. end
  170. if spawned then
  171. ambiance.sound_play("nether_portal_ignite", origin, 1.0, 64)
  172. end
  173. end
  174. -- Determine whether the gateway has active portal liquid.
  175. function obsidian_gateway.have_liquid(origin, northsouth)
  176. local airpoints = obsidian_gateway.door_positions(origin, northsouth)
  177. local total = 0
  178. local count = #airpoints
  179. for k = 1, count, 1 do
  180. local p = airpoints[k]
  181. local node = minetest.get_node(p)
  182. if node.name == "nether:portal_liquid" then
  183. total = total + 1
  184. end
  185. end
  186. return (total == 6)
  187. end
  188. -- Get gate's origin and northsouth/eastwest orientation.
  189. function obsidian_gateway.get_origin_and_dir(pos)
  190. local meta = minetest.get_meta(pos)
  191. local str_origin = meta:get_string("gate_origin")
  192. local str_northsouth = meta:get_string("gate_northsouth")
  193. local origin = minetest.string_to_pos(str_origin)
  194. if origin then
  195. -- Returns origin, true/false.
  196. return origin, (str_northsouth == "true")
  197. end
  198. end
  199. function obsidian_gateway.remove_liquid(pos, points)
  200. local node = {name="air"}
  201. local removed = false
  202. local count = #points
  203. for k = 1, count, 1 do
  204. local v = points[k]
  205. local n = minetest.get_node(v)
  206. if n.name == "nether:portal_liquid" then
  207. -- Must use 'swap_node' to avoid triggering further callbacks on the
  208. -- portal liquid node (and nearby portal liquid nodes).
  209. minetest.swap_node(v, node)
  210. removed = true
  211. elseif n.name == "nether:portal_hidden" then
  212. minetest.swap_node(v, node)
  213. -- Node is hidden, so do not set 'removed' flag (removal makes no sound).
  214. end
  215. end
  216. if removed then
  217. ambiance.sound_play("nether_portal_extinguish", pos, 1.0, 64)
  218. end
  219. end
  220. function obsidian_gateway.regenerate_liquid(target, northsouth)
  221. local success, so, ap, ns, key, po =
  222. obsidian_gateway.find_gate(target, northsouth)
  223. -- Spawn portal liquid only if there is a gate here with the expected
  224. -- orientation. Force liquid placement over hidden portal nodes.
  225. if success then
  226. local meta = minetest.get_meta(so)
  227. local isreturngate = (meta:get_int("obsidian_gateway_return_gate_" .. key) == 1)
  228. obsidian_gateway.spawn_liquid(so, ns, isreturngate, true)
  229. end
  230. end
  231. function obsidian_gateway.find_gate(pos, require_ns)
  232. local result
  233. local points
  234. local counts
  235. local origin
  236. local northsouth
  237. local ns_key
  238. local playerorigin
  239. -- Find the gateway (threshold under player)!
  240. result, points, counts, origin =
  241. schematic_find.detect_schematic(pos, gate_northsouth)
  242. northsouth = true
  243. ns_key = "ns"
  244. if result then
  245. playerorigin = vector.add(origin, {x=1, y=1, z=0})
  246. end
  247. if not result then
  248. -- Couldn't find northsouth gateway, so try to find eastwest.
  249. result, points, counts, origin =
  250. schematic_find.detect_schematic(pos, gate_eastwest)
  251. northsouth = false
  252. ns_key = "ew"
  253. if result then
  254. playerorigin = vector.add(origin, {x=0, y=1, z=1})
  255. end
  256. end
  257. -- Early exit.
  258. if not result then
  259. return
  260. end
  261. -- If a specific orientation is required, then check that.
  262. if require_ns ~= nil then
  263. if northsouth ~= require_ns then
  264. return
  265. end
  266. end
  267. -- Store locations of air/portal nodes inside the gateway.
  268. local airpoints = {}
  269. if result then
  270. for k, v in ipairs(points) do
  271. local nn = minetest.get_node(v).name
  272. if nn == "air" or nn == "nether:portal_liquid" or
  273. nn == "nether:portal_hidden" then
  274. airpoints[#airpoints+1] = v
  275. end
  276. end
  277. end
  278. -- Did we find a working gateway?
  279. local yes = false
  280. if result then
  281. local o = counts["default:obsidian"] or 0
  282. local d = counts["cavestuff:dark_obsidian"] or 0
  283. local c = counts["cavestuff:glow_obsidian"] or 0
  284. local g = counts["griefer:grieferstone"] or 0
  285. local a = (#airpoints == 6)
  286. if (o + d + c) == 12 and g == 2 and a == true then
  287. yes = true
  288. end
  289. end
  290. if yes then
  291. return true, origin, airpoints, northsouth, ns_key, playerorigin
  292. end
  293. end
  294. function obsidian_gateway.attempt_activation(pos, player)
  295. local pname = player:get_player_name()
  296. local ppos = vector_round(player:get_pos())
  297. local under = utility.node_under_pos(player:get_pos())
  298. local inside = vector.add(under, {x=0, y=1, z=0})
  299. local nodeunder = minetest.get_node(under).name
  300. -- Player must be standing on one of these.
  301. if nodeunder ~= "default:obsidian" and
  302. nodeunder ~= "griefer:grieferstone" and
  303. nodeunder ~= "cavestuff:dark_obsidian" and
  304. nodeunder ~= "cavestuff:glow_obsidian" then
  305. -- This triggers when other types of portals are used, so is incorrect to display this chat.
  306. --minetest.chat_send_player(pname, "# Server: You need to be standing in the gateway for it to work!")
  307. return
  308. end
  309. local success
  310. local origin
  311. local northsouth
  312. local ns_key
  313. local playerorigin
  314. local airpoints
  315. success, origin, airpoints, northsouth, ns_key, playerorigin =
  316. obsidian_gateway.find_gate(pos)
  317. if not success then
  318. return
  319. end
  320. -- Add/update sound beacon.
  321. ambiance.spawn_sound_beacon("soundbeacon:gate", origin, 20, 1)
  322. ambiance.replay_nearby_sound_beacons(origin, 6)
  323. if sheriff.is_cheater(pname) then
  324. if sheriff.punish_probability(pname) then
  325. sheriff.punish_player(pname)
  326. return
  327. end
  328. end
  329. local target
  330. local meta = minetest.get_meta(origin)
  331. -- By spliting the key names by ns/ew, I ensure connected portals don't
  332. -- stomp on each other's data.
  333. target = minetest.string_to_pos(meta:get_string("obsidian_gateway_destination_" .. ns_key))
  334. --if not target then
  335. -- minetest.chat_send_player(pname, "# Server: Gateway has no destination! Aborting.")
  336. -- return
  337. --end
  338. -- Enable this if any serious problems occur.
  339. --if pname ~= gdac.name_of_admin then
  340. -- minetest.chat_send_player(pname, "# Server: Safety abort! Gateways are locked until further notice due to an error in the code.")
  341. -- return
  342. --end
  343. -- If activating the gate in the OUTBACK, and player previously died in
  344. -- MIDFELD, send them back to MIDFELD, do NOT send them to the overworld.
  345. if rc.current_realm_at_pos(origin) == "abyss" then
  346. if player:get_meta():get_int("abyss_return_midfeld") == 1 then
  347. target = obsidian_gateway.get_midfeld_spawn()
  348. end
  349. end
  350. -- Gates CANNOT be initialized in the Abyss!
  351. -- (Only the outgoing realm-gate is useable.)
  352. -- This prevents players from building their own gates in the Abyss.
  353. if not target and rc.current_realm_at_pos(origin) == "abyss" then
  354. minetest.after(0, function()
  355. -- Detonate some TNT!
  356. tnt.boom(vector.add(ppos, {x=math_random(-3, 3), y=0, z=math_random(-3, 3)}), {
  357. radius = 3,
  358. ignore_protection = false,
  359. ignore_on_blast = false,
  360. damage_radius = 5,
  361. disable_drops = true,
  362. })
  363. end)
  364. return
  365. end
  366. local isreturngate = (meta:get_int("obsidian_gateway_return_gate_" .. ns_key) == 1)
  367. local actual_owner = meta:get_string("obsidian_gateway_owner_" .. ns_key)
  368. local isowner = (actual_owner == pname)
  369. local first_time_init = false
  370. -- Event horizon color depends on whether we are a return gate.
  371. obsidian_gateway.spawn_liquid(origin, northsouth, isreturngate)
  372. minetest.log("action", pname .. " activated gateway @ " .. minetest.pos_to_string(pos))
  373. -- Initialize gateway for the first time.
  374. if not target or (meta:get_string("obsidian_gateway_success_" .. ns_key) ~= "yes" and not isreturngate) then
  375. -- Target is valid then this could be an OLD gate with old metadata.
  376. if target and not isreturngate and meta:get_string("obsidian_gateway_success_" .. ns_key) == "" then
  377. minetest.chat_send_player(pname, "# Server: It looks like this could possibly be an OLD gate! Aborting for safety reasons.")
  378. minetest.chat_send_player(pname, "# Server: If this Gateway was previously functioning normally, please mail the admin with the coordinates.")
  379. minetest.chat_send_player(pname, "# Server: If this is a Gate that you have just constructed, you can safely ignore this message.")
  380. minetest.chat_send_player(pname, "# Server: The Gateway's EXIT location is @ " .. rc.pos_to_namestr(target) .. ".")
  381. minetest.after(1.5, function() easyvend.sound_error(pname) end)
  382. return
  383. end
  384. -- Algorithm for locating the destination.
  385. -- Get a potential gate location.
  386. target = rc.get_random_realm_gate_position(pname, origin)
  387. -- Is target outside bounds?
  388. local bad = function(target, origin)
  389. -- Handle nil.
  390. if not target then
  391. return true
  392. end
  393. -- Don't allow exit points near the colonies.
  394. if vector_distance(target, {x=0, y=0, z=0}) < 1000 or
  395. vector_distance(target, {x=0, y=-30790, z=0}) < 1000 then
  396. return true
  397. end
  398. -- Exit must not be too close to start.
  399. if vector_distance(target, origin) < 100 then
  400. return true
  401. end
  402. -- Or too far.
  403. -- This causes too many failures.
  404. -- Note: this is now handled by the 'rc' mod.
  405. --if vector_distance(target, origin) > 7000 then
  406. -- return true
  407. --end
  408. if not rc.is_valid_gateway_region(target) then
  409. return true
  410. end
  411. end
  412. -- Keep trying until the target is within bounds.
  413. local num_tries = 0
  414. while bad(target, origin) do
  415. target = rc.get_random_realm_gate_position(pname, origin)
  416. num_tries = num_tries + 1
  417. -- Max 3 tries.
  418. if num_tries >= 2 then
  419. ---[[
  420. minetest.after(0, function()
  421. -- Detonate some TNT!
  422. tnt.boom(vector.add(ppos, {x=math_random(-3, 3), y=0, z=math_random(-3, 3)}), {
  423. radius = 3,
  424. ignore_protection = false,
  425. ignore_on_blast = false,
  426. damage_radius = 5,
  427. disable_drops = true,
  428. })
  429. end)
  430. --]]
  431. return
  432. end
  433. end
  434. meta:set_string("obsidian_gateway_destination_" .. ns_key, minetest.pos_to_string(target))
  435. meta:set_string("obsidian_gateway_owner_" .. ns_key, pname)
  436. first_time_init = true
  437. isowner = true
  438. end
  439. if gdac.player_is_admin(pname) then
  440. isowner = true
  441. end
  442. -- Let everyone use gates owned by the admin.
  443. if actual_owner == gdac.name_of_admin then
  444. isowner = true
  445. end
  446. -- Slightly randomize player's exit coordinates.
  447. -- Without changing the coordinates of the gateway.
  448. local pdest
  449. if northsouth then
  450. pdest = vector.add(target, {x=math_random(0, 1), y=0, z=0})
  451. else
  452. pdest = vector.add(target, {x=0, y=0, z=math_random(0, 1)})
  453. end
  454. pdest = vector_round(pdest)
  455. -- Make sure target is within some realm.
  456. if not rc.is_valid_realm_pos(pdest) then
  457. return
  458. end
  459. -- Collect any friends to bring along.
  460. local friendstobring = {}
  461. local allplayers = minetest.get_connected_players()
  462. for k, v in ipairs(allplayers) do
  463. if v:get_player_name() ~= pname then
  464. if vector_distance(v:get_pos(), player:get_pos()) < 3 then
  465. friendstobring[#friendstobring+1] = v:get_player_name()
  466. end
  467. end
  468. end
  469. portal_cb.call_before_use({
  470. gate_origin = origin,
  471. gate_orientation = ns_key, -- "ns" or "ew"
  472. player_name = pname,
  473. teleport_destination = table.copy(pdest),
  474. })
  475. -- Create a gateway at the player's destination.
  476. -- This gateway links back to the first.
  477. -- If it is destroyed, the player is stuck!
  478. preload_tp.execute({
  479. player_name = pname,
  480. target_position = pdest,
  481. emerge_radius = 32,
  482. particle_effects = true,
  483. pre_teleport_callback = function()
  484. -- Cancel teleport if origin gate does not have portal liquid.
  485. if not obsidian_gateway.have_liquid(origin, northsouth) then
  486. minetest.chat_send_player(pname, "# Server: Portal disrupted.")
  487. -- Cancel transport.
  488. return true
  489. end
  490. -- Don't build return portal on top of someone's protected stuff.
  491. if first_time_init then
  492. if check_protection(vector.add(target, {x=0, y=3, z=0}), 5) then
  493. minetest.chat_send_player(pname, "# Server: Return-gate construction FAILED due to protection near " .. rc.pos_to_namestr(target) .. ".")
  494. -- Clear data for the initial gate. This will permit the player to retry without tearing everything down and building it again.
  495. local meta = minetest.get_meta(origin)
  496. meta:set_string("obsidian_gateway_success_" .. ns_key, "")
  497. meta:set_string("obsidian_gateway_destination_" .. ns_key, "")
  498. meta:set_string("obsidian_gateway_owner_" .. ns_key, "")
  499. -- Cancel transport.
  500. return true
  501. end
  502. end
  503. -- Build return portal (only if not already using a return portal).
  504. -- Also, only build return portal on first use of the initial portal.
  505. if not isreturngate and first_time_init then
  506. if northsouth then
  507. -- Place northsouth gateway.
  508. local path = obsidian_gateway.modpath .. "/obsidian_gateway_northsouth.mts"
  509. local gpos = vector.add(target, {x=-1, y=-1, z=0})
  510. minetest.place_schematic(gpos, path, "0", nil, true)
  511. local meta = minetest.get_meta(gpos)
  512. meta:set_string("obsidian_gateway_destination_" .. ns_key, minetest.pos_to_string(playerorigin))
  513. meta:set_string("obsidian_gateway_owner_" .. ns_key, pname)
  514. meta:set_int("obsidian_gateway_return_gate_" .. ns_key, 1)
  515. else
  516. -- Place eastwest gateway.
  517. local path = obsidian_gateway.modpath .. "/obsidian_gateway_eastwest.mts"
  518. local gpos = vector.add(target, {x=0, y=-1, z=-1})
  519. minetest.place_schematic(gpos, path, "0", nil, true)
  520. local meta = minetest.get_meta(gpos)
  521. meta:set_string("obsidian_gateway_destination_" .. ns_key, minetest.pos_to_string(playerorigin))
  522. meta:set_string("obsidian_gateway_owner_" .. ns_key, pname)
  523. meta:set_int("obsidian_gateway_return_gate_" .. ns_key, 1)
  524. end
  525. end
  526. -- Mark the initial gate as success.
  527. -- If this is not done, then gate will assume it is not initialized
  528. -- the next time it is used. This fixes a bug where the return gate is
  529. -- not properly constructed if the player moves during transport
  530. -- (because this callback function doesn't get called).
  531. if not isreturngate and first_time_init then
  532. local meta = minetest.get_meta(origin)
  533. meta:set_string("obsidian_gateway_success_" .. ns_key, "yes")
  534. meta:mark_as_private("obsidian_gateway_success_" .. ns_key)
  535. end
  536. -- If the destination is the Abyss, then kill player first.
  537. -- This helps to prevent player from bringing any foreign items into this realm.
  538. -- Note: this relies on the teleport code already checking all other preconditions
  539. -- first. I.e., if this callback returns 'false', then the player absolutely
  540. -- will be teleported.
  541. if rc.current_realm_at_pos(pdest) == "abyss" then
  542. -- Dump player bones, as if they died.
  543. -- This should behave exactly as if the player died, with the exception of
  544. -- setting the player's health to 0.
  545. bones.dump_bones(pname, true)
  546. local pref = minetest.get_player_by_name(pname)
  547. pref:set_hp(pova.get_active_modifier(pref, "properties").hp_max)
  548. pref:get_meta():set_string("last_death_pos", "") -- Fake death.
  549. give_initial_stuff.give(pref)
  550. end
  551. -- Always regenerate portal liquid in the destination portal.
  552. -- (It will often be missing since no one was near it.)
  553. -- This function will check if there actually is a gate, here.
  554. obsidian_gateway.regenerate_liquid(target, northsouth)
  555. -- If the player is someone other than the owner, using this Gate has consequences.
  556. if not isowner then
  557. -- This function is already called normally, when a Gate is used.
  558. -- Calling it again here, effectively doubles the chance that the user
  559. -- starts feeling rather ill.
  560. portal_sickness.on_use_portal(pname)
  561. end
  562. end,
  563. post_teleport_callback = function()
  564. portal_cb.call_after_use({
  565. gate_origin = origin,
  566. gate_orientation = ns_key, -- "ns" or "ew"
  567. player_name = pname,
  568. teleport_destination = table.copy(pdest),
  569. })
  570. -- Any others in area get brought along, too.
  571. for k, v in ipairs(friendstobring) do
  572. local friend = minetest.get_player_by_name(v)
  573. if friend then
  574. local fname = friend:get_player_name()
  575. preload_tp.execute({
  576. player_name = fname,
  577. target_position = pdest,
  578. particle_effects = true,
  579. pre_teleport_callback = function()
  580. -- If the destination is the Abyss, then kill player first.
  581. -- Note: this relies on the teleport code already checking all other preconditions
  582. -- first. I.e., if this callback returns 'false', then the player absolutely
  583. -- will be teleported.
  584. if rc.current_realm_at_pos(pdest) == "abyss" then
  585. -- Dump player bones, as if they died.
  586. -- This should behave exactly as if the player died, with the exception of
  587. -- setting the player's health to 0.
  588. bones.dump_bones(fname, true)
  589. local pref = minetest.get_player_by_name(fname)
  590. pref:set_hp(pova.get_active_modifier(pref, "properties").hp_max)
  591. pref:get_meta():set_string("last_death_pos", "") -- Fake death.
  592. give_initial_stuff.give(pref)
  593. end
  594. end,
  595. force_teleport = true,
  596. send_blocks = true,
  597. })
  598. portal_sickness.on_use_portal(fname)
  599. end
  600. end
  601. -- Update liquids around on first init.
  602. if first_time_init then
  603. minetest.after(2, function()
  604. mapfix.execute(target, 10)
  605. end)
  606. end
  607. ambiance.spawn_sound_beacon("soundbeacon:gate", target, 20, 1)
  608. ambiance.replay_nearby_sound_beacons(target, 6)
  609. portal_sickness.on_use_portal(pname)
  610. -- Clear player's "died in MIDFELD" flag, once transport to MIDFELD succeeded.
  611. if rc.current_realm_at_pos(target) == "midfeld" then
  612. local pref = minetest.get_player_by_name(pname)
  613. if pref then
  614. pref:get_meta():set_int("abyss_return_midfeld", 0)
  615. end
  616. end
  617. end,
  618. teleport_sound = "nether_portal_usual",
  619. send_blocks = true,
  620. })
  621. end
  622. -- To be called inside node's 'on_destruct' callback.
  623. -- Note: 'transient' is ONLY true when node to be destructed is portal liquid!
  624. function obsidian_gateway.on_damage_gate(pos, transient)
  625. -- First, perform some cheap checks to see if there could possibly be a gate
  626. -- at this location. We only perform the expensive checks if the cheap checks
  627. -- pass!
  628. local minp = vector.add(pos, {x=-4, y=-4, z=-4})
  629. local maxp = vector.add(pos, {x=4, y=4, z=4})
  630. local names = {
  631. "default:obsidian",
  632. "cavestuff:dark_obsidian",
  633. "cavestuff:glow_obsidian",
  634. "griefer:grieferstone",
  635. "nether:portal_liquid",
  636. "nether:portal_hidden",
  637. }
  638. local points, counts = minetest.find_nodes_in_area(minp, maxp, names)
  639. if #points == 0 then
  640. return
  641. end
  642. local doorpoints = points
  643. -- Remove all portal-liquid nodes. (Will play sound if any removed.)
  644. -- First, try to get gate origin from meta. If this fails, then we use the
  645. -- 'points' array as a fallback (old behavior).
  646. local origin, northsouth = obsidian_gateway.get_origin_and_dir(pos)
  647. if origin then
  648. doorpoints = obsidian_gateway.door_positions(origin, northsouth)
  649. end
  650. minetest.after(0, obsidian_gateway.remove_liquid, pos, doorpoints)
  651. -- If transient "pop" of portal liquid nodes, then do not continue further to
  652. -- actually damage the gate.
  653. if transient then
  654. return
  655. end
  656. -- A valid gate requires exactly 2 oerkki stone.
  657. -- (There may be additional oerkki stone not part of the gate.)
  658. if counts["griefer:grieferstone"] < 2 then
  659. return
  660. end
  661. do
  662. local o = counts["default:obsidian"] or 0
  663. local d = counts["cavestuff:dark_obsidian"] or 0
  664. local c = counts["cavestuff:glow_obsidian"] or 0
  665. -- Including the node that will be removed (we should have been called
  666. -- inside of 'on_destruct' for a given node), there should be 12 obsidian
  667. -- remaining, otherwise cannot be a valid gate. (There may be additional
  668. -- obsidian nearby not part of the gate.)
  669. if (o + d + c) < 12 then
  670. return
  671. end
  672. end
  673. -- Cheap checks completed, now do the expensive check.
  674. if not obsidian_gateway.find_gate(pos) then
  675. return
  676. end
  677. -- If we reach here, we know we have a valid gate attached to this position.
  678. -- We don't care if it is a NS or EW-facing gate.
  679. -- This has a chance to destroy one of the oerkki stones, which costs some
  680. -- resources to craft again. But don't spawn lava in overworld. The point of
  681. -- this is to make it a bit more costly to constantly reset the gate if you
  682. -- don't like where it goes. Note: using 'swap_node' first in order to prevent
  683. -- calling additional callbacks.
  684. local idx = math.random(1, #points)
  685. local tar = points[idx]
  686. minetest.swap_node(tar, {name="air"})
  687. minetest.set_node(tar, {name="fire:basic_flame"})
  688. ambiance.sound_play("nether_rack_destroy", pos, 1.0, 64)
  689. end
  690. dofile(obsidian_gateway.modpath .. "/flame_staff.lua")
  691. if not obsidian_gateway.run_once then
  692. local c = "obsidian_gateway:core"
  693. local f = obsidian_gateway.modpath .. "/init.lua"
  694. reload.register_file(c, f, false)
  695. obsidian_gateway.run_once = true
  696. end