HTML5 player movement like in popular iOS game “Back to Square One” made with Phaser

Talking about Back to Square One game, Game development, HTML5, Javascript and Phaser.

One of the latest Umbrella Games hits on the App Store is Back to Square One. It’s a platform game with a single tap control scheme that’s a perfect fit for mobile devices. Roll from tile to tile, while avoiding numerous obstacles and collecting coins. Sounds easy? Think again! You’ll need to pay close attention, or you’re back to square one! Umbrella games always publishes simple but awesome games, I already blogged about Boom Dots and Down the Mountain so have a look at them too! Back to our game, it’s an horizontal endless runner, and like most of the endless runner the player actually does not run, it’s the whole environment which moves towards the player to give the illusion of movement. The basic prototype was very easy to create, the only difficult thing was handling pivot points in player movement as it’s a rotating square. Have a look at the prototype:
Click or tap on the canvas to make the player move and the terrain scroll. Have a look at the fully commented source code:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
// the game
var game;
 
// size of each square, in pixels
var squareSize = 40;
 
// square where the hero will start to walk
var startingSquare = 2;
 
// this is the time required to make a move, in milliseconds
var moveTime = 250;
 
// giving some color to ground squares
var squareColors = [0x888888, 0xaaaaaa];
 
// creation of the game, portrait mode 320x480
window.onload = function() {   
    game = new Phaser.Game(320, 480);
     game.state.add("PlayGame", playGame);
     game.state.start("PlayGame");
}
 
 
var playGame = function(game){}
 
playGame.prototype = {
     preload: function(){
           
          // preloading the assets
          game.load.image("square", "square.png");         
     },
      
     create: function(){
          // adding a group which will contain all terrain squares
          this.terrainGroup = game.add.group();
           
          // filling the field with squares to create the terrain
          for(var i = 0; i < game.width / squareSize + 2; i++){
                
               // adding a terrain square
               var square = game.add.sprite(i * squareSize, game.height / 3 * 2, "square");
                
               // settings its anchor on the center
               square.anchor.set(0.5);
                
               // giving the square a tint color
               square.tint = squareColors[i % 2];
                
               // finally adding the square to the group
               this.terrainGroup.add(square);
          }
           
          // adding the hero
          this.hero = game.add.sprite(startingSquare * squareSize, game.height / 3 * 2 - squareSize, "square");
           
          // setting the anchor on its center
          this.hero.anchor.set(0.5);
           
          // flag to see if the hero can move
          this.hero.canMove = true;
           
          // input listener waiting for mouse or touch input, then calling moveSquare method
          game.input.onDown.add(this.moveSquare, this);                                                                                         
     },
      
     // this method will be called each time the player tries to move the square
     moveSquare: function(){
      
          // can the hero move?
          if(this.hero.canMove){
           
               // the hero is about to be moved so we aren't considering more inputs
               this.hero.canMove = false;
                
               // according to current square angle, we have to change its position and registration point
               // in order to make it rotate along the required pivot point
               // this part needs to be optimized but at the moment it works
               switch(this.hero.angle){
                    case 0:
                         this.hero.x += squareSize / 2;
                         this.hero.y += squareSize / 2;
                         this.hero.pivot.x = squareSize / 2;
                         this.hero.pivot.y = squareSize / 2;
                         break;
                    case 90:
                         this.hero.x += squareSize;
                         this.hero.pivot.x = squareSize / 2;
                         this.hero.pivot.y = -squareSize / 2;
                         break;
                    case -180:
                         this.hero.x += squareSize;
                         this.hero.pivot.x = -squareSize / 2;
                         this.hero.pivot.y = -squareSize / 2;
                         break;
                    case -90:
                         this.hero.x += squareSize;
                         this.hero.pivot.x = -squareSize / 2;
                         this.hero.pivot.y = squareSize / 2;                        
               }
                
               // tween to scroll the terrain to the left by squareSize pixels
               var scrollTween = game.add.tween(this.terrainGroup).to({
                    x: this.terrainGroup.x - squareSize
               }, moveTime, Phaser.Easing.Linear.None, true);
                
               // tween to rotate and move the hero
               var moveTween = game.add.tween(this.hero).to({
                    angle: this.hero.angle + 90,
                    x: this.hero.x - squareSize
               }, moveTime, Phaser.Easing.Linear.None, true);
                
               // once the tween has been completed...
               moveTween.onComplete.add(function(){
                
                    // the hero can move again
                    this.hero.canMove = true;
                     
                    // if hero's angle is zero, that is if we rotated by 360 degrees...
                    if(this.hero.angle == 0){
                     
                         // we restore hero original pivot point and position
                         this.hero.pivot.x = 0;
                         this.hero.pivot.y = 0;
                         this.hero.x += squareSize / 2;
                         this.hero.y -= squareSize / 2;
                    }
                     
                    // we order all terrain squares according to their x position
                    this.terrainGroup.sort("x", Phaser.Group.SORT_ASCENDING);
                     
                    // we don't want to destroy assets to create new ones, we prefer to move the leftmost square
                    // which is not visible anymore to the right.
                    this.terrainGroup.getChildAt(0).x += this.terrainGroup.length * squareSize;
               
               }, this);
          }
     }    
}
Next time we’ll add different terrain types, coins and enemies. Meanwhile download the source code of the entire project.