114 lines
3.5 KiB
Markdown
114 lines
3.5 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
|
|
|
|
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.
|
|
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" })
|
|
end
|
|
})
|
|
```
|
|
|
|
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()
|
|
```
|
|
|
|
## 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.
|