mirror of
https://github.com/minetest-mods/technic.git
synced 2025-01-26 18:00:32 +01:00
Replace mining laser ray tracer with a simpler iterable one
This commit is contained in:
parent
4d1f9753e3
commit
42d0081367
@ -32,26 +32,70 @@ minetest.register_craft({
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
local function table_icontains(t, v)
|
-- Based on code by Uberi: https://gist.github.com/Uberi/3125280
|
||||||
for i = 1,#t do
|
local function rayIter(pos, dir, range)
|
||||||
if v == t[i] then
|
local p = vector.round(pos)
|
||||||
return true
|
local x_step, y_step, z_step = 0, 0, 0
|
||||||
|
local x_component, y_component, z_component = 0, 0, 0
|
||||||
|
local x_intersect, y_intersect, z_intersect = 0, 0, 0
|
||||||
|
|
||||||
|
if dir.x == 0 then
|
||||||
|
x_intersect = math.huge
|
||||||
|
elseif dir.x > 0 then
|
||||||
|
x_step = 1
|
||||||
|
x_component = 1 / dir.x
|
||||||
|
x_intersect = x_component
|
||||||
|
else
|
||||||
|
x_step = -1
|
||||||
|
x_component = 1 / -dir.x
|
||||||
end
|
end
|
||||||
|
if dir.y == 0 then
|
||||||
|
y_intersect = math.huge
|
||||||
|
elseif dir.y > 0 then
|
||||||
|
y_step = 1
|
||||||
|
y_component = 1 / dir.y
|
||||||
|
y_intersect = y_component
|
||||||
|
else
|
||||||
|
y_step = -1
|
||||||
|
y_component = 1 / -dir.y
|
||||||
end
|
end
|
||||||
return false
|
if dir.z == 0 then
|
||||||
|
z_intersect = math.huge
|
||||||
|
elseif dir.z > 0 then
|
||||||
|
z_step = 1
|
||||||
|
z_component = 1 / dir.z
|
||||||
|
z_intersect = z_component
|
||||||
|
else
|
||||||
|
z_step = -1
|
||||||
|
z_component = 1 / -dir.z
|
||||||
end
|
end
|
||||||
|
|
||||||
local function laser_node(pos, player)
|
return function()
|
||||||
local node = minetest.get_node(pos)
|
if x_intersect < y_intersect then
|
||||||
if table_icontains({"air", "ignore", "default:lava_source", "default:lava_flowing"}, node.name) then
|
if x_intersect < z_intersect then
|
||||||
return
|
p.x = p.x + x_step
|
||||||
|
x_intersect = x_intersect + x_component
|
||||||
|
else
|
||||||
|
p.z = p.z + z_step
|
||||||
|
z_intersect = z_intersect + z_component
|
||||||
end
|
end
|
||||||
local pname = player:get_player_name()
|
elseif y_intersect < z_intersect then
|
||||||
if minetest.is_protected(pos, pname) then
|
p.y = p.y + y_step
|
||||||
minetest.record_protection_violation(pos, pname)
|
y_intersect = y_intersect + y_component
|
||||||
return
|
else
|
||||||
|
p.z = p.z + z_step
|
||||||
|
z_intersect = z_intersect + z_component
|
||||||
end
|
end
|
||||||
if table_icontains({"default:water_flowing", "default:water_source"}, node.name) then
|
if vector.distance(pos, p) > range then
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
return p
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function laser_node(pos, node, player)
|
||||||
|
local def = minetest.registered_nodes[node.name]
|
||||||
|
if def and def.liquidtype ~= "none" then
|
||||||
minetest.remove_node(pos)
|
minetest.remove_node(pos)
|
||||||
minetest.add_particle({
|
minetest.add_particle({
|
||||||
pos = pos,
|
pos = pos,
|
||||||
@ -63,33 +107,44 @@ local function laser_node(pos, player)
|
|||||||
})
|
})
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
if player then
|
|
||||||
minetest.node_dig(pos, node, player)
|
minetest.node_dig(pos, node, player)
|
||||||
end
|
end
|
||||||
end
|
|
||||||
|
|
||||||
if not vector.line then
|
|
||||||
dofile(technic.modpath.."/tools/vector_line.lua")
|
|
||||||
end
|
|
||||||
|
|
||||||
|
local no_destroy = {
|
||||||
|
["air"] = true,
|
||||||
|
["default:lava_source"] = true,
|
||||||
|
["default:lava_flowing"] = true,
|
||||||
|
}
|
||||||
local function laser_shoot(player, range, particle_texture, sound)
|
local function laser_shoot(player, range, particle_texture, sound)
|
||||||
local playerpos = player:getpos()
|
local player_pos = player:getpos()
|
||||||
|
local player_name = player:get_player_name()
|
||||||
local dir = player:get_look_dir()
|
local dir = player:get_look_dir()
|
||||||
|
|
||||||
local startpos = {x = playerpos.x, y = playerpos.y + 1.625, z = playerpos.z}
|
local start_pos = vector.new(player_pos)
|
||||||
local mult_dir = vector.multiply(dir, 50)
|
-- Adjust to head height
|
||||||
|
start_pos.y = start_pos.y + 1.9
|
||||||
minetest.add_particle({
|
minetest.add_particle({
|
||||||
pos = startpos,
|
pos = startpos,
|
||||||
vel = dir,
|
vel = dir,
|
||||||
acc = mult_dir,
|
acc = vector.multiply(dir, 50),
|
||||||
expirationtime = range / 11,
|
expirationtime = range / 11,
|
||||||
size = 1,
|
size = 1,
|
||||||
texture = particle_texture .. "^[transform" .. math.random(0, 7),
|
texture = particle_texture .. "^[transform" .. math.random(0, 7),
|
||||||
})
|
})
|
||||||
for _,pos in ipairs(vector.line(vector.round(startpos), dir, range)) do
|
minetest.sound_play(sound, {pos = player_pos, max_hear_distance = range})
|
||||||
laser_node(pos, player)
|
for pos in rayIter(start_pos, dir, range) do
|
||||||
|
if minetest.is_protected(pos, player_name) then
|
||||||
|
minetest.record_protection_violation(pos, player_name)
|
||||||
|
break
|
||||||
|
end
|
||||||
|
local node = minetest.get_node_or_nil(pos)
|
||||||
|
if not node then
|
||||||
|
break
|
||||||
|
end
|
||||||
|
if not no_destroy[node.name] then
|
||||||
|
laser_node(pos, node, player)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
minetest.sound_play(sound, {pos = playerpos, max_hear_distance = range})
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,83 +0,0 @@
|
|||||||
local twolines = {}
|
|
||||||
function vector.twoline(x, y)
|
|
||||||
local pstr = x.." "..y
|
|
||||||
local line = twolines[pstr]
|
|
||||||
if line then
|
|
||||||
return line
|
|
||||||
end
|
|
||||||
line = {}
|
|
||||||
local n = 1
|
|
||||||
local dirx = 1
|
|
||||||
if x < 0 then
|
|
||||||
dirx = -dirx
|
|
||||||
end
|
|
||||||
local ymin, ymax = 0, y
|
|
||||||
if y < 0 then
|
|
||||||
ymin, ymax = ymax, ymin
|
|
||||||
end
|
|
||||||
local m = y/x --y/0 works too
|
|
||||||
local dir = 1
|
|
||||||
if m < 0 then
|
|
||||||
dir = -dir
|
|
||||||
end
|
|
||||||
for i = 0,x,dirx do
|
|
||||||
local p1 = math.max(math.min(math.floor((i-0.5)*m+0.5), ymax), ymin)
|
|
||||||
local p2 = math.max(math.min(math.floor((i+0.5)*m+0.5), ymax), ymin)
|
|
||||||
for j = p1,p2,dir do
|
|
||||||
line[n] = {i, j}
|
|
||||||
n = n+1
|
|
||||||
end
|
|
||||||
end
|
|
||||||
twolines[pstr] = line
|
|
||||||
return line
|
|
||||||
end
|
|
||||||
|
|
||||||
local threelines = {}
|
|
||||||
function vector.threeline(x, y, z)
|
|
||||||
local pstr = x.." "..y.." "..z
|
|
||||||
local line = threelines[pstr]
|
|
||||||
if line then
|
|
||||||
return line
|
|
||||||
end
|
|
||||||
if x ~= math.floor(x) then
|
|
||||||
print("[technic] INFO: The position used for vector.threeline isn't round.")
|
|
||||||
end
|
|
||||||
local two_line = vector.twoline(x, y)
|
|
||||||
line = {}
|
|
||||||
local n = 1
|
|
||||||
local zmin, zmax = 0, z
|
|
||||||
if z < 0 then
|
|
||||||
zmin, zmax = zmax, zmin
|
|
||||||
end
|
|
||||||
local m = z/math.hypot(x, y)
|
|
||||||
local dir = 1
|
|
||||||
if m < 0 then
|
|
||||||
dir = -dir
|
|
||||||
end
|
|
||||||
for _,i in ipairs(two_line) do
|
|
||||||
local px, py = unpack(i)
|
|
||||||
local ph = math.hypot(px, py)
|
|
||||||
local z1 = math.max(math.min(math.floor((ph-0.5)*m+0.5), zmax), zmin)
|
|
||||||
local z2 = math.max(math.min(math.floor((ph+0.5)*m+0.5), zmax), zmin)
|
|
||||||
for pz = z1,z2,dir do
|
|
||||||
line[n] = {px, py, pz}
|
|
||||||
n = n+1
|
|
||||||
end
|
|
||||||
end
|
|
||||||
threelines[pstr] = line
|
|
||||||
return line
|
|
||||||
end
|
|
||||||
|
|
||||||
function vector.line(pos, dir, range)
|
|
||||||
if range then --dir = pos2
|
|
||||||
dir = vector.round(vector.multiply(dir, range))
|
|
||||||
else
|
|
||||||
dir = vector.subtract(dir, pos)
|
|
||||||
end
|
|
||||||
local line,n = {},1
|
|
||||||
for _,i in ipairs(vector.threeline(dir.x, dir.y, dir.z)) do
|
|
||||||
line[n] = {x=pos.x+i[1], y=pos.y+i[2], z=pos.z+i[3]}
|
|
||||||
n = n+1
|
|
||||||
end
|
|
||||||
return line
|
|
||||||
end
|
|
Loading…
Reference in New Issue
Block a user