This is the documentation for the subsection of Triangle.js that is directly connected to gameplay. For the full documentation, see the main README.
This page assumes you have created a Client named client
that is in a room.
client.game.start
Runs when a game is started. Contains information about first to, win by, and the players that are playing.
client.game.end
Runs when a game ends gracefully. Contains information about the winner, the players, and the scoreboard.
client.game.abort
Runs when a game is aborted through the /abort
command.
client.game.over
Runs in any game over scenario. This includes when a player disconnects, when a game is aborted, and when a game ends gracefully. If the game ends gracefully, contains information about how the game ended.
client.game.round.start
Runs when a round starts. Contains a ticker callback and a reference to the game's internal engine.
Opponent information is available through client.game.opponents
.
client.game.round.end
Runs when a round ends. Contains the gameid of the winning player.
client.room.start();
You should listen for the client.game.start
event. In the handler, this is where you should initialize your gameplay engine, keyfinder, etc.
client.on("client.game.start", (data) => {
// initialize your engine here
console.log(
"playing a game against",
data.players.map((p) => p.name).join(", ")
);
});
You should listen for the client.game.round.start
event. In the handler, this is where you should pass in your tick callback. You can also process engine data here.
client.on("client.game.round.start", ([tick, engine]) => {
console.log(
"the game board has a width of",
engine.board.width,
"and a height of",
engine.board.height
);
tick(tickerCallback);
});
The tick callback is called every frame. It can be asynchronous, but you should optimize it to be as fast as possible. The tick callback takes in the engine and incoming events, and should return data about keys pressed, etc.
const tickerCallback = async ({
gameid,
frame,
events,
engine
}: Types.Game.Tick.In): Promise<Types.Game.Tick.Out> => {
// First, process incoming events
let garbageAdded = false;
for (const event of events) {
if (event.type === "garbage") garbageAdded = true;
}
if (garbageAdded)
// If your solve maintains an internal state, you should update it here
// Tell your solver (what you use to calculate moves) that garbage has been tanked and needs to be updated
solver.update(engine);
// For example, playing 1 piece every second.
let keys: Types.Game.Tick.Keypress[] = [];
if (frame % 60 === 0) {
// If your solver does not maintain an internal state, you can pass in the engine to the solver
const move = solver.getMove();
// Any move must have a set of keys
const solvedKeys = solver.findKeys(move);
let runningSubframe = 0;
solvedKeys.forEach((key) => {
keys.push({
frame,
type: "keydown",
data: {
key,
subframe: runningSubframe
}
});
if (key === "softDrop") {
runningSubframe += 0.1;
}
keys.push({
frame: frame,
type: "keyup",
data: {
key,
subframe: runningSubframe
}
});
});
}
return { keys };
};
If you need to run something right after a frame is processed internally, you can return a runAfter property from the tick callback. This will run after the frame is processed, but before the tick is called. This is useful for things like logging.
const tickerCallback = async ({
gameid,
frame,
events,
engine
}: Types.Game.Tick.In): Promise<Types.Game.Tick.Out> => {
// your code here
return { keys, runAfter: [() => console.log("Frame processed")] };
};