Bowling Challenge in Ruby

  • Feel free to use google, your notes, books, etc. but work on your own
  • If you refer to the solution of another coach or student, please put a link to that in your README
  • If you have a partial solution, still check in a partial solution
  • You must submit a pull request to this repo with your code by 9am Monday week

The Task

THIS IS NOT A BOWLING GAME, IT IS A BOWLING SCORECARD PROGRAM. DO NOT GENERATE RANDOM ROLLS. THE USER INPUTS THE ROLLS.

Count and sum the scores of a bowling game for one player. For this challenge, you do not need to build a web app with a UI, instead, just focus on the logic for bowling (you also don't need a database). Next end-of-unit challenge, you will have the chance to translate the logic to Javascript and build a user interface.

A bowling game consists of 10 frames in which the player tries to knock down the 10 pins. In every frame the player can roll one or two times. The actual number depends on strikes and spares. The score of a frame is the number of knocked down pins plus bonuses for strikes and spares. After every frame the 10 pins are reset.

As usual please start by

  • Forking this repo

  • Finally submit a pull request before Monday week at 9am with your solution or partial solution. However much or little amount of code you wrote please please please submit a pull request before Monday week at 9am.

STRONG HINT, IGNORE AT YOUR PERIL: Bowling is a deceptively complex game. Careful thought and thorough diagramming — both before and throughout — will save you literal hours of your life.

Focus for this challenge

The focus for this challenge is to write high-quality code.

In order to do this, you may pay particular attention to the following:

  • Using diagramming to plan your approach to the challenge
  • TDD your code
  • Focus on testing behaviour rather than state
  • Commit often, with good commit messages
  • Single Responsibility Principle and encapsulation
  • Clear and readable code

Bowling — how does it work?

Strikes

The player has a strike if he knocks down all 10 pins with the first roll in a frame. The frame ends immediately (since there are no pins left for a second roll). The bonus for that frame is the number of pins knocked down by the next two rolls. That would be the next frame, unless the player rolls another strike.

Spares

The player has a spare if the knocks down all 10 pins with the two rolls of a frame. The bonus for that frame is the number of pins knocked down by the next roll (first roll of next frame).

10th frame

If the player rolls a strike or spare in the 10th frame they can roll the additional balls for the bonus. But they can never roll more than 3 balls in the 10th frame. The additional rolls only count for the bonus not for the regular frame count.

10, 10, 10 in the 10th frame gives 30 points (10 points for the regular first strike and 20 points for the bonus).
1, 9, 10 in the 10th frame gives 20 points (10 points for the regular spare and 10 points for the bonus).

Gutter Game

A Gutter Game is when the player never hits a pin (20 zero scores).

Perfect Game

A Perfect Game is when the player rolls 12 strikes (10 regular strikes and 2 strikes for the bonus in the 10th frame). The Perfect Game scores 300 points.

In the image below you can find some score examples.

More about ten pin bowling here: http://en.wikipedia.org/wiki/Ten-pin_bowling

Ten Pin Score Example

Planning

I/O Notes

Input: An array of ten frames, each of which is an array of two rolls Output: The sum total of points during that game

Input: A score of ten, i.e. a ‘strike’ has occured during a roll Output: The points from the next two rolls are added to the total score

Input: A score of ten has occured during a frame, i.e. a spare Output: The points from the next roll is added to the total score

Thoughts

Input is the bowling scorecard contains 10/11 frames where each frame contains two rolls i.e. Techinically, this is a multi-dimensional array containing of 10/11 sub-arrays each with two elements

Need to focus on iterating over a frame not all the frames or induvidual rolls Why? If we focus on rolls or flatten the whole array then we cannot determine the bonus accurate for strike or spares

Methods

attr_accessor This allows us to 'get and set' the instance variable, frames from outside the class.

def initialize(scorecard) This method will receive the scorecard, the multi-dimensional array described above as the only input. Please see bowling_spec.rb for examples of the scorecard. We will use the .self method to gain direct access to the elements within the scorecard object.

def sum_frame(frame) This method will summate all the elements within a frame using the 'inject' method

def first_roll(frame) This method identifies the first roll within a frame. This helps the total method score a spare, where the first roll of the subsequent frame is added as a points bonus.

def strike?(frame) This method identifies if the frame contains a strike.

def spare?(frame) This method identifies if the frame contains a spare.

def total This method returns the total score achieved.