Scoreboard Timers
Introduction
Sourced By Bedrock Commands Community Discord
This system allows you to run your desired commands at specific intervals with any amount of delay that you wish to add.
- Some Examples:
- Sending a message in chat every 2 hours.
- Running a 'lag clear' function every 10 minutes.
- Effecting players with 'speed' every 30 seconds.
This system is especially useful when you need to set up multiple timers on your world. When working with command blocks, you may use the Tick Delay option to delay the time taken for your commands to run. However, when working with functions, you will need to use a system like this.
It is recommended to use this system while working with command blocks, as well if you wish to run all your world-timers in sync with one another, ie. with the same start time.
Setup
To be typed in chat:
/scoreboard objectives add ticks dummy
/scoreboard objectives add events dummy
Once you have created these two objectives, you will need to define the interval for each repeating event you need on your world in the ticks
objective.
To do that, first, you must know that 1 second is approximately 20 game-ticks in Minecraft. Based on this knowledge, you will need to do some basic calculations to obtain the equivalent ticks for each interval you want to define.
# 2h = 20(t) × 60(s) × 60(m) × 2(h) = 144000t
/scoreboard players set 2h ticks 144000
# 10m = 20(t) × 60(s) × 10(m) = 12000t
/scoreboard players set 10m ticks 12000
# 30s = 20(t) × 30(s) = 600t
/scoreboard players set 30s ticks 600
We will now use this scoreboard data to make our timers function.
System
# World Timer/Clock
scoreboard players add timer ticks 1
scoreboard players operation * events = timer ticks
# Chat Message (every 2h)
scoreboard players operation chatMessage events %= 2h ticks
execute if score chatMessage events matches 0 run say Technoblade never dies!
# Lag Clear (every 10m)
scoreboard players operation lagClear events %= 10m ticks
execute if score lagClear events matches 0 run function clear_lag
# Speed Effect (every 30s)
scoreboard players operation speedEffect events %= 30s ticks
execute if score speedEffect events matches 0 run effect @a speed 10 2 true
Here, we have taken 3 examples to show how to implement them, but you can add any timer you prefer and as many as you need.
Just make sure to follow the given order and properly apply the /execute if score
condition as shown for your desired commands.
Explanation
events
on this objective we label all the repeating events we want on our world.chatMessage
lagClear
speedEffect
- Note: All 3 are fake-player-names in the objective.
ticks
on this objective we define all the intervals for our events and also run our scoreboard timer.2h
interval (static score: 144000)10m
interval (static score: 12000)30s
interval (static score: 600)timer
clock (variable score: n+1)- Note: All 4 are fake-player-names in the objective
Command 1: This command adds +1 score every tick to the fake-player-name 'timer' indicating a tick has passed in the game. This is basically our scoreboard timer/clock which we will use for all the repeating events on our world.
Command 2: Here, we copy 'timer' score to all our events using the
*
wildcard selector. This will allow us to perform operations to determine if the interval has been reached to run the commands for that particular event. Example:- If
timer
score is 1200, it means 1200 game-ticks have passed. - And this command makes it so all our events fake-player-names 'chatMessage', 'lagClear', 'speedEffect' scores are also 1200.
- If
Command 3: We will use the
%=
modulo operation to check if our events scores are divisible by their corresponding interval. ie., if the remainder is equal 0.- Chat Message:
1200/144000
- Q=0, R=1200
- hence, interval not reached.
- Lag Clear:
1200/12000
- Q=0, R=1200
- hence, interval not reached.
- Speed Effect:
1200/600
- Q=2, R=0
- hence, interval reached and event commands can be executed.
- Here, we can note that the first 2 events are yet to happen, but the 3rd event is happening for the second time.
- Chat Message:
TIP
In Minecraft, scoreboard division is only calculated up to whole numbers and decimal values are ignored.
- Command 4: the remainder obtained from the calculation is applied to the corresponding event's fake-player-name. Based on this knowledge, we can run our commands if the it's score is equal to
0
.
The rest of the commands are identical in structure and only the event labels and interval durations are changed.
Defining Events With Limited Intervals
To limit how many times an event occurs, you need to create a new objective called intervals
and define how many times that event should occur, as shown below.
/scoreboard objectives set chatMessage intervals 5
/scoreboard objectives set speedEffect intervals 10
Once you have done that, modify your system as shown below.
scoreboard players add timer ticks 1
scoreboard players operation * events = timer ticks
# Chat Message (every 10m)
scoreboard players operation chatMessage events %= 2h ticks
execute if score chatMessage events matches 0 if score chatMessage intervals matches 1.. run say Technoblade never dies!
execute if score chatMessage events matches 0 if score chatMessage intervals matches 1.. run scoreboard players remove chatMessage intervals 1
# Speed Effect (every 30s)
scoreboard players operation speedEffect events %= 30s ticks
execute if score speedEffect events matches 0 if score speedEffect intervals matches 1.. run effect @a speed 10 2 true
execute if score speedEffect events matches 0 if score speedEffect intervals matches 1.. run scoreboard players remove speedEffect intervals 1
Executing Commands During Timeframe
To run commands during the timeframe between intervals for a particular system, you may employ the technique shown below.
# Speed Effect (every 30s) + Particle (every tick)
scoreboard players operation speedEffect events %= 30s ticks
execute if score speedEffect intervals matches 1.. as @a at @s run particle minecraft:shulker_bullet ~~~
execute if score speedEffect events matches 0 if score speedEffect intervals matches 1.. run effect @a speed 10 2 true
execute if score speedEffect events matches 0 if score speedEffect intervals matches 1.. run scoreboard players remove speedEffect intervals 1
As shown in line 3, to run commands while the timer is running, all you need to do is remove the "if score" testing if the interval has been reached. And instead, only test if any interval remains to run our commands.
Let's say we had set the intervals
for this event to 10
. Then players would also have had particle trails for 300 seconds as repeating a 30s event 10 times will total 300 seconds.
Entity Timers
In some cases, such as an entity despawn event, you will need to run timers for each entity separately rather than a synchronised timer which could cause the event to trigger too soon. In such cases, an Async Timer can be helpful.
Let's say we want to perform the following actions:
- kill all entities named "station" 5 minutes after they've been summoned.
- play a shulker particle around them during that timeframe.
- play a flame particle around them in the first 10 seconds.
- play a pling sound to nearby players when the timer reaches half way.
- stop the timer if a passive mob is nearby.
- loop the timer if a hostile mob is nearby.
# Clock
scoreboard players add @e [name=station, scores={ticks=0..}] ticks 1
# Executing Commands while timer is running
execute as @e [name=station, scores={ticks=0..}] at @s run particle minecraft:shulker_bullet ~~~
# Executing commands within a timeframe
execute as @e [name=station, scores={ticks=0..200}] at @s run particle minecraft:basic_flame_particle ~~~
# Executing commands at specific intervals
execute as @e [name=station, scores={ticks=3600}] at @s run playsound note.pling @a [r=10]
# Stopping the timer
execute as @e [name=station] at @s if entity @e [family=pacified, r=10, c=1] run scoreboard players set @s ticks -1
# Looping the timer
execute as @e [name=station, scores={ticks=6000}] at @s if entity @e [family=monster, r=10, c=1] run scoreboard players set @s ticks 0
# End
/kill @e [name=station, scores={ticks=6000}]
As shown, setting the score to 0
when it completes the timeframe will loop the timer. And setting the score to -1
will stop/disable it. You can still set the score to 0
to start the timer again.