Up All Knight is a chess app built on Ruby on Rails, JavaScript, jQuery and has special features: blitz chess (chess with a timer) or standard (no timer), captured pieces and rankings. Includes TDD/RSpec testing with 83% coverage.
Player's Create a Game
Chess Landing Page
Chessboard Game
Explanation and Challenges
- Work on Agile/Scrum team with six developers to build a working chess app with features:
- chessboard visualizations,
- color choices (black or white),
- chess pieces moves,
- timers,
- captured pieces,
- player readiness, draw, forfeit
- rankings
- Understand Logic behind Chess piece moves:
- How would you validate that a queen can move to a particular location on the board, given where the other pieces are on the board?
Results
class Queen < Piece
def valid_move?(destination_x, destination_y)
valid = super(destination_x, destination_y)
valid && (
horizontal(destination_y) ||
vertical(destination_x) ||
diagonal(destination_x, destination_y)
)
end
def unicode_symbol
if self.get_color == WHITE
return "♕"
else
return "♛"
end
end
end
- How would you enable the castling move?
class King < Piece
def valid_move?(destination_x, destination_y)
valid = super(destination_x, destination_y)
if valid
if can_castle?(destination_x, destination_y)
return true
elsif (destination_x - self.x_position).abs > 1 || (destination_y - self.y_position).abs > 1
valid = false
end
end
return valid
end
def move_to(destination_x, destination_y)
valid = super
if valid
castle!(destination_x, destination_y)
end
valid
end
def can_castle?(destination_x, destination_y)
return self.moves == 0 && (castling_kingside?(destination_x, destination_y) ||
castling_queenside?(destination_x, destination_y)) && (game.check?(player)) == false
end
def castle!(destination_x, destination_y)
if castling_kingside?(destination_x, destination_y)
rook = game.pieces.where(x_position: 7, y_position: destination_y, player_id: player_id, type: "Rook").last
rook.update_attributes(x_position: 5)
end
if castling_queenside?(destination_x, destination_y)
rook = game.pieces.where(x_position: 0, y_position: destination_y, player_id: player_id, type: "Rook").last
rook.update_attributes(x_position: 3)
end
end
def castling_kingside?(destination_x, destination_y)
return (destination_x == 6 && destination_y == y_position) &&
(is_rook_kingside?(destination_y))
end
def castling_queenside?(destination_x, destination_y)
return (destination_x == 2 && destination_y == y_position) &&
(is_rook_queenside?(destination_y))
end
def is_rook_queenside?(destination_y)
rook = game.pieces.where(x_position: 0, y_position: destination_y, player_id: player_id, type: "Rook").last
if rook != nil
if rook.moves == 0
return true
end
end
return false
end
def is_rook_kingside?(destination_y)
rook = game.pieces.where(x_position: 7, y_position: destination_y, player_id: player_id, type: "Rook").last
if rook != nil
if rook.moves == 0
return true
end
end
return false
end
def unicode_symbol
if self.get_color == WHITE
return "♔"
else
return "♚"
end
end
end
- How would you update the board for user #2, when user #1 moves the piece without requiring a page refresh. We used pusher, a websocket applications with social features that enables realtime user activity.
class PusherController < ApplicationController
skip_before_filter :verify_authenticity_token
def auth
if current_player && params[:channel_name] == "private-user_#{current_player.id}"
render :json => Pusher[params[:channel_name]].authenticate(params[:socket_id])
else
render :text => "Not authorized", :status => '403'
end
end
end
def player_ready
if @game.is_blitz
if !@game.white_ready && !@game.black_ready
Pusher["private-user_#{@opponent_id}"].trigger!('start_ready_timer', {
:message => "Start Ready Timer."
})
end
if params[:current_player].to_i == @game.white_player_id
Pusher["private-user_#{@opponent_id}"].trigger!( "message", {
:message => "White Player is ready for blitz chess game."
})
@game.white_ready = true
@game.save
elsif params[:current_player].to_i == @game.black_player_id
Pusher["private-user_#{@opponent_id}"].trigger!( "message", {
:message => "Black Player is ready for blitz chess game."
})
@game.black_ready = true
@game.save
end
if @game.white_ready && @game.black_ready
@game.has_started = true
@game.save
Pusher["broadcast_#{@game.id}"].trigger!('start_game', {
has_started: true })
Pusher["broadcast_#{@game.id}"].trigger!('clear_ready_timer', {
})
Pusher["broadcast_#{@game.id}"].trigger!('hide_ready_buttons', {})
end
end
render json: {}
end
- How would you check for checkmate/check?
def check?(player)
check = false
king = pieces.where(type: 'King', player: player).last
opponents_pieces = pieces.where(player: opponent_player(player), captured: false)
if king !=nil
return opponents_pieces.any? { |piece| piece.valid_move?(king.x_position, king.y_position) }
end
return false
end
def checkmate?(player)
if check?(player)
checkmate = true
king = pieces.where(type: 'King', player: player).last
(0..7).each do |x|
(0..7).each do |y|
if king.valid_move?(x, y)
original_x = king.x_position
original_y = king.y_position
captured_piece = pieces.where(x_position: x,y_position: y, game_id: id).last
king.move_to(x, y)
checkmate = false if !check?(player)
king.move_to(original_x, original_y)
if captured_piece != nil
captured_piece.reload.update_attributes(x_position: x, y_position: y, captured: false)
end
end
end
end
return checkmate
end
return false
end
Test using RSpec
require 'rails_helper'
RSpec.describe Game, type: :model do
let(:white_player) { FactoryGirl.create(:player, email: 'blah@blah.com', password: 'SPACECAT') }
let(:black_player) { FactoryGirl.create(:player, email: 'meow@meow.com', password: 'MONORAILCAT') }
let(:game) { FactoryGirl.create(:game, :populated, white_player_id: white_player.id, black_player_id: black_player.id) }
let(:king) { game.pieces.where(:player_id => black_player.id, :type => "King").last }
let(:rook) { FactoryGirl.create(:rook, player_id: white_player.id) }
describe "check?" do
let(:game) { FactoryGirl.create(:game, white_player: white_player, black_player: black_player) }
context "the black king" do
it "should be in check" do
king = FactoryGirl.create(:king, x_position: 4, y_position: 1, player: black_player, game: game)
rook = FactoryGirl.create(:rook, x_position: 4, y_position: 7, player: white_player, game: game)
expect(game.check?(black_player)).to eq (true)
end
it "should not be in check" do
king = FactoryGirl.create(:king, x_position: 4, y_position: 1, player: black_player, game: game)
rook = FactoryGirl.create(:rook, x_position: 6, y_position: 7, player: white_player, game: game)
expect(game.check?(black_player)).to eq (false)
end
end
context "the white king" do
it "should be in check" do
king = FactoryGirl.create(:king, x_position: 4, y_position: 1, player: white_player, game: game)
rook = FactoryGirl.create(:rook, x_position: 4, y_position: 7, player: black_player, game: game)
expect(game.check?(white_player)).to eq (true)
end
it "should not be in check" do
king = FactoryGirl.create(:king, x_position: 4, y_position: 1, player: white_player, game: game)
rook = FactoryGirl.create(:rook, x_position: 6, y_position: 7, player: black_player, game: game)
expect(game.check?(white_player)).to eq (false)
end
end
end
describe "checkmate?" do
let(:game) { FactoryGirl.create(:game, white_player_id: white_player.id, black_player_id: black_player.id) }
it "should be in checkmate if king is currently in check and cannot move out of check" do
king = FactoryGirl.create(:king, x_position: 0, y_position: 0, player: black_player, game: game)
queen = FactoryGirl.create(:queen, x_position: 0, y_position: 1, player: white_player, game: game)
queen_2 = FactoryGirl.create(:queen, x_position: 1, y_position: 1, player: white_player, game: game)
expect(game.reload.checkmate?(black_player)).to eq(true)
end
it "should have checkmate value equal to false if king is not in check" do
game.check?(black_player) == false
expect(game.checkmate?(black_player)).to eq(false)
end
end
Player Ready/Timer
- Use Javascript to signal player is ready and turn on timer (if playing blitz chess)
$(document).ready(function() {
$("#player-ready").click(function() {
$.ajax({
type: 'PUT',
url: $("#player-ready").data("update-url"),
dataType: 'json',
data: { current_player: window.player_id }
});
});
var timerId = null;
window.private_channel.bind('start_ready_timer', function(data) {
var updateURL = $(event.currentTarget).data("update-url");
var timeLeft = 30;
timerId = setInterval(countdown, 1000);
function countdown() {
$('.ready-timer').text(timeLeft + ' seconds remaining until you forfeit')
if (timeLeft === 0) {
clearTimeout(timerId);
$.ajax({
type: 'PUT',
url: $("#player-ready").data('player-not-ready-url'),
dataType: 'json',
data: { current_player: window.player_id }
});
} else {
timeLeft--;
}
}
});
window.broadcast_channel.bind('hide_ready_buttons', function(data) {
$('#player-ready').css('display', 'none')
$('.ready-timer').css('display', 'none')
});
window.broadcast_channel.bind('clear_ready_timer', function(data) {
clearTimeout(timerId);
});
});
Conclusion
On the team, we wanted the chess app to have solid logic to move the pieces as conventional chess pieces and also have the game be asychronous, using JavaScript/jQuery. We also wanted the front end to be clean, inviting and easy to navigate.