HOME

Navigate back to the homepage

Perplexity

Spenser Naor
July 5th, 2021 · 3 min read
Check out the github repo for more information about the project

About

When I started work on Perplexity in 2013, I knew it would be the largest scale development project I’d worked on up to that point. I had growing familiarity with OOP and was ready to tackle something full scale. The “match 3” app genre was in full swing, so I decided to give it a little twist - opting for a staggared circular playfield in contrast tot he typical rectangular grid. I learned an enormous amont on this project, and developement my skills a several new areas.

Skills and Tools utilized in this project:

  • Procedural puzzle generator
  • Time complexity optimization
  • Depth first search
  • Physics engine implementation (box2d)
  • Collision detection
  • Persistent user data
  • A/B testing with live users
  • In-app purchases
  • Flurry Analytics
  • Game Center implementation

Overview

Like in a typical match 3 game, players match 3 or more tiles with adjacent tiles of the same color. Once matched, tiles disappear and new tiles fall into the playfield. The game sports power-ups, special tiles and a couple different modes.

Gameplay

Player manipulate the playfield in a way completely unique to match 3 style games. Instead of tiles moving from top to bottom, they move from the outer rings toward the inner rings. This theme is carried through to the gameplay. Players can swap a tile with another tile further from (or closer to) the center of the playboard.

Perplexity Radial Swapping

One of the cool side-effects of playing this game, is that you get more comfortable thiking in a polar grid - a space in which most people aren’t particularly familiar.

I tried to lean into the circular format - so instead of just swaping adjacent tiles on the same ring, the player can also spin and entire ring in order to find matches.

Perplexity Angular Spinning

You’ll notice in the above image that tiles will fall toward the center of the game board if there are no supporting tiles stopping them. This leads to an entire world of potential puzzle options.

Modes

Perplexity Main Menu

Perplexity has a simple menu system. Sound effects and music controls, settings stored as persistant user data, game center integration, and two primary modes - Voyage and Arcade:

Voyage

Perplexity Puzzle Mode

The voyage mode was my game’s “campaign,” where players could explore themed puzzle packs, trying to clear the board in the face of a growing number of obstacles. These obstacles may be unusable rings, or locked tiles as in the image below:

Arcade

Perplexity Arcade Mode

In the Arcade mode, players battle the clock to score as many points as they can. Perplexity uses recursive search functions to identify tile adjacency across the board, and then trigger the appropriate response. There are a variety of triggered events throughout the game - matches disappearing from the board, generating special tiles, awarding more points to the player, etc.

Under the hood

Each square represents a polar coordinate, and each tile an instance of a NSObject gamepiece class. That gamepiece object has properties which allowed me to define each distinct gamepiece as a collection of properties.

I used a physics engine called Box2d to manage the gamepiece class and have natural falling animations built-in.

Iteration vs Recursion

One interesting problem arose while programming the logic for finding collections of matching game pieces. I followed a recursive method initially bcause this came the most naturally, but I found that there were times when this method couldn’t evaluate every piece on the board within a single time step, resulting in missed collisions within the physics engine. I had to find a faster way.

Below is a sample code block of my final iterative approach with most of the app-specific logic cleaned out:

1for (int *row; row < sizeOf(currentBoardState); ++row)
2{
3 for (int *column = 0; column<sizeOf([currentBoardState objectAtIndex:row]); column = column + 1 ){
4 //this records the total matches for this entire iteration.
5 //At the end, I'll check if this is greater than two, and then trigger the destruction of its contents.
6 NSMutableArray *totalMatches;
7
8 //Identify and store the active matches to iterate through. the origin gamepiece is its own first match
9 NSMutableArray *activeMatches;
10 [activeMatches addObject:[[currentBoardState objectAtIndex:row] objectAtIndex:column];];
11
12 while (sizeOf(adjacentMatches)>0){
13 b2Body *gamePiece = [adjacentMatches objectAtIndex:0];
14 bodyUserData *gamePieceUserData = (bodyUserData *)gamePiece->GetUserData();
15
16 //check if the current gamepiece is already tagged to be destroyed due to a match
17 if (gamePieceUserData->destroy != true){
18 //Check for adjacent matches
19 NSMutableArray *adjacentPieces;
20
21 [adjacentPieces addObject: (b2Body *)[self findGamePieceLeft:gamePiece]];
22 [adjacentPieces addObject: (b2Body *)[self findGamePieceRight:gamePiece]];
23 [adjacentPieces addObject: (b2Body *)[self findGamePieceAbove:gamePiece]];
24 [adjacentPieces addObject: (b2Body *)[self findGamePieceBelow:gamePiece]];
25
26 for (int *adjacentPieceIndex = 0; adjacentPieceIndex < sizeOf(adjacentPieces); ++adjacentPieceIndex){
27 b2Body *gamePieceAdjacent = [adjacentPieces objectAtIndex:adjacentPieceIndex];
28 bodyUserData *gamePieceAdjacentUserData = (bodyUserData*)gamePieceAdjacent->GetUserData();
29 if (gamePieceUserData->color == gamePieceAdjacentUserData -> color && gamePieceAdjacentUserData->destroy != true && ![totalMatches containsObject:gamePieceAdjacent]){
30 [activeMatches addObject:gamePieceAdjacent];
31 [totalMatches addObject:gamePieceAdjacent];
32 }
33 }
34 [activeMatches removeObject:gamePiece];
35 }
36 }
37 if (sizeOf(totalMatches)>2){
38 for (int destroyIndex = 0; destroyIndex < sizeOf(totalMatches); ++destroyIndex)
39 {
40 b2Body *pieceToDestroy = [totalMatches objectAtIndex:destroyIndex];
41 pieceToDestroy->destroy = true;
42 }
43 }
44 }
45}

It’s bulky, but it gets the job done. I didn’t know anything about depth-first search algorithms before diving into this project so this was a fun case study.

Coordinates

This app uses a lot of math! In order to determine force vectors, I stored each gamepiece’s angle once it locks into a given space.

Concurrently, I managed the board state as a 2d array. You’ll see in the comments that since number of columns decreases as you get closer to the center of the grid, special care needs to be taken to translate between the rows.

1//This is only an illustration.
2//The actual array contans objects, not integers.
3NSMutableArray* currentBoardState = (NSMutableArray*)
4[[1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1],
5[1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1], //multiply by 2 to find column
6[1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1], //multiply by 2 to find column
7[1,1,1,1,1,1,1,1] //multiply by 4 to find column
8]

More articles from Spenser Naor

F Tools for Maya

A collection of helper tools for CG asset creation

July 1st, 2021 · 1 min read

goPlay Social

goPlay is a social app designed to incentivize outdoor activities with discounts on products and services.

July 6th, 2021 · 3 min read
© 2021 Spenser Naor
Link to $https://github.com/spenser-naorLink to $https://www.linkedin.com/in/spenser-naor-90726014/