Build a Multiplayer Quiz Game With Vue.js
As developers, deploying applications and websites can be a pain point at times and we generally tend to prefer solutions that are easy and scalable.
Hosting solutions that require SSHing and having to make configurations and a million things to do before deploying can (and will) get tedious after a while. This is where Firebase Hosting comes in.
We’ll build a static website and deploy it to the web using Firebase Hosting. We are going to build a realtime multiplayer trivia game, the game is going to work like the popular multiplayer game QuizUp.
Here’s how the game is going to work. The questions are going to be gotten from the lyrics from Hamilton: An American Musical and users have to answer correctly the name of the character in the musical who sang that line. The game tests the user’s knowledge of Hamilton and users can play for as long as they like. See a demo below.
https://vimeo.com/230121610
Our game will be built with Vue.js and will use Pusher’s Client Events and Presence Channels to make sure a user’s move is shown realtime.
What are Presence Channels?
Pusher’s Presence channels expose the additional feature of an awareness of who is subscribed to that channel. This gives developers the ability to build features like chat rooms, collaborators on a document and multiplayer games.
All Presence Channels must be subscribed with the prefix of presence-
and as with private channels a HTTP Request is made to a configurable authentication URL to determine if the current user has permissions to access the channel.
What are Client Events?
Client events are simply a way in which events can be triggered directly from the client-side as opposed to triggering from the backend server. Client Events are used in instances where some actions may not need validation or persistence and can go directly via the socket to all the other clients connected to the channel.
In our case, we use the Client Events to update a user on the current score of the other user playing the game.
Client Events have a number of enforced restrictions to ensure that the user subscribing to the channel is an authenticated user:
- Client events must be enabled for the application. You can do this in the Settings tab for your app within the Pusher dashboard.
- The user must be subscribed to the channel that the event is being triggered on.
- Client events can only be triggered on private and presence channels because they require authentication.
- Client events must be prefixed by
client-
. Events with any other prefix will be rejected by the Pusher server, as will events sent to channels to which the client is not subscribed.
You can read more on Client Events by going through the documentation.
Setting Up Pusher
Log in to your dashboard (or create a free account if you don’t already have one) and create a new app. Copy your app_id
, key
, secret
and cluster
and store them somewhere as we’ll be needing them later. One more thing. In your dashboard, navigate to the App Settings tab and make sure the Enable Client Events checkbox is checked. This lets clients communicate directly with each other.
Initializing the Vue.js Application
Vue.js is the preferred JavaScript framework to build the game because of its popularity and low barrier to getting started. There are some Vue.js tutorials on Pusher which you can read here, here and here.
We’re going to be using the vue-cli to scaffold a Vue.js project. The vue-cli
is a simple CLI for scaffolding Vue.js projects. It ships with many templates like webpack, browserify, pwa and simple. We’ll install vue-cli
and then use it to bootstrap the app using the webpack template, with the following commands:
npm install -g vue-cli
vue init webpack hamiltonlyrics
This creates a Vue.js app inside the a folder titled hamiltonlyrics
. Navigate into the folder and run the command npm run dev
to see the Vue.js application.
Setting up a Node.js Server
As explained above, Client Events require authentication to make sure a user is subscribed to the channel. Therefore, we are going to create a Node.js server so that Client Events can have an authentication route.
Let’s install the modules we’ll need for the Node.js server.
npm i express body-parser pusher
In the root of the project directory, create a file named server.js
and type in the following code:
// server.js
const express = require('express')
const path = require('path')
const bodyParser = require('body-parser')
const app = express()
const Pusher = require('pusher')
const crypto = require('crypto')
//initialize Pusher with your appId, key and secret
const pusher = new Pusher({
appId: 'APP_ID',
key: 'APP_KEY',
secret: 'APP_SECRET',
cluster: 'YOUR_CLUSTER',
encrypted: true
})
// Body parser middleware
app.use(bodyParser.json())
app.use(bodyParser.urlencoded({ extended: false }))
// The code below helps to fix any potential CORS issue.
app.use((req, res, next) => {
// Website you wish to allow to connect
res.setHeader('Access-Control-Allow-Origin', '*')
// Request methods you wish to allow
res.setHeader('Access-Control-Allow-Methods', 'GET, POST, OPTIONS, PUT, PATCH, DELETE')
// Request headers you wish to allow
res.setHeader('Access-Control-Allow-Headers', 'X-Requested-With,content-type')
// Set to true if you need the website to include cookies in the requests sent
// to the API (e.g. in case you use sessions)
res.setHeader('Access-Control-Allow-Credentials', true)
// Pass to next layer of middleware
next()
})
// Index API route for the Express app
app.get('/', (req, res) => {
res.send('Welcome')
})
// API route used by Pusher as a way of authenticating users
app.post('/pusher/auth', (req, res) => {
let socketId = req.body.socket_id
let channel = req.body.channel_name
// Generate a random string and use as presence channel user_id
let presenceData = {
user_id: crypto.randomBytes(16).toString('hex')
}
let auth = pusher.authenticate(socketId, channel, presenceData)
res.send(auth)
})
// Set port to be used by Node.js
app.set('port', (5000))
app.listen(app.get('port'), () => {
console.log('Node app is running on port', app.get('port'))
})
In the code block above, Pusher is initialized with the dashboard credentials. The /pusher/auth
route is also created.
Now we can simply run node server.js
and the Node.js app should be up and running. Before we go on let’s add the command above to the existing scripts
object in the package.json
file so we don’t have to type in the command every time. Open up the package.json
file and edit the dev
line inside the scripts
object with the one below.
"dev": "nodemon server.js & node build/dev-server.js"
Building the game
Let’s get started on building the game. We’ll be working with two files throughout the course of this tutorial, a Home.vue
file and a ChannelDetails.vue
file.
Navigate to the components
folder inside the src
folder and create a new file called ChannelDetails.vue
. This will contain the JavaScript code that establishes a connection to Pusher from the client side. Open the file and type in the following code:
<script>
import Pusher from 'pusher-js'
const pusher = new Pusher('APP_KEY', {
cluster: 'YOUR_CLUSTER',
encrypted: true,
authEndpoint: 'http://localhost:5000/pusher/auth'
})
export default ({
getPresenceID () {
// This function checks the address bar of the browser for params
let getQueryString = (field, url) => {
let href = url ? url : window.location.href
let reg = new RegExp('[?&]' + field + '=([^&#]*)', 'i')
let string = reg.exec(href)
return string ? string[1] : null
}
// Appends 'presence' to the result
let id = getQueryString('id')
id = 'presence-' + id
return id
},
subscribeToPusher () {
let presenceid = this.getPresenceID()
let channel = pusher.subscribe(presenceid)
return channel
}
})
</script>
If you use ESLint, you should be getting a warning that the Pusher JS library has not been installed. That can be installed by running this command npm i pusher-js
. So what’s happening up there?
Firstly, Pusher is imported and a connection is established using credentials like APP_KEY, and CLUSTER. An authEndpoint is added to the Pusher instance. The authEndpoint is the endpoint Pusher uses to authenticate users.
Secondly, there are two functions above which are being exported. The first function getPresenceID()
checks the address bar of the browser for URL parameters and then appends presence
to the result. This is done so that the channel name will always have a prefix of presence-
since we are using Presence channels.
The second function uses the result of the getPresenceID()
function and uses it to subscribe to a channel (a Presence Channel to be specific).
The next thing to do is to start writing code for the game itself. Open up the App.vue
file inside the src
folder and make sure its content is similar to the code below:
<template>
<div id="app">
<router-view></router-view>
</div>
</template>
<script>
export default {
name: 'app'
}
</script>
<style>a
html {
background: #7fd4d3;
}
body {
background: #7fd4d3;
padding: 20px;
}
#app {
height: 100vh;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
text-align: center;
color: #fff;
}
.fade-enter-active, .fade-leave-active {
transition: opacity .5s
}
.fade-enter, .fade-leave-to /* .fade-leave-active in <2.1.8 */ {
opacity: 0
}
</style>
Navigate to the src/components
directory and you should see a Hello.vue
file. You can either delete that file or rename to Home.vue
as we will be needing a Home.vue
file inside the components folder. Open up the file and type in/replace with the code below:
<script>
// Import ChannelDetails component created above
import ChannelDetails from '@/components/ChannelDetails'
// An array that holds the lyrics questions and their correct answers. All questions can be seen here https://gist.github.com/yomete/2d851c2adc008a9763a0db9f85879083
const lyrics = [
{
lyric: 'When he was ten his father split, full of it, debt-ridden. Two years later, see Alex and his mother bed-ridden. Half-dead sittin\' in their own sick, the scent thick',
options: [{name: 'Aaron Burr', correct: false}, {name: 'James Madison', correct: false}, {name: 'John Laurens', correct: false}, {name: 'Eliza Hamilton', correct: true}],
answer: 'Eliza Hamilton'
},
{
lyric: 'I am sailing off to London. I’m accompanied by someone who always pays. I have found a wealthy husband. Who will keep me in comfort for all my days. He is not a lot of fun, but there’s no one',
options: [{name: 'Eliza', correct: false}, {name: 'Peggy', correct: false}, {name: 'Angelica', correct: true}, {name: 'Maria', correct: false}],
answer: 'Angelica'
}
]
export default {
name: 'home',
data () {
return {
// This holds the current presence-id
presenceid: null,
// This checks if a question has been answered, default to false
hasAnswered: false,
// This holds the current question
question: null,
// This holds the options for the current question
options: null,
// This holds the correct answer for the current question
correctanswer: null,
// This is used for a countdown timer
count: null,
// Number of players in the game
players: 1,
// This checks if there's a second player, it becomes true when players = 2
secondplayer: false,
// This holds the player data for both players
playerdata: {
one: {
id: null,
score: 0,
userid: null
},
two: {
id: null,
score: 0,
userid: null
}
},
// This holds the userid for the current player
userid: null,
// This holds the current URL of the game
url: null
}
},
created () {
this.fetchData()
},
methods: {
fetchData () {},
getUniqueId () {}
checkPresenceID () {}
checkAnswer (item) {}
getRandomQuestions (array, count) {}
getNewQuestion () {}
}
}
</script>
<!-- Add "scoped" attribute to limit CSS to this component only -->
<style scoped>
.home {
display: flex;
align-items: center;
justify-content: center;
height: 100vh;
}
h1 {
font-size: 3rem;
font-weight: bold;
}
p {
font-size: 1.5rem;
margin: 0 0 20px 0;
}
.play--button {
background-color: white;
color: #7fd4d3;
font-weight: bold;
border-radius: 20px;
letter-spacing: 1px;
padding: 20px;
transition: all .3s ease;
text-shadow: 0 1px 3px rgba(36,180,126,.4);
text-transform: uppercase;
box-shadow: 0 4px 6px rgba(50,50,93,.11), 0 1px 3px rgba(0,0,0,.08);
}
.play--button:hover {
background-color: white;
color: #7fd4d3;
transform: translateY(-1px);
box-shadow: 0 7px 14px rgba(50,50,93,.1), 0 3px 6px rgba(0,0,0,.08);
}
.fade-enter-active, .fade-leave-active {
transition: opacity .5s
}
.fade-enter, .fade-leave-to {
opacity: 0
}
a {
color: #fff;
}
p {
color: #fff;
}
h1 {
font-size: 3rem;
font-weight: bold;
text-align: center;
}
.fade-enter-active, .fade-leave-active {
transition: opacity .5s
}
.fade-enter, .fade-leave-to /* .fade-leave-active in <2.1.8 */ {
opacity: 0
}
.play--button {
background-color: white;
color: #7fd4d3;
font-weight: bold;
border-radius: 20px;
letter-spacing: 1px;
padding: 20px;
transition: all .3s ease;
text-shadow: 0 1px 3px rgba(36,180,126,.4);
text-transform: uppercase;
box-shadow: 0 4px 6px rgba(50,50,93,.11), 0 1px 3px rgba(0,0,0,.08);
position: absolute;
top: 20px;
right: 20px;
z-index: 5;
}
.play--button:hover {
background-color: white;
color: #7fd4d3;
transform: translateY(-1px);
box-shadow: 0 7px 14px rgba(50,50,93,.1), 0 3px 6px rgba(0,0,0,.08);
}
.hamilton--header--text {
margin-top: 50px;
}
.hamilton--inner {
margin-top: 20px;
}
.hamilton--inner .left{
text-align: left;
}
.hamilton--inner .right{
text-align: right;
}
.title {
font-weight: bold;
}
.hamilton--lyrics--text {
width: 600px;
margin: 0 auto;
}
.hamilton--lyrics--text p {
font-weight: bold;
}
.hamilton--answers a{
display: block;
border: 3px solid white;
border-radius: 50px;
margin: 20px auto;
width: 500px;
padding: 10px;
}
.wronganswer {
background-color: #ec6969;
border: none !important;
opacity: 0.4;
transition: background-color 0.5s ease;
}
.correctanswer {
background-color: #00c4a7;
border: none !important;
transition: background-color 0.5s ease;
}
</style>
In the code block above, we set up the foundation for the game and how it’s going to work. Inside the <style>
tag and inside the <script>
tag, there are a couple of functions that we will need to create and add logic to.
Let’s take a look at the functions we need to create.
fetchData()
This function is called inside the created
hook and that means it will always be called whenever the instance has been created. Let’s write the code for this function.
// Sets the data instance presenceid variable to the result of the getUniqueId function
this.presenceid = this.getUniqueId()
// This checks if there's no presence ID in the URL via the checkPresenceID function and appends the presenceid to the current URL so that we can have the URL end with a parameter like this https://hamilton-lyrics.firebaseapp.com/#/?id=agbew0gz
if (!this.checkPresenceID()) {
var separator = (window.location.href.indexOf('?') === -1) ? '?' : '&'
window.location.href = window.location.href + separator + this.presenceid
}
// Sets the data instance url variable to the current URL.
this.url = window.location.href
// Gets a new question via the getNewQuestion() function
this.getNewQuestion()
// The channel variable is set to to the subscribeToPusher function in ChannelDetails.
let channel = ChannelDetails.subscribeToPusher()
// The pusher:member_added event is triggered when a user joins a channel. We increase the number of players by one and also set the secondplayer boolean to true.
channel.bind('pusher:member_added', members => {
this.players += 1
this.secondplayer = true
})
// Once a subscription has been made to a presence channel, an event is triggered with a members iterator.
channel.bind('pusher:subscription_succeeded', members => {
// This checks if its just one player online and sets them up as player one and the required info for the game
if (members.count === 1 && !this.playerdata.one.id) {
this.playerdata.one.id = members.myID
this.playerdata.one.userid = 1
this.userid = 1
// This checks if there's a player online already and sets the new player as player two.
} else if (members.count === 2) {
this.secondplayer = true
this.playerdata.two.id = members.myID
this.playerdata.two.userid = 2
this.userid = 2
}
})
// The pusher:member_removed is triggered when a user leaves a channel. We decrease the number of players by one and also set the secondplayer boolean to false.
channel.bind('pusher:member_removed', member => {
this.players -= 1
if (member.count === 1) {
this.secondplayer = false
}
})
// This function receives new data from Pusher and updates the exisiting scores. This is what updates each player's score in realtime.
channel.bind('client-send', (data) => {
if (this.userid === 1) {
this.playerdata.two.score = data.data.two.score
} else if (this.userid === 2) {
this.playerdata.one.score = data.data.one.score
}
})
getUniqueId()
This function simply generates random alphanumeric characters and adds a prefix of id=
to the result.
getUniqueId () { return 'id=' + Math.random().toString(36).substr(2, 8)}
checkPresenceId()
This function checks the address bar of the browser for URL parameters, in this case, any parameter that starts with ?id=
prefix and then returns the alphanumeric character at the end of the prefix. For example, this URL https://hamilton-lyrics.firebaseapp.com/#/?id=agbew0gz
will return agbew0gz
.
checkPresenceID () {
let getQueryString = (field, url) => {
let href = url ? url : window.location.href
let reg = new RegExp('[?&]' + field + '=([^&#]*)', 'i')
let string = reg.exec(href)
return string ? string[1] : null
}
let id = getQueryString('id')
return id
}
checkAnswer()
This function is used to check if the answer chosen is correct or incorrect. If the chosen answer is correct, 10 points will be added to the current score and if the answer is incorrect, 10 points will be deducted. The score is also sent to other subscribers of the channel via the channel.trigger()
function. At the end of it all, a new question is gotten via the getNewQuestion()
function.
checkAnswer (item) {
let channel = ChannelDetails.subscribeToPusher()
this.hasAnswered = true
if (item.name === this.correctanswer) {
if (this.userid === 1) {
this.playerdata.one.score += 10
} else if (this.userid === 2) {
this.playerdata.two.score += 10
}
} else {
if (this.userid === 1) {
this.playerdata.one.score = Math.max(0, this.playerdata.one.score -= 10)
} else if (this.userid === 2) {
this.playerdata.two.score = Math.max(0, this.playerdata.two.score -= 10)
}
}
channel.trigger('client-send', {data: this.playerdata})
this.count = 3
let countdown = setInterval(() => {
this.count -= 1
if (this.count === 0) {
clearInterval(countdown)
this.getNewQuestion()
}
}, 1000)
}
getRandomQuestions()
This function is used to select questions randomly from the lyrics
array that holds the various questions. It takes in two arguments, array
and count
. array
would be the array we are trying to pick a random item from (lyrics
) and count
is the number of item to be chosen (1 item). This function is called in the getNewQuestion()
function below.
getRandomQuestions (array, count) {
let length = array.length
let randomIndexes = []
let randomItems = []
let index, item
count = count | 1
while (count) {
index = Math.floor(Math.random() * length)
if (randomIndexes.indexOf(index) === -1) {
count--
randomIndexes.push(index)
}
}
randomIndexes.forEach((index) => {
item = array.slice(index, index + 1).pop()
randomItems.push(item)
})
if (randomItems.length === 1) {
return randomItems.pop()
} else {
return randomItems
}
}
getNewQuestion()
This function is used to get a new question for the game. It utilizes the getRandomQuestions()
to get a new random question and sets it to the question
variable. It also uses the question’s data in the question
variable to initialize the various data instances.
getNewQuestion () {
let question = this.getRandomQuestions(lyrics, 1)
this.question = question
this.options = question.options
this.correctanswer = question.answer
this.hasAnswered = false
}
We are done with the functions. Let’s create the template
tag and write the HTML code that will display the view for the game.
<template>
<transition name="fade">
<div class="home" v-if="!secondplayer">
<div class="inner">
<h1>Do you know your Hamilton Lyrics?</h1>
<p>Test your knowledge of Hamilton: An American Musical by guessing who sang what lyric.</p>
<p>Invite a second player by sending them this link {{url}}.</p>
</div>
</div>
<div class="play" v-if="secondplayer">
<div>
<div class="container hamilton--header--text">
<h1>Do you know your Hamilton Lyrics?</h1>
<div class="columns hamilton--inner">
<div class="column is-half left">
<p class="title">User 1</p>
<p class="subtitle">Total Score: {{playerdata.one.score}}</p>
</div>
<div v-if="secondplayer" class="column is-half right">
<p class="title">User 2</p>
<p class="subtitle">Total Score: {{playerdata.two.score}}</p>
</div>
</div>
<div class="hamilton--lyrics--text">
<p>{{question.lyric}}
</p>
<div class="hamilton--answers">
<a v-bind:class="{ 'wronganswer': hasAnswered && !item.correct, 'correctanswer': hasAnswered && item.correct}" @click="checkAnswer(item)" v-for="(item, index) in options">{{item.name}}</a>
</div>
</div>
</div>
</div>
</div>
</transition>
</template>
In the code above, we enclose everything in a transition
tag with an attribute of fade
. There are two div
sections that display conditionally. The div
tag with a class of home
is shown when there’s just one player online and the div
tag with a class of play
is only shown when two players are online.
At the end of it all, your Home.vue
file should look like this.
Setting up Firebase Hosting
Now that our application is ready and working well, let’s deploy the application using Firebase Hosting.
Getting started with Firebase Hosting is straightforward. Go over to console.firebase.google.com and create a new account or sign in if you already have an account.
Since this is a platform hosted by Google, you’ll need a Gmail account to be able to sign up and use the Firebase Console
Your dashboard should look like this (if you’re a new user). Let’s add a new project by clicking on the Add Project button.
That opens up a modal box that asks you to give your project a name and also choose your region. Once that’s done, you should be redirected to the project’s dashboard which looks like this.
The dashboard menu on the left shows all the Firebase services you can use in your application. Before we start deploying apps with Firebase Hosting, we need to install the Firebase CLI using npm.
npm install -g firebase-tools
If you’ve previously installed Firebase command line tools, run the install command again to make sure you have the latest version. Once the Firebase CLI has been successfully installed, we can deploy apps to Firebase Hosting with a single command.
Next step is to sign in to Google from the terminal so Firebase knows which account to use. Run the command firebase login
in your terminal. This process takes you to a login page in a browser where you enter your Google credentials and you are then logged in. Now Firebase is installed on our computer and we can begin deployment.
Preparing the Vue.js app for deployment
Now that we are done with the development of the app, it’s time to deploy the application to production via Firebase Hosting. How exactly do we do that? First of all, we need to build the Vue.js app for production and then run the Firebase deploy command. Let’s get started on that by running the command below.
npm run build
The command above helps to minify JS, HTML, and CSS. All static assets are also compiled with version hashes for efficient long-term caching, and a production index.html
is auto-generated with proper URLs to these generated assets.
Once the command is done with its process, the production-ready app can be found in the dist
folder. That is where the firebase deploy
command will be used.
Deploying to Firebase
We’ll need to initiate Firebase for this project, specifically inside the dist
folder. So run the command firebase init
. That command prompts the following:
- You’ll be prompted to choose which of the Firebase CLI feature you want to use, choose Hosting.
- You’ll be prompted to associate the current project directory with a Firebase project. Choose a Firebase project or create a new one.
- You’ll be prompted to type in the name of the folder you want to use as a public directory. This public directory is the folder (relative to your project directory) that will contain Hosting assets to be uploaded with firebase deploy. In this case, the name of the folder is
dist
. - You’ll be prompted to choose whether to configure the project as a single-page app. Choose Y.
The initialization process should be completed and we can run the firebase deploy
command now. When the deploy process is done, a live URL will be generated automatically by Firebase, in this case, hamilton-lyrics.firebaseapp.com. Firebase allows you to connect a domain to your Hosting instance so you can use a custom domain for your applications.
Conclusion
In this tutorial, we learnt how to deploy static pages to Firebase Hosting and making it realtime by using Pusher. We saw how to implement Pusher’s Presence Channels by using it to identify two different users online and then using it to build a multiplayer game. We also learnt how Client Events work, how to trigger events from the client side as opposed to triggering from a backend server.
Firebase offers a slew of services that can help you to build apps faster and you can read about them on the Firebase site.
If you want to go through the source code for the game above, you can do that on Github. You can also see the live demo at hamilton-lyrics.firebaseapp.com.
This post first appeared on the Pusher blog.