mirror of
https://github.com/osmarks/random-stuff
synced 2024-11-08 13:39:53 +00:00
104 lines
3.1 KiB
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)
|