Vanessa Dannenberg
2018-10-31 44cb8df048e09b64214f59db73a3fd23cfe12e77
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
--- Forcefield generator.
-- @author ShadowNinja
--
-- Forcefields are powerful barriers but they consume huge amounts of power.
-- The forcefield Generator is an HV machine.
 
-- How expensive is the generator?
-- Leaves room for upgrades lowering the power drain?
local digilines_path = minetest.get_modpath("digilines")
 
local forcefield_power_drain   = 10
 
local S = technic.getter
 
local cable_entry = "^technic_cable_connection_overlay.png"
 
minetest.register_craft({
    output = "technic:forcefield_emitter_off",
    recipe = {
        {"default:mese",         "basic_materials:motor",          "default:mese"        },
        {"technic:deployer_off", "technic:machine_casing", "technic:deployer_off"},
        {"default:mese",         "technic:hv_cable",       "default:mese"        },
    }
})
 
 
local replaceable_cids = {}
 
minetest.after(0, function()
    for name, ndef in pairs(minetest.registered_nodes) do
        if ndef.buildable_to == true and name ~= "ignore" then
            replaceable_cids[minetest.get_content_id(name)] = true
        end
    end
end)
 
 
-- Idea: Let forcefields have different colors by upgrade slot.
-- Idea: Let forcefields add up by detecting if one hits another.
--    ___   __
--   /   \/   \
--  |          |
--   \___/\___/
 
local function update_forcefield(pos, meta, active, first)
    local shape = meta:get_int("shape")
    local range = meta:get_int("range")
    local vm = VoxelManip()
    local MinEdge, MaxEdge = vm:read_from_map(vector.subtract(pos, range),
            vector.add(pos, range))
    local area = VoxelArea:new({MinEdge = MinEdge, MaxEdge = MaxEdge})
    local data = vm:get_data()
 
    local c_air = minetest.get_content_id("air")
    local c_field = minetest.get_content_id("technic:forcefield")
 
    for z = -range, range do
    for y = -range, range do
    local vi = area:index(pos.x + (-range), pos.y + y, pos.z + z)
    for x = -range, range do
        local relevant
        if shape == 0 then
            local squared = x * x + y * y + z * z
            relevant =
                squared <= range       *  range      +  range and
                squared >= (range - 1) * (range - 1) + (range - 1)
        else
            relevant =
                x == -range or x == range or
                y == -range or y == range or
                z == -range or z == range
        end
        if relevant then
            local cid = data[vi]
            if active and replaceable_cids[cid] then
                data[vi] = c_field
            elseif not active and cid == c_field then
                data[vi] = c_air
            end
        end
        vi = vi + 1
    end
    end
    end
 
    vm:set_data(data)
    vm:update_liquids()
    vm:write_to_map()
    -- update_map is very slow, but if we don't call it we'll
    -- get phantom blocks on the client.
    if not active or first then
        vm:update_map()
    end
end
 
local function set_forcefield_formspec(meta)
    local formspec
    if digilines_path then
        formspec = "size[5,3.25]"..
            "field[0.3,3;5,1;channel;Digiline Channel;"..meta:get_string("channel").."]"
    else
        formspec = "size[5,2.25]"
    end
    formspec = formspec..
        "field[0.3,0.5;2,1;range;"..S("Range")..";"..meta:get_int("range").."]"
    -- The names for these toggle buttons are explicit about which
    -- state they'll switch to, so that multiple presses (arising
    -- from the ambiguity between lag and a missed press) only make
    -- the single change that the user expects.
    if meta:get_int("shape") == 0 then
        formspec = formspec.."button[3,0.2;2,1;shape1;"..S("Sphere").."]"
    else
        formspec = formspec.."button[3,0.2;2,1;shape0;"..S("Cube").."]"
    end
    if meta:get_int("mesecon_mode") == 0 then
        formspec = formspec.."button[0,1;5,1;mesecon_mode_1;"..S("Ignoring Mesecon Signal").."]"
    else
        formspec = formspec.."button[0,1;5,1;mesecon_mode_0;"..S("Controlled by Mesecon Signal").."]"
    end
    if meta:get_int("enabled") == 0 then
        formspec = formspec.."button[0,1.75;5,1;enable;"..S("%s Disabled"):format(S("%s Forcefield Emitter"):format("HV")).."]"
    else
        formspec = formspec.."button[0,1.75;5,1;disable;"..S("%s Enabled"):format(S("%s Forcefield Emitter"):format("HV")).."]"
    end
    meta:set_string("formspec", formspec)
end
 
local forcefield_receive_fields = function(pos, formname, fields, sender)
    local meta = minetest.get_meta(pos)
    local range = nil
    if fields.range then
        range = tonumber(fields.range) or 0
        -- Smallest field is 5. Anything less is asking for trouble.
        -- Largest is 20. It is a matter of pratical node handling.
        -- At the maximim range updating the forcefield takes about 0.2s
        range = math.max(range, 5)
        range = math.min(range, 20)
        if range == meta:get_int("range") then range = nil end
    end
    if fields.shape0 or fields.shape1 or range then
        update_forcefield(pos, meta, false)
    end
    if range then meta:set_int("range", range) end
    if fields.channel then meta:set_string("channel", fields.channel) end
    if fields.shape0  then meta:set_int("shape", 0) end
    if fields.shape1  then meta:set_int("shape", 1) end
    if fields.enable  then meta:set_int("enabled", 1) end
    if fields.disable then meta:set_int("enabled", 0) end
    if fields.mesecon_mode_0 then meta:set_int("mesecon_mode", 0) end
    if fields.mesecon_mode_1 then meta:set_int("mesecon_mode", 1) end
    set_forcefield_formspec(meta)
end
 
local mesecons = {
    effector = {
        action_on = function(pos, node)
            minetest.get_meta(pos):set_int("mesecon_effect", 1)
        end,
        action_off = function(pos, node)
            minetest.get_meta(pos):set_int("mesecon_effect", 0)
        end
    }
}
 
local digiline_def = {
    receptor = {action = function() end},
    effector = {
        action = function(pos, node, channel, msg)
            local meta = minetest.get_meta(pos)
            if channel ~= meta:get_string("channel") then
                return
            end
            local msgt = type(msg)
            if msgt == "string" then
                local smsg = msg:lower()
                msg = {}
                if smsg == "get" then
                    msg.command = "get"
                elseif smsg == "off" then
                    msg.command = "off"
                elseif smsg == "on" then
                    msg.command = "on"
                elseif smsg == "toggle" then
                    msg.command = "toggle"
                elseif smsg:sub(1, 5) == "range" then
                    msg.command = "range"
                    msg.value = tonumber(smsg:sub(7))
                elseif smsg:sub(1, 5) == "shape" then
                    msg.command = "shape"
                    msg.value = smsg:sub(7):lower()
                    msg.value = tonumber(msg.value) or msg.value
                end
            elseif msgt ~= "table" then
                return
            end
            if msg.command == "get" then
                digilines.receptor_send(pos, digilines.rules.default, channel, {
                    enabled = meta:get_int("enabled"),
                    range   = meta:get_int("range"),
                    shape   = meta:get_int("shape")
                })
                return
            elseif msg.command == "off" then
                meta:set_int("enabled", 0)
            elseif msg.command == "on" then
                meta:set_int("enabled", 1)
            elseif msg.command == "toggle" then
                local onn = meta:get_int("enabled")
                onn = 1-onn -- Mirror onn with pivot 0.5, so switch between 1 and 0.
                meta:set_int("enabled", onn)
            elseif msg.command == "range" then
                if type(msg.value) ~= "number" then
                    return
                end
                msg.value = math.max(msg.value, 5)
                msg.value = math.min(msg.value, 20)
                update_forcefield(pos, meta, false)
                meta:set_int("range", msg.value)
            elseif msg.command == "shape" then
                local valuet = type(msg.value)
                if valuet == "string" then
                    if msg.value == "sphere" then
                        msg.value = 0
                    elseif msg.value == "cube" then
                        msg.value = 1
                    end
                elseif valuet ~= "number" then
                    return
                end
                if not msg.value then
                    return
                end
                update_forcefield(pos, meta, false)
                meta:set_int("shape", msg.value)
            else
                return
            end
            set_forcefield_formspec(meta)
        end
    },
}
 
local function run(pos, node)
    local meta = minetest.get_meta(pos)
    local eu_input   = meta:get_int("HV_EU_input")
    local enabled = meta:get_int("enabled") ~= 0 and
        (meta:get_int("mesecon_mode") == 0 or meta:get_int("mesecon_effect") ~= 0)
    local machine_name = S("%s Forcefield Emitter"):format("HV")
 
    local range = meta:get_int("range")
    local power_requirement
    if meta:get_int("shape") == 0 then
        power_requirement = math.floor(4 * math.pi * range * range)
    else
        power_requirement = 24 * range * range
    end
    power_requirement = power_requirement * forcefield_power_drain
 
    if not enabled then
        if node.name == "technic:forcefield_emitter_on" then
            update_forcefield(pos, meta, false)
            technic.swap_node(pos, "technic:forcefield_emitter_off")
            meta:set_string("infotext", S("%s Disabled"):format(machine_name))
        end
        meta:set_int("HV_EU_demand", 0)
        return
    end
    meta:set_int("HV_EU_demand", power_requirement)
    if eu_input < power_requirement then
        meta:set_string("infotext", S("%s Unpowered"):format(machine_name))
        if node.name == "technic:forcefield_emitter_on" then
            update_forcefield(pos, meta, false)
            technic.swap_node(pos, "technic:forcefield_emitter_off")
        end
    elseif eu_input >= power_requirement then
        local first = false
        if node.name == "technic:forcefield_emitter_off" then
            first = true
            technic.swap_node(pos, "technic:forcefield_emitter_on")
            meta:set_string("infotext", S("%s Active"):format(machine_name))
        end
        update_forcefield(pos, meta, true, first)
    end
end
 
minetest.register_node("technic:forcefield_emitter_off", {
    description = S("%s Forcefield Emitter"):format("HV"),
    tiles = {
        "technic_forcefield_emitter_off.png",
        "technic_machine_bottom.png"..cable_entry,
        "technic_forcefield_emitter_off.png",
        "technic_forcefield_emitter_off.png",
        "technic_forcefield_emitter_off.png",
        "technic_forcefield_emitter_off.png"
    },
    groups = {cracky = 1, technic_machine = 1, technic_hv = 1},
    on_receive_fields = forcefield_receive_fields,
    on_construct = function(pos)
        local meta = minetest.get_meta(pos)
        meta:set_int("HV_EU_input", 0)
        meta:set_int("HV_EU_demand", 0)
        meta:set_int("range", 10)
        meta:set_int("enabled", 0)
        meta:set_int("mesecon_mode", 0)
        meta:set_int("mesecon_effect", 0)
        if digilines_path then
            meta:set_string("channel", "forcefield"..minetest.pos_to_string(pos))
        end
        meta:set_string("infotext", S("%s Forcefield Emitter"):format("HV"))
        set_forcefield_formspec(meta)
    end,
    mesecons = mesecons,
    digiline = digiline_def,
    technic_run = run,
})
 
minetest.register_node("technic:forcefield_emitter_on", {
    description = S("%s Forcefield Emitter"):format("HV"),
    tiles = {
        "technic_forcefield_emitter_on.png",
        "technic_machine_bottom.png"..cable_entry,
        "technic_forcefield_emitter_on.png",
        "technic_forcefield_emitter_on.png",
        "technic_forcefield_emitter_on.png",
        "technic_forcefield_emitter_on.png"
    },
    groups = {cracky = 1, technic_machine = 1, technic_hv = 1,
            not_in_creative_inventory=1},
    drop = "technic:forcefield_emitter_off",
    on_receive_fields = forcefield_receive_fields,
    on_destruct = function(pos)
        local meta = minetest.get_meta(pos)
        update_forcefield(pos, meta, false)
    end,
    mesecons = mesecons,
    digiline = digiline_def,
    technic_run = run,
    technic_on_disable = function (pos, node)
        local meta = minetest.get_meta(pos)
        update_forcefield(pos, meta, false)
        technic.swap_node(pos, "technic:forcefield_emitter_off")
    end,
    on_blast = function(pos, intensity)
        minetest.dig_node(pos)
        return {"technic:forcefield_emitter_off"}
    end,
})
 
minetest.register_node("technic:forcefield", {
    description = S("%s Forcefield"):format("HV"),
    sunlight_propagates = true,
    drawtype = "glasslike",
    groups = {not_in_creative_inventory=1},
    paramtype = "light",
    light_source = default.LIGHT_MAX,
    diggable = false,
    drop = '',
    tiles = {{
        name = "technic_forcefield_animated.png",
        animation = {
            type = "vertical_frames",
            aspect_w = 16,
            aspect_h = 16,
            length = 1.0,
        },
    }},
    on_blast = function(pos, intensity)
    end,
})
 
 
if minetest.get_modpath("mesecons_mvps") then
    mesecon.register_mvps_stopper("technic:forcefield")
end
 
technic.register_machine("HV", "technic:forcefield_emitter_on",  technic.receiver)
technic.register_machine("HV", "technic:forcefield_emitter_off", technic.receiver)