Snake Game Using Flutter
4 min readJul 23, 2021
how to code snake game using flutter
In This Lesson, I’ll show you how to build a Snake Game using Flutter so let’s get started :
You must watch entire the Lesson on youtube to understand every step
import :
import 'dart:async';
import 'dart:math';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
hide system bar you can use this line of code :
SystemChrome.setEnabledSystemUIMode(SystemUiMode.manual, overlays: []);
First of all, let’s add some variables :
List<int> snakePosition = [42, 62, 82, 102];
int numberOfSquares = 760;static var randomNumber = Random();
int food = randomNumber.nextInt(700);var speed = 300;bool playing = false;var direction = 'down';bool x1 = false;
bool x2 = false;
bool x3 = false;bool endGame = false;
Then Let’s create some functions for our Game :
the first one to Start our Game :
startGame() {
setState(() {
playing = true;
});
endGame = false;
snakePosition = [42, 62, 82, 102];
var duration = Duration(milliseconds: speed);
Timer.periodic(duration, (Timer timer) {
updateSnake();
if (gameOver() || endGame) {
timer.cancel();
showGameOverDialog();
playing = false;
x1 = false;
x2 = false;
x3 = false;
}
});
}
and create Game Over Function it returns bool value
gameOver() {
for (int i = 0; i < snakePosition.length; i++) {
int count = 0;
for (int j = 0; j < snakePosition.length; j++) {
if (snakePosition[i] == snakePosition[j]) {
count += 1;
}
if (count == 2) {
setState(() {
playing = false;
});
return true;
}
}
}
return false;
}
and this function to show a message in case the user end the game or snake crush itself
showGameOverDialog() {
showDialog(
context: context,
builder: (BuildContext context) {
return AlertDialog(
title: const Text('Game Over'),
content: Text('your score is ' + snakePosition.length.toString()),
actions: [
TextButton(
onPressed: () {
startGame();
Navigator.of(context).pop(true);
},
child: const Text('Play Again'))
],
);
});
}
this function to generate food randomly
generateNewFood() {
food = randomNumber.nextInt(700);
}
and the last function to update snake position with out timer in start game function :
updateSnake() {
setState(() {
switch (direction) {
case 'down':
if (snakePosition.last > 740) {
snakePosition.add(snakePosition.last + 20 - 760);
} else {
snakePosition.add(snakePosition.last + 20);
}
break; case 'up':
if (snakePosition.last < 20) {
snakePosition.add(snakePosition.last - 20 + 760);
} else {
snakePosition.add(snakePosition.last - 20);
}
break; case 'left':
if (snakePosition.last % 20 == 0) {
snakePosition.add(snakePosition.last - 1 + 20);
} else {
snakePosition.add(snakePosition.last - 1);
}
break; case 'right':
if ((snakePosition.last + 1) % 20 == 0) {
snakePosition.add(snakePosition.last + 1 - 20);
} else {
snakePosition.add(snakePosition.last + 1);
}
break; default:
}
if (snakePosition.last == food) {
generateNewFood();
} else {
snakePosition.removeAt(0);
}
});
}
Now let’s build our game design :
@override
Widget build(BuildContext context) {
return Scaffold(
backgroundColor: Colors.white12,
body: Column(
children: [
Expanded(
child: GestureDetector(
onVerticalDragUpdate: (details) {
if (direction != 'up' && details.delta.dy > 0) {
direction = 'down';
} else if (direction != 'down' && details.delta.dy < 0) {
direction = 'up';
}
},
onHorizontalDragUpdate: (details) {
if (direction != 'left' && details.delta.dx > 0) {
direction = 'right';
} else if (direction != 'right' && details.delta.dx < 0) {
direction = 'left';
}
},
child: Stack(
children: [
Center(
child: Image.asset('assets/images/snake.png',
fit: BoxFit.contain)),
GridView.builder(
itemCount: numberOfSquares,
physics: const NeverScrollableScrollPhysics(),
gridDelegate:
const SliverGridDelegateWithFixedCrossAxisCount(
crossAxisCount: 20,
),
itemBuilder: (BuildContext context, int index) {
if (snakePosition.contains(index)) {
return Center(
child: Container(
padding: const EdgeInsets.all(2),
child: ClipRRect(
borderRadius: BorderRadius.circular(5),
child: Container(
color: Colors.white,
),
),
),
);
}
if (index == food) {
return Container(
padding: const EdgeInsets.all(2),
child: ClipRRect(
borderRadius: BorderRadius.circular(5),
child: const Center(
child: Icon(
Icons.fastfood,
size: 15,
color: Colors.yellow,
)),
),
);
} else {
return Container(
padding: const EdgeInsets.all(2),
child: ClipRRect(
borderRadius: BorderRadius.circular(5),
child: Container(
color: Colors.transparent,
),
),
);
}
},
),
],
),
),
),
!playing
? Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
Container(
decoration: BoxDecoration(
borderRadius: BorderRadius.circular(50),
color: x1 ? Colors.green : Colors.transparent,
),
margin: const EdgeInsets.all(10),
child: TextButton(
onPressed: () {
setState(() {
x1 = true;
x2 = false;
x3 = false;
speed = 300;
});
},
child: const Text(
'X1',
style: TextStyle(
color: Colors.white,
),
)),
),
Container(
decoration: BoxDecoration(
borderRadius: BorderRadius.circular(50),
color: x2 ? Colors.green : Colors.transparent,
),
margin: const EdgeInsets.all(10),
child: TextButton(
onPressed: () {
setState(() {
x2 = true;
x1 = false;
x3 = false;
speed = 200;
});
},
child: const Text(
'X2',
style: TextStyle(
color: Colors.white,
),
)),
),
Container(
decoration: BoxDecoration(
borderRadius: BorderRadius.circular(50),
color: x3 ? Colors.green : Colors.transparent,
),
margin: const EdgeInsets.all(10),
child: TextButton(
onPressed: () {
setState(() {
x3 = true;
x1 = false;
x2 = false;
speed = 100;
});
},
child: const Text(
'X3',
style: TextStyle(
color: Colors.white,
),
)),
),
Container(
height: 20,
width: 1,
color: Colors.white70,
),
OutlinedButton(
onPressed: () {
startGame();
},
child: Row(
children: const [
Text(
'Start',
style: TextStyle(color: Colors.yellow),
),
SizedBox(
width: 5,
),
Icon(Icons.play_arrow, color: Colors.yellow),
],
))
],
)
: Container(
height: 50,
color: Colors.white12,
child: Center(
child: OutlinedButton(
onPressed: () {
setState(() {
endGame = true;
});
},
child: Text(
'End the Game and show result',
style: Theme.of(context)
.textTheme
.subtitle2!
.copyWith(color: Colors.white),
),
),
),
),
],
),
);
}