forceload.lua 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. --[[
  2. Tube Library
  3. ============
  4. Copyright (C) 2017-2020 Joachim Stolberg
  5. AGPL v3
  6. See LICENSE.txt for more information
  7. forceload.lua:
  8. ]]--
  9. -- Load support for I18n
  10. local S = tubelib.S
  11. -- for lazy programmers
  12. local P2S = function(pos) if pos then return minetest.pos_to_string(pos) end end
  13. local P = minetest.string_to_pos
  14. local M = minetest.get_meta
  15. local function calc_area(pos)
  16. local xpos = (math.floor(pos.x / 16) * 16)
  17. local ypos = (math.floor(pos.y / 16) * 16)
  18. local zpos = (math.floor(pos.z / 16) * 16)
  19. local pos1 = {x=xpos, y=ypos, z=zpos}
  20. local pos2 = {x=xpos+15, y=ypos+15, z=zpos+15}
  21. return pos1, pos2
  22. end
  23. local function in_list(list, x)
  24. local pos1 = calc_area(x)
  25. for _,v in ipairs(list) do
  26. local pos2 = calc_area(v)
  27. if vector.equals(pos1, pos2) then return true end
  28. end
  29. return false
  30. end
  31. local function remove_list_elem(list, x)
  32. local n = nil
  33. for idx, v in ipairs(list) do
  34. if vector.equals(v, x) then
  35. n = idx
  36. break
  37. end
  38. end
  39. if n then
  40. table.remove(list, n)
  41. end
  42. return list
  43. end
  44. local function chat(player, text)
  45. minetest.chat_send_player(player:get_player_name(), "[Tubelib] "..text)
  46. end
  47. local function get_node_lvm(pos)
  48. local node = minetest.get_node_or_nil(pos)
  49. if node then
  50. return node
  51. end
  52. local vm = minetest.get_voxel_manip()
  53. local MinEdge, MaxEdge = vm:read_from_map(pos, pos)
  54. local data = vm:get_data()
  55. local param2_data = vm:get_param2_data()
  56. local area = VoxelArea:new({MinEdge = MinEdge, MaxEdge = MaxEdge})
  57. local idx = area:index(pos.x, pos.y, pos.z)
  58. node = {
  59. name = minetest.get_name_from_content_id(data[idx]),
  60. param2 = param2_data[idx]
  61. }
  62. return node
  63. end
  64. local function add_pos(pos, player)
  65. local lPos = minetest.deserialize(player:get_attribute("tubelib_forceload_blocks")) or {}
  66. if not in_list(lPos, pos) and #lPos < tubelib.max_num_forceload_blocks then
  67. lPos[#lPos+1] = pos
  68. player:set_attribute("tubelib_forceload_blocks", minetest.serialize(lPos))
  69. return true
  70. end
  71. return false
  72. end
  73. local function del_pos(pos, player)
  74. local lPos = minetest.deserialize(player:get_attribute("tubelib_forceload_blocks")) or {}
  75. lPos = remove_list_elem(lPos, pos)
  76. player:set_attribute("tubelib_forceload_blocks", minetest.serialize(lPos))
  77. end
  78. local function get_pos_list(player)
  79. return minetest.deserialize(player:get_attribute("tubelib_forceload_blocks")) or {}
  80. end
  81. local function set_pos_list(player, lPos)
  82. player:set_attribute("tubelib_forceload_blocks", minetest.serialize(lPos))
  83. end
  84. local function get_data(pos, player)
  85. local pos1, pos2 = calc_area(pos)
  86. local num = #minetest.deserialize(player:get_attribute("tubelib_forceload_blocks")) or 0
  87. local max = tubelib.max_num_forceload_blocks
  88. return pos1, pos2, num, max
  89. end
  90. local function formspec(player)
  91. local lPos = get_pos_list(player)
  92. local tRes = {}
  93. tRes[1] = "size[7,9]"..
  94. default.gui_bg..
  95. default.gui_bg_img..
  96. default.gui_slots..
  97. "label[0,0;"..S("List of your Forceload Blocks")..":]"
  98. for idx,pos in ipairs(lPos) do
  99. local pos1, pos2 = calc_area(pos)
  100. local ypos = 0.2 + idx * 0.4
  101. tRes[#tRes+1] = "label[0,"..ypos..";"..idx.."]"
  102. tRes[#tRes+1] = "label[0.8,"..ypos..";"..P2S(pos1).."]"
  103. tRes[#tRes+1] = "label[3.2,"..ypos..";"..S("to").."]"
  104. tRes[#tRes+1] = "label[4,"..ypos..";"..P2S(pos2).."]"
  105. end
  106. return table.concat(tRes)
  107. end
  108. minetest.register_node("tubelib:forceload", {
  109. description = S("Tubelib Forceload Block"),
  110. tiles = {
  111. -- up, down, right, left, back, front
  112. 'tubelib_front.png',
  113. 'tubelib_front.png',
  114. {
  115. image = "tubelib_forceload.png",
  116. backface_culling = false,
  117. animation = {
  118. type = "vertical_frames",
  119. aspect_w = 32,
  120. aspect_h = 32,
  121. length = 0.5,
  122. },
  123. },
  124. },
  125. after_place_node = function(pos, placer, itemstack)
  126. if add_pos(pos, placer) then
  127. minetest.forceload_block(pos, true)
  128. local pos1, pos2, num, max = get_data(pos, placer)
  129. M(pos):set_string("infotext", S("Area").." "..P2S(pos1).." "..S("to").." "..P2S(pos2).." "..S("loaded!").."\n"..
  130. S("Punch the block to make the area visible."))
  131. chat(placer, S("Area").." ("..num.."/"..max..") "..P2S(pos1).." "..S("to").." "..P2S(pos2).." "..S("loaded!"))
  132. tubelib.mark_region(placer:get_player_name(), pos1, pos2)
  133. M(pos):set_string("owner", placer:get_player_name())
  134. else
  135. chat(placer, S("Area already loaded or max. number of Forceload Blocks reached!"))
  136. minetest.remove_node(pos)
  137. return itemstack
  138. end
  139. end,
  140. after_dig_node = function(pos, oldnode, oldmetadata, digger)
  141. local player = minetest.get_player_by_name(oldmetadata.fields.owner)
  142. if player then
  143. del_pos(pos, player)
  144. end
  145. minetest.forceload_free_block(pos, true)
  146. tubelib.unmark_region(oldmetadata.fields.owner)
  147. end,
  148. on_rightclick = function(pos, node, clicker, itemstack, pointed_thing)
  149. if M(pos):get_string("owner") == clicker:get_player_name() or
  150. minetest.check_player_privs(clicker:get_player_name(), "server") then
  151. local s = formspec(clicker)
  152. minetest.show_formspec(clicker:get_player_name(), "tubelib:forceload", s)
  153. end
  154. end,
  155. on_punch = function(pos, node, puncher, pointed_thing)
  156. local pos1, pos2 = calc_area(pos)
  157. tubelib.switch_region(puncher:get_player_name(), pos1, pos2)
  158. end,
  159. paramtype = "light",
  160. sunlight_propagates = true,
  161. groups = {choppy=2, cracky=2, crumbly=2,
  162. not_in_creative_inventory = tubelib.max_num_forceload_blocks == 0 and 1 or 0},
  163. is_ground_content = false,
  164. sounds = default.node_sound_wood_defaults(),
  165. })
  166. if tubelib.max_num_forceload_blocks > 0 then
  167. minetest.register_craft({
  168. output = "tubelib:forceload",
  169. recipe = {
  170. {"group:wood", "", "group:wood"},
  171. {"", "basic_materials:energy_crystal_simple", ""},
  172. {"group:wood", "tubelib:wlanchip", "group:wood"},
  173. },
  174. })
  175. end
  176. minetest.register_on_joinplayer(function(player)
  177. local lPos = {}
  178. for _,pos in ipairs(get_pos_list(player)) do
  179. local node = get_node_lvm(pos)
  180. if node.name == "tubelib:forceload" then
  181. minetest.forceload_block(pos, true)
  182. lPos[#lPos+1] = pos
  183. end
  184. end
  185. set_pos_list(player, lPos)
  186. end)
  187. minetest.register_on_leaveplayer(function(player)
  188. for _,pos in ipairs(get_pos_list(player)) do
  189. minetest.forceload_free_block(pos, true)
  190. end
  191. end)