I’m surprised that I managed to get this done in about 20 minutes last night given that I’ve come back to Godot after nearly 3 years!
Anyways, we’ll be writing a barebones random encounter mechanic here. From my experience with some 90s to mid 2000s JRPGs, I think random encounters generally work as such:
Start a timer of a random duration that counts down to a random encounter
Pause this timer if the player stops moving / is idle
Resume the same timer when the player starts moving again
This is what we shall implement.
Project setup
We’ll have a total of 3 scenes: one for the player, one for the “main” scene, and one that contains the code for the random encounters.
Player scene
I went a bit overkill with the player scene, since I set up animations with this spritesheet and added a sprinting mechanic. You don’t need to do this, but you can if you want.
This is how I have my player scene set up:
“player” is a CharacterBody2D
Writing player movement isn’t the focus of this article, so I’ll just put the code that handles player movement below.
If you want this code to work via copy/pasting, then you’ll have to set up the animations for the sprite and the input map yourself.
Also, this code is probably pretty convoluted; I just needed something quick.
Adding signals
We’ll emit a signal whenever the player moves or stops moving. These signals we shall connect in our random encounter script, which we’ll write in just a bit.
First, define two signals.
1signalplayer_moving2signalplayer_stopped
Then, at line 49, add player_stopped.emit() and at line 61, player_moving.emit(). That’s it!
Random encounter scene
There are only two things here: a regular ol’ Node and a timer (which I’ve named to “random_encounter_timer”). You’ll want to attach a script to the node.
Connecting the signals we defined
Firstly, create a “main” scene and add your player and this new random encounter scene in it. Then, select your player and find the signals we defined on the right-hand side.
Double click on each signal, and connect it to the script we created in the random encounter scene.
We’ll add one more signal to our random encounter script, which is the timer’s timeout signal. You can connect that in a similar fashion to the other signals. There should be 3 functions now in that script.
Now comes the actual meat of this article.
Remember what we read about random encounters above:
Start a timer of a random duration that counts down to a random encounter
Pause this timer if the player stops moving / is idle
Resume the same timer when the player starts moving again
Let’s start with the timer. Remember, we don’t want the timer to run if the player is idle, so it makes sense to start it when we get the signal of player movement.
The main issue now is that we’ll be starting a new timer every time the player moves. We want to allow any timer we set to complete first, before we start a new one. So, we’ll check if we currently have a timer going in the background.
1func_on_player_player_movement():
2# if no timer is running, start a new one.3ifrandom_encounter_timer.is_stopped():
4timer_duration=randf.range(0.5, 1.0) *25random_encounter_timer.start(timer_duration)
67# if we have a paused timer, resume it.8ifrandom_encounter_timer.is_paushed():
9random_encounter_timer.set_paused(false)