121 lines
3.8 KiB
Markdown
121 lines
3.8 KiB
Markdown
|
---
|
||
|
title: Node Timers and ABMs
|
||
|
layout: default
|
||
|
root: ../..
|
||
|
idx: 3.2
|
||
|
description: Learn how to make ABMs to change blocks.
|
||
|
redirect_from:
|
||
|
- /en/chapters/abms.html
|
||
|
- /en/map/abms.html
|
||
|
---
|
||
|
|
||
|
## Introduction <!-- omit in toc -->
|
||
|
|
||
|
Periodically running a function on certain nodes is a common task.
|
||
|
Minetest provides two methods of doing this: Active Block Modifiers (ABMs) and node timers.
|
||
|
|
||
|
ABMs scan all loaded MapBlocks looking for nodes that match a criteria.
|
||
|
They are best suited for nodes which are frequently found in the world,
|
||
|
such as grass.
|
||
|
They have a high CPU overhead, but a low memory and storage overhead.
|
||
|
|
||
|
For nodes that are uncommon or already use metadata, such as furnaces
|
||
|
and machines, node timers should be used instead.
|
||
|
Node timers work by keeping track of pending timers in each MapBlock, and then
|
||
|
running them when they expire.
|
||
|
This means that timers don't need to search all loaded nodes to find matches,
|
||
|
but instead require slightly more memory and storage for the tracking
|
||
|
of pending timers.
|
||
|
|
||
|
- [Node Timers](#node-timers)
|
||
|
- [Active Block Modifiers](#active-block-modifiers)
|
||
|
- [Your Turn](#your-turn)
|
||
|
|
||
|
## Node Timers
|
||
|
|
||
|
Node timers are directly tied to a single node.
|
||
|
You can manage node timers by obtaining a NodeTimerRef object.
|
||
|
|
||
|
```lua
|
||
|
local timer = minetest.get_node_timer(pos)
|
||
|
timer:start(10.5) -- in seconds
|
||
|
```
|
||
|
|
||
|
You can also check the status or stop the timer:
|
||
|
|
||
|
```lua
|
||
|
if timer:is_started() then
|
||
|
print("The timer is running, and has " .. timer:get_timeout() .. "s remaining!")
|
||
|
print(timer:get_elapsed() .. "s has elapsed.")
|
||
|
end
|
||
|
|
||
|
timer:stop()
|
||
|
```
|
||
|
|
||
|
When a node timer is up, the `on_timer` method in the node's definition table will
|
||
|
be called.
|
||
|
The method only takes a single parameter, the position of the node.
|
||
|
|
||
|
```lua
|
||
|
minetest.register_node("autodoors:door_open", {
|
||
|
on_timer = function(pos)
|
||
|
minetest.set_node(pos, { name = "autodoors:door" })
|
||
|
return false
|
||
|
end
|
||
|
})
|
||
|
```
|
||
|
|
||
|
Returning true in `on_timer` will cause the timer to run again for the same interval.
|
||
|
|
||
|
You may have noticed a limitation with timers: for optimisation reasons, it's
|
||
|
only possible to have one type of timer per node type, and only one timer running per node.
|
||
|
|
||
|
|
||
|
## Active Block Modifiers
|
||
|
|
||
|
Alien grass, for the purposes of this chapter, is a type of grass which
|
||
|
has a chance to appear near water.
|
||
|
|
||
|
|
||
|
```lua
|
||
|
minetest.register_node("aliens:grass", {
|
||
|
description = "Alien Grass",
|
||
|
light_source = 3, -- The node radiates light. Min 0, max 14
|
||
|
tiles = {"aliens_grass.png"},
|
||
|
groups = {choppy=1},
|
||
|
on_use = minetest.item_eat(20)
|
||
|
})
|
||
|
|
||
|
minetest.register_abm({
|
||
|
nodenames = {"default:dirt_with_grass"},
|
||
|
neighbors = {"default:water_source", "default:water_flowing"},
|
||
|
interval = 10.0, -- Run every 10 seconds
|
||
|
chance = 50, -- Select every 1 in 50 nodes
|
||
|
action = function(pos, node, active_object_count,
|
||
|
active_object_count_wider)
|
||
|
local pos = {x = pos.x, y = pos.y + 1, z = pos.z}
|
||
|
minetest.set_node(pos, {name = "aliens:grass"})
|
||
|
end
|
||
|
})
|
||
|
```
|
||
|
|
||
|
This ABM runs every ten seconds, and for each matching node, there is
|
||
|
a 1 in 50 chance of it running.
|
||
|
If the ABM runs on a node, an alien grass node is placed above it.
|
||
|
Please be warned, this will delete any node previously located in that position.
|
||
|
To prevent this you should include a check using minetest.get_node to make sure there is space for the grass.
|
||
|
|
||
|
Specifying a neighbour is optional.
|
||
|
If you specify multiple neighbours, only one of them needs to be
|
||
|
present to meet the requirements.
|
||
|
|
||
|
Specifying chance is also optional.
|
||
|
If you don't specify the chance, the ABM will always run when the other conditions are met.
|
||
|
|
||
|
## Your Turn
|
||
|
|
||
|
* Midas touch: Make water turn to gold blocks with a 1 in 100 chance, every 5 seconds.
|
||
|
* Decay: Make wood turn into dirt when water is a neighbour.
|
||
|
* Burnin': Make every air node catch on fire. (Tip: "air" and "fire:basic_flame").
|
||
|
Warning: expect the game to crash.
|