Over my last posts I talked about Windows Phone 7, we saw some real examples for Location Service, Maps, Ads, Accelerometer and more.
Because I saw how many of you download my demo projects I understand the need for more demos around Windows Phone 7.
So Today, we’ll build a Puzzle Game for Windows Phone 7 (Including the Sources
)

Download Demo Project
Step 1: The Puzzle Base
There are several ways to build the puzzle layer, In my demo I chose a simple way of a Canvas with 16 children's type of StackPanel.
Basically I have a lower canvas with the Board image, on top of it I put another canvas with 16 Stack Panels, each panel spread to 100X100

Now, when I have the main puzzle structure I added an image of size 95x95 to each Stack Panel. (the reason it’s not 100X100 – is to leave a space between each), for each image I set the Tag property with the value of the image – 1.png Tag = 1

I’ve also add a timer for counting the time took to solve this puzzle and another int property to count the moves the user made.
Step 2: Find
One of the most common things we’ll use in our code, is FIND:
- Find Stack Panel by Image Id
- Find the Empty Panel
- Find the Value by position
Find the parent of image with a specific Tag value
StackPanel FindStackPanelByTagId(int tag)
{
if (tag == 16)
{
return (from stackPanel in ContentPanel.Children.OfType<StackPanel>()
where stackPanel.Children.Count == 0 select stackPanel).First();
}
else
{
return (from stackPanel in ContentPanel.Children.OfType<StackPanel>()
from img in stackPanel.Children.OfType<Image>()
where Convert.ToInt32(img.Tag) == tag
select stackPanel).First();
}
}
Find the position of StackPanel without children.
int FindEmptyItemPosition()
{
int index = 15;
for (int i = 0; i < 15; i++)
{
if (((StackPanel)ContentPanel.Children[i]).Children.Count == 0)
return index;
index--;
}
return 0;
}
Get the Tag value by StackPanel position.
int FindItemValueByPosition(int position)
{
return ((StackPanel)ContentPanel.Children[position]).Children.Count > 0
? Convert.ToInt32(((Image)((StackPanel)ContentPanel.Children
[position]).Children[0]).Tag) : 16;
}
Step 3: Scrambles
Now we have are puzzle structure and Find method helpers, the first thing is to Scramble or puzzle.
So I wrote a method that runs n times and generate random numbers from 1 to 16, for each number find the current StackPanel that hold him. (FindStackPanelByTagId) .
If First and Second number are smaller then 16 then - swipe the images and tag values.
If One of the values is 16 the swipe - One Stackpanel will be cleared of Items
void Scrambles()
{
var count = 0;
while (count < 25)
{
var a = _rnd.Next(1, 17);
var b = _rnd.Next(1, 17);
if (a == b) continue;
var stack1 = FindStackPanelByTagId(a);
var stack2 = FindStackPanelByTagId(b);
if (a == 16)
{
var image2 = stack2.Children[0];
stack2.Children.Clear();
stack1.Children.Add(image2);
}
else if (b == 16)
{
var image1 = stack1.Children[0];
stack1.Children.Clear();
stack2.Children.Add(image1);
}
else
{
var image1 = stack1.Children[0];
var image2 = stack2.Children[0];
stack1.Children.Clear();
stack2.Children.Clear();
stack1.Children.Add(image2);
stack2.Children.Add(image1);
}
count++;
}
}
Step 4: Check Board
Each move the user do, perform a loop and checks values from 1 to 16. if the numbers are not in the correct order than nothing happed.
Else You stop the game timer and display a win message.
void CheckBoard()
{
var index = 1;
for (var i = 15; i > 0; i--)
{
if (FindItemValueByPosition(i) != index) return;
index++;
}
_timer.Stop();
WinGrid.Visibility = System.Windows.Visibility.Visible;
}
Step 5: Move Items
Before we can apply move of items we need to check several things, the first thing is:
Check if the Item Can move, Checking all panels around the specific item with -1 +1 -4 +4, if one of them is empty then he can move.
StackPanel CanMove(UIElement itemToMove)
{
var count = ContentPanel.Children.Count;
for (var i = 0; i < count; i++)
{
if (!(ContentPanel.Children[i] is StackPanel)) continue;
var stackPanel = (StackPanel)ContentPanel.Children[i];
if (!stackPanel.Children.Contains(itemToMove)) continue;
if (!IsBorderSwich(i, i + 1) && i + 1 <= 15 && ContentPanel.
Children[i + 1] != null &&
((StackPanel)ContentPanel.Children[i + 1]).Children.Count == 0)
return ((StackPanel)ContentPanel.Children[i + 1]);
if (!IsBorderSwich(i, i - 1) && i - 1 > -1 && ContentPanel.
Children[i - 1] != null &&
((StackPanel)ContentPanel.Children[i - 1]).Children.Count == 0)
return ((StackPanel)ContentPanel.Children[i - 1]);
if (i + 4 <= 15 && ContentPanel.Children[i + 4] != null &&
((StackPanel)ContentPanel.Children[i + 4]).Children.Count == 0)
return ((StackPanel)ContentPanel.Children[i + 4]);
if (i - 4 > -1 && ContentPanel.Children[i - 4] != null &&
((StackPanel)ContentPanel.Children[i - 4]).Children.Count == 0)
return ((StackPanel)ContentPanel.Children[i - 4]);
}
return null;
}
The Second - if both of the items you want to swipe are in the Board borders do nothing.
private readonly int[] _bordersNums = { 0, 4, 8, 12, 3, 7, 11, 15 };
bool IsBorderSwich(int a, int b)
{
return _bordersNums.Contains(a) && _bordersNums.Contains(b);
}

Now after we have those safety methods we can move the items based on user clicks:
Just register to ItemManipulationStarted on the entire windows, for each event check if the item isn’t image do nothing, if it does, call the CanMove method to verify that this item can move around.
private void ItemManipulationStarted(object sender,
ManipulationStartedEventArgs e)
{
var item = (UIElement)e.OriginalSource;
if (!(item is Image)) return;
var to = CanMove(item);
if (to != null)
{
_moves++;
txtMoves.Text = _moves.ToString();
MoveItem(item, to);
CheckBoard();
}
e.Handled = true;
e.Complete();
}
MoveItem - Move Item From One StackPanel to Another.
void MoveItem(UIElement item, StackPanel targetPanel)
{
foreach (var stackPanel in
ContentPanel.Children.OfType<StackPanel>().Where(stackPanel =>
stackPanel.Children.Count > 0 && stackPanel.Children.Contains(item)))
{
stackPanel.Children.Remove(item);
}
targetPanel.Children.Add(item);
}
Step 6: Check If Puzzle Solvable
This part of very important, because half of the starting positions for the n-puzzle are impossible to resolve.
Johnson & Story (1879) used a parity argument to show that half of the starting positions for the n-puzzle are impossible to resolve, no matter how many moves are made. This is done by considering a function of the tile configuration that is invariant under any valid move, and then using this to partition the space of all possible labeled states into two equivalence classes of reachable and unreachable states.
The Puzzle 15 (n-puzzle) is a classical problem for modeling algorithms involving heuristics. Commonly used heuristics for this problem include counting the number of misplaced tiles and finding the sum of the Manhattan distances between each block and its position in the goal configuration. Note that both are admissible, i.e., they never overestimate the number of moves left, which ensures optimality for certain search algorithms such as A*.
bool CheckIfSolvable()
{
var n = 0;
for (var i = 1; i <= 16; i++)
{
if (!(ContentPanel.Children[i] is StackPanel)) continue;
var num1 = FindItemValueByPosition(i);
var num2 = FindItemValueByPosition(i - 1);
if (num1 > num2)
{
n++;
}
}
var emptyPos = FindEmptyItemPosition();
return n % 2 == (emptyPos + emptyPos / 4) % 2 ? true : false;
}
Step 7: Setup a New Game
Now, when we defined everything we need, let’s write the New Game method, reset all timer and moves number back to 0, call the Scrambles method, and while the game is unsolvable continue scramble the game, once it’s solvable start the timer.
public void NewGame()
{
_moves = 0;
txtMoves.Text = "0";
txtTime.Text = Const.DefaultTimeValue;
Scrambles();
while (!CheckIfSolvable())
{
Scrambles();
}
_startTime = DateTime.Now.AddSeconds(1);
_timer.Start();
GridScrambling.Visibility = System.Windows.Visibility.Collapsed;
}

Download Demo Project
Enjoy