There were three different versions of the original code. I refactored the first one.
This is the original code:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class TennisGame | |
def initialize(player1_name, player2_name) | |
@player1_name = player1_name | |
@player2_name = player2_name | |
@p1points = 0 | |
@p2points = 0 | |
end | |
def won_point(player_name) | |
if player_name == @player1_name | |
@p1points += 1 | |
else | |
@p2points += 1 | |
end | |
end | |
def score | |
result = "" | |
tempScore=0 | |
if (@p1points==@p2points) | |
result = { | |
0 => "Love-All", | |
1 => "Fifteen-All", | |
2 => "Thirty-All", | |
}.fetch(@p1points, "Deuce") | |
elsif (@p1points>=4 or @p2points>=4) | |
minusResult = @p1points-@p2points | |
if (minusResult==1) | |
result ="Advantage " + @player1_name | |
elsif (minusResult ==-1) | |
result ="Advantage " + @player2_name | |
elsif (minusResult>=2) | |
result = "Win for " + @player1_name | |
else | |
result ="Win for " + @player2_name | |
end | |
else | |
(1...3).each do |i| | |
if (i==1) | |
tempScore = @p1points | |
else | |
result+="-" | |
tempScore = @p2points | |
end | |
result += { | |
0 => "Love", | |
1 => "Fifteen", | |
2 => "Thirty", | |
3 => "Forty", | |
}[tempScore] | |
end | |
end | |
result | |
end | |
end |
First, I tried to refactor the code to the State Pattern as I did once before in C++. Then I realized checking the tests that the code was not meant to model a real tennis game. It's just focused in displaying the current score.
So I decided to try to separate the code that was keeping the tennis game score from the code that was displaying its score. I also wanted to be able to display the score in different languages.
This is the resulting code after the refactoring:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class TennisGame | |
attr_reader :game_score, :score_displayer | |
def initialize(game_score, score_displayer) | |
@game_score = game_score | |
@score_displayer = score_displayer | |
end | |
def won_point(player_name) | |
game_score.won_point(player_name) | |
end | |
def score | |
score_displayer.display(game_score) | |
end | |
end | |
class GameScore | |
attr_reader :first_player, :second_player | |
def initialize(first_player, second_player) | |
@first_player = first_player | |
@second_player = second_player | |
end | |
def tied? | |
first_player.points == second_player.points | |
end | |
def advantage_for_any_player? | |
both_with_forty_or_more? and points_difference() == 1 | |
end | |
def over? | |
any_over_forty? and points_difference() >= 2 | |
end | |
def won_point(player_name) | |
if player_name == first_player.name | |
first_player.won_point | |
else | |
second_player.won_point | |
end | |
end | |
def current_winner | |
first_player.points > second_player.points ? first_player : second_player | |
end | |
private | |
def points_difference | |
(first_player.points - second_player.points).abs | |
end | |
def both_with_forty_or_more? | |
first_player.points >= 3 and second_player.points >= 3 | |
end | |
def any_over_forty? | |
first_player.points >= 4 or second_player.points >= 4 | |
end | |
end | |
class Player | |
attr_reader :points, :name | |
def initialize(name) | |
@name = name | |
@points = 0 | |
end | |
def won_point | |
@points += 1 | |
end | |
end | |
class ScoreDisplayer | |
attr_accessor :vocabulary | |
def initialize | |
@vocabulary = GameVocabulary.new({ | |
:zero_all => "Love-All", | |
:fifteen_all => "Fifteen-All", | |
:thirty_all => "Thirty-All", | |
:deuce => "Deuce", | |
:zero => "Love", | |
:fifteen => "Fifteen", | |
:thirty => "Thirty", | |
:forty => "Forty", | |
:advantage => "Advantage", | |
:game_over => "Win for" | |
}) | |
end | |
def display(game_score) | |
if game_score.tied? | |
display_tie(game_score.first_player) | |
elsif game_score.over? | |
display_game_over(game_score.current_winner()) | |
elsif game_score.advantage_for_any_player? | |
display_advantage(game_score.current_winner()) | |
else | |
display_default(game_score.first_player, game_score.second_player) | |
end | |
end | |
def display_game_over(winner) | |
@vocabulary.word_for(:game_over) + " " + winner.name | |
end | |
def display_advantage(player_with_advantage) | |
@vocabulary.word_for(:advantage) + " " + player_with_advantage.name | |
end | |
def display_tie(any_player) | |
key = { | |
0 => :zero_all, | |
1 => :fifteen_all, | |
2 => :thirty_all, | |
}.fetch(any_player.points, :deuce) | |
@vocabulary.word_for(key) | |
end | |
def display_default(first_player, second_player) | |
key_by_points = { | |
0 => :zero, | |
1 => :fifteen, | |
2 => :thirty, | |
3 => :forty, | |
} | |
@vocabulary.word_for(key_by_points[first_player.points]) + | |
"-" + | |
@vocabulary.word_for(key_by_points[second_player.points]) | |
end | |
end | |
class GameVocabulary | |
def initialize(vocabulary) | |
@vocabulary = vocabulary | |
end | |
def word_for(key) | |
@vocabulary[key] | |
end | |
end |
The TennisGame has two collaborators: the GameScore and the ScoreDisplayer.
The GameScore keeps track of the two Players in order to answer questions about the game score, whereas the ScoreDisplayer is in charge of displaying the score.
By default the ScoreDisplayer uses an English GameVocabulary, but the GameVocabulary can be injected through its accessor in order to display the score a different language (check the tests to see how the score is displayed in Spanish).
You can check the code and its tests in this GitHub repository.
I also committed after every tiny refactoring step so you can follow the process (especially the changes of mind I had and the dead-ends I found).
This kata was a great puzzle to practise and think.
I'd like to thank David Vrensk for his great work facilitating the kata and the interesting talk about Deliberate Practice that he gave before.
No comments:
Post a Comment