Coder Social home page Coder Social logo

ps-wise-2021-program-description's Introduction

Programming Styles -- WiSe 20-21


Minesweeper

All the assignments of the Programming Style WiSe 20-21 consist in re-implementing the same program over and over, using a different programming style every time. This follows the same approach proposed by Cristina Lopes in her book "Exercises in Programming Style."

The program we will implement is a single-player, interactive puzzle-game called "Minesweeper" which is modeled after the classic video-game available on old PCs. The first iterations of Minesweeper as we know it was built by Robert Donner who was hired by Microsoft in 1989. The game is now one of the most iconic computer games next to Pong, Tetris, and PacMan to name a few.

According to Wikipedia, there are many variants of this game. Below I summarize the rules of the variant you will implement for the assignments. In particular, we will not make use of the mouse to interact with the game, instead we will play by reading the inputs from the console.

In this document, I will describe the specifications of the Minesweeper program from this and this other links. You can find more information about grading here.

NOTE: these specifications might be incomplete or unclear. For example, handling of expected inputs or corner cases might not be fully specified. If you see anything to be fixed, instead of complaining about it, pro-actively work to fix the problem. Open issues, propose updates, refinements, and write more public test cases (very good for corner cases!!).

In the initial assignments, you'll be asked to implement a basic version of the game. However, later assignments, you will be asked to implement add-os and "enhancements" to it.

NOTE: All the changes to the specifications will be also notified on StudIP.

Goal and Rules of the Game

The objective of the game is to clear a rectangular board containing hidden "mines" or bombs without detonating any of them. The players can make use of clues about the number of neighboring mines in each field. So a number on a block indicates how many mines are around that block. For example, if there are two squares touching each other and one of the squares has "1" on it, you know that the square next to it has a mine beneath it.

If the player selects a square containing a bomb, the game is over and the player has lost it. Otherwise, if the player manages to select all the squares (without clicking on any bombs), the game is won.

In some versions of Minesweeper, the first move is "protected," that is the boards are set up by never placing a mine on the first square revealed. For the assignment, we do not implement such a rule. The initial board is "empty" and it does not matter whether or not the players detonates a mine after the first move.

Game Setup

Each game starts with a rectangular board of unmarked squares. Some contains mines others do not. The size of the board and the number and placement of the mines are specified in a text file passed as input parameter. The program takes ONLY this file as input parameter.

The file must follow this format:

  1. It must contain only text. The number of lines identifies the height of the board while the length of the lines identify its width.
  2. Each line follows this format: a . indicates an empty square, a * identifies a mine. No other characters (except new line [\n]) shall be present in this file.
  3. There are no empty lines.
  4. All the lines must have the same length and must be terminated by [\n].

A valid file defining a 3x3 board containing 2 mines might look like this:

..*[\n]
...[\n]
..*[\n]

Valid boards must have more than one square, and do not contain only mines.

The board configuration files must have the .cfg extension.

Playing the game

The game is usually played using the mouse to click covered squares that don't contain mines (left button) and flag squares that potentially contain mines (right button). For the assignment, we do not rely on mouse, but input the coordinates of the block we intent to click and the action to perform R for reveal and F for flag. Clicking squares that have been already uncovered does not cause any change in the game.

In the following, "(left-)click" means input the coordinate of the square with action R, while "flag" means input the coordinate of the square with action F.

A player left-clicks a square to uncover it. If a player uncovers a mine, the mine detonates, and the game is lost. Otherwise, the uncovered squares displays either a number, indicating the quantity of mines adjacent to it, or a blank tile. All the adjacent covered and empty squares are automatically uncovered. Players' job is to use these numbers to figure out which of the blank squares have mines and which are safe to click.

NOTE: A squares' "neighbours" are the squares adjacent above, below, left, right, and all 4 diagonals. Squares on the sides of the board or in a corner have fewer neighbors. The board does not wrap around the edges.

Right-clicking on a square will flag it, causing a flag () to appear on it. Flagged squares are still covered, and a player can click on them to uncover them. There is not need to "unflag" squares to uncover them.

So squares can be covered, uncovered and flagged. A covered square is blank and "clickable", while an uncovered square is exposed. Flagged squares are those marked by the player to indicate a potential mine location. Additionally, squares might be empty, contain a number, or a mine.

End of the Game

A game ends when the player uncover all non-mine squares (win) or a mine is detonated (lost). Flagging all the mined squares is not required to finish the game; however, when the game is lost all the mines must be clearly shown on the board.

The Text-based Graphical User Interface (GUI)

The GUI of the game is made entirely by characters, like in very old games. This is mostly because it is easy to automatically test whether your implementation is correct or not. There are no colors, but you need to generate the right characters (see below).

Minesweeper text-based GUI is composed of three parts: the board, the message banner, and the console input.

The board

The board is a "box/table" aligned to the left (so no space on the left), made of cells. Each cell corresponds to a square. Squares are indexed starting from (1,1) on the top left corner to (N,M) on the bottom right corner. The first coordinate identifies the row, the second the column. The content of the cell is the central char: * identifies a mine, 1 (or any digit) is the number of mines in neighboring cells, :nothing: (is an uncovered square), is a covered-but-empty square, while is a flag.

An example of a 3x3 board with those symbols is show below:

┌───┬───┬───┐[\n]
│ * │ 1 │   │[\n]
├───┼───┼───┤[\n]
│   │   │ 2 │[\n]
├───┼───┼───┤[\n]
│   │ ▓ │ ¶ │[\n]
└───┴───┴───┘[\n]

In the above board, the mine is in position (1,1), the 1 is in position (1,2), the 2 is in position (2,3), the square in position (3,2) is empty (), while the square in position (3,3) is flagged. All the remaining squares are uncovered.

NOTE: The board above serves only as illustration, such a configuration would be not possible because a mine has been found.

Here are some specifications of the boards:

  • The height of each square is 3 line
  • The width of each square is 3 chars
  • Nearby squares share borders
  • An uncovered square is "empty"
  • An flanges square contains a flag
  • Covered squares contain a shaded cell unless it contains a mine (*) or a digit
  • Each line is terminated by \n and there are no spaces before or after it

The message banner

During the game, Minesweeper might output messages to the player. For example, if the provided input is invalid, or the game is over. Messages are displayed inside the message banner, which is positioned below the board.

┌───┬───┬───┐[\n]
│   │   │   │[\n]
├───┼───┼───┤[\n]
│   │   │   │[\n]
├───┼───┼───┤[\n]
│   │   │   │[\n]
└───┴───┴───┘[\n]
╔═══════════╗[\n]
║           ║[\n]
╚═══════════╝[\n]

Both the message banner and the messages are left justified (no spaces on the left). The message banner must stretch to fit all the messages on one line and cannot be shorter than the board itself.

An example of short message when the player wins the game is:

└───┴───┴───┘[\n]
╔═══════════╗[\n]
║You Won!   ║[\n]
╚═══════════╝[\n]

An example of a long message when the player input an invalid input is:

└───┴───┴───┘[\n]
╔════════════════════════════════╗[\n]
║The provided input is not valid!║[\n]
╚════════════════════════════════╝[\n]

As you might have noted, there is not space also on the right side of the message when the message is longer than the banner.

Console Input

The console input starts with the char > (yes exactly this character!) and it is not terminated by a newline.

>

All together

Putting the elements of the GUI all together results and assuming an 3x3 initial board results in:

┌───┬───┬───┐[\n]
│   │   │   │[\n]
├───┼───┼───┤[\n]
│   │   │   │[\n]
├───┼───┼───┤[\n]
│   │   │   │[\n]
└───┴───┴───┘[\n]
╔═══════════╗[\n]
║           ║[\n]
╚═══════════╝[\n]
>

Extended ASCII

Note that to generate the GUI you do not need necessary to use the extended ASCII characters if you can use other encodings.

Please read this StackOverflow question.

Processing User Inputs

The player inputs three values at the time, and the program reacts to them refreshing the GUI and potentially showing a message in the message banner.

Inputs must follow the specified format to be considered valid. Invalid inputs are rejected by the system (i.e., they do not cause a change in the state of the board), but the system notifies the use about them.

Input format

Valid inputs consist of three values separated by one or more spaces.

The first two values identify a square in the board (target for the action) while the last input identifies the action to perform (i.e., reveal or flag). Spaces in front and in the back of the inputs does not matter. Any input with a different format is considered invalid and reported with the message: The provided input is not valid!

Examples of valid inputs are:

1`` ``1`` ``F

1`` ``2`` `` ``F

1`` ``1`` `` ``F`` ``

`` `` ``1`` ``1`` `` ``F`` ``

Examples of invalid inputs are:

1`` ``1`` ``1`` `` ``F`` `` (more than three inputs)

111`` `` ``F`` `` (only two inputs)

``111F`` (only one input)

Not only the input format matters, but also the values provided by the player. Valid inputs must have a positive integer less or equal the numbers of rows as first value, a positive integer less or equal to the number of columns as second value, and either the letter R or F (capitalized) as the last input. Inputs that do not met those constraints are invalid and must be reported with the message: The provided input is not valid!

Assuming a 3x3 board, the following inputs are invalid:

1`` ``1`` ``f (non capitalized R)

1`` ``5`` ``R (5 is invalid as the board contains only 3 columns)

-1`` ``1`` ``F (-1 is not positive)

1.0`` ``1`` ``F (1.0 is not integer)

Examples of games

This section exemplifies how the game develops.

Simple game (win case)

We start the game by pointing it to the board configuration. Assume the following configuration for the board stored in a file called simple.cfg. Note that there is only one mine.

..*[\n]
...[\n]
...[\n]

For the java implementation, invoking the program looks like this:

java Minesweeper simple.cfg

For the javascript implementation, invoking the program looks like this:

node minesweeper.js simple.cfg

After starting the program, we should see the following output on the console:

┌───┬───┬───┐[\n]
│   │   │   │[\n]
├───┼───┼───┤[\n]
│   │   │   │[\n]
├───┼───┼───┤[\n]
│   │   │   │[\n]
└───┴───┴───┘[\n]
╔═══════════╗[\n]
║           ║[\n]
╚═══════════╝[\n]
>

At this point, we input ``1`` ``1`` R to reveal the top left corner. This causes the GUI to refresh and we obtain:

┌───┬───┬───┐[\n]
│ ▓ │ 1 │   │[\n]
├───┼───┼───┤[\n]
│ ▓ │ 1 │ 1 │[\n]
├───┼───┼───┤[\n]
│ ▓ │ ▓ │ ▓ │[\n]
└───┴───┴───┘[\n]
╔═══════════╗[\n]
║You Won!   ║[\n]
╚═══════════╝[\n]

What happened is that that squares closed by (1,1) do not contain a mines so they are automatically revealed (empty or with 1). Because there are no more square left to uncover except the one that contain the mine, the game automatically ends. As the message says, the player won the game.

NOTE: Because the game ended, no additional inputs must be provided; hence the input console is NOT shown.

Simple game (lost case)

We start the game using the same simple.cfg used before, but this time we provide 1`` ``3`` R as first input to reveal the top right corner. Since the square contains the mine, we lost the game and the program shows the following output.

┌───┬───┬───┐[\n]
│   │   │ * │[\n]
├───┼───┼───┤[\n]
│   │   │   │[\n]
├───┼───┼───┤[\n]
│   │   │   │[\n]
└───┴───┴───┘[\n]
╔═══════════╗[\n]
║You Lost!  ║[\n]
╚═══════════╝[\n]

Once again, since the game is over, the input console is not shows anymore. Since the game is lost, all the mines (in this case only one) are shown on the board.

Corner cases, Exceptions, and the like

Invalid board configuration

If the configuration file is not present, or the file does not contain a valid board definition, the program must end with an error. If the file is missing, the exit code must be 1; if the file is not valid, the exit code is 2.

An empty configuration file is considered as a non valid file, so the exit code must be 2. Other invalid files are files that contains characters different from . and *, or files that do not correspond to rectangular boards. Spaces are NOT allowed in the board configuration files.

Not exception message must be raised !

Additional cases

In general, additional corner cases might exist. If you find some, please list them here. Ideally, you can define public tests which capture the corner cases, so all the other students can update their code (and you can get bonus points!!)

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.