dfcaverns/df_trees/blood_thorn.lua
FaceDeer dbc5dd38a3 Squashed commit of the following:
commit 0a61781b99ec16166ba66f8cb19cd679c8a04f03
Author: FaceDeer <derksenmobile@gmail.com>
Date:   Mon Aug 1 14:50:07 2022 -0600

    add an additional check to ensure old timers don't cause inappropriate growth

commit 1d7b6010c36119e2a183478833e5e44759caeaca
Author: FaceDeer <derksenmobile@gmail.com>
Date:   Mon Aug 1 13:21:29 2022 -0600

    stop timers when seeds are picked up

commit c8fa25ccd79b775d98e32db2a6937f27e1d1ed2c
Author: FaceDeer <derksenmobile@gmail.com>
Date:   Mon Aug 1 13:05:24 2022 -0600

    fix replacements for dwarven syrup taffy recipe

commit 4de45bb6d728b0696a1e968744b06191e9283c14
Author: FaceDeer <derksenmobile@gmail.com>
Date:   Mon Aug 1 11:09:48 2022 -0600

    account for some additional mod dependencies

commit 83ea06bbaa709daab22ff934e901ef0e9dc85767
Author: FaceDeer <derksenmobile@gmail.com>
Date:   Mon Aug 1 11:09:14 2022 -0600

    update cooking recipes to be more specific.

commit 302da3ec5155c5be1a29e781cd90b7e84274f831
Author: FaceDeer <derksenmobile@gmail.com>
Date:   Fri Jul 29 17:12:59 2022 -0600

    add location logging for debugging purposes

commit 11667e184ebeb9f9d43666b3ccbc1bc71d0bcc70
Author: FaceDeer <derksenmobile@gmail.com>
Date:   Sun Jul 24 16:54:21 2022 -0600

    add checks for submods being present

    the df_trees and df_farming checks are likely redundant, but if primordial layers are disabled someone might not have df_primordial_items installed.

commit 5906308d87e48841a46855e023201ea6d8e6159c
Author: FaceDeer <derksenmobile@gmail.com>
Date:   Sun Jul 24 16:49:23 2022 -0600

    add config settings for biome restrictions, vastly reduce copy and paste in code

commit e52820c2822baf71e4647cf3181b247eac9703aa
Author: FaceDeer <derksenmobile@gmail.com>
Date:   Sat Jul 23 20:45:26 2022 -0600

    add initial stab at growing conditions - biome restrictions for trees

commit 7b99556df9c8f95f5311b9822eb6210ac3038d34
Author: FaceDeer <derksenmobile@gmail.com>
Date:   Sat Jul 23 12:08:41 2022 -0600

    adding biome API. Not yet tested.

commit bf82b3b3fed9bd62b280cb549d972a924f514208
Author: FaceDeer <derksenmobile@gmail.com>
Date:   Fri Jul 22 21:22:37 2022 -0600

    added stubs for growth permission for farming plants

commit 46765df3ef01cf031cbe5074d781792a8218e1a8
Author: FaceDeer <derksenmobile@gmail.com>
Date:   Fri Jul 22 18:36:45 2022 -0600

    initial work for restricted plant growth. split out growth conditions for trees, and reworked torchspine to not use ABMs while I was at it.
2022-08-01 14:55:46 -06:00

335 lines
10 KiB
Lua

--------------------------------------------------
-- Blood thorn
-- Max trunk height 5
-- red with purple mottling
-- High density wood
-- Depth 3
local S = minetest.get_translator(minetest.get_current_modname())
local spike_directions = {
{dir={x=0,y=0,z=1}, facedir=2},
{dir={x=0,y=0,z=-1}, facedir=0},
{dir={x=1,y=0,z=0}, facedir=3},
{dir={x=-1,y=0,z=0}, facedir=1}
}
local spikes = {["df_trees:blood_thorn_spike"] = true, ["df_trees:blood_thorn_spike_dead"] = true}
local blood_thorn_after_dig = function(pos, oldnode, oldmetadata, digger)
for _, spike_dir in pairs(spike_directions) do
local loc = vector.add(pos,spike_dir.dir)
local node = minetest.get_node(loc)
if spikes[node.name] and spike_dir.facedir == node.param2 then
minetest.node_dig(loc, node, digger)
end
end
end
local blood_thorn_min_delay = df_trees.config.blood_thorn_delay_multiplier*df_trees.config.tree_min_growth_delay
local blood_thorn_max_delay = df_trees.config.blood_thorn_delay_multiplier*df_trees.config.tree_max_growth_delay
minetest.register_node("df_trees:blood_thorn", {
description = S("Blood Thorn Stem"),
_doc_items_longdesc = df_trees.doc.blood_thorn_desc,
_doc_items_usagehelp = df_trees.doc.blood_thorn_usage,
tiles = {"dfcaverns_blood_thorn_top.png", "dfcaverns_blood_thorn_top.png",
"dfcaverns_blood_thorn_side.png", "dfcaverns_blood_thorn_side.png", "dfcaverns_blood_thorn_side.png", "dfcaverns_blood_thorn_side.png"},
paramtype2 = "facedir",
paramtype = "light",
groups = {choppy = 3, tree = 1, flammable = 2, light_sensitive_fungus = 11},
_dfcaverns_dead_node = "df_trees:blood_thorn_dead",
sounds = df_trees.sounds.wood,
is_ground_content = false,
on_place = minetest.rotate_node,
after_dig_node = blood_thorn_after_dig,
on_construct = function(pos)
minetest.get_node_timer(pos):start(math.random(blood_thorn_min_delay, blood_thorn_max_delay))
end,
on_destruct = function(pos)
minetest.get_node_timer(pos):stop()
local below = {x=pos.x, y=pos.y-1, z=pos.z}
local below_node = minetest.get_node(below)
while below_node.name == "df_trees:blood_thorn" do
-- if you chop down a bloodthorn it can start growing again
minetest.get_node_timer(below):start(math.random(blood_thorn_min_delay, blood_thorn_max_delay))
below.y = below.y-1
below_node = minetest.get_node(below)
end
end,
on_timer = function(pos, elapsed)
while elapsed > blood_thorn_max_delay do
-- catch up if the block was unloaded for a long time
elapsed = elapsed - blood_thorn_max_delay
if not df_trees.grow_blood_thorn(vector.new(pos), minetest.get_node(pos)) then
return
end
end
if df_trees.grow_blood_thorn(vector.new(pos), minetest.get_node(pos)) then
minetest.get_node_timer(pos):start(math.random(blood_thorn_min_delay, blood_thorn_max_delay))
end
end,
})
minetest.register_node("df_trees:blood_thorn_dead", {
description = S("Dead Blood Thorn Stem"),
_doc_items_longdesc = df_trees.doc.blood_thorn_desc,
_doc_items_usagehelp = df_trees.doc.blood_thorn_usage,
tiles = {"dfcaverns_blood_thorn_top.png^[multiply:#804000", "dfcaverns_blood_thorn_top.png^[multiply:#804000",
"dfcaverns_blood_thorn_side.png^[multiply:#804000"},
paramtype2 = "facedir",
paramtype = "light",
groups = {choppy = 3, tree = 1, flammable = 2},
sounds = df_trees.sounds.wood,
is_ground_content = false,
on_place = minetest.rotate_node,
after_dig_node = blood_thorn_after_dig,
})
minetest.register_node("df_trees:blood_thorn_spike", {
description = S("Blood Thorn Spike"),
_doc_items_longdesc = df_trees.doc.blood_thorn_spike_desc,
_doc_items_usagehelp = df_trees.doc.blood_thorn_spike_usage,
tiles = {
"dfcaverns_blood_thorn_spike_side.png^[transformR90",
"dfcaverns_blood_thorn_spike_side.png^[transformR270",
"dfcaverns_blood_thorn_spike_side.png",
"dfcaverns_blood_thorn_spike_side.png^[transformR180",
"dfcaverns_blood_thorn_spike_front.png",
"dfcaverns_blood_thorn_spike_front.png"
},
groups = {choppy = 3, flammable = 2, fall_damage_add_percent=100, light_sensitive_fungus = 11},
_dfcaverns_dead_node = "df_trees:blood_thorn_spike_dead",
sounds = df_trees.sounds.wood,
drawtype = "nodebox",
climbable = true,
is_ground_content = false,
damage_per_second = 1,
paramtype = "light",
paramtype2 = "facedir",
node_box = {
type = "fixed",
fixed = {
{-0.1875, -0.1875, 0.1875, 0.1875, 0.1875, 0.5}, -- base
{-0.125, -0.125, -0.125, 0.125, 0.125, 0.1875}, -- mid
{-0.0625, -0.0625, -0.5, 0.0625, 0.0625, -0.125}, -- tip
}
},
})
minetest.register_node("df_trees:blood_thorn_spike_dead", {
description = S("Dead Blood Thorn Spike"),
_doc_items_longdesc = df_trees.doc.blood_thorn_spike_desc,
_doc_items_usagehelp = df_trees.doc.blood_thorn_spike_usage,
tiles = {
"dfcaverns_blood_thorn_spike_side.png^[transformR90^[multiply:#804000",
"dfcaverns_blood_thorn_spike_side.png^[transformR270^[multiply:#804000",
"dfcaverns_blood_thorn_spike_side.png^[multiply:#804000",
"dfcaverns_blood_thorn_spike_side.png^[transformR180^[multiply:#804000",
"dfcaverns_blood_thorn_spike_front.png^[multiply:#804000",
"dfcaverns_blood_thorn_spike_front.png^[multiply:#804000"
},
groups = {choppy = 3, flammable = 2, fall_damage_add_percent=100},
sounds = df_trees.sounds.wood,
drawtype = "nodebox",
climbable = true,
is_ground_content = false,
paramtype = "light",
paramtype2 = "facedir",
node_box = {
type = "fixed",
fixed = {
{-0.1875, -0.1875, 0.1875, 0.1875, 0.1875, 0.5}, -- base
{-0.125, -0.125, -0.125, 0.125, 0.125, 0.1875}, -- mid
{-0.0625, -0.0625, -0.5, 0.0625, 0.0625, -0.125}, -- tip
}
},
})
--Wood
minetest.register_craft({
output = 'df_trees:blood_thorn_wood 4',
recipe = {
{'df_trees:blood_thorn'},
}
})
minetest.register_craft({
output = 'df_trees:blood_thorn_wood 4',
recipe = {
{'df_trees:blood_thorn_dead'},
}
})
minetest.register_node("df_trees:blood_thorn_wood", {
description = S("Blood Thorn Planks"),
_doc_items_longdesc = df_trees.doc.blood_thorn_desc,
_doc_items_usagehelp = df_trees.doc.blood_thorn_usage,
paramtype2 = "facedir",
place_param2 = 0,
tiles = {"dfcaverns_blood_thorn_wood.png"},
is_ground_content = false,
groups = {choppy = 2, oddly_breakable_by_hand = 2, flammable = 2, wood = 1},
sounds = df_trees.sounds.wood,
})
df_trees.register.all_stairs("blood_thorn_wood")
df_trees.register.all_fences("blood_thorn_wood", {burntime = 40})
minetest.register_craft({
type = "fuel",
recipe = "df_trees:blood_thorn_wood",
burntime = 40,
})
minetest.register_craft({
type = "fuel",
recipe = "df_trees:blood_thorn",
burntime = 150,
})
minetest.register_craft({
type = "fuel",
recipe = "df_trees:blood_thorn_dead",
burntime = 120,
})
minetest.register_craft({
type = "fuel",
recipe = "df_trees:blood_thorn_spike",
burntime = 5,
})
minetest.register_craft({
type = "fuel",
recipe = "df_trees:blood_thorn_spike_dead",
burntime = 5,
})
-- This ensures consistent random maximum to bloodthorn growth
local max_bloodthorn_height = function(pos)
local next_seed = math.floor(math.random() * 2^31)
math.randomseed(pos.x + pos.z * 2 ^ 8)
local output = math.random(3,6)
math.randomseed(next_seed)
return output
end
function df_trees.grow_blood_thorn(pos, node)
if df_farming and df_farming.kill_if_sunlit(pos) then
return
end
-- node is tipped over
if node.param2 >= 4 then
return
end
if not df_trees.blood_thorn_growth_permitted(pos) then
return
end
local height = 0
local max_height = max_bloodthorn_height(pos)
while node.name == "df_trees:blood_thorn" and height < max_height do
height = height + 1
pos.y = pos.y + 1
node = minetest.get_node(pos)
end
if height == 7 or node.name ~= "air" then
return
end
minetest.set_node(pos, {name = "df_trees:blood_thorn"})
local dir = spike_directions[math.random(1,4)]
local spike_pos = vector.add(pos, dir.dir)
if minetest.get_node(spike_pos).name == "air" then
minetest.set_node(spike_pos, {name="df_trees:blood_thorn_spike", param2=dir.facedir})
end
dir = spike_directions[math.random(1,4)]
spike_pos = vector.add(pos, dir.dir)
if minetest.get_node(spike_pos).name == "air" then
minetest.set_node(spike_pos, {name="df_trees:blood_thorn_spike", param2=dir.facedir})
end
return true
end
minetest.register_lbm({
label = "Start timers for bloodthorn nodes that used to depend on the ABM",
name = "df_trees:start_bloodthorn_timers",
nodenames = {"df_trees:blood_thorn"},
run_at_every_load = false,
action = function(pos, node)
minetest.get_node_timer(pos):start(math.random(blood_thorn_min_delay, blood_thorn_max_delay))
end,
})
function df_trees.spawn_blood_thorn(pos)
local height = max_bloodthorn_height(pos)
local x, y, z = pos.x, pos.y, pos.z
local maxy = y + height -- Trunk top
local vm = minetest.get_voxel_manip()
local minp, maxp = vm:read_from_map(
{x = x - 1, y = y, z = z - 1},
{x = x + 1, y = maxy, z = z + 1}
)
local area = VoxelArea:new({MinEdge = minp, MaxEdge = maxp})
local data = vm:get_data()
local data_param2 = vm:get_param2_data()
df_trees.spawn_blood_thorn_vm(area:indexp(pos), area, data, data_param2, height)
vm:set_data(data)
vm:set_param2_data(data_param2)
vm:write_to_map()
vm:update_map()
end
local c_air = minetest.get_content_id("air")
local c_ignore = minetest.get_content_id("ignore")
local c_blood_thorn = minetest.get_content_id("df_trees:blood_thorn")
local c_blood_thorn_spike = minetest.get_content_id("df_trees:blood_thorn_spike")
local facedir_to_increment = function(direction, area)
if direction == 0 then
return -area.zstride
elseif direction == 1 then
return -1
elseif direction == 2 then
return area.zstride
elseif direction == 3 then
return 1
end
end
df_trees.spawn_blood_thorn_vm = function(vi, area, data, data_param2, height)
local pos = area:position(vi)
if height == nil then height = max_bloodthorn_height(pos) end
for i = 1, height do
local node_id = data[vi]
if node_id == c_air or node_id == c_ignore then
data[vi] = c_blood_thorn
for i = 1, 2 do
local facedir = math.random(1,4)-1
local spike_vi = vi + facedir_to_increment(facedir, area)
if data[spike_vi] == c_air or data[spike_vi] == c_ignore then
data[spike_vi] = c_blood_thorn_spike
data_param2[spike_vi] = facedir
end
end
else
break
end
vi = vi + area.ystride
end
end