264 lines
6.5 KiB
GDScript
264 lines
6.5 KiB
GDScript
extends GridMap
|
|
|
|
const ExplodingLine = preload("res://ExplodingLine/ExplodingLine.tscn")
|
|
const Tetromino = preload("res://Tetrominos/Tetromino.gd")
|
|
const TetroI = preload("res://Tetrominos/TetroI.tscn")
|
|
const TetroJ = preload("res://Tetrominos/TetroJ.tscn")
|
|
const TetroL = preload("res://Tetrominos/TetroL.tscn")
|
|
const TetroO = preload("res://Tetrominos/TetroO.tscn")
|
|
const TetroS = preload("res://Tetrominos/TetroS.tscn")
|
|
const TetroT = preload("res://Tetrominos/TetroT.tscn")
|
|
const TetroZ = preload("res://Tetrominos/TetroZ.tscn")
|
|
|
|
const NB_LINES = 20
|
|
const NB_COLLUMNS = 10
|
|
|
|
const EMPTY_CELL = -1
|
|
|
|
const NEXT_POSITION = Vector3(13, 16, 0)
|
|
const START_POSITION = Vector3(5, 20, 0)
|
|
const HOLD_POSITION = Vector3(-5, 16, 0)
|
|
|
|
const movements = {
|
|
"move_right": Vector3(1, 0, 0),
|
|
"move_left": Vector3(-1, 0, 0),
|
|
"soft_drop": Vector3(0, -1, 0)
|
|
}
|
|
|
|
const SCORES = [
|
|
[0, 4, 1],
|
|
[1, 8, 2],
|
|
[3, 12],
|
|
[5, 16],
|
|
[8]
|
|
]
|
|
const LINES_CLEARED_NAMES = ["", "SINGLE", "DOUBLE", "TRIPLE", "TETRIS"]
|
|
const T_SPIN_NAMES = ["", "T-SPIN", "MINI T-SPIN"]
|
|
|
|
const LINE_CLEAR_MIDI_CHANNELS = [2, 6]
|
|
|
|
var next_piece = random_piece()
|
|
var current_piece
|
|
var held_piece
|
|
var current_piece_held = false
|
|
|
|
var autoshift_action = ""
|
|
|
|
var exploding_lines = []
|
|
var random_bag = []
|
|
var playing = true
|
|
|
|
var level = 0
|
|
var goal = 0
|
|
var score = 0
|
|
|
|
func _ready():
|
|
randomize()
|
|
for y in range(NB_LINES):
|
|
exploding_lines.append(ExplodingLine.instance())
|
|
add_child(exploding_lines[y])
|
|
exploding_lines[y].translation = Vector3(NB_COLLUMNS/2, y, 1)
|
|
resume()
|
|
new_level()
|
|
|
|
func new_level():
|
|
level += 1
|
|
goal += 5 * level
|
|
$DropTimer.wait_time = pow(0.8 - ((level - 1) * 0.007), level - 1)
|
|
if level > 15:
|
|
$LockDelay.wait_time = 0.5 * pow(0.9, level-15)
|
|
print("LEVEL ", level, " Goal ", goal)
|
|
new_piece()
|
|
|
|
func random_piece():
|
|
if not random_bag:
|
|
random_bag = [
|
|
TetroI, TetroJ, TetroL, TetroO,
|
|
TetroS, TetroT, TetroZ
|
|
]
|
|
var choice = randi() % random_bag.size()
|
|
var piece = random_bag[choice].instance()
|
|
random_bag.remove(choice)
|
|
add_child(piece)
|
|
return piece
|
|
|
|
func new_piece():
|
|
current_piece = next_piece
|
|
current_piece.translation = START_POSITION
|
|
current_piece.emit_trail(true)
|
|
autoshift_action = ""
|
|
next_piece = random_piece()
|
|
next_piece.translation = NEXT_POSITION
|
|
if move(movements["soft_drop"]):
|
|
$DropTimer.start()
|
|
$LockDelay.start()
|
|
current_piece_held = false
|
|
else:
|
|
game_over()
|
|
|
|
func _process(delta):
|
|
if autoshift_action:
|
|
if not Input.is_action_pressed(autoshift_action):
|
|
$AutoShiftDelay.stop()
|
|
$AutoShiftTimer.stop()
|
|
autoshift_action = ""
|
|
if Input.is_action_just_pressed("pause"):
|
|
if playing:
|
|
pause()
|
|
else:
|
|
resume()
|
|
if playing:
|
|
process_actions()
|
|
|
|
func process_actions():
|
|
for action in movements:
|
|
if action != autoshift_action:
|
|
if Input.is_action_pressed(action):
|
|
move(movements[action])
|
|
autoshift_action = action
|
|
$AutoShiftTimer.stop()
|
|
$AutoShiftDelay.start()
|
|
if Input.is_action_just_pressed("hard_drop"):
|
|
while move(movements["soft_drop"]):
|
|
pass
|
|
lock_piece()
|
|
if Input.is_action_just_pressed("rotate_clockwise"):
|
|
rotate(Tetromino.CLOCKWISE)
|
|
if Input.is_action_just_pressed("rotate_counterclockwise"):
|
|
rotate(Tetromino.COUNTERCLOCKWISE)
|
|
if Input.is_action_just_pressed("hold"):
|
|
hold()
|
|
|
|
func _on_AutoShiftDelay_timeout():
|
|
if playing and autoshift_action:
|
|
move(movements[autoshift_action])
|
|
$AutoShiftTimer.start()
|
|
|
|
func _on_AutoShiftTimer_timeout():
|
|
if playing and autoshift_action:
|
|
move(movements[autoshift_action])
|
|
|
|
func is_free_cell(position):
|
|
return (
|
|
0 <= position.x and position.x < NB_COLLUMNS
|
|
and position.y >= 0
|
|
and get_cell_item(position.x, position.y, 0) == GridMap.INVALID_CELL_ITEM
|
|
)
|
|
|
|
func possible_positions(initial_positions, movement):
|
|
var position
|
|
var test_positions = []
|
|
for i in range(4):
|
|
position = initial_positions[i] + movement
|
|
if is_free_cell(position):
|
|
test_positions.append(position)
|
|
if test_positions.size() == Tetromino.NB_MINOES:
|
|
return test_positions
|
|
else:
|
|
return []
|
|
|
|
func move(movement):
|
|
if current_piece.move(movement):
|
|
$LockDelay.start()
|
|
return true
|
|
else:
|
|
return false
|
|
|
|
func rotate(direction):
|
|
if current_piece.rotate(direction):
|
|
$LockDelay.start()
|
|
return true
|
|
else:
|
|
return false
|
|
|
|
func _on_DropTimer_timeout():
|
|
move(movements["soft_drop"])
|
|
|
|
func _on_LockDelay_timeout():
|
|
if not move(movements["soft_drop"]):
|
|
lock_piece()
|
|
|
|
func lock_piece():
|
|
for mino in current_piece.minoes:
|
|
set_cell_item(current_piece.to_global(mino.translation).x, current_piece.to_global(mino.translation).y, 0, 0)
|
|
remove_child(current_piece)
|
|
line_clear()
|
|
|
|
func line_clear():
|
|
var NB_MINOES
|
|
var lines_cleared = 0
|
|
for y in range(NB_LINES-1, -1, -1):
|
|
NB_MINOES = 0
|
|
for x in range(NB_COLLUMNS):
|
|
if get_cell_item(x, y, 0) == 0:
|
|
NB_MINOES += 1
|
|
if NB_MINOES == NB_COLLUMNS:
|
|
for y2 in range(y, NB_LINES+2):
|
|
for x in range(NB_COLLUMNS):
|
|
set_cell_item(x, y2, 0, get_cell_item(x, y2+1, 0))
|
|
lines_cleared += 1
|
|
exploding_lines[y].restart()
|
|
if lines_cleared or current_piece.t_spin:
|
|
var s = SCORES[lines_cleared][current_piece.t_spin]
|
|
score += 100 * s
|
|
goal -= s
|
|
print(T_SPIN_NAMES[current_piece.t_spin], ' ', LINES_CLEARED_NAMES[lines_cleared], " Score ", score)
|
|
|
|
if lines_cleared == Tetromino.NB_MINOES:
|
|
for channel in LINE_CLEAR_MIDI_CHANNELS:
|
|
$MidiPlayer.channel_status[channel].vomume = 127
|
|
$MidiPlayer/LineCLearTimer.wait_time = 0.86
|
|
else:
|
|
for channel in LINE_CLEAR_MIDI_CHANNELS:
|
|
$MidiPlayer.channel_status[channel].vomume = 100
|
|
$MidiPlayer/LineCLearTimer.wait_time = 0.43
|
|
$MidiPlayer.unmute_channels(LINE_CLEAR_MIDI_CHANNELS)
|
|
$MidiPlayer/LineCLearTimer.start()
|
|
if goal <= 0:
|
|
new_level()
|
|
else:
|
|
new_piece()
|
|
|
|
func hold():
|
|
if not current_piece_held:
|
|
current_piece.emit_trail(false)
|
|
if held_piece:
|
|
var tmp = held_piece
|
|
held_piece = current_piece
|
|
current_piece = tmp
|
|
current_piece.translation = START_POSITION
|
|
current_piece.emit_trail(true)
|
|
else:
|
|
held_piece = current_piece
|
|
new_piece()
|
|
held_piece.translation = HOLD_POSITION
|
|
current_piece_held = true
|
|
|
|
func resume():
|
|
playing = true
|
|
$DropTimer.start()
|
|
$LockDelay.start()
|
|
$MidiPlayer.resume()
|
|
$MidiPlayer.mute_channels(LINE_CLEAR_MIDI_CHANNELS)
|
|
print("RESUME")
|
|
|
|
func pause():
|
|
playing = false
|
|
$DropTimer.stop()
|
|
$LockDelay.stop()
|
|
$MidiPlayer.stop()
|
|
print("PAUSE")
|
|
|
|
func game_over():
|
|
pause()
|
|
print("GAME OVER")
|
|
|
|
func _notification(what):
|
|
if what == MainLoop.NOTIFICATION_WM_FOCUS_OUT:
|
|
pause()
|
|
if what == MainLoop.NOTIFICATION_WM_FOCUS_IN:
|
|
resume()
|
|
|
|
func _on_LineCLearTimer_timeout():
|
|
$MidiPlayer.mute_channels(LINE_CLEAR_MIDI_CHANNELS)
|