Rust语言俄罗斯方块(漂亮的界面案例+详细的代码解说+完美运行)

tetris-demo A Tetris example written in Rust using Piston in under 500 lines of code

项目地址: https://gitcode.com/gh_mirrors/te/tetris-demo

项目介绍

"Tetris Example in Rust, v2" 是一个用Rust语言编写的俄罗斯方块游戏示例。这个项目不仅是一个简单的游戏实现,更是一个展示Rust编程基础的绝佳范例。通过414行代码,开发者可以深入了解Rust的基本语法和编程思想。此外,项目还提供了一个清晰的Git历史记录,展示了功能的逐步迭代过程,非常适合初学者和有经验的开发者学习参考。

完整代码

use piston_window::{WindowSettings, PistonWindow, Event, RenderEvent, PressEvent};
use piston_window::{Rectangle, DrawState, Context, Graphics};
use piston_window::{Button, Key};use rand::Rng;use std::time::{Duration, Instant};
use std::collections::HashMap;enum DrawEffect<'a> {None,Darker,Flash(&'a Vec<i8>),
}#[derive(Copy, Clone)]
enum Color {Red, Green, Blue, Magenta, Cyan, Yellow, Orange,
}#[derive(Default, Clone)]
struct Board(HashMap<(i8, i8), Color>);impl Board {fn new(v: &[(i8, i8)], color: Color) -> Self {Board(v.iter().cloned().map(|(x, y)| ((x, y), color)).collect())}fn modified<F>(&self, f: F) -> Selfwhere F: Fn((i8, i8)) -> (i8, i8){Board(self.0.iter().map(|((x, y), color)| (f((*x, *y)), *color)).collect())}fn modified_filter<F>(&self, f: F) -> Selfwhere F: Fn((i8, i8)) -> Option<(i8, i8)>{Board(self.0.iter().filter_map(|((x, y), color)| f((*x, *y)).map(|p| (p, *color))).collect())}fn transposed(&self) -> Self {self.modified(|(ox, oy)| (oy, ox))}fn mirrored_y(&self) -> Self {self.modified(|(ox, oy)| (ox, -oy))}fn rotated(&self) -> Self {self.mirrored_y().transposed()}fn rotated_counter(&self) -> Self {self.rotated().rotated().rotated()}fn negative_shift(&self) -> (i8, i8) {use std::cmp::min;self.0.keys().into_iter().cloned().fold((0, 0), |(mx, my), (ox, oy)| (min(mx, ox), min(my, oy)))}fn shifted(&self, (x, y): (i8, i8)) -> Self {self.modified(|(ox, oy)| (ox + x, oy + y))}fn merged(&self, other: &Board) -> Option<Self> {let mut hashmap = HashMap::new();hashmap.extend(other.0.iter());hashmap.extend(self.0.iter());if hashmap.len() != self.0.len() + other.0.len() {return None;}Some(Self(hashmap))}fn contained(&self, x: i8, y: i8) -> bool {self.0.keys().into_iter().cloned().fold(true, |b, (ox, oy)| b && ox < x && oy < y && ox >= 0 && oy >= 0)}fn whole_lines(&self, x: i8, y: i8) -> Vec<i8> {let mut idxs = vec![];for oy in 0 .. y {if (0 .. x).filter_map(|ox| self.0.get(&(ox, oy))).count() == x as usize {idxs.push(oy)}}idxs}fn kill_line(&self, y: i8) -> Self {self.modified_filter(|(ox, oy)|if oy > y {Some((ox, oy))} else if oy == y {None} else {Some((ox, oy + 1))})}fn render<'a, G>(&self,metrics: &Metrics,c: &Context,g: &mut G,draw_effect: DrawEffect<'a>,)where G: Graphics{let mut draw = |color, rect: [f64; 4]| {Rectangle::new(color).draw(rect, &DrawState::default(), c.transform, g);};for x in 0 .. metrics.board_x {for y in 0 .. metrics.board_y {let block_pixels = metrics.block_pixels as f64;let border_size = block_pixels / 20.0;let outer = [block_pixels * (x as f64), block_pixels * (y as f64), block_pixels, block_pixels];let inner = [outer[0] + border_size, outer[1] + border_size,outer[2] - border_size * 2.0, outer[3] - border_size * 2.0];draw([0.2, 0.2, 0.2, 1.0], outer);draw([0.1, 0.1, 0.1, 1.0], inner);if let Some(color) = self.0.get(&(x as i8, y as i8)) {let code = match color {Color::Red     => [1.0, 0.0, 0.0, 1.0],Color::Green   => [0.0, 1.0, 0.0, 1.0],Color::Blue    => [0.5, 0.5, 1.0, 1.0],Color::Magenta => [1.0, 0.0, 1.0, 1.0],Color::Cyan    => [0.0, 1.0, 1.0, 1.0],Color::Yellow  => [1.0, 1.0, 0.0, 1.0],Color::Orange  => [1.0, 0.5, 0.0, 1.0],};draw(code, outer);let code = [code[0]*0.8, code[1]*0.8, code[2]*0.8, code[3]];draw(code, inner);}match draw_effect {DrawEffect::None => {},DrawEffect::Flash(lines) => {if lines.contains(&(y as i8)) {draw([1.0, 1.0, 1.0, 0.5], outer);}}DrawEffect::Darker => {draw([0.0, 0.0, 0.0, 0.9], outer);}}}}}
}#[derive(Default)]
struct Metrics {block_pixels: usize,board_x: usize,board_y: usize,
}impl Metrics {fn resolution(&self) -> [u32; 2] {[(self.board_x * self.block_pixels) as u32,(self.board_y * self.block_pixels) as u32]}
}enum State {Flashing(isize, Instant, Vec<i8>),Falling(Board),GameOver,
}struct Game {board: Board,metrics: Metrics,state: State,shift: (i8, i8),possible_pieces: Vec<Board>,time_since_fall: Instant,
}impl Game {fn new(metrics: Metrics) -> Self {Self {metrics,board: Default::default(),state: State::Falling(Default::default()),time_since_fall: Instant::now(),shift: (0, 0),possible_pieces: vec![Board::new(&[(0, 0), (0, 1), (1, 0), (1, 1), ][..], Color::Red),Board::new(&[(0, 0), (1, 0), (1, 1), (2, 0), ][..], Color::Green),Board::new(&[(0, 0), (1, 0), (2, 0), (3, 0), ][..], Color::Blue),Board::new(&[(0, 0), (1, 0), (2, 0), (0, 1), ][..], Color::Orange),Board::new(&[(0, 0), (1, 0), (2, 0), (2, 1), ][..], Color::Yellow),Board::new(&[(0, 0), (1, 0), (1, 1), (2, 1), ][..], Color::Cyan),Board::new(&[(1, 0), (2, 0), (0, 1), (1, 1), ][..], Color::Magenta),]}}fn new_falling(&mut self) {let mut rng = rand::thread_rng();let idx = rng.gen_range(0, self.possible_pieces.len());self.state = State::Falling(self.possible_pieces[idx].clone());self.shift = (0, 0);if self.board.merged(&self.falling_shifted()).is_none() {self.state = State::GameOver;} else {for _ in 0 .. rng.gen_range(0, 4usize) {self.rotate(false)}}}fn render(&self, window: &mut PistonWindow, event: &Event) {window.draw_2d(event, |c, g, _| {let (board, draw_effect) = match &self.state {State::Flashing(stage, _, lines) => {let draw_effect = if *stage % 2 == 0 {DrawEffect::None} else {DrawEffect::Flash(lines)};(self.board.clone(), draw_effect)}State::GameOver => (self.board.clone(), DrawEffect::Darker),State::Falling(_) => (self.board.merged(&self.falling_shifted()).unwrap(), DrawEffect::None),};board.render(&self.metrics, &c, g, draw_effect);});}fn falling_shifted(&self) -> Board {match &self.state {State::Falling(state_falling) => {state_falling.shifted(self.shift)}State::GameOver { ..  } => panic!(),State::Flashing { ..  } => panic!(),}}fn progress(&mut self) {match &mut self.state {State::Falling(_) => {if self.time_since_fall.elapsed() <= Duration::from_millis(700) {return;}self.move_falling(0, 1);self.time_since_fall = Instant::now();}State::Flashing(stage, last_stage_switch, lines) => {if last_stage_switch.elapsed() <= Duration::from_millis(50) {return;}if *stage < 18 {*stage += 1;*last_stage_switch = Instant::now();return;} else {for idx in lines {self.board = self.board.kill_line(*idx);}self.new_falling()}}State::GameOver { } => {},}}fn move_falling(&mut self, x: i8, y: i8) {let falling = self.falling_shifted().shifted((x, y));let merged = self.board.merged(&falling);let contained = falling.contained(self.metrics.board_x as i8,self.metrics.board_y as i8);if merged.is_some() && contained {// Allow the movementself.shift.0 += x;self.shift.1 += y;return}if let (0, 1) = (x, y) {self.board = self.board.merged(&self.falling_shifted()).unwrap();let completed = self.board.whole_lines(self.metrics.board_x as i8,self.metrics.board_y as i8);if completed.is_empty() {self.new_falling();} else {self.state = State::Flashing(0, Instant::now(), completed);}}}fn on_press(&mut self, args: &Button) {match args {Button::Keyboard(key) => { self.on_key(key); }_ => {},}}fn on_key(&mut self, key: &Key) {match &mut self.state {State::Flashing {..} => {},State::Falling {..} => {let movement = match key {Key::Right => Some((1, 0)),Key::Left => Some((-1, 0)),Key::Down => Some((0, 1)),_ => None,};if let Some(movement) = movement {self.move_falling(movement.0, movement.1);return;}match key {Key::Up => self.rotate(false),Key::NumPad5 => self.rotate(true),_ => return,}}State::GameOver { } => {match key {Key::Return => {self.board.0.clear();self.new_falling();},_ => return,}},}}fn rotate(&mut self, counter: bool) {match &mut self.state {State::Falling(state_falling) => {let rotated = if counter {state_falling.rotated()} else {state_falling.rotated_counter()};let (x, y) = rotated.negative_shift();let falling = rotated.shifted((-x, -y));for d in &[(0, 0), (-1, 0)] {let mut shift = self.shift;shift.0 += d.0;shift.1 += d.1;if let Some(merged) = self.board.merged(&falling.shifted(shift)) {if merged.contained(self.metrics.board_x as i8,self.metrics.board_y as i8){// Allow the rotation*state_falling = falling;self.shift = shift;return}}}}State::GameOver {..} => panic!(),State::Flashing {..} => panic!(),}}
}fn main() {let metrics = Metrics {block_pixels: 20,board_x: 8,board_y: 20,};let mut window: PistonWindow = WindowSettings::new("Tetris", metrics.resolution()).exit_on_esc(true).build().unwrap();let mut game = Game::new(metrics);game.new_falling();while let Some(e) = window.next() {game.progress();if let Some(_) = e.render_args() {game.render(&mut window, &e);}if let Some(args) = e.press_args() {game.on_press(&args);}}
}

以下是将上述代码拆分为几段并分别给出注释的内容:

1. 导入相关库和模块

rust

// 导入 `piston_window` 库中的相关模块,用于创建游戏窗口、处理事件、图形绘制等功能
use piston_window::{WindowSettings, PistonWindow, Event, RenderEvent, PressEvent};
use piston_window::{Rectangle, DrawState, Context, Graphics};
use piston_window::{Button, Key};// 导入 `rand` 库,用于生成随机数
use rand::Rng;// 导入标准库中用于处理时间的模块
use std::time::{Duration, Instant};
// 导入标准库中用于处理哈希表数据结构的模块
use std::collections::HashMap;

这段代码主要是导入了实现俄罗斯方块游戏所需的各种库和模块。piston_window相关模块用于创建游戏窗口、处理窗口事件以及进行图形绘制等操作。rand库用于生成随机数,以便在游戏中随机生成方块形状等。std::time中的DurationInstant用于处理时间相关的操作,比如控制方块下落的时间间隔等。HashMap则用于存储游戏板上方块的位置和颜色等信息。

2. 定义绘制效果和颜色枚举

rust

// `DrawEffect` 枚举定义了绘制方块时可能的效果
// `'a` 是生命周期参数,用于确保引用的有效性
enum DrawEffect<'a> {// 无特殊绘制效果None,// 使方块颜色变深的绘制效果Darker,// 使指定行的方块闪烁的绘制效果,接受一个 `i8` 类型向量的引用Flash(&'a Vec<i8>),
}// `Color` 枚举定义了方块可能的颜色
#[derive(Copy, Clone)]
enum Color {Red, Green, Blue, Magenta, Cyan, Yellow, Orange,
}

这里定义了两个枚举类型。DrawEffect枚举用于指定在绘制游戏板上的方块时可能采用的不同效果,比如无效果、颜色变深或者使某些行的方块闪烁等。Color枚举则明确了方块可能出现的各种颜色,以便在游戏中区分不同的方块形状或状态。

3. 定义游戏板结构体及相关方法

rust

// `Board` 结构体用于表示游戏板的状态
// 它包含一个 `HashMap`,用于存储游戏板上每个方块的位置(以 `(i8, i8)` 坐标表示)和对应的颜色
#[derive(Default, Clone)]
struct Board(HashMap<(i8, i8), Color>);// `Board` 结构体的实现块,包含了一系列处理游戏板相关操作的方法
impl Board {// `new` 方法用于创建一个新的 `Board` 实例// 接受一个坐标切片 `v` 和一个颜色 `color`,将每个坐标位置设置为指定的颜色fn new(v: &[(i8, i8)], color: Color) -> Self {Board(v.iter().cloned().map(|(x, y)| ((x, y), color)).collect())}// `modified` 方法根据传入的函数 `f` 对游戏板上的每个方块位置进行变换// `f` 函数接受一个坐标并返回一个新的坐标,用于修改方块在游戏板上的位置fn modified<F>(&self, f: F) -> Selfwhere F: Fn((i8, i8)) -> (i8, i8){Board(self.0.iter().map(|((x, y), color)| (f((*x, *y)), *color)).collect())}// `modified_filter` 方法根据传入的函数 `f` 对游戏板上的方块位置进行过滤和变换// `f` 函数接受一个坐标并返回一个可选的新坐标,如果返回 `Some`,则保留该方块并应用变换;如果返回 `None`,则移除该方块fn modified_filter<F>(&self, f: F) -> Selfwhere F: Fn((i8, i8)) -> Option<(i8, i8)>{Board(self.0.iter().filter_map(|((x, y), color)| f((*x, y)).map(|p| (p, *color))).collect())}// `transposed` 方法用于对游戏板进行转置操作,即将 `x` 与 `y` 坐标互换fn transposed(&self) -> Self {self.modified(|(ox, oy)| (oy, ox))}// `mirrored_y` 方法用于对游戏板在 `y` 轴上进行镜像翻转操作,即将 `y` 坐标取反fn mirrored_y(&self) -> Self {self.modified(|(ox, oy)| (ox, -oy))}// `rotated` 方法用于对游戏板进行顺时针旋转操作// 先在 `y` 轴上镜像翻转,然后再进行转置操作来实现旋转效果fn rotated(&self) -> Self {self.mirrored_y().transposed()}// `rotated_counter` 方法用于对游戏板进行逆时针旋转操作// 通过连续三次调用 `rotated` 方法来实现逆时针旋转效果fn rotated_counter(&self) -> Self {self.rotated().rotated().rotated()}// `negative_shift` 方法用于获取游戏板上所有方块位置的最小 `x` 和 `y` 坐标值// 返回一个 `(i8, i8)` 类型的元组,表示最小的偏移量fn negative_shift(&self) -> (i8, i8) {use std::cmp::min;self.0.keys().into_iter().cloned().fold((0, 0), |(mx, my), (ox, oy)| (min(mx, ox), min(my, oy)))}// `shifted` 方法用于根据传入的偏移量 `(x, y)` 对游戏板上的所有方块位置进行平移操作fn shifted(&self, (x, y): (i8, y)) -> Self {self.modified(|(ox, oy)| (ox + x, oy + y))}// `merged` 方法用于将当前游戏板与另一个游戏板 `other` 进行合并操作// 如果合并后的哈希表长度不等于两个游戏板原来哈希表长度之和,说明有位置冲突,返回 `None`;否则返回合并后的新游戏板 `Some(Self)`fn merged(&self, other: &Board) -> Option<Self> {let mut hashmap = HashMap::new();hashmap.extend(other.0.iter());hashmap.extend(self.0.iter());if hashmap.len()!= self.0.len() + other.0.len() {return None;}Some(Self(hashmap))}// `contained` 方法用于检查给定的坐标 `(x, y)` 是否在游戏板的有效范围内// 如果所有方块的坐标都满足小于 `x` 且小于 `y`,且大于等于 `0`,则返回 `true`,否则返回 `false`fn contained(&self, x: i8, y: i8) -> bool {self.0.keys().into_iter().cloned().fold(true, |b, (ox, oy)| b && ox < x && oy < y && ox >= 0 && oy >= 0)}// `whole_lines` 方法用于查找游戏板上给定范围内完整的行// 遍历 `y` 坐标从 `0` 到 `y - 1` 的行,对于每一行,如果该行在 `x` 坐标从 `0` 到 `x - 1` 的范围内所有方块都存在(通过 `filter_map` 和 `count` 来判断),则将该行的 `y` 坐标添加到结果向量中fn whole_lines(&self, x: i8, y: i8) -> Vec<i8> {let mut idxs = vec![];for oy in 0.. y {if (0.. x).filter_map(|ox| self.0.get(&(ox, oy))).count() == x as usize {idxs.push(oy)}}idxs}// `kill_line` 方法用于清除游戏板上指定的行 `y`// 通过 `modified_filter` 方法根据条件对游戏板上的方块位置进行过滤,保留不在指定行 `y` 的方块,并将在指定行上方的方块 `y` 坐标加 `1`,从而实现清除行的效果fn kill_line(&self, y: i8) -> Self {self.modified_filter(|(ox, oy)|if oy > y {Some((ox, oy))} else if oy == y {None} else {Some((ox, oy + 1))})}// `render` 方法用于在给定的图形上下文 `c` 和图形绘制对象 `g` 下,根据指定的绘制效果 `draw_effect` 绘制游戏板// 遍历游戏板的每个方块位置,根据是否有方块以及方块的颜色,按照不同的绘制效果进行绘制操作fn render<'a, G>(&self,metrics: &Metrics,c: &Context,g: &mut G,draw_effect: DrawEffect<'a>,)where G: Graphics{let mut draw = |color, rect: [f64; 4]| {Rectangle::new(color).draw(rect, &DrawState::default(), c.transform, g);};for x in 0.. metrics.board_x {for y in 0.. metrics.board_y {let block_pixels = metrics.block_pixels as f64;let border_size = block_pixels / 20.0;let outer = [block_pixels * (x as f64), block_pixels * (y as f64), block_pixels, block_pixels];let inner = [outer[0] + border_size, outer[1] + border_size,outer[2] - border_size * 2.0, outer[3] - border_size * 2.0];draw([0.2, 0.2, 0.2, 1.0], outer);draw([0.1, 0.1, 0.1, 1.0], inner);if let Some(color) = self.0.get(&(x as i8, y as i8)) {let code = match color {Color::Red     => [1.0, 0.0, 0.0, 1.0],Color::Green   => [0.0, 1.0, 0.0, 1.0],Color::Blue    => [0.5, 0.5, 1.0, 1.0],Color::Magenta => [1.0, 0.0, 1.0, 1.0],Color::Cyan    => [0.0, 1.0, 1.0, 1.0],Color::Yellow  => [1.0, 1.0, 0.0, 1.0],Color::Orange  => [1.0, 0.5, 0.0, 1.0],};draw(code, outer);let code = [code[0]*0.8, code[1]*0.8, code[2]*0.8, code[3]];draw(code, inner);}match draw_effect {DrawEffect::None => {},DrawEffect::Flash(lines) => {if lines.contains(&(y as i8)) {draw([1.0, 1.0, 1.0, 0.5], outer);}}DrawEffect::Darker => {draw([0.0, 0.0, 0.0, 0.9], outer);}}}}}
}

这部分代码定义了Board结构体来表示游戏板的状态,其中使用HashMap存储每个方块在游戏板上的位置和对应的颜色。同时,为Board结构体实现了一系列方法,用于对游戏板进行各种操作,比如创建新的游戏板实例、对游戏板上的方块位置进行变换(平移、旋转、翻转等)、合并两个游戏板、检查坐标是否在游戏板范围内、查找完整的行以及清除指定行等,并且还实现了绘制游戏板的方法,根据不同的绘制效果和方块颜色进行绘制。

4. 定义游戏度量结构体及相关方法

rust

// `Metrics` 结构体用于存储游戏的一些度量信息,如每个方块的像素大小、游戏板的行数和列数等
#[derive(Default)]
struct Metrics {block_pixels: usize,board_x: usize,board_y: usize,
}// `Metrics` 结构体的实现块,包含了一个用于计算游戏窗口分辨率的方法
impl Metrics {// `resolution` 方法根据游戏板的行数、列数和每个方块的像素大小计算并返回游戏窗口的分辨率,以 `[u32; 2]` 类型的数组表示(分别为宽度和高度)fn resolution(&self) -> [u32; 2] {[(self.board_x * self.block_pixels) as u32,(self.board_y * self.block_pixels) as u32]}
}

这里定义了Metrics结构体,用于存储游戏相关的度量信息,比如每个方块在屏幕上显示的像素大小、游戏板的行数和列数等。并且为该结构体实现了resolution方法,用于根据存储的度量信息计算并返回游戏窗口的分辨率,以便后续创建合适大小的游戏窗口。

5. 定义游戏状态枚举和游戏结构体及相关方法

rust

// `State` 枚举用于表示游戏的不同状态
enum State {// 闪烁状态,用于处理满行消除时的闪烁效果// 包含当前闪烁阶段(`isize` 类型)、上次阶段切换的时间点(`Instant` 类型)以及需要闪烁的行索引向量(`Vec<i8>` 类型)Flashing(isize, Instant, Vec<i8>),// 方块下落状态,包含当前正在下落的方块信息(`Board` 类型)Falling(Board),// 游戏结束状态GameOver,
}// `Game` 结构体用于表示整个游戏的状态和逻辑
struct Game {// 游戏板对象,用于存储游戏板的当前状态board: Board,// 游戏度量信息对象,用于存储游戏的相关度量参数metrics: Metrics,// 当前游戏状态,使用 `State` 枚举表示state: State,// 方块在游戏板上的偏移量,以 `(i8, i8)` 坐标表示shift: (i8, i8),// 可能出现的方块形状列表,每个元素都是一个 `Board` 类型,表示不同形状的方块possible_pieces: Vec<Board>,// 记录上次方块下落的时间点,用于控制方块下落的速度time_since_fall: Instant,
}// `Game` 结构体的实现块,包含了一系列处理游戏逻辑的方法
impl Game {// `new` 方法用于创建一个新的 `Game` 实例// 接受一个 `Metrics` 类型的参数,用于初始化游戏的度量信息,并设置游戏板、状态、偏移量等初始值fn new(metrics: Metrics) -> Self {Self {metrics,board: Default::default(),state: State::Falling(Default::default()),time_since_fall: Instant::now(),shift: (0, 0),possible_pieces: vec![Board::new(&[(0, 0), (0, 1), (1, 0), (1, 1), ][..], Color::Red),Board::new(&[(0, 0), (1, 0), (1, 1), (2, 0), ][..], Color::Green),Board::new(&[(0, 0), (1, 0), (2, 0), (3, 0), ][..], Color::Blue),Board::new(&[(0, 0), (1, 0), (2, 0), (0, 1), ][..], Color::Orange),Board::new(&[(0, 0), (1, 0), (2, 0), (2, 1), ][..], Color::Yellow),Board::new(&[(0, 0), (1, 0), (1, 1), (2, 1), ][..], Color::Cyan),Board::new(&[(1, 0), (2, 0), (0, 1), (1, 1), ][..], Color::Magenta),]}}// `new_falling` 方法用于生成一个新的下落方块// 通过随机数生成器选择一个可能的方块形状,并设置为当前下落状态// 如果当前游戏

6. Game结构体相关方法

rust

// `falling_shifted` 方法用于获取当前下落状态下经过偏移后的方块信息
// 根据当前游戏状态中的下落方块和偏移量,返回偏移后的方块
// 如果当前状态不是下落状态,则触发 `panic!`,表示程序出现错误情况
fn falling_shifted(&self) -> Board {match &self.state {State::Falling(state_falling) => {state_falling.shifted(self.shift)}State::GameOver {..  } => panic!(),State::Flashing {..  } => panic!(),}
}

作用

  • 此方法的目的是根据游戏当前状态获取经过偏移后的正在下落的方块信息。它通过匹配当前游戏状态,如果处于Falling状态,就利用Board结构体的shifted方法对下落方块按照当前的偏移量self.shift进行偏移操作,从而得到准确位置的下落方块。
  • 若当前状态不是Falling状态(如GameOverFlashing状态),则触发panic!,这是因为该方法假设只有在方块处于下落状态时才会被调用以获取正确偏移后的方块,其他状态下调用此方法是不符合预期逻辑的,所以通过panic!来提示程序出现了错误情况。

rust

// `progress` 方法用于推进游戏的进程,根据当前游戏状态执行不同的操作
fn progress(&mut self) {match &mut self.state {State::Falling(_) => {if self.time_since_fall.elapsed() <= Duration::from_millis(700) {return;}self.move_falling(0, 1);self.time_since_fall = Instant::now();}State::Flashing(stage, last_stage_switch, lines) => {if last_stage_switch.elapsed() <= Duration::from_millis(50) {return;}if *stage < 18 {*stage += 1;*last_stage_switch = Instant::now();return;} else {for idx in lines {self.board = self.board.kill_line(*idx);}self.new_falling();}}State::GameOver { } => {},}
}

作用

  • 该方法是游戏逻辑的核心部分之一,用于根据游戏当前所处的不同状态来推进游戏进程。
  • 当游戏处于Falling状态时,它首先检查从上一次方块下落至今所经过的时间是否小于等于 700 毫秒,如果是,则直接返回,不进行任何操作,这是为了控制方块下落的速度,避免下落过快。若时间超过了限制,就调用move_falling方法让方块向下移动一格(传入参数(0, 1)表示在x方向移动 0 格,在y方向移动 1 格),然后更新time_since_fall为当前时间,以便下一次准确计算方块下落间隔时间。
  • 当游戏处于Flashing状态时,它先检查从上一次阶段切换至今所经过的时间是否小于等于 50 毫秒,如果是则返回。若超过了限制,并且当前闪烁阶段*stage小于 18,就将闪烁阶段加 1,并更新last_stage_switch为当前时间,继续进行闪烁效果的展示。而当闪烁阶段达到 18 时,意味着闪烁效果结束,此时会遍历需要闪烁的行索引lines,通过调用boardkill_line方法清除这些行,然后调用new_falling方法生成一个新的下落方块,继续游戏流程。
  • 当游戏处于GameOver状态时,此方法不执行任何操作,因为游戏已经结束,无需再进行其他逻辑处理。

rust

// `move_falling` 方法用于移动正在下落的方块
fn move_falling(&mut self, x: i8, y: i8) {let falling = self.falling_shifted().shifted((x, y));let merged = self.board.merged(&falling);let contained = falling.contained(self.metrics.board_x as i8,self.metrics.board_y as i8);if merged.is_some() && contained {// Allow the movementself.shift.0 += x;self.shift.1 += y;return}if let (0, 1) = (x, y) {self.board = self.board.merged(&self.falling_shifted()).unwrap();let completed = self.board.whole_lines(self.metrics.board_x as i8,self.metrics.board_y as i8);if completed.is_empty() {self.new_falling();} else {self.state = State::Flashing(0, Instant::now(), completed);}}
}

作用

  • 此方法用于处理正在下落的方块的移动操作。首先,它通过falling_shifted方法获取当前经过偏移的下落方块,然后再根据传入的参数(x, y)对该方块进行进一步的偏移操作,得到新的下落方块位置falling
  • 接着,它检查新位置的方块能否与游戏板self.board进行合并(通过merged方法)以及是否在游戏板的有效范围内(通过contained方法)。如果这两个条件都满足,说明方块可以移动到新位置,那么就更新方块在游戏板上的偏移量self.shift,完成方块的移动操作并返回。
  • 如果传入的参数(x, y)(0, 1),表示方块是向下移动一格,此时会先将当前下落方块与游戏板进行合并(通过merged方法获取合并后的游戏板),然后检查游戏板上是否有完整的行(通过whole_lines方法)。如果没有完整的行,就调用new_falling方法生成一个新的下落方块继续游戏;如果有完整的行,就将游戏状态设置为Flashing状态,并传入当前时间和需要闪烁的行索引,开始处理满行消除的闪烁效果。

rust

// `on_press` 方法用于处理按键按下事件,根据按下的按钮类型进行不同的处理
fn on_press(&mut self, args: &Button) {match args {Button::Keyboard(key) => { self.on_key(key); }_ => {},}
}

作用

  • 该方法是游戏处理输入事件的入口点之一,用于接收一个Button类型的参数,表示按下的按钮信息。它通过匹配按钮类型,如果是键盘按钮(Button::Keyboard),就调用on_key方法进一步处理具体的键盘按键操作;如果不是键盘按钮,则不进行任何操作,直接返回。

rust

// `on_key` 方法用于处理具体的键盘按键操作,根据当前游戏状态和按下的键盘按键执行不同的操作
fn on_key(&mut self, key: &Key) {match &mut self.state {State::Flashing {..} => {},State::Falling {..} => {let movement = match key {Key::Right => Some((1, 0)),Key::Left => Some((-1, 0)),Key::Down => Some((0, 1)),_ => None,};if let Some(movement) = movement {self.move_falling(movement.0, movement.1);return;}match key {Key::Up => self.rotate(false),Key::NumPad5 => self.rotate(true),_ => return,}}State::GameOver { } => {match key {Key::Return => {self.board.0.clear();self.new_falling();},_ => return,}},}
}

作用

  • 此方法根据当前游戏状态和按下的具体键盘按键来执行相应的游戏操作。
  • 当游戏处于Flashing状态时,不执行任何操作,因为在满行消除闪烁期间,通常不希望玩家进行其他操作干扰闪烁效果的处理。
  • 当游戏处于Falling状态时,首先通过匹配按下的键盘按键来确定方块的移动方向或旋转操作。如果按下的是Key::RightKey::LeftKey::Down,就分别对应方块向右、向左或向下移动一格的操作,通过调用move_falling方法并传入相应的移动参数来实现方块的移动。如果按下的是Key::UpKey::NumPad5,则分别对应方块的逆时针或顺时针旋转操作,通过调用rotate方法并传入相应的旋转方向参数来实现方块的旋转。
  • 当游戏处于GameOver状态时,只有当按下Key::Return键时,会清除游戏板上的所有方块(通过board.0.clear()),然后调用new_falling方法生成一个新的下落方块,重新开始游戏;按下其他键则不执行任何操作,直接返回。

rust

// `rotate` 方法用于旋转正在下落的方块
fn rotate(&mut self, counter: bool) {match &mut self.state {State::Falling(state_falling) => {let rotated = if counter {state_falling.rotated()} else {state_falling.rotated_counter()}let (x, y) = rotated.negative_shift();let falling = rotated.shifted((-x, -y));for d in &[(0, 0), (-1, 0)] {let mut shift = self.shift;shift.0 += d.0;shift.1 += d.1;if let Some(merged) = self.board.merged(&falling.shifted(shift)) {if merged.contained(self.metrics.board_x as i8,self.metrics.board_y as i8){// Allow the rotation*state_falling = falling;self.shift = shift;return}}}}State::GameOver {..} => panic!(),State::Flashing {..} => panic!(),}
}

作用

  • 该方法用于处理正在下落的方块的旋转操作。当游戏处于Falling状态时,首先根据传入的参数counter确定是顺时针还是逆时针旋转方块。如果countertrue,就通过state_falling.rotated()方法对下落方块进行顺时针旋转;如果counterfalse,则通过state_failing.rotated_counter()方法对下落方块进行逆时针旋转。
  • 然后获取旋转后方块的最小偏移量(x, y)(通过negative_shift方法),并将旋转后的方块按照这个偏移量的相反数进行反向偏移,得到falling,以便将方块放置在合适的位置进行后续的合并操作。
  • 接着,通过遍历[(0, 0), (-1, 0)]这两个偏移量,对旋转后的方块在不同的偏移位置尝试与游戏板进行合并操作(通过merged方法),并检查合并后的方块是否在游戏板的有效范围内(通过contained方法)。如果在某个偏移位置满足这两个条件,说明方块可以旋转到该位置,就更新下落方块的状态*state_fallingfalling,并更新方块在游戏板上的偏移量self.shift,完成方块的旋转操作并返回。
  • 如果当前游戏状态不是Falling状态(如GameOverFlashing状态),则触发panic!,因为该方法假设只有在方块处于下落状态时才会被调用以进行正确的旋转操作,其他状态下调用此方法是不符合预期逻辑的,所以通过panic!来提示程序出现了错误情况。

7. main函数

rust

fn main() {let metrics = Metrics {block_pixels: 20,board_x: 8,board_y: 20,};let mut window: PistonWindow = WindowSettings::new("Tetris", metrics.resolution()).exit_on_esc(true).build().unwrap();let mut game = Game::new(metrics);game.new_falling();while let Some(e) = window.next() {game.progress();if let Some(_) = e.render_args() {game.render(&mut window, &e);}if let Some(args) = e.press_args() {game.on_press(&args);}}
}

作用

  • main函数中,首先创建了一个Metrics结构体实例,设置了每个方块的像素大小为 20,游戏板的行数为 20,列数为 8,这些参数定义了游戏的基本布局和显示效果。
  • 然后使用PistonWindow库创建了一个游戏窗口,通过WindowSettings设置窗口的标题为 "Tetris",并根据metrics.resolution()计算出的窗口分辨率来设置窗口大小,同时设置了按下Esc键时退出游戏。
  • 接着创建了一个Game结构体实例,并调用new_falling方法生成第一个下落方块,开始游戏。
  • 之后进入一个循环,不断从游戏窗口获取事件(通过window.next())。对于每个获取到的事件:
    • 如果事件有渲染相关的参数(通过e.render_args()),就调用game.render方法在窗口中绘制游戏的当前状态。
    • 如果事件有按键按下相关的参数(通过e.press_args()),就调用game.on_press方法处理按键按下事件。
  • 这样,游戏就能够不断地根据玩家的操作和游戏自身的逻辑进行更新和绘制,实现俄罗斯方块游戏的基本功能。

综上所述,这段代码通过定义一系列结构体、枚举和相关方法,完整地实现了一个俄罗斯方块游戏的核心逻辑,包括游戏板的操作、方块的生成、移动、旋转、满行消除以及游戏状态的管理和窗口事件的处理等功能。

效果如下

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/887567.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring Boot 与 Spring Cloud Alibaba 版本兼容对照

版本选择要点 Spring Boot 3.x 与 Spring Cloud Alibaba 2022.0.x Spring Boot 3.x 基于 Jakarta EE&#xff0c;javax.* 更换为 jakarta.*。 需要使用 Spring Cloud 2022.0.x 和 Spring Cloud Alibaba 2022.0.x。 Alibaba 2022.0.x 对 Spring Boot 3.x 的支持在其发行说明中…

(免费送源码)计算机毕业设计原创定制:Java+ssm+JSP+Ajax SSM棕榈校园论坛的开发

摘要 随着计算机科学技术的高速发展,计算机成了人们日常生活的必需品&#xff0c;从而也带动了一系列与此相关产业&#xff0c;是人们的生活发生了翻天覆地的变化&#xff0c;而网络化的出现也在改变着人们传统的生活方式&#xff0c;包括工作&#xff0c;学习&#xff0c;社交…

Ubuntu Opencv 源码包安装

说明&#xff1a; ubuntu20.04 建议 使用 opencv-4.6.0版本 ubuntu18.04 建议 使用 opencv-4.5.2-版本 安装包准备 1、下载源码包 OpenCV官网 下载相关版本源码 Sources # 克隆方式 OpenCV 源码git clone https://github.com/opencv/opencv.gitcd opencvgit checkout 4.5.2 …

Linux 下自动化之路:达梦数据库定期备份并推送至 GitLab 全攻略

目录 环境准备 生成SSH 密钥对 数据库备份并推送到gitlab脚本 设置定时任务 环境准备 服务器要有安装达梦数据库&#xff08;达梦安装这里就不示例了&#xff09;&#xff0c;git 安装Git 1、首先&#xff0c;确保包列表是最新的&#xff0c;运行以下命令&#xff1a; …

<项目代码>YOLOv8 停车场空位识别<目标检测>

YOLOv8是一种单阶段&#xff08;one-stage&#xff09;检测算法&#xff0c;它将目标检测问题转化为一个回归问题&#xff0c;能够在一次前向传播过程中同时完成目标的分类和定位任务。相较于两阶段检测算法&#xff08;如Faster R-CNN&#xff09;&#xff0c;YOLOv8具有更高的…

Spring Boot 集成 Knife4j 的 Swagger 文档

在开发微服务应用时&#xff0c;API 文档的生成和维护是非常重要的一环。Swagger 是一个非常流行的 API 文档工具&#xff0c;可以帮助我们自动生成 RESTful API 的文档&#xff0c;并提供了一个友好的界面供开发者测试 API。本文将介绍如何在 Spring Boot 项目中集成 Knife4j …

微信小程序中会议列表页面的前后端实现

题外话&#xff1a;想通过集成腾讯IM来解决即时聊天的问题&#xff0c;如果含语音视频&#xff0c;腾讯组件一年5万起步&#xff0c;贵了&#xff01;后面我们改为自己实现这个功能&#xff0c;这里只是个总结而已。 图文会诊需求 首先是个图文列表界面 同个界面可以查看具体…

git(Linux)

1.git 三板斧 基本准备工作&#xff1a; 把远端仓库拉拉取到本地了 .git --> 本地仓库 git在提交的时候&#xff0c;只会提交变化的部分 就可以在当前目录下新增代码了 test.c 并没有被仓库管理起来 怎么添加&#xff1f; 1.1 git add test.c 也不算完全添加到仓库里面&…

【动手学电机驱动】STM32-FOC(8)MCSDK Profiler 电机参数辨识

STM32-FOC&#xff08;1&#xff09;STM32 电机控制的软件开发环境 STM32-FOC&#xff08;2&#xff09;STM32 导入和创建项目 STM32-FOC&#xff08;3&#xff09;STM32 三路互补 PWM 输出 STM32-FOC&#xff08;4&#xff09;IHM03 电机控制套件介绍 STM32-FOC&#xff08;5&…

5G NR:带宽与采样率的计算

100M 带宽是122.88Mhz sampling rate这是我们都知道的&#xff0c;那它是怎么来的呢&#xff1f; 采样率 子载波间隔 * 采样长度 38.211中对于Tc的定义&#xff0c; 在LTE是定义了Ts&#xff0c;在NR也就是5G定义了Tc。 定义这个单位会对我们以后工作中的计算至关重要。 就是在…

【湿度数据处理】中国地面气候资料日值数据集(V3.0)(MATLAB全代码)

【湿度数据处理】中国地面气候资料日值数据集 处理1:数据范围筛选处理2:缺测数据筛查处理3:缺测数据插补参考基于此博客完成各要素数据提取后-【数据集处理】中国地面气候资料日值数据集(V3.0)(含MATLAB全代码),进行后续数据筛选及缺测处理,此处以湿度数据为例。 提取到的…

MySQL面试-1

InnoDB中ACID的实现 先说一下原子性是怎么实现的。 事务要么失败&#xff0c;要么成功&#xff0c;不能做一半。聪明的InnoDB&#xff0c;在干活儿之前&#xff0c;先将要做的事情记录到一个叫undo log的日志文件中&#xff0c;如果失败了或者主动rollback&#xff0c;就可以通…

大数据-231 离线数仓 - DWS 层、ADS 层的创建 Hive 执行脚本

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; Java篇开始了&#xff01; 目前开始更新 MyBatis&#xff0c;一起深入浅出&#xff01; 目前已经更新到了&#xff1a; Hadoop&#xff0…

leetcode_有序数组中的单一元素

540. 有序数组中的单一元素 - 力扣&#xff08;LeetCode&#xff09; 二分查找 使用条件 &#xff1a; 有序 &#xff0c; log n class Solution { public:int singleNonDuplicate(vector<int>& nums) {int left 0, right nums.size() - 1, mid;while (left <…

Python中的简单爬虫

文章目录 一. 基于FastAPI之Web站点开发1. 基于FastAPI搭建Web服务器2. Web服务器和浏览器的通讯流程3. 浏览器访问Web服务器的通讯流程4. 加载图片资源代码 二. 基于Web请求的FastAPI通用配置1. 目前Web服务器存在问题2. 基于Web请求的FastAPI通用配置 三. Python爬虫介绍1. 什…

USRP:B205mini-i

USRP B205mini-i B205mini-i都是采用工业级的FPGA芯片(-I表示industrial-grade)&#xff0c;所以价格贵。 这个工业级会让工作温度从原来 0 – 45 C 变为 -40 – 75 C. 温度的扩宽&#xff0c;会让工作的稳定性变好。但是前提是你需要配合NI的外壳才行&#xff0c;你如果只买一…

基于Redis内核的热key统计实现方案|得物技术

一、Redis热key介绍 Redis热key问题是指单位时间内&#xff0c;某个特定key的访问量特别高&#xff0c;占用大量的CPU资源&#xff0c;影响其他请求并导致整体性能降低。而且&#xff0c;如果访问热key的命令是时间复杂度较高的命令&#xff0c;会使得CPU消耗变得更加严重&…

鸿蒙安全控件之位置控件简介

位置控件使用直观且易懂的通用标识&#xff0c;让用户明确地知道这是一个获取位置信息的按钮。这满足了授权场景需要匹配用户真实意图的需求。只有当用户主观愿意&#xff0c;并且明确了解使用场景后点击位置控件&#xff0c;应用才会获得临时的授权&#xff0c;获取位置信息并…

鸿蒙主流路由详解

鸿蒙主流路由详解 Navigation Navigation更适合于一次开发,多端部署,也是官方主流推荐的一种路由控制方式,但是,使用起来入侵耦合度高,所以,一般会使用HMRouter,这也是官方主流推荐的路由 Navigation官网地址 个人源码地址 路由跳转 第一步-定义路由栈 Provide(PageInfo) pag…

Jackson库中JsonInclude的使用

简介 JsonInclude是 Jackson 库&#xff08;Java 中用于处理 JSON 数据的流行库&#xff09;中的一个注解。它用于控制在序列化 Java 对象为 JSON 时&#xff0c;哪些属性应该被包含在 JSON 输出中。这个注解提供了多种策略来决定属性的包含与否&#xff0c;帮助减少不必要的数…