Retour d'expérience
Sur FarmAttack
FarmAttack
Un jeu multi-joueurs en temps réel en 3D isométrique
data:image/s3,"s3://crabby-images/bb27f/bb27f2f6e3ba6f717013dbab1b71c5cb68a46e09" alt=""
Problématiques
- Un jeu sur navigateur
- 3D isométrique
- Une plateforme pour gérer le contenu du jeu
- Authentification
- Multi-joueurs, interaction entre joueurs.
- Un monde persistant, les joueurs évoluent dans un environnement dynamique.
- Carte infinie
Technologies
Une plateforme pour gérer le contenu du jeu
data:image/s3,"s3://crabby-images/bd43a/bd43a7a685a3dfb59f18727fad5eddf53393ea72" alt=""
Node.js® is a platform built on Chrome's JavaScript runtime for easily building fast, scalable network applications. Node.js uses an event-driven, non-blocking I/O model that makes it lightweight and efficient, perfect for data-intensive real-time applications that run across distributed devices.
NodeJS
console.log("first");
setTimeout(function() {
console.log("second");
}, 0);
console.log("third");
// output
first
third
second
ExpressJS
// app.js
var express = require('express');
var app = express();
app.get('/', function (req, res) {
res.send('<h1>Hello World!</h1>');
});
var server = app.listen(3000, function () {
console.log('Example app listening at http://localhost:3000');
});
# Install
$ npm install express
# Launch application
$ npm app.js
data:image/s3,"s3://crabby-images/9fd04/9fd04c3640db2cb07cd2e50ef1beb0e869344ef9" alt=""
Un moteur de template ?
// app.js
var express = require('express');
var app = express();
app.set('view engine', 'jade'); // default value
Twig aussi !
# Install
$ npm install twig
// app.js
require("twig");
var express = require('express');
var app = express();
app.set('view engine', 'twig');
app.set('views', __dirname + '/../view');
// app.js
// ...
app.get('/', function(req, res){
res.render('index', {
message : "Hello World"
});
});
data:image/s3,"s3://crabby-images/9d221/9d22118f7a2bd0e8389bc4a3b9b94edc1abb8e44" alt=""
Message of the moment: <b>{{ message }}</b>
view/index.twig
Authentification ?
data:image/s3,"s3://crabby-images/77ddd/77ddd5d01aa59bea77de99877abde2b7d47929bf" alt=""
$ wget http://download.redis.io/releases/redis-3.0.0.tar.gz
$ tar xzf redis-3.0.0.tar.gz
$ cd redis-3.0.0
$ make
# or
$ brew install redis
# etc..
$ redis-server
C'est tout !
Installation
$ npm install redis
Ba voui ba non !
// app.js
var express = require('express');
var app = express();
var redis = require('redis');
app.get('/login', function (req, res) {
res.render('login');
});
app.post('/login', function (req, res) {
player = playerModel.findOne( //... some logic);
var token = generateToken(32);
redis.set(player.username, token);
res.cookie('__secretTokenDontStealMeDontInjectMePlease#Yolo', token);
res.redirect('/');
});
Multi-joueurs, interaction entre joueurs
Socket.IO enables real-time bidirectional event-based communication.
# Install
$ npm install socket.io
// app.js
var app = require('express')();
var http = require('http').Server(app);
var io = require('socket.io')(http);
app.get('/', function(req, res){
res.render('index');
});
io.on('connection', function(socket){
console.log('a user connected');
socket.on('disconnect', function(){
console.log('user disconnected');
});
});
<html>
<body>
<script src="https://cdn.socket.io/socket.io-1.2.0.js"></script>
<script>
var socket = io();
</script>
<body>
</html>
data:image/s3,"s3://crabby-images/638a0/638a00b0283cacff592dd3ada9b99f9b652c1a19" alt=""
Et maintenant ?
// app.js
var app = require('express')();
var http = require('http').Server(app);
var io = require('socket.io')(http);
app.get('/', function(req, res){
res.sendFile(__dirname + '/index.html');
});
io.on('connection', function(socket){
socket.on('chat message', function(msg){
// Broadcast
io.emit('chat message', msg);
});
});
http.listen(3000, function(){
console.log('listening on *:3000');
});
<body>
<ul id="messages"></ul>
<form action="">
<input id="m" autocomplete="off" /><button>Send</button>
</form>
<script src="https://cdn.socket.io/socket.io-1.2.0.js"></script>
<script src="http://code.jquery.com/jquery-1.11.1.js"></script>
<script>
var socket = io();
$('form').submit(function(){
socket.emit('chat message', $('#m').val());
$('#m').val('');
return false;
});
socket.on('chat message', function(msg){
$('#messages').append($('<li>').text(msg));
});
</script>
</body>
3D isométrique
data:image/s3,"s3://crabby-images/1256b/1256b779b64e302fe9199872e0bd0aaad4ac3012" alt=""
data:image/s3,"s3://crabby-images/bb3d5/bb3d52d8d5aea21f7d570eedc8f61f71ba4365a7" alt=""
data:image/s3,"s3://crabby-images/ce259/ce259c454b4ce76bb669b8977a4c27dfc490df06" alt=""
data:image/s3,"s3://crabby-images/580f9/580f993f0a288d026fb2c35a7bf91ca6b0a35f21" alt=""
data:image/s3,"s3://crabby-images/07729/07729be096a3a9d74c97538635b4284e67c74c18" alt=""
- Représentation d'un objet 3D en 2D
- Pas de distortions
- Nécessite peu de ressources
- Plus simple que la 3D
data:image/s3,"s3://crabby-images/006c0/006c09d845237aa4aef4f085d032533c34bc78e9" alt=""
Les tuiles
data:image/s3,"s3://crabby-images/f8f9f/f8f9ff96483425897eb904432a30e94c4a63c371" alt=""
data:image/s3,"s3://crabby-images/45fea/45fea4e068ac486d2b53d9d1293ed740ae8fd0f0" alt=""
Dessiner une tuile
data:image/s3,"s3://crabby-images/29ae1/29ae12a8b4ad3a09fac6130dd63aea721fb40415" alt=""
Tileset
Générer la carte
isoToCart: function(isoX, isoY)
{
var height = 50;
var width = 100;
var x = (isoX - isoY) * width / 2;
var y = (isoX + isoY) * height / 2;
return {
x: x,
y: y
};
},
data:image/s3,"s3://crabby-images/034bc/034bcddf189e8ede96a81c62cf39f6a3a8eaa41f" alt=""
data:image/s3,"s3://crabby-images/034bc/034bcddf189e8ede96a81c62cf39f6a3a8eaa41f" alt=""
data:image/s3,"s3://crabby-images/034bc/034bcddf189e8ede96a81c62cf39f6a3a8eaa41f" alt=""
data:image/s3,"s3://crabby-images/034bc/034bcddf189e8ede96a81c62cf39f6a3a8eaa41f" alt=""
data:image/s3,"s3://crabby-images/034bc/034bcddf189e8ede96a81c62cf39f6a3a8eaa41f" alt=""
data:image/s3,"s3://crabby-images/034bc/034bcddf189e8ede96a81c62cf39f6a3a8eaa41f" alt=""
Gestion de la profondeur :
i + j
data:image/s3,"s3://crabby-images/034bc/034bcddf189e8ede96a81c62cf39f6a3a8eaa41f" alt=""
data:image/s3,"s3://crabby-images/034bc/034bcddf189e8ede96a81c62cf39f6a3a8eaa41f" alt=""
8 directions
data:image/s3,"s3://crabby-images/8c76c/8c76c4800f423d047996447020332dd885146414" alt=""
data:image/s3,"s3://crabby-images/0ec14/0ec148c8d0a8775809956ad2943150ae1d2d8050" alt=""
Les personnages
data:image/s3,"s3://crabby-images/d4df9/d4df9a80186ee0ed59ebfcad2455676d37ce2226" alt=""
Déplacements et colision
- Algorithme A*
- Tuiles non-walkable
EaselJS
data:image/s3,"s3://crabby-images/a26f6/a26f632afbd7f98b8a2fe40dbcd81b176fda5d1f" alt=""
DisplayObject
- Tout les objets visibles sont des DisplayObject.
- Images : Bitmap
- Vecteurs : Shape
- Animations : SpriteSheet
- Text
- Container
var bitmap = new createjs.Bitmap("imagePath.jpg");
var shape = new createjs.Shape();
shape.graphics.beginFill("#ff0000").drawRect(0, 0, 100, 100);
Container
- Permet de grouper des éléments graphiques
- Permet de cacher un groupe d'objets
- Permet d'effectuer du sorting
var container = new createjs.Container();
container.addChild(bitmapInstance, shapeInstance);
container.x = 100;
stage = new createjs.Stage("canvas");
Stage
- Wrap l'element canvas
- Container principal
- stage.update();
Ticker
var stage, circle;
function init() {
stage = new createjs.Stage("canvas");
circle = new createjs.Shape();
circle.graphics.beginFill("red").drawCircle(0, 0, 40);
circle.y = 50;
stage.addChild(circle);
createjs.Ticker.addEventListener("tick", tick);
createjs.Ticker.setFPS(30);
}
function tick(event) {
circle.x = circle.x + 5;
if (circle.x > stage.canvas.width) { circle.x = 0; }
stage.update();
}
Architecture
data:image/s3,"s3://crabby-images/a2feb/a2feb1b9ed808001888b2817fb370e776cfbd68c" alt=""
Carte infini !?
data:image/s3,"s3://crabby-images/f058f/f058f935f028c2b46e4a67e86412c2359dbb00b2" alt=""
# Install
$ npm install mongoose
// db.js
var mongoose = require('mongoose');
mongoose.connect('mongodb://localhost/test');
var Cat = mongoose.model('Cat', { name: String });
var kitty = new Cat({ name: 'Zildjian' });
kitty.save(function (err) {
if (err) // ...
console.log('meow');
});
// model/tile.js
var tileSchema = new mongoose.Schema({
_groupTileParent: {type: Schema.Types.ObjectId, ref: 'groupTile'},
x: Number,
y: Number,
walkable: {type: Boolean, default: true},
owner: {type: String, default: 'NEUTRAL'},
fertility: Number,
maxFertility: Number,
humidity: Number,
crop: String,
cropMaturity: Number,
imageIndex: Number,
building: {
name: String,
crop: [{
name: String,
quantity: Number
}]
}
});
var groupTileSchema = new mongoose.Schema({
player : {type: String, default: 'NULL'},
enable : {type: Boolean, default: false},
tiles : [{type: Schema.Types.ObjectId, ref: 'tile'}],
created_at: {type: Date, default: Date.now}
});
FarmAttack
By skigun
FarmAttack
- 479