2016-03-10 23:01:04 +01:00
|
|
|
enum Direction {
|
2016-10-11 22:48:25 +02:00
|
|
|
//% block=right
|
2016-03-10 23:01:04 +01:00
|
|
|
Right,
|
2016-03-29 07:16:07 +02:00
|
|
|
//% block=left
|
2016-03-10 23:01:04 +01:00
|
|
|
Left
|
|
|
|
}
|
|
|
|
|
|
|
|
enum LedSpriteProperty {
|
2016-03-29 07:16:07 +02:00
|
|
|
//% block=x
|
2016-03-10 23:01:04 +01:00
|
|
|
X,
|
2016-03-29 07:16:07 +02:00
|
|
|
//% block=y
|
2016-03-10 23:01:04 +01:00
|
|
|
Y,
|
2016-03-29 07:16:07 +02:00
|
|
|
//% block=direction
|
2016-03-10 23:01:04 +01:00
|
|
|
Direction,
|
2016-03-29 07:16:07 +02:00
|
|
|
//% block=brightness
|
2016-03-10 23:01:04 +01:00
|
|
|
Brightness,
|
2016-03-29 07:16:07 +02:00
|
|
|
//% block=blink
|
2016-03-10 23:01:04 +01:00
|
|
|
Blink
|
|
|
|
}
|
|
|
|
|
2016-04-06 00:59:25 +02:00
|
|
|
/**
|
|
|
|
* A single-LED sprite game engine
|
|
|
|
*/
|
2017-01-20 05:55:31 +01:00
|
|
|
//% color=#008272 weight=32 icon="\uf11b"
|
2016-10-11 22:48:25 +02:00
|
|
|
//% advanced=true
|
2016-03-10 23:01:04 +01:00
|
|
|
namespace game {
|
2016-05-19 20:59:57 +02:00
|
|
|
let _score: number = 0;
|
|
|
|
let _life: number = 3;
|
|
|
|
let _startTime: number = 0;
|
|
|
|
let _endTime: number = 0;
|
|
|
|
let _isGameOver: boolean = false;
|
|
|
|
let _countdownPause: number = 0;
|
|
|
|
let _level: number = 1;
|
|
|
|
let _gameId: number = 0;
|
2017-12-14 19:34:32 +01:00
|
|
|
let _img: Image;
|
|
|
|
let _sprites: LedSprite[];
|
|
|
|
let _paused: boolean = false;
|
|
|
|
let _backgroundAnimation = false; // indicates if an auxiliary animation (and fiber) is already running
|
2016-05-19 20:59:57 +02:00
|
|
|
|
2016-03-10 23:01:04 +01:00
|
|
|
/**
|
|
|
|
* Creates a new LED sprite pointing to the right.
|
|
|
|
* @param x sprite horizontal coordinate, eg: 2
|
|
|
|
* @param y sprite vertical coordinate, eg: 2
|
|
|
|
*/
|
2017-12-14 19:34:32 +01:00
|
|
|
//% weight=60 blockGap=8 help=game/create-sprite
|
2016-03-10 23:01:04 +01:00
|
|
|
//% blockId=game_create_sprite block="create sprite at|x: %x|y: %y"
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
export function createSprite(x: number, y: number): LedSprite {
|
|
|
|
init();
|
|
|
|
let p = new LedSprite(x, y);
|
|
|
|
return p;
|
2016-05-19 20:59:57 +02:00
|
|
|
}
|
2016-03-10 23:01:04 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the current score
|
|
|
|
*/
|
2016-03-22 06:13:39 +01:00
|
|
|
//% weight=9 help=game/score
|
2016-03-10 23:01:04 +01:00
|
|
|
//% blockId=game_score block="score" blockGap=8
|
|
|
|
export function score(): number {
|
|
|
|
return _score;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-12-14 19:34:32 +01:00
|
|
|
* Adds points to the current score and shows an animation
|
2016-03-10 23:01:04 +01:00
|
|
|
* @param points amount of points to change, eg: 1
|
|
|
|
*/
|
2016-03-22 06:13:39 +01:00
|
|
|
//% weight=10 help=game/add-score
|
2016-03-10 23:01:04 +01:00
|
|
|
//% blockId=game_add_score block="change score by|%points" blockGap=8
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
export function addScore(points: number): void {
|
|
|
|
setScore(_score + points);
|
2017-12-14 19:34:32 +01:00
|
|
|
if (!_paused && !_backgroundAnimation) {
|
|
|
|
_backgroundAnimation = true;
|
|
|
|
control.inBackground(() => {
|
|
|
|
led.stopAnimation();
|
|
|
|
basic.showAnimation(`0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 1 1 1 0 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1 0 0 0 1 0 0 0 0 0
|
|
|
|
0 0 0 0 0 0 0 1 0 0 0 1 1 1 0 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1 0 0 0 1 0 0 0 0 0 0 0 0 0 0
|
|
|
|
0 0 1 0 0 0 1 1 1 0 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
|
|
|
|
0 0 0 0 0 0 0 1 0 0 0 1 1 1 0 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1 0 0 0 1 0 0 0 0 0 0 0 0 0 0
|
|
|
|
0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 1 1 1 0 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1 0 0 0 1 0 0 0 0 0`, 20);
|
|
|
|
_backgroundAnimation = false;
|
|
|
|
});
|
|
|
|
}
|
2016-03-10 23:01:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-12-14 19:34:32 +01:00
|
|
|
* Shows an animation, then starts a game countdown timer, which causes Game Over when it reaches 0
|
2016-03-10 23:01:04 +01:00
|
|
|
* @param ms countdown duration in milliseconds, eg: 10000
|
|
|
|
*/
|
2016-03-22 06:13:39 +01:00
|
|
|
//% weight=9 help=game/start-countdown
|
2016-03-10 23:01:04 +01:00
|
|
|
//% blockId=game_start_countdown block="start countdown|(ms) %duration" blockGap=8
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
export function startCountdown(ms: number): void {
|
|
|
|
if (checkStart()) {
|
|
|
|
basic.showAnimation(`1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 1 0 0 0 0 0
|
|
|
|
0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 1 1 0 0 0 0 0
|
|
|
|
1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 1 0 0 0 0 0
|
|
|
|
0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 1 1 0 0 0 0 0
|
|
|
|
1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 1 0 0 0 0 0`, 400);
|
|
|
|
_countdownPause = Math.max(500, ms);
|
|
|
|
_startTime = -1;
|
|
|
|
_endTime = input.runningTime() + _countdownPause;
|
2017-12-14 19:34:32 +01:00
|
|
|
_paused = false;
|
2016-03-10 23:01:04 +01:00
|
|
|
control.inBackground(() => {
|
|
|
|
basic.pause(_countdownPause);
|
|
|
|
gameOver();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-12-14 19:34:32 +01:00
|
|
|
* Displays a game over animation and the score.
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2016-03-22 06:13:39 +01:00
|
|
|
//% weight=8 help=game/game-over
|
2016-03-10 23:01:04 +01:00
|
|
|
//% blockId=game_game_over block="game over"
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
export function gameOver(): void {
|
|
|
|
if (!_isGameOver) {
|
|
|
|
_isGameOver = true;
|
|
|
|
unplugEvents();
|
|
|
|
led.stopAnimation();
|
|
|
|
led.setBrightness(255);
|
|
|
|
while (true) {
|
|
|
|
for (let i = 0; i < 8; i++) {
|
|
|
|
basic.clearScreen();
|
|
|
|
basic.pause(100);
|
|
|
|
basic.showLeds(`1 1 1 1 1
|
|
|
|
1 1 1 1 1
|
|
|
|
1 1 1 1 1
|
|
|
|
1 1 1 1 1
|
|
|
|
1 1 1 1 1`, 300);
|
|
|
|
}
|
|
|
|
basic.showAnimation(`1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1 0 0 1 1 0 0 0 1 0 0 0 0 0 0 0 0 0
|
|
|
|
1 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1 0 0 1 1 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
|
|
|
|
1 1 0 1 1 1 0 0 0 1 1 0 0 0 1 1 0 0 0 1 1 0 0 0 1 1 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
|
|
|
|
1 1 1 1 1 1 1 1 1 1 1 0 1 1 1 1 0 0 1 1 1 0 0 0 1 1 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
|
|
|
|
1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1 1 0 0 1 1 1 0 0 0 1 1 0 0 0 0 1 0 0 0 0 0`, 100);
|
|
|
|
for (let j = 0; j < 3; j++) {
|
|
|
|
basic.showString(" GAMEOVER ", 100);
|
|
|
|
showScore();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// already in game over mode in another fiber
|
|
|
|
while (true) {
|
|
|
|
basic.pause(10000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the current score value
|
2017-12-14 19:34:32 +01:00
|
|
|
* @param value new score value.
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2017-12-14 19:34:32 +01:00
|
|
|
//% blockId=game_set_score block="set score %points" blockGap=8
|
2016-03-22 06:13:39 +01:00
|
|
|
//% weight=10 help=game/set-score
|
2016-03-10 23:01:04 +01:00
|
|
|
export function setScore(value: number): void {
|
|
|
|
_score = Math.max(0, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the current life
|
|
|
|
*/
|
2016-03-22 06:13:39 +01:00
|
|
|
//% weight=10
|
2016-03-10 23:01:04 +01:00
|
|
|
export function life(): number {
|
|
|
|
return _life;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the current life value
|
2019-12-02 05:58:26 +01:00
|
|
|
* @param value current life value
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2019-12-02 05:58:26 +01:00
|
|
|
//% weight=10 help=game/set-life
|
|
|
|
//% blockId=game_set_life block="set life %value" blockGap=8
|
2016-03-10 23:01:04 +01:00
|
|
|
export function setLife(value: number): void {
|
|
|
|
_life = Math.max(0, value);
|
|
|
|
if (_life <= 0) {
|
|
|
|
gameOver();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-12-02 05:58:26 +01:00
|
|
|
* Add life points to the current life amount
|
|
|
|
* @param lives amount of lives to add
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2019-12-02 05:58:26 +01:00
|
|
|
//% weight=10 help=game/add-life
|
|
|
|
//% blockId=game_add_life block="add life %lives" blockGap=8
|
2016-03-10 23:01:04 +01:00
|
|
|
export function addLife(lives: number): void {
|
|
|
|
setLife(_life + lives);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the remaining time (since `start countdown`) or current time (since the device started or `start stopwatch`) in milliseconds.
|
|
|
|
*/
|
2016-03-22 06:13:39 +01:00
|
|
|
//% weight=10
|
2016-03-10 23:01:04 +01:00
|
|
|
export function currentTime(): number {
|
|
|
|
if (_endTime > 0) {
|
|
|
|
return Math.max(0, _endTime - input.runningTime());
|
|
|
|
} else {
|
|
|
|
return input.runningTime() - _startTime;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-12-02 05:58:26 +01:00
|
|
|
* Remove some life
|
|
|
|
* @param life amount of life to remove
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2019-12-02 05:58:26 +01:00
|
|
|
//% weight=10 help=game/remove-life
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2019-12-02 05:58:26 +01:00
|
|
|
//% blockId=game_remove_life block="remove life %life" blockGap=8
|
2016-03-10 23:01:04 +01:00
|
|
|
export function removeLife(life: number): void {
|
|
|
|
setLife(_life - life);
|
2019-12-02 05:58:26 +01:00
|
|
|
if (!_paused && !_backgroundAnimation) {
|
|
|
|
_backgroundAnimation = true;
|
2017-12-14 19:34:32 +01:00
|
|
|
control.inBackground(() => {
|
|
|
|
led.stopAnimation();
|
|
|
|
basic.showAnimation(`1 0 0 0 1 0 0 0 0 0 1 0 0 0 1 0 0 0 0 0
|
2016-03-10 23:01:04 +01:00
|
|
|
0 1 0 1 0 0 0 0 0 0 0 1 0 1 0 0 0 0 0 0
|
|
|
|
0 0 1 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0
|
|
|
|
0 1 0 1 0 0 0 0 0 0 0 1 0 1 0 0 0 0 0 0
|
|
|
|
1 0 0 0 1 0 0 0 0 0 1 0 0 0 1 0 0 0 0 0`, 40);
|
2019-12-02 05:58:26 +01:00
|
|
|
_backgroundAnimation = false;
|
2017-12-14 19:34:32 +01:00
|
|
|
});
|
2019-12-02 05:58:26 +01:00
|
|
|
}
|
2016-03-10 23:01:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Increments the level and display a message.
|
|
|
|
*/
|
2016-03-22 06:13:39 +01:00
|
|
|
//% weight=10
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
export function levelUp(): void {
|
|
|
|
_level = _level + 1;
|
|
|
|
basic.showString("LEVEL:", 150);
|
|
|
|
basic.showNumber(_level, 150);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the current level
|
|
|
|
*/
|
2016-03-22 06:13:39 +01:00
|
|
|
//% weight=10
|
2016-03-10 23:01:04 +01:00
|
|
|
export function level(): number {
|
|
|
|
return _level;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Starts a stopwatch timer. `current time` will return the elapsed time.
|
|
|
|
*/
|
2016-03-22 06:13:39 +01:00
|
|
|
//% weight=10
|
2016-03-10 23:01:04 +01:00
|
|
|
export function startStopwatch(): void {
|
|
|
|
_startTime = input.runningTime();
|
|
|
|
_endTime = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-12-02 05:58:26 +01:00
|
|
|
* Indicates if the game is still running. Returns `false` if the game is over or paused.
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2019-12-02 05:58:26 +01:00
|
|
|
//% weight=5 help=game/is-running
|
|
|
|
//% blockId=game_isrunning block="is running" blockGap=8
|
2016-03-10 23:01:04 +01:00
|
|
|
export function isRunning(): boolean {
|
2019-12-02 05:58:26 +01:00
|
|
|
return !_isGameOver && !_paused && !!_img;
|
2016-03-10 23:01:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Displays the score on the screen.
|
|
|
|
*/
|
2016-03-22 06:13:39 +01:00
|
|
|
//% weight=60
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
export function showScore(): void {
|
|
|
|
basic.showString(" SCORE ", 100);
|
|
|
|
basic.showNumber(_score, 150);
|
|
|
|
basic.showString(" ", 150);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-12-02 05:58:26 +01:00
|
|
|
* Indicates if the game is over and displaying the game over sequence.
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2019-12-02 05:58:26 +01:00
|
|
|
//% weight=7 help=game/is-game-over
|
|
|
|
//% blockId=game_isgameover block="is game over" blockGap=8
|
2016-03-10 23:01:04 +01:00
|
|
|
export function isGameOver(): boolean {
|
|
|
|
return _isGameOver;
|
|
|
|
}
|
|
|
|
|
2017-12-14 19:34:32 +01:00
|
|
|
/**
|
|
|
|
* Indicates if the game rendering is paused to allow other animations
|
|
|
|
*/
|
2019-12-02 05:58:26 +01:00
|
|
|
//% weight=6 help=game/is-paused
|
|
|
|
//% blockId=game_ispaused block="is paused" blockGap=8
|
2017-12-14 19:34:32 +01:00
|
|
|
export function isPaused(): boolean {
|
|
|
|
return _paused;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Pauses the game rendering engine to allow other animations
|
|
|
|
*/
|
|
|
|
//% blockId=game_pause block="pause"
|
|
|
|
//% advanced=true blockGap=8 help=game/pause
|
|
|
|
export function pause(): void {
|
|
|
|
plot()
|
|
|
|
_paused = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resumes the game rendering engine
|
|
|
|
*/
|
|
|
|
//% blockId=game_resume block="resume"
|
|
|
|
//% advanced=true blockGap=8 help=game/resumeP
|
|
|
|
export function resume(): void {
|
|
|
|
_paused = false;
|
|
|
|
plot();
|
|
|
|
}
|
|
|
|
|
2016-03-10 23:01:04 +01:00
|
|
|
/**
|
|
|
|
* returns false if game can't start
|
|
|
|
*/
|
|
|
|
function checkStart(): boolean {
|
|
|
|
if (_countdownPause > 0 || _startTime > 0) {
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function unplugEvents(): void {
|
2023-01-11 18:51:27 +01:00
|
|
|
input.onButtonEvent(Button.A, input.buttonEventValue(ButtonEvent.Click), () => { });
|
|
|
|
input.onButtonEvent(Button.B, input.buttonEventValue(ButtonEvent.Click), () => { });
|
|
|
|
input.onButtonEvent(Button.AB, input.buttonEventValue(ButtonEvent.Click), () => {
|
2016-03-10 23:01:04 +01:00
|
|
|
control.reset();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-12-14 19:34:32 +01:00
|
|
|
/**
|
|
|
|
* A game sprite rendered as a single LED
|
|
|
|
*/
|
|
|
|
//%
|
2016-03-10 23:01:04 +01:00
|
|
|
export class LedSprite {
|
|
|
|
private _x: number;
|
|
|
|
private _y: number;
|
|
|
|
private _dir: number;
|
|
|
|
private _brightness: number;
|
|
|
|
private _blink: number;
|
2017-12-14 19:34:32 +01:00
|
|
|
private _enabled: boolean;
|
2016-03-10 23:01:04 +01:00
|
|
|
|
|
|
|
constructor(x: number, y: number) {
|
2016-04-02 07:32:33 +02:00
|
|
|
this._x = Math.clamp(0, 4, x);
|
|
|
|
this._y = Math.clamp(0, 4, y);
|
2016-03-10 23:01:04 +01:00
|
|
|
this._dir = 90;
|
|
|
|
this._brightness = 255;
|
2017-12-14 19:34:32 +01:00
|
|
|
this._enabled = true;
|
2016-03-10 23:01:04 +01:00
|
|
|
init();
|
2017-12-14 19:34:32 +01:00
|
|
|
_sprites.push(this);
|
2016-03-10 23:01:04 +01:00
|
|
|
plot();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-12-14 19:34:32 +01:00
|
|
|
* Move a certain number of LEDs in the current direction
|
2016-03-10 23:01:04 +01:00
|
|
|
* @param this the sprite to move
|
|
|
|
* @param leds number of leds to move, eg: 1, -1
|
|
|
|
*/
|
2017-12-14 19:34:32 +01:00
|
|
|
//% weight=50 help=game/move
|
2016-03-10 23:01:04 +01:00
|
|
|
//% blockId=game_move_sprite block="%sprite|move by %leds" blockGap=8
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
public move(leds: number): void {
|
|
|
|
if (this._dir == 0) {
|
|
|
|
this._y = this._y - leds;
|
|
|
|
} else if (this._dir == 45) {
|
|
|
|
this._x = this._x + leds;
|
|
|
|
this._y = this._y - leds;
|
|
|
|
} else if (this._dir == 90) {
|
|
|
|
this._x = this._x + leds;
|
|
|
|
} else if (this._dir == 135) {
|
|
|
|
this._x = this._x + leds;
|
|
|
|
this._y = this._y + leds;
|
|
|
|
} else if (this._dir == 180) {
|
|
|
|
this._y = this._y + leds;
|
|
|
|
} else if (this._dir == -45) {
|
|
|
|
this._x = this._x - leds;
|
|
|
|
this._y = this._y - leds;
|
|
|
|
} else if (this._dir == -90) {
|
|
|
|
this._x = this._x - leds;
|
|
|
|
} else {
|
|
|
|
this._x = this._x - leds;
|
|
|
|
this._y = this._y + leds;
|
|
|
|
}
|
2016-04-02 07:32:33 +02:00
|
|
|
this._x = Math.clamp(0, 4, this._x);
|
|
|
|
this._y = Math.clamp(0, 4, this._y);
|
2016-03-10 23:01:04 +01:00
|
|
|
plot();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Go to this position on the screen
|
|
|
|
* @param this TODO
|
|
|
|
* @param x TODO
|
|
|
|
* @param y TODO
|
|
|
|
*/
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
public goTo(x: number, y: number): void {
|
|
|
|
this._x = x;
|
|
|
|
this._y = y;
|
2016-04-02 07:32:33 +02:00
|
|
|
this._x = Math.clamp(0, 4, this._x);
|
|
|
|
this._y = Math.clamp(0, 4, this._y);
|
2016-03-10 23:01:04 +01:00
|
|
|
plot();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-12-14 19:34:32 +01:00
|
|
|
* If touching the edge of the stage and facing towards it, then turn away.
|
2019-12-02 05:58:26 +01:00
|
|
|
* @param this the sprite to check for bounce
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2017-12-14 19:34:32 +01:00
|
|
|
//% weight=18 help=game/if-on-edge-bounce
|
2016-03-10 23:01:04 +01:00
|
|
|
//% blockId=game_sprite_bounce block="%sprite|if on edge, bounce"
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
public ifOnEdgeBounce(): void {
|
|
|
|
if (this._dir == 0 && this._y == 0) {
|
|
|
|
this._dir = 180;
|
|
|
|
} else if (this._dir == 45 && (this._x == 4 || this._y == 0)) {
|
|
|
|
if (this._x == 0 && this._y == 0) {
|
|
|
|
this._dir = -135;
|
|
|
|
} else if (this._y == 0) {
|
|
|
|
this._dir = 135;
|
|
|
|
} else {
|
|
|
|
this._dir = -45;
|
|
|
|
}
|
|
|
|
} else if (this._dir == 90 && this._x == 4) {
|
|
|
|
this._dir = -90;
|
|
|
|
} else if (this._dir == 135 && (this._x == 4 || this._y == 4)) {
|
|
|
|
if (this.x() == 4 && this.y() == 4) {
|
|
|
|
this._dir = -45;
|
|
|
|
} else if (this._y == 4) {
|
|
|
|
this._dir = 45;
|
|
|
|
} else {
|
|
|
|
this._dir = -135;
|
|
|
|
}
|
|
|
|
} else if (this._dir == 180 && this._y == 4) {
|
|
|
|
this._dir = 0;
|
|
|
|
} else if (this._dir == -45 && (this._x == 0 || this._y == 0)) {
|
|
|
|
if (this.x() == 0 && this.y() == 0) {
|
|
|
|
this._dir = 135;
|
|
|
|
} else if (this._y == 0) {
|
|
|
|
this._dir = -135;
|
|
|
|
} else {
|
|
|
|
this._dir = 45;
|
|
|
|
}
|
|
|
|
} else if (this._dir == -90 && this._x == 0) {
|
|
|
|
this._dir = 90;
|
|
|
|
} else if (this._dir == -135 && (this._x == 0 || this._y == 4)) {
|
|
|
|
if (this._x == 0 && this._y == 4) {
|
|
|
|
this._dir = 45;
|
|
|
|
} else if (this._y == 4) {
|
|
|
|
this._dir = -45;
|
|
|
|
} else {
|
|
|
|
this._dir = 135;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
plot();
|
|
|
|
}
|
2016-05-19 20:59:57 +02:00
|
|
|
|
2016-03-10 23:01:04 +01:00
|
|
|
/**
|
|
|
|
* Turn the sprite
|
2019-12-02 05:58:26 +01:00
|
|
|
* @param this the sprite to trun
|
2016-03-10 23:01:04 +01:00
|
|
|
* @param direction left or right
|
|
|
|
* @param degrees angle in degrees to turn, eg: 45, 90, 180, 135
|
|
|
|
*/
|
2017-12-14 19:34:32 +01:00
|
|
|
//% weight=49 help=game/turn
|
2016-03-10 23:01:04 +01:00
|
|
|
//% blockId=game_turn_sprite block="%sprite|turn %direction|by (°) %degrees"
|
2016-05-19 20:59:57 +02:00
|
|
|
public turn(direction: Direction, degrees: number) {
|
2016-03-10 23:01:04 +01:00
|
|
|
if (direction == Direction.Right)
|
|
|
|
this.setDirection(this._dir + degrees);
|
|
|
|
else
|
|
|
|
this.setDirection(this._dir - degrees);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Turn to the right (clockwise)
|
2019-12-02 05:58:26 +01:00
|
|
|
* @param this the sprite to turn
|
2016-03-10 23:01:04 +01:00
|
|
|
* @param degrees TODO
|
|
|
|
*/
|
|
|
|
public turnRight(degrees: number): void {
|
|
|
|
this.turn(Direction.Right, degrees);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Turn to the left (counter-clockwise)
|
2019-12-02 05:58:26 +01:00
|
|
|
* @param this the sprite to turn
|
2016-03-10 23:01:04 +01:00
|
|
|
* @param degrees TODO
|
|
|
|
*/
|
|
|
|
public turnLeft(degrees: number): void {
|
|
|
|
this.turn(Direction.Left, degrees);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets a property of the sprite
|
|
|
|
* @param property the name of the property to change
|
|
|
|
* @param the updated value
|
|
|
|
*/
|
2017-12-14 19:34:32 +01:00
|
|
|
//% weight=29 help=game/set
|
2016-03-10 23:01:04 +01:00
|
|
|
//% blockId=game_sprite_set_property block="%sprite|set %property|to %value" blockGap=8
|
2016-05-19 20:59:57 +02:00
|
|
|
public set(property: LedSpriteProperty, value: number) {
|
|
|
|
switch (property) {
|
2016-03-10 23:01:04 +01:00
|
|
|
case LedSpriteProperty.X: this.setX(value); break;
|
|
|
|
case LedSpriteProperty.Y: this.setY(value); break;
|
|
|
|
case LedSpriteProperty.Direction: this.setDirection(value); break;
|
|
|
|
case LedSpriteProperty.Brightness: this.setBrightness(value); break;
|
|
|
|
case LedSpriteProperty.Blink: this.setBlink(value); break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Changes a property of the sprite
|
|
|
|
* @param property the name of the property to change
|
|
|
|
* @param value amount of change, eg: 1
|
|
|
|
*/
|
2017-12-14 19:34:32 +01:00
|
|
|
//% weight=30 help=game/change
|
2016-03-10 23:01:04 +01:00
|
|
|
//% blockId=game_sprite_change_xy block="%sprite|change %property|by %value" blockGap=8
|
2016-05-19 20:59:57 +02:00
|
|
|
public change(property: LedSpriteProperty, value: number) {
|
|
|
|
switch (property) {
|
2016-03-10 23:01:04 +01:00
|
|
|
case LedSpriteProperty.X: this.changeXBy(value); break;
|
|
|
|
case LedSpriteProperty.Y: this.changeYBy(value); break;
|
|
|
|
case LedSpriteProperty.Direction: this.changeDirectionBy(value); break;
|
|
|
|
case LedSpriteProperty.Brightness: this.changeBrightnessBy(value); break;
|
|
|
|
case LedSpriteProperty.Blink: this.changeBlinkBy(value); break;
|
|
|
|
}
|
|
|
|
}
|
2016-05-19 20:59:57 +02:00
|
|
|
|
2016-03-10 23:01:04 +01:00
|
|
|
/**
|
|
|
|
* Gets a property of the sprite
|
|
|
|
* @param property the name of the property to change
|
|
|
|
*/
|
2017-12-14 19:34:32 +01:00
|
|
|
//% weight=28 help=game/get
|
2016-03-10 23:01:04 +01:00
|
|
|
//% blockId=game_sprite_property block="%sprite|%property"
|
2016-05-19 20:59:57 +02:00
|
|
|
public get(property: LedSpriteProperty) {
|
|
|
|
switch (property) {
|
2016-03-10 23:01:04 +01:00
|
|
|
case LedSpriteProperty.X: return this.x();
|
|
|
|
case LedSpriteProperty.Y: return this.y();
|
|
|
|
case LedSpriteProperty.Direction: return this.direction()
|
|
|
|
case LedSpriteProperty.Brightness: return this.brightness();
|
|
|
|
case LedSpriteProperty.Blink: return this.blink();
|
|
|
|
default: return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the direction of the current sprite, rounded to the nearest multiple of 45
|
2019-12-02 05:58:26 +01:00
|
|
|
* @param this the sprite to set direction for
|
|
|
|
* @param degrees new direction in degrees
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
public setDirection(degrees: number): void {
|
2019-12-02 05:58:26 +01:00
|
|
|
this._dir = (Math.floor(degrees / 45) % 8) * 45;
|
2016-03-10 23:01:04 +01:00
|
|
|
if (this._dir <= -180) {
|
|
|
|
this._dir = this._dir + 360;
|
|
|
|
} else if (this._dir > 180) {
|
|
|
|
this._dir = this._dir - 360;
|
|
|
|
}
|
|
|
|
plot();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reports the ``x`` position of a sprite on the LED screen
|
|
|
|
* @param this TODO
|
|
|
|
*/
|
|
|
|
public x(): number {
|
|
|
|
return this._x;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reports the ``y`` position of a sprite on the LED screen
|
|
|
|
* @param this TODO
|
|
|
|
*/
|
|
|
|
public y(): number {
|
|
|
|
return this._y;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reports the current direction of a sprite
|
|
|
|
* @param this TODO
|
|
|
|
*/
|
|
|
|
public direction(): number {
|
|
|
|
return this._dir;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the ``x`` position of a sprite
|
|
|
|
* @param this TODO
|
|
|
|
* @param x TODO
|
|
|
|
*/
|
|
|
|
public setX(x: number): void {
|
|
|
|
this.goTo(x, this._y);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the ``y`` position of a sprite
|
|
|
|
* @param this TODO
|
|
|
|
* @param y TODO
|
|
|
|
*/
|
|
|
|
public setY(y: number): void {
|
|
|
|
this.goTo(this._x, y);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Changes the ``y`` position by the given amount
|
|
|
|
* @param this TODO
|
|
|
|
* @param y TODO
|
|
|
|
*/
|
|
|
|
public changeYBy(y: number): void {
|
|
|
|
this.goTo(this._x, this._y + y);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Changes the ``x`` position by the given amount
|
|
|
|
* @param this TODO
|
|
|
|
* @param x TODO
|
|
|
|
*/
|
|
|
|
public changeXBy(x: number): void {
|
|
|
|
this.goTo(this._x + x, this._y);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-12-14 19:34:32 +01:00
|
|
|
* Reports true if sprite has the same position as specified sprite
|
2019-12-02 05:58:26 +01:00
|
|
|
* @param this the sprite to check overlap or touch
|
|
|
|
* @param other the other sprite to check overlap or touch
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2017-12-14 19:34:32 +01:00
|
|
|
//% weight=20 help=game/is-touching
|
2019-12-02 05:58:26 +01:00
|
|
|
//% blockId=game_sprite_touching_sprite block="is %sprite|touching %other" blockGap=8
|
2016-03-10 23:01:04 +01:00
|
|
|
public isTouching(other: LedSprite): boolean {
|
2017-12-14 19:34:32 +01:00
|
|
|
return this._enabled && other._enabled && this._x == other._x && this._y == other._y;
|
2016-03-10 23:01:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reports true if sprite is touching an edge
|
2019-12-02 05:58:26 +01:00
|
|
|
* @param this the sprite to check for an edge contact
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2017-12-14 19:34:32 +01:00
|
|
|
//% weight=19 help=game/is-touching-edge
|
2019-12-02 05:58:26 +01:00
|
|
|
//% blockId=game_sprite_touching_edge block="is %sprite|touching edge" blockGap=8
|
2016-03-10 23:01:04 +01:00
|
|
|
public isTouchingEdge(): boolean {
|
2019-12-02 05:58:26 +01:00
|
|
|
return this._enabled && (this._x == 0 || this._x == 4 || this._y == 0 || this._y == 4);
|
2016-03-10 23:01:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Turns on the sprite (on by default)
|
2017-12-14 19:34:32 +01:00
|
|
|
* @param this the sprite
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
|
|
|
public on(): void {
|
|
|
|
this.setBrightness(255);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Turns off the sprite (on by default)
|
2017-12-14 19:34:32 +01:00
|
|
|
* @param this the sprite
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
|
|
|
public off(): void {
|
|
|
|
this.setBrightness(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the ``brightness`` of a sprite
|
2017-12-14 19:34:32 +01:00
|
|
|
* @param this the sprite
|
|
|
|
* @param brightness the brightness from 0 (off) to 255 (on), eg: 255.
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
public setBrightness(brightness: number): void {
|
2016-04-02 07:32:33 +02:00
|
|
|
this._brightness = Math.clamp(0, 255, brightness);
|
2016-03-10 23:01:04 +01:00
|
|
|
plot();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reports the ``brightness` of a sprite on the LED screen
|
2017-12-14 19:34:32 +01:00
|
|
|
* @param this the sprite
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2017-12-14 19:34:32 +01:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
public brightness(): number {
|
|
|
|
let r: number;
|
|
|
|
return this._brightness;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Changes the ``y`` position by the given amount
|
2017-12-14 19:34:32 +01:00
|
|
|
* @param this the sprite
|
|
|
|
* @param value the value to change brightness
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
|
|
|
public changeBrightnessBy(value: number): void {
|
|
|
|
this.setBrightness(this._brightness + value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Changes the ``direction`` position by the given amount by turning right
|
|
|
|
* @param this TODO
|
|
|
|
* @param angle TODO
|
|
|
|
*/
|
|
|
|
public changeDirectionBy(angle: number): void {
|
|
|
|
this.turnRight(angle);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-12-14 19:34:32 +01:00
|
|
|
* Deletes the sprite from the game engine. The sprite will no longer appear on the screen or interact with other sprites.
|
|
|
|
* @param this sprite to delete
|
2016-03-10 23:01:04 +01:00
|
|
|
*/
|
2019-12-02 05:58:26 +01:00
|
|
|
//% weight=59 blockGap=8 help=game/delete
|
|
|
|
//% blockId="game_delete_sprite" block="delete %this(sprite)"
|
2017-12-14 19:34:32 +01:00
|
|
|
public delete(): void {
|
|
|
|
this._enabled = false;
|
|
|
|
if (_sprites.removeElement(this))
|
|
|
|
plot();
|
2016-03-10 23:01:04 +01:00
|
|
|
}
|
|
|
|
|
2019-12-02 05:58:26 +01:00
|
|
|
/**
|
|
|
|
* Reports whether the sprite has been deleted from the game engine.
|
|
|
|
*/
|
|
|
|
//% weight=58 help=game/is-deleted
|
|
|
|
//% blockId="game_sprite_is_deleted" block="is %sprite|deleted"
|
|
|
|
public isDeleted(): boolean {
|
|
|
|
return !this._enabled;
|
|
|
|
}
|
|
|
|
|
2016-03-10 23:01:04 +01:00
|
|
|
/**
|
|
|
|
* Sets the blink duration interval in millisecond.
|
|
|
|
* @param sprite TODO
|
|
|
|
* @param ms TODO
|
|
|
|
*/
|
|
|
|
public setBlink(ms: number): void {
|
2016-04-02 07:32:33 +02:00
|
|
|
this._blink = Math.clamp(0, 10000, ms);
|
2016-03-10 23:01:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Changes the ``blink`` duration by the given amount of millisecons
|
|
|
|
* @param this TODO
|
|
|
|
* @param ms TODO
|
|
|
|
*/
|
|
|
|
public changeBlinkBy(ms: number): void {
|
|
|
|
this.setBlink(this._blink + ms);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reports the ``blink`` duration of a sprite
|
|
|
|
* @param this TODO
|
|
|
|
*/
|
|
|
|
public blink(): number {
|
|
|
|
return this._blink;
|
|
|
|
}
|
|
|
|
|
|
|
|
//% weight=-1
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
public _plot(now: number) {
|
|
|
|
let ps = this
|
|
|
|
if (ps._brightness > 0) {
|
|
|
|
let r = 0;
|
|
|
|
if (ps._blink > 0) {
|
2019-12-02 05:58:26 +01:00
|
|
|
r = Math.floor(now / ps._blink) % 2;
|
2016-03-10 23:01:04 +01:00
|
|
|
}
|
|
|
|
if (r == 0) {
|
2017-12-14 19:34:32 +01:00
|
|
|
_img.setPixelBrightness(ps._x, ps._y, _img.pixelBrightness(ps._x, ps._y) + ps._brightness);
|
2016-03-10 23:01:04 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function init(): void {
|
2017-12-14 19:34:32 +01:00
|
|
|
if (_img) return;
|
|
|
|
const img = images.createImage(
|
|
|
|
`0 0 0 0 0
|
2016-03-10 23:01:04 +01:00
|
|
|
0 0 0 0 0
|
|
|
|
0 0 0 0 0
|
|
|
|
0 0 0 0 0
|
|
|
|
0 0 0 0 0`);
|
2017-12-14 19:34:32 +01:00
|
|
|
_sprites = (<LedSprite[]>[]);
|
|
|
|
basic.forever(() => {
|
|
|
|
basic.pause(30);
|
|
|
|
plot();
|
|
|
|
if (game.isGameOver()) {
|
|
|
|
basic.pause(600);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
_img = img;
|
2016-03-10 23:01:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Plots the current sprites on the screen
|
|
|
|
*/
|
2016-08-22 17:48:48 +02:00
|
|
|
//% parts="ledmatrix"
|
2016-03-10 23:01:04 +01:00
|
|
|
function plot(): void {
|
2017-12-14 19:34:32 +01:00
|
|
|
if (game.isGameOver() || game.isPaused() || !_img || _backgroundAnimation) {
|
2016-03-10 23:01:04 +01:00
|
|
|
return;
|
|
|
|
}
|
2017-12-14 19:34:32 +01:00
|
|
|
// ensure greyscale mode
|
|
|
|
const dm = led.displayMode();
|
2019-12-02 05:58:26 +01:00
|
|
|
if (dm != DisplayMode.Greyscale)
|
2017-12-14 19:34:32 +01:00
|
|
|
led.setDisplayMode(DisplayMode.Greyscale);
|
|
|
|
// render sprites
|
|
|
|
const now = input.runningTime();
|
|
|
|
_img.clear();
|
|
|
|
for (let i = 0; i < _sprites.length; i++) {
|
|
|
|
_sprites[i]._plot(now);
|
|
|
|
}
|
|
|
|
_img.plotImage(0);
|
2016-03-10 23:01:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets an invalid sprite; used to initialize locals.
|
|
|
|
*/
|
|
|
|
//% weight=0
|
|
|
|
export function invalidSprite(): LedSprite {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2019-12-02 05:58:26 +01:00
|
|
|
|