not-gwent-online/server/Battleside.js

926 lines
24 KiB
JavaScript

var DeckData = require("../assets/data/deck");
var Deck = require("./Deck");
var Hand = require("./Hand");
var Card = require("./Card");
var Field = require("./Field");
var _ = require("underscore");
var Promise = require("jquery-deferred");
var Battleside;
Battleside = (function() {
var Battleside = function(user, n, battle) {
if(!(this instanceof Battleside)) {
return (new Battleside(user, n, battle));
}
/**
* constructor here
*/
var deck = user.getDeck();
var self = this;
this._isWaiting = true;
this.socket = user.socket;
this.cm = battle.cm;
this.field = {};
this.field[Card.TYPE.LEADER] = Field(this);
this.field[Card.TYPE.CLOSE_COMBAT] = Field(this, true);
this.field[Card.TYPE.RANGED] = Field(this, true);
this.field[Card.TYPE.SIEGE] = Field(this, true);
this.n = n ? "p2" : "p1";
this._name = user.getName();
this.battle = battle;
this.hand = Hand();
this.deck = Deck(deck, this);
this._discard = [];
this.runEvent = this.battle.runEvent.bind(this.battle);
this.on = this.battle.on.bind(this.battle);
this.off = this.battle.off.bind(this.battle);
this.receive("activate:leader", function() {
if(self._isWaiting) return;
if(self.isPassing()) return;
var leaderCard = self.getLeader();
if(leaderCard.isDisabled()) return;
//console.log("leader activated");
var ability = leaderCard.getAbility();
ability.onActivate.apply(self, [leaderCard]);
leaderCard.setDisabled(true);
self.battle.sendNotification(self.getName() + " activated " + leaderCard.getName() + "! (leadercard)");
self.update();
if(ability.waitResponse) {
return;
}
//self.runEvent("NextTurn", null, [self.foe]);
self.endTurn();
})
this.receive("play:cardFromHand", function(data) {
if(self._isWaiting) return;
if(self.isPassing()) return;
var cardID = data.id;
var card = self.hand.getCard(cardID);
self.playCard(card);
})
this.receive("decoy:replaceWith", function(data) {
if(self._isWaiting) return;
var card = self.findCardOnFieldByID(data.cardID);
/*if(card === -1) throw new Error("decoy:replace | unknown card");*/
if(card === -1) {
console.log("decoy:replace | unknown card: ", card);
self.sendNotificationTo(self, "Possible bug occured: unknown card was chosen by playing decoy ability.");
//self.endTurn();
return;
}
self.runEvent("Decoy:replaceWith", self, [card]);
})
this.receive("cancel:decoy", function() {
self.off("Decoy:replaceWith");
})
this.receive("set:passing", function() {
self.setPassing(true);
self.update();
self.battle.sendNotification(self.getName() + " passed!");
//self.runEvent("NextTurn", null, [self.foe]);
self.endTurn();
})
this.receive("medic:chooseCardFromDiscard", function(data) {
if(!data) {
//self.runEvent("NextTurn", null, [self.foe]);
self.endTurn();
return;
}
var cardID = data.cardID;
var card = self.getCardFromDiscard(cardID);
if(card === -1) {
console.log("medic:chooseCardFromDiscard | unknown card: ", card);
self.sendNotificationTo(self, "Possible bug occured: unknown card was chosen by playing medic ability.");
self.endTurn();
return;
}
self.removeFromDiscard(card);
self.playCard(card);
})
this.receive("emreis_leader4:chooseCardFromDiscard", function(data) {
if(!data) {
self.endTurn();
self.sendNotificationTo(self.foe, self.getName() + " takes no card from your discard pile (or there wasn't any card to choose)");
//self.runEvent("NextTurn", null, [self.foe]);
return;
}
var cardID = data.cardID;
var card = self.foe.getCardFromDiscard(cardID);
if(card === -1) {
console.log("emreis_leader4:chooseCardFromDiscard | unknown card: ", card);
self.sendNotificationTo(self, "Possible bug occured: unknown card was chosen by playing nilfgaardian leader ability.");
self.endTurn();
return;
}
self.foe.removeFromDiscard(card);
//self.placeCard(card);
self.sendNotificationTo(self.foe, self.getName() + " takes " + card.getName() + " from your discard pile into his hand.");
self.hand.add(card);
self.endTurn();
// self.runEvent("NextTurn", null, [self.foe]);
})
this.receive("agile:field", function(data) {
var fieldType = data.field;
if(!(fieldType in [0, 1])) throw new Error("set field agile: false fieldtype " + fieldType);
self.runEvent("agile:setField", null, [fieldType]);
self.endTurn();
//self.runEvent("NextTurn", null, [self.foe]);
})
this.receive("cancel:agile", function() {
self.off("agile:setField");
})
this.receive("horn:field", function(data) {
var fieldType = data.field;
if(!(fieldType in [0, 1, 2])) throw new Error("set field horn: false fieldtype " + fieldType);
self.runEvent("horn:setField", null, [fieldType]);
self.endTurn();
//self.runEvent("NextTurn", null, [self.foe]);
})
this.receive("cancel:horn", function() {
self.off("horn:setField");
})
this.on("Turn" + this.getID(), this.onTurnStart, this);
};
var r = Battleside.prototype;
/**
* methods && properties here
* r.property = null;
* r.getProperty = function() {...}
*/
r._name = null;
r._discard = null;
r._rubies = 2;
r._score = 0;
r._isWaiting = null;
r._passing = null;
r.field = null;
r.socket = null;
r.n = null;
r.cm = null;
r.foe = null;
r.hand = null;
r.battle = null;
r.deck = null;
r.createCard = function(key) {
return this.cm.create(key, this.n);
}
r.isPassing = function() {
return this._passing;
}
r.isWaiting = function() {
return this._isWaiting;
}
r.setUpWeatherFieldWith = function(p2) {
this.field[Card.TYPE.WEATHER] = p2.field[Card.TYPE.WEATHER] = Field(this);
}
r.findCardOnFieldByID = function(id) {
for(var key in this.field) {
var field = this.field[key];
var card = field.getCard(id);
if(card !== -1) return card;
}
/*
for(var i = 0; i < this._discard.length; i++) {
var c = this._discard[i];
if(c.getID() === id) return c;
}*/
return -1;
}
r.getRandomCardOnField = function() {
var allCards = this.getFieldCards();
var rnd = (Math.random() * allCards.length) | 0;
return allCards[rnd];
}
r.getCardFromDiscard = function(id) {
for(var i = 0; i < this._discard.length; i++) {
var c = this._discard[i];
if(c.getID() === id) return c;
}
return -1;
}
r.getFieldCards = function() {
var close, range, siege;
close = this.field[Card.TYPE.CLOSE_COMBAT].get();
range = this.field[Card.TYPE.RANGED].get();
siege = this.field[Card.TYPE.SIEGE].get();
return close.concat(range.concat(siege));
}
r.setPassing = function(b) {
this._passing = b;
this.send("set:passing", {passing: this._passing}, true);
}
r.wait = function() {
this._isWaiting = true;
this.send("set:waiting", {waiting: this._isWaiting}, true);
}
r.turn = function() {
this._isWaiting = false;
this.send("set:waiting", {waiting: this._isWaiting}, true);
}
r.setLeadercard = function() {
var leaderCard = this.deck.find("type", Card.TYPE.LEADER);
this.deck.removeFromDeck(leaderCard[0]);
/*
this.getYourside().setField("leader", leaderCard[0]);*/
this.field[Card.TYPE.LEADER].add(leaderCard[0]);
}
r.getLeader = function() {
return this.field[Card.TYPE.LEADER].get()[0];
}
r.getID = function() {
return this.n;
}
r.draw = function(times) {
while(times--) {
var card = this.deck.draw();
this.hand.add(card);
}
}
r.calcScore = function() {
var score = 0;
for(var key in this.field) {
score += +this.field[key].getScore();
}
return this._score = score;
}
r.getInfo = function() {
return {
name: this.getName(),
lives: this._rubies,
score: this.calcScore(),
hand: this.hand.length(),
deck: this.deck.length(),
discard: this.getDiscard(true),
passing: this._passing
}
}
r.getRubies = function() {
return this._rubies;
}
r.getScore = function() {
return +this.calcScore();
}
r.removeRuby = function() {
this._rubies--;
}
r.getName = function() {
return this._name;
}
r.send = function(event, msg, isPrivate) {
msg = msg || {};
isPrivate = typeof isPrivate === "undefined" ? false : isPrivate;
msg._roomSide = this.n;
if(isPrivate) {
return this.socket.emit(event, msg);
}
this.battle.send(event, msg);
}
r.receive = function(event, cb) {
this.socket.on(event, cb);
}
r.update = function(self) {
self = self || false;
this.runEvent("Update", null, [self]);
}
r.onTurnStart = function() {
this.foe.wait();
this.turn();
//wait for cardplay event
};
r.playCard = function(card) {
if(card === null || card === -1) return;
if(this.isWaiting()) return;
if(this.isPassing()) return;
if(!this.placeCard(card)) return;
this.hand.remove(card);
this.update();
//this.runEvent("NextTurn", null, [this.foe]);
this.endTurn();
}
r.endTurn = function() {
this.update();
this.runEvent("NextTurn", null, [this.foe]);
}
r.placeCard = function(card, obj) {
obj = _.extend({}, obj);
if(typeof card === "string") {
//card = Card(card);
card = this.createCard(card);
}
this.checkAbilities(card, obj);
if(obj._cancelPlacement && !obj.forceField) {
//this.battle.sendNotification(this.getName() + " played " + card.getName() + "!");
return 0;
}
if(obj._nextTurn && !obj.forceField) {
this.update();
//this.runEvent("NextTurn", null, [this.foe]);
this.endTurn();
return 0;
}
var field = obj.forceField || null;
if(typeof obj.isHorn !== "undefined") {
if(!field) {
field = obj.targetSide.field[obj.isHorn];
}
field.add(card, true);
}
else {
if(!field) {
field = obj.targetSide.field[card.getType()];
}
field.add(card);
}
this.runEvent("EachCardPlace");
this.checkAbilityOnAfterPlace(card, obj);
if(obj._waitResponse) {
this.hand.remove(card);
this.update();
return 0;
}
this.update();
return 1;
}
r.setHorn = function(card, field) {
var self = this;
field = typeof field === "undefined" ? null : field;
if(typeof card === "string") {
//card = Card(card);
//card = this.cm.create(card);
card = this.createCard(card);
}
if(typeof field === "number") {
card.changeType(field);
this.placeCard(card, {
isHorn: field,
forcePlace: true
});
self.hand.remove(card);
return;
}
this.send("played:horn", {cardID: card.getID()}, true)
this.on("horn:setField", function(type) {
self.off("horn:setField");
card.changeType(type);
self.placeCard(card, {
isHorn: type,
disabled: true
});
self.hand.remove(card);
self.battle.sendNotification(self.getName() + " played " + card.getName());
})
}
r.commanderHornAbility = function(card) {
var field = this.field[card.getType()];
var id = "commanders_horn";
if(typeof field === "undefined") {
//console.log("field unknown | %s", card.getName());
return;
}
if(!field.isOnField(card)) {
field.get().forEach(function(_card) {
if(_card.getID() === id) return;
if(_card.getID() === card.getID()) return;
if(_card.getType() !== card.getType()) return;
if(_card.hasAbility("hero")) return;
_card.setBoost(id, 0);
})
this.off("EachCardPlace", card.getUidEvents("EachCardPlace"));
return;
}
field.get().forEach(function(_card) {
if(_card.getID() === id) return;
if(_card.getID() === card.getID()) return;
if(_card.getType() != card.getType()) return;
if(_card.hasAbility("hero")) return;
_card.setBoost(id, 0);
_card.setBoost(id, _card.getPower());
})
}
r.setTightBond = function(card) {
var field = this.field[card.getType()];
/*
var pos = field.getPosition(card);*/
var cards = field.get();
card.resetTightBond();
cards.forEach(function(c) {
if(c.getID() === card.getID()) return;
if(c.getName() !== card.getName()) return;
card.setBoost(card.getID() + "|tight_bond|" + c.getID(), "tight_bond");
});
/*if(pos < 0) return;
if(pos >= 1 && cards[pos - 1].getName() === cards[pos].getName()){
cards[pos].setBoost(cards[pos].getID() + "|left", "tight_bond");
}
else {
cards[pos].setBoost(cards[pos].getID() + "|left", 0);
}
if(pos < cards.length - 1 && cards[pos + 1].getName() === cards[pos].getName()){
cards[pos].setBoost(cards[pos].getID() + "|right", "tight_bond");
}
else {
cards[pos].setBoost(cards[pos].getID() + "|right", 0);
}*/
}
r.checkAbilities = function(card, obj, __flag) {
var self = this;
obj.targetSide = this;
if(obj.disabled) return;
var ability = Array.isArray(__flag) ? __flag : card.getAbility();
if(Array.isArray(ability) && ability.length) {
var ret = ability.slice();
ret.splice(0, 1);
this.checkAbilities(card, obj, ret);
ability = ability[0];
}
/*if(ability && ability.name === obj.suppress){
//this.update();
}*/
if(ability && !Array.isArray(ability)) {
if(ability.onBeforePlace) {
ability.onBeforePlace.apply(this, [card]);
}
if(ability.isCommandersHornCard && typeof obj.isHorn === "undefined") {
this.setHorn(card);
}
if(ability.commandersHorn) {
ability.onEachCardPlace = this.commanderHornAbility;
ability.onWeatherChange = this.commanderHornAbility;
}
if(ability.cancelPlacement && !obj.forcePlace) {
obj._cancelPlacement = true;
}
if(ability.nextTurn) {
obj._nextTurn = ability.nextTurn;
}
if(ability.tightBond) {
//this.setTightBond(card);
ability.onAfterPlace = this.setTightBond;
ability.onEachCardPlace = this.setTightBond;
//ability.onWeatherChange = this.setTightBond;
}
if(ability.scorch) {
this.scorch(card);
}
if(ability.scorchMelee) {
this.scorchMelee(card);
}
if(ability.removeImmediately) {
this.hand.remove(card);
this.addToDiscard(card);
}
if(ability.waitResponse && !obj.forcePlace) {
obj._waitResponse = true;
}
if(ability.changeSide) {
obj.targetSide = this.foe;
}
if(typeof ability.weather !== "undefined") {
ability.onEachTurn = this.setWeather.bind(this, ability.weather);
ability.onEachCardPlace = this.setWeather.bind(this, ability.weather);
}
if(ability.replaceWith && !obj.forcePlace) {
obj._cancelPlacement = true;
this.on("Decoy:replaceWith", function(replaceCard) {
if(replaceCard.getType() == Card.TYPE.LEADER ||
replaceCard.getType() == Card.TYPE.WEATHER ||
replaceCard.getType() == Card.TYPE.SPECIAL) {
return;
}
if(replaceCard.getName() === card.getName()) return;
if(replaceCard.hasAbility("hero")) return;
self.off("Decoy:replaceWith");
var field = self.field[replaceCard.getType()];
field.replaceWith(replaceCard, card);
self.runEvent("EachCardPlace");
self.hand.add(replaceCard);
self.hand.remove(card);
self.update();
//self.runEvent("NextTurn", null, [self.foe]);
self.endTurn();
self.battle.sendNotification(self.getName() + " played Decoy!");
})
}
if(ability.onEachTurn) {
var uid = this.on("EachTurn", ability.onEachTurn, this, [card])
card._uidEvents["EachTurn"] = uid;
}
if(ability.onEachCardPlace) {
var uid = this.on("EachCardPlace", ability.onEachCardPlace, this, [card]);
card._uidEvents["EachCardPlace"] = uid;
}
if(ability.onWeatherChange) {
var uid = this.on("WeatherChange", ability.onWeatherChange, this, [card]);
card._uidEvents["WeatherChange"] = uid;
}
//this.update();
}
}
r.checkAbilityOnAfterPlace = function(card, obj, __flag) {
//var ability = card.getAbility();
var ability = Array.isArray(__flag) ? __flag : card.getAbility();
if(Array.isArray(ability) && ability.length) {
var ret = ability.slice();
ret.splice(0, 1);
this.checkAbilityOnAfterPlace(card, obj, ret);
ability = ability[0];
}
if(ability && !Array.isArray(ability)) {
if(ability.name && ability.name === obj.suppress) {
//this.update();
return;
}
if(ability.onAfterPlace) {
ability.onAfterPlace.call(this, card)
}
}
}
r.setWeather = function(weather, opt) {
var targetRow = weather;
var field;
if(typeof targetRow === "undefined") {
console.log("setWeather: targetRow undefined", targetRow);
console.trace(this);
return;
}
opt = opt || {};
var onRoundEnd = opt.onTurnEnd || false;
if(targetRow === Card.TYPE.WEATHER) {
if(!onRoundEnd) {
this.battle.sendNotification(this.getName() + " played Clear Weather!");
}
field = this.field[targetRow];
field.removeAll();
for(var i = Card.TYPE.CLOSE_COMBAT; i <= Card.TYPE.SIEGE; i++) {
var _field1, _field2, _field;
_field1 = this.field[i].get();
_field2 = this.foe.field[i].get();
_field = _field1.concat(_field2);
_field.forEach(function(_card) {
if(_card.hasAbility("hero")) return;
_card.setForcedPower(-1);
});
}
this.runEvent("WeatherChange");
return;
}
var forcedPower = 1;
if(typeof targetRow === "undefined") {
console.trace(this);
}
var field1 = this.field[targetRow].get();
var field2 = this.foe.field[targetRow].get();
field = field1.concat(field2);
field.forEach(function(_card) {
if(_card.hasAbility("hero")) return;
_card.setForcedPower(forcedPower);
});
this.runEvent("WeatherChange");
}
r.scorchMelee = function(card) {
var side = this.foe;
var field = side.field[Card.TYPE.CLOSE_COMBAT];
this.battle.sendNotification(this.getName() + " played " + card.getName());
if(field.getScore() < 10) {
this.battle.sendNotification("Scorch: Score is under 10! Nothing happens.");
return;
}
var cards = field.getHighestCards(true);
var removeCards = field.removeCard(cards);
var txt = "Scorch destroyed:";
for(var i = 0; i < removeCards.length; i++) {
var c = removeCards[i];
txt += "\n" + c.getName();
}
this.battle.sendNotification(txt);
side.addToDiscard(removeCards);
}
r.scorch = function(card) {/*
var side = this.foe;
var field = side.field[Card.TYPE.CLOSE_COMBAT];
var cards = field.getHighestCards(true);
var removeCards = field.removeCard(cards);*/
var cards = this.getFieldCards();
cards = cards.concat(this.foe.getFieldCards());
var noHeroes = true;
var res = [];
var highest = 0;
var self = this;
this.battle.sendNotification(this.getName() + " played " + card.getName());
cards.forEach(function(card) {
if(noHeroes && card.hasAbility("hero")) return;
highest = card.getPower() > highest ? card.getPower() : highest;
})
cards.forEach(function(card) {
if(noHeroes && card.hasAbility("hero")) return;
if(card.getPower() === highest) res.push(card);
});
res.forEach(function(card) {
var side = self;
if(self.foe.field[card.getType()].isOnField(card)) {
side = self.foe;
}
var removed = side.field[card.getType()].removeCard(card);
side.addToDiscard(removed);
})
var txt = "Scorch destroyed:";
for(var i = 0; i < res.length; i++) {
var c = res[i];
txt += "\n" + c.getName();
}
this.battle.sendNotification(txt);
}
r.clearMainFields = function() {
var rndCard = null;
if(this.deck.getFaction() === Deck.FACTION.MONSTERS) {
rndCard = this.getRandomCardOnField();
if(rndCard) {
rndCard.__lock = true;
this.sendNotification(this.getName() + ": Monsters faction ability triggered! " + rndCard.getName());
}
else {
this.sendNotification(this.getName() + ": Monsters faction ability triggered! But no card found.");
}
}
var cards1 = this.field[Card.TYPE.CLOSE_COMBAT].removeAll();
var cards2 = this.field[Card.TYPE.RANGED].removeAll();
var cards3 = this.field[Card.TYPE.SIEGE].removeAll();
var cards4 = this.field[Card.TYPE.WEATHER].removeAll();
var cards = cards1.concat(cards2.concat(cards3.concat(cards4)));
this.addToDiscard(cards);
}
r.addToDiscard = function(cards) {
var self = this;
if(!Array.isArray(cards)) {
cards = [cards];
}
cards.forEach(function(_card) {
if(_card.__lock) {
delete _card.__lock;
return;
}
self._discard.push(_card);
});
}
r.removeFromDiscard = function(card) {
for(var i = 0; i < this._discard.length; i++) {
var c = this._discard[i];
if(c.getID() === card.getID()) {
this._discard.splice(i, 1);
return
}
}
}
r.getDiscard = function(json) {
if(json) {
return JSON.stringify(this._discard);
}
return this._discard;
}
r.resetNewRound = function() {
this.clearMainFields();
this.setWeather(5, {
onTurnEnd: true
}); //clear weather
this.setPassing(false);
}
r.filter = function(arrCards, opt) {
var arr = arrCards.slice();
for(var key in opt) {
var res = [];
var prop = key, val = opt[key];
arrCards.forEach(function(card) {
var property = card.getProperty(prop);
if(_.isArray(property)) {
var _f = false;
for(var i = 0; i < property.length; i++) {
if(property[i] === val) {
_f = true;
break;
}
}
if(!_f) {
res.push(card);
}
}
else if(_.isArray(val)) {
var _f = false;
for(var i = 0; i < val.length; i++) {
if(property === val[i]) {
_f = true;
break;
}
}
if(!_f) {
res.push(card);
}
}
else if(card.getProperty(prop) !== val) {
res.push(card);
}
})
arr = _.intersection(arr, res);
}
return arr;
}
r.reDraw = function(n) {
//var hand = this.hand.getCards();
var self = this;
var left = n;
var deferred = Promise.Deferred();
this.send("redraw:cards", null, true);
this.receive("redraw:reDrawCard", function(data) {
var id = data.cardID;
if(!left) return;
left--;
var card = self.hand.remove(id)[0];
self.deck.add(card);
self.deck.shuffle();
self.draw(1);
if(!left) {
self.send("redraw:close", null, true);
self.wait();
deferred.resolve("done");
self.sendNotificationTo(self.foe, self.getName() + " finished redraw phase.")
}
self.battle.updateSelf(self);
})
this.receive("redraw:close_client", function() {
self.wait();
deferred.resolve("done");
self.sendNotificationTo(self.foe, self.getName() + " finished redraw phase.")
})
return deferred;
}
r.sendNotificationTo = function(side, msg) {
this.battle.sendNotificationTo(side, msg);
}
r.sendNotification = function(msg) {
this.battle.sendNotification(msg);
}
return Battleside;
})();
module.exports = Battleside;