random-stuff/computercraft/panel-turtle.lua

104 lines
3.1 KiB
Lua

local sensor = peripheral.wrap "right"
local laser = peripheral.wrap "left"
local modem = peripheral.find "modem"
local targets = {}
local function scan()
local nearby = sensor.sense()
local ret = {}
for k, v in pairs(nearby) do
v.s = vector.new(v.x, v.y, v.z)
v.v = vector.new(v.motionX, v.motionY, v.motionZ)
v.distance = v.s:length()
if v.displayName == v.name then ret[v.name] = v end
end
return ret
end
local function calc_yaw_pitch(v)
local x, y, z = v.x, v.y, v.z
local pitch = -math.atan2(y, math.sqrt(x * x + z * z))
local yaw = math.atan2(-x, z)
return math.deg(yaw), math.deg(pitch)
end
local function vector_sqlength(self)
return self.x * self.x + self.y * self.y + self.z * self.z
end
local function project(line_start, line_dir, point)
local t = (point - line_start):dot(line_dir) / vector_sqlength(line_dir)
return line_start + line_dir * t, t
end
local function angles_to_axis(yaw, pitch)
return vector.new(
-math.sin(math.rad(yaw)) * math.cos(math.rad(pitch)),
-math.sin(math.rad(pitch)),
math.cos(math.rad(yaw)) * math.cos(math.rad(pitch))
)
end
local laser_origin = vector.new(0, 0, 0)
local function would_hit(beam_line, safe_position, target_position)
local point, t = project(laser_origin, beam_line, safe_position)
return t > 0 and (point - safe_position):length() < 1.5 and safe_position:length() < target_position:length()
end
local function lase(entity, others)
local target_location = entity.s - vector.new(0, 1, 0)
for i = 1, 5 do
target_location = entity.s + entity.v * (target_location:length() / 1.5)
end
local y, p = calc_yaw_pitch(target_location)
local line = angles_to_axis(y, p)
for _, other in pairs(others) do
if would_hit(line, other.s, target_location) then
--print("would hit", other.name)
return false
end
end
if would_hit(line, vector.new(-3, 0, 2), target_location) then
return false
end
print(y)
if y > (-90 + 45) or y < (-90 - 45) then
return false
end
rs.setOutput("front", true)
laser.fire(y, p, 1)
sleep(0.05)
rs.setOutput("front", false)
end
local function laser_defense()
while true do
local entities = scan()
local now = os.epoch "utc"
local action_taken = false
for _, entity in pairs(entities) do
local targeted_at = targets[entity.name]
if targeted_at and targeted_at > (now - 60000) then
print("lasing", entity.displayName, entity.s)
lase(entity, entities)
action_taken = true
end
end
if not action_taken then sleep(0.2) end
end
end
local function laser_commands()
modem.open(55)
while true do
local _, _, c, rc, m = os.pullEvent "modem_message"
if c == 55 and type(m) == "table" and m[1] == "lase" and type(m[2]) == "string" then
targets[m[2]] = os.epoch "utc"
print("command to lase", m[2], "remotely")
end
end
end
parallel.waitForAll(laser_defense, laser_commands)