'use strict'; function Game (cfg) { // 基本属性 this.seed = cfg.seed; this.onGameover = cfg.onGameover; this.winner = null; this.cards = []; // 私有 this._r = new Random(Random.engines.mt19937().seed(cfg.seed)); this._frameCount = 0; this._sources = []; this._framedBeforeBlock = false; // 表示处理块前,是否有帧发生. // 在驱逐力量0以下的 SIGNI 前置 false, 在 handleFrameEnd() 中置 true. // 因为第一次驱逐力量0以下的 SIGNI 后,可能发生常时效果改变, // 导致剩下的 SIGNI 力量也变为0以下. // 于是不断驱逐力量0以下的 SIGNI 直至 _framedBeforeBlock 为 false. // 储存的数据 this.objList = []; this.hostMsgObjs = []; this.guestMsgObjs = []; this.dataObj = {}; // 储存游戏对象(card,player 等)绑定的数据,回合结束时清空 this.trashingCards = []; this.trashingCharms = []; // 注册 this.register(this); // 玩家 this.hostPlayer = new Player(this,cfg.hostIO,cfg.hostMainDeck,cfg.hostLrigDeck); this.guestPlayer = new Player(this,cfg.guestIO,cfg.guestMainDeck,cfg.guestLrigDeck); this.hostPlayer.opponent = this.guestPlayer; this.guestPlayer.opponent = this.hostPlayer; this.turnPlayer = this.hostPlayer; // 组件 this.phase = new Phase(this); this.constEffectManager = new ConstEffectManager(this); this.effectManager = new EffectManager(this); this.triggeringEffects = []; this.triggeringEvents = []; // 附加属性 this.trashWhenPowerBelowZero = false; this.spellToCutIn = null; // <ブルー・パニッシュ> this.lastTurnCoinSkillsDisabled = false; }; Game.checkDeck = function (cfg,mayusRoom) { if (!isObj(cfg)) return false; if (!isArr(cfg.mainDeck) || !isArr(cfg.lrigDeck)) return false; // 主卡组正好40张 if (cfg.mainDeck.length !== 40) return false; // LRIG卡组最多10张 if (cfg.lrigDeck.length > 10) return false; // toInfo var mainDeckInfos = []; var lrigDeckInfos = []; var burstCount = 0; var flagLrig = false; for (var i = 0; i < cfg.mainDeck.length; i++) { var pid = cfg.mainDeck[i]; var info = CardInfo[pid]; if (!info) return false; info = CardInfo[info.cid]; if (!info) return false; if (info.cardType === 'LRIG') return false; if (info.cardType === 'ARTS') return false; if (info.cardType === 'RESONA') return false; if (info.burstEffect) { burstCount++; } mainDeckInfos.push(info); } for (var i = 0; i < cfg.lrigDeck.length; i++) { var pid = cfg.lrigDeck[i]; var info = CardInfo[pid]; if (!info) return false; info = CardInfo[info.cid]; if (!info) return false; if (info.cardType === 'SIGNI') return false; if (info.cardType === 'SPELL') return false; if ((info.cardType === 'LRIG') && (info.level === 0)) { flagLrig = true; } lrigDeckInfos.push(info); } var infos = mainDeckInfos.concat(lrigDeckInfos); // LRIG卡组至少有一张0级LRIG if (!flagLrig) return false; // 生命迸发恰好20张 if (burstCount !== 20) return false; // 相同的卡最多放入4张 var legal = [mainDeckInfos,lrigDeckInfos].every(function (infos) { var bucket = {}; infos.forEach(function (info) { if (info.sideA) { info = CardInfo[info.sideA] } if (info.cid in bucket) { bucket[info.cid]++; } else { bucket[info.cid] = 1; } }); for (var cid in bucket) { if (bucket[cid] > 4) return false; } return true; },this); if (!legal) return false; if (!mayusRoom) return true; return Game.checkMayusRoom(infos); }; Game.checkMayusRoom = function (infos) { // 禁止 狐狸+修复 和 狐狸+Three out if (infos.some(function (info) { return info.cid === 33; // 狐狸 }) && infos.some(function (info) { return (info.cid === 34) || (info.cid === 84); // 修复, three out })) { return false; } // 禁止 V・@・C+共鸣・进军 和 V・@・C+共鸣 if (infos.some(function (info) { return info.cid === 1202; // V・@・C }) && infos.some(function (info) { return (info.cid === 884) || (info.cid === 1369); // 共鸣・进军, 共鸣 })) { return false; } // 禁止 Lock+割裂 和 Lock+精挖 if (infos.some(function (info) { return info.cid === 534; // Lock }) && infos.some(function (info) { return (info.cid === 408) || (info.cid === 570); // 割裂, 精挖 })) { return false; } // 禁止 Ar+魔术手 if (infos.some(function (info) { return info.cid === 814; // Ar }) && infos.some(function (info) { return (info.cid === 1090); // 魔术手 })) { return false; } // 禁止 双麻油 if (infos.some(function (info) { return info.cid === 649; // 創世の巫女 マユ }) && infos.some(function (info) { return (info.cid === 1562); // 真名の巫女 マユ })) { return false; } // 禁止 台风蛇 if (infos.some(function (info) { return info.cid === 957; // 台風一過 }) && infos.some(function (info) { return (info.cid === 1652); // コードアンシエンツ ヘルボロス })) { return false; } // 限制 var limitMap = { 37: 2, // <忘得ぬ幻想 ヴァルキリー> 34: 2, // <修復> 178: 2, // <先駆の大天使 アークゲイン> 1501: 2, // <幻竜 アパト> 534: 1, // <ロック・ユー> // 689: 1, // <RAINY> 474: 0, // <ノー・ゲイン> 23: 0, // <大器晩成> 689: 0, // <RAINY> 1030: 0, // <四面楚火> 1457: 0, // <サーバント Z> 1212: 0, // <コードアート C・L> }; for (var i = 0; i < infos.length; i++) { var info = infos[i]; var cid = info.cid; if (cid in limitMap) { limitMap[cid]--; if (limitMap[cid] < 0) return false; } } return true; }; Game.prototype.start = function () { this.allocateSid(this.hostPlayer,this.objList); this.allocateSid(this.guestPlayer,this.objList); this.setupEffects(); this.outputInitMsg(this.hostPlayer); this.outputInitMsg(this.guestPlayer); this.phase.setup(); this.sendMsgQueue(); }; Game.prototype.setupEffects = function () { [this.hostPlayer,this.guestPlayer].forEach(function (player) { concat(player.mainDeck.cards,player.lrigDeck.cards).forEach(function (card) { card.setupEffects(); },this); },this); }; Game.prototype.outputInitMsg = function (player) { var opponent = player.opponent; player.output({ type: 'INIT', content: { player: player, opponent: opponent, playerZones: { mainDeck: player.mainDeck, lrigDeck: player.lrigDeck, handZone: player.handZone, lrigZone: player.lrigZone, signiZones: player.signiZones, enerZone: player.enerZone, checkZone: player.checkZone, trashZone: player.trashZone, lrigTrashZone: player.lrigTrashZone, lifeClothZone: player.lifeClothZone, excludedZone: player.excludedZone, mainDeckCards: player.mainDeck.cards, lrigDeckCards: player.lrigDeck.cards, lrigDeckCardInfos: player.lrigDeck.cards.map(function (card) { return { pid: card.pid, isSide: !!card.sideA } },this) }, opponentZones: { mainDeck: opponent.mainDeck, lrigDeck: opponent.lrigDeck, handZone: opponent.handZone, lrigZone: opponent.lrigZone, signiZones: opponent.signiZones, enerZone: opponent.enerZone, checkZone: opponent.checkZone, trashZone: opponent.trashZone, lrigTrashZone: opponent.lrigTrashZone, lifeClothZone: opponent.lifeClothZone, excludedZone: opponent.excludedZone, mainDeckCards: opponent.mainDeck.cards, lrigDeckCards: opponent.lrigDeck.cards, lrigDeckCardInfos: opponent.lrigDeck.cards.map(function (card) { return { pid: 0, isSide: !!card.sideA } },this) } } }); }; Game.prototype.register = function (obj) { if (!obj) throw new TypeError(); obj.gid = obj._asid = obj._bsid = this.objList.push(obj); }; Game.prototype.allocateSid = function (player,objs) { var len = objs.length; objs.forEach(function (obj,i) { var r = this.rand(i,len-1); if ((player === this.hostPlayer)) { var tmp = obj._asid; obj._asid = objs[r]._asid; objs[r]._asid = tmp; } else { var tmp = obj._bsid; obj._bsid = objs[r]._bsid; objs[r]._bsid = tmp; } },this); }; Game.prototype.getSid = function (player,obj) { return (player === this.hostPlayer)? obj._asid : obj._bsid; }; Game.prototype.setSid = function (player,obj,sid) { if (player === this.hostPlayer) { obj._asid = sid; } else { obj._bsid = sid; } }; Game.prototype.getObjectBySid = function (player,sid) { if (!isNum(sid)) return null; for (var i = this.objList.length - 1; i >= 0; i--) { var obj = this.objList[i]; if (player === this.hostPlayer) { if (obj._asid === sid) return obj; } else { if (obj._bsid === sid) return obj; } } return null; }; Game.prototype.decideFirstPlayerAsyn = function () { return Callback.immediately().callback(this,function () { var firstPlayer = this.rand(0,1)? this.hostPlayer : this.guestPlayer; return firstPlayer; }); // var a,b; // return this.hostPlayer.rockPaperScissorsAsyn(this,function (v) { // a = v; // }).callback(this,function () { // return this.guestPlayer.rockPaperScissorsAsyn(this,function (v) { // b = v; // }); // }).callback(this,function () { // if (a === b) return this.decideFirstPlayerAsyn(); // var aWin = // ((a === 0) && (b === 2)) || // ((a === 1) && (b === 0)) || // ((a === 2) && (b === 2)); // var firstPlayer = aWin? this.hostPlayer : this.guestPlayer; // return firstPlayer; // }); }; Game.prototype.win = function (player) { if (player.opponent.wontLoseGame) return false; player.output({ type: 'WIN', content: {} }); player.opponent.output({ type: 'LOSE', content: {} }); this.winner = player; return true; }; Game.prototype.destroy = function () { this.hostPlayer.io.listener = null; this.guestPlayer.io.listener = null; }; Game.prototype.output = function (msgObj) { this.hostPlayer.output(msgObj); this.guestPlayer.output(msgObj); }; Game.prototype.packOutputs = function (func,thisp) { this.output({ type: 'PACKED_MSG_START', content: {} }); var rtn = func.call(thisp || this); this.output({ type: 'PACKED_MSG_END', content: {} }); return rtn; }; Game.prototype.outputColor = function () { var playerColor = this.turnPlayer.lrig.color; var opponentColor = this.turnPlayer.opponent.lrig.color; if (playerColor === 'colorless') { playerColor = 'white'; } if (opponentColor === 'colorless') { opponentColor = 'white'; } this.turnPlayer.output({ type: 'SET_COLOR', content: { selfColor: playerColor, opponentColor: opponentColor } }); this.turnPlayer.opponent.output({ type: 'SET_COLOR', content: { selfColor: opponentColor, opponentColor: playerColor } }); }; Game.prototype.sendMsgQueue = function () { this.hostPlayer.sendMsgQueue(); this.guestPlayer.sendMsgQueue(); }; Game.prototype.rand = function (min,max) { // return Math.floor(Math.random() * (max+1-min)) + min; return this._r.integer(min,max); }; Game.prototype.moveCards = function (cards,zone,arg) { if (!cards.length) return []; cards = cards.slice(); this.packOutputs(function () { this.frameStart(); cards = cards.filter(function (card) { return card.moveTo(zone,arg); },this); this.frameEnd(); }); return cards; }; Game.prototype.moveCardsAdvancedAsyn = function (cards,zones,args,force) { cards = cards.slice(); this.frameStart(); var source = this.getEffectSource(); var signis = []; // 可以被保护的 SIGNI var protectedSignis = []; // 确实被保护了的 SIGNI var succs = []; // 返回值,表示是否成功移动. 注: 被保护了也算成功移动. var protectedFlags = []; // 返回值,表示是否被保护 if (!force && source && !source.powerChangeBanned) { // <幻兽神 狮王> // 获得以被保护的 SIGNI signis = cards.filter(function (card,i) { return card.protectingShironakujis.length && (source.player === card.player.opponent) && inArr(card,card.player.signis) && !card.isEffectFiltered(source); },this); } return Callback.forEach(signis,function (signi) { // 获得保护该 SIGNI 的<幻水 蓝鲸> var protectingShironakujis = signi.protectingShironakujis; if (!protectingShironakujis.length) return; return signi.player.selectOptionalAsyn('PROTECT',[signi]).callback(this,function (c) { if (!c) return; signi.beSelectedAsTarget(); return signi.player.selectOptionalAsyn('_SHIRONAKUJI',protectingShironakujis).callback(this,function (shironakuji) { if (!shironakuji) return; shironakuji.beSelectedAsTarget(); // 保护 protectedSignis.push(signi); this.tillTurnEndAdd(source,shironakuji,'power',-6000); // 力量-6000,注意效果源 this.constEffectManager.compute(); // 强制计算,即使不是帧结束 }); }); },this).callback(this,function () { this.packOutputs(function () { cards.forEach(function (card,i) { if (inArr(card,protectedSignis)) { protectedFlags.push(true); succs.push(true); } else { protectedFlags.push(false); succs.push(card.moveTo(zones[i],args[i])); } },this); },this); this.frameEnd(); return {protectedFlags: protectedFlags, succs: succs}; }); }; // 只要有1只 SIGNI 被驱逐成功(包括代替)就返回 true. Game.prototype.banishCardsAsyn = function (cards,force,arg) { if (!arg) arg = {}; var attackingSigni = arg.attackingSigni || null; var control = { someBanished: false, }; var source = this.getEffectSource(); cards = cards.filter(function (card) { if (card.isEffectFiltered(source)) return false; if (!card.canBeBanished()) return false; return true; },this); if (!cards.length) return Callback.immediately(false); this.frameStart(); return this.protectBanishAsyn(cards,this.turnPlayer,control).callback(this,function (cards) { var zones = cards.map(function (card) { // <绿叁游 水滑梯> if (card.resonaBanishToTrash) return card.player.lrigTrashZone; // 原枪 if (card.player.banishTrash) return card.player.trashZone; // <原槍 エナジェ> if (inArr(source,card.player._RanergeOriginalSpear)) return card.player.trashZone; if (inArr(attackingSigni,card.player._RanergeOriginalSpear)) return card.player.trashZone; return card.player.enerZone; },this); var opposingSignis = cards.map(function (card) { return card.getOpposingSigni(); },this); var accedCards = cards.map(function (card) { return card.getAccedCards(); },this); return this.moveCardsAdvancedAsyn(cards,zones,[],force).callback(this,function (arg) { arg.protectedFlags.forEach(function (isProtected,i) { if (isProtected) return; if (!arg.succs[i]) return; var signi = cards[i]; // <幻獣神 ウルティム> var banishSource = attackingSigni || source; var count; if (banishSource) { if (banishSource.hasClass('空獣') || banishSource.hasClass('地獣')) { if (signi.player === banishSource.player.opponent) { count = this.getData(banishSource.player,'_UltimPhantomBeastDeity') || 0; this.setData(banishSource.player,'_UltimPhantomBeastDeity',++count); } } } var event = { card: signi, opposingSigni: opposingSignis[i], accedCards: accedCards[i], attackingSigni: attackingSigni, source: banishSource }; this.setData(signi.player,'flagSigniBanished',true); signi.onBanish.trigger(event); signi.player.onSigniBanished.trigger(event); },this); this.frameEnd(); return control.someBanished || !!arg.succs.length; }); }); }; Game.prototype.protectBanishAsyn = function (cards,player,control) { // 过滤不被驱逐的卡 cards = cards.filter(function (card) { if (card.isEffectFiltered()) return false; if (!card.canBeBanished()) return false; return true; },this); // 获得玩家受保护的卡及其保护措施 var cardList = []; var protectionTable = {}; var optional = true; cards.forEach(function (card) { if (card.player !== player) return; card.banishProtections.forEach(function (protection) { if (!protection.condition.call(protection.source,card)) return; if (!protection.optional) optional = false; if (protectionTable[card.gid]) { protectionTable[card.gid].push(protection); } else { cardList.push(card); protectionTable[card.gid] = [protection]; } },this); },this); // 选择要保护的卡以及保护措施 return player.selectAsyn('PROTECT',cardList,optional).callback(this,function (card) { if (!card) { // 回合玩家处理完毕,处理非回合玩家. if (player === this.turnPlayer) { return this.protectBanishAsyn(cards,player.opponent,control); } // 非回合玩家处理完毕,结束处理. return cards; } card.beSelectedAsTarget(); var protections = protectionTable[card.gid]; return player.selectAsyn('CHOOSE_EFFECT',protections).callback(this,function (protection) { protection.source.activate(); return protection.actionAsyn.call(protection.source,card).callback(this,function () { control.someBanished = true; this.constEffectManager.compute(); // 强制计算,即使不是帧结束 cards = cards.filter(function (c) { if (c === card) return false; // 排除已被保护的卡 if (!inArr(c,c.player.signis)) return false; // 排除已不在场的卡 return true; },this); // 继续处理当前玩家. return this.protectBanishAsyn(cards,player,control); }); }); }); }; // Game.prototype.protectBanishAsyn = function () { // // 代替驱逐 // return Callback.forEach(cards,function (card) { // if (force) return; // if (card.isEffectFiltered()) return; // if (card.canNotBeBanished) return; // if (card.power <= 0 ) return; // // <堕落虚无 派蒙> // if (card.trashCharmInsteadOfBanish && card.charm) { // return card.player.selectOptionalAsyn('TRASH_CHARM',[card]).callback(this,function (c) { // if (!c) return; // someBanished = true; // card.charm.trash(); // removeFromArr(card,cards); // }); // } // // <核心代号 S・W・T> // if (card.discardSpellInsteadOfBanish) { // var spells = card.player.hands.filter(function (card) { // return card.type === 'SPELL'; // },this); // if (!spells.length) return; // return card.player.selectOptionalAsyn('PROTECT',[card]).callback(this,function (c) { // if (!c) return; // return card.player.selectAsyn('TRASH',spells).callback(this,function (spell) { // if (!spell) return; // someBanished = true; // spell.trash(); // removeFromArr(card,cards); // }); // }); // } // // <コードハート M・P・P> // if (card.protectingMpps.length) { // // 获得保护该 SIGNI 的<M・P・P> // var protectingMpps = card.protectingMpps.filter(function (mpp) { // if (!inArr(mpp,card.player.signis)) return false; // var spells = mpp.zone.cards.slice(1).filter(function (card) { // return (card !== mpp.charm); // },this); // return (spells.length >= 2); // },this); // if (!protectingMpps.length) return; // card.beSelectedAsTarget(); // 被驱逐的SIGNI闪烁 // return card.player.selectOptionalAsyn('PROTECT',protectingMpps).callback(this,function (mpp) { // if (!mpp) return; // var spells = mpp.zone.cards.slice(1).filter(function (card) { // return (card !== mpp.charm); // },this); // return card.player.selectSomeAsyn('TRASH',spells,2,2).callback(this,function (spells) { // mpp.activate(); // this.trashCards(spells); // someBanished = true; // removeFromArr(card,cards); // }); // }); // } // },this) // }; // Game.prototype.banishCards = function (cards) { // if (!cards.length) return; // cards = cards.slice(); // this.packOutputs(function () { // this.frameStart(); // cards.forEach(function (card) { // card.beBanished(); // },this); // this.frameEnd(); // }); // }; // Game.prototype.doBanishCards = function (cards) { // if (!cards.length) return; // cards = cards.slice(); // this.packOutputs(function () { // this.frameStart(); // cards.forEach(function (card) { // card.doBanish(); // },this); // this.frameEnd(); // }); // }; // Game.prototype.banishCardsAsyn = function (cards) { // if (!cards.length) return Callback.immediately(); // cards = cards.slice(); // return this.packOutputs(function () { // this.frameStart(); // return Callback.forEach(cards,function (card) { // return card.banishAsyn(); // },this).callback(this,function () { // this.frameEnd(); // }); // }); // }; Game.prototype.trashCardsAsyn = function (cards,arg) { if (!cards.length) return Callback.immediately(); cards = cards.slice(); var zones = cards.map(function (card) { if (inArr(card.type,['LRIG','ARTS'])) return card.player.lrigTrashZone; return card.player.trashZone; },this); var args = cards.map(function (card) { return arg || {}; },this); return this.moveCardsAdvancedAsyn(cards,zones,[],args); }; Game.prototype.bounceCardsAsyn = function (cards) { if (!cards.length) return Callback.immediately(); cards = cards.slice(); var zones = cards.map(function (card) { return card.player.handZone; },this); return this.moveCardsAdvancedAsyn(cards,zones,[]); }; Game.prototype.bounceCardsToDeckAsyn = function (cards) { if (!cards.length) return Callback.immediately(); cards = cards.slice(); var zones = cards.map(function (card) { if (inArr(this.type,['LRIG','ARTS'])) return card.player.lrigDeck; return card.player.mainDeck; },this); return this.moveCardsAdvancedAsyn(cards,zones,[]); }; Game.prototype.trashCards = function (cards,arg) { if (!cards.length) return; cards = cards.slice(); this.packOutputs(function () { this.frameStart(); cards = cards.filter(function (card) { return card.trash(arg); },this); this.frameEnd(); }); return cards; }; Game.prototype.excludeCards = function (cards) { if (!cards.length) return; cards = cards.slice(); this.packOutputs(function () { this.frameStart(); cards = cards.filter(function (card) { card.exclude(); },this); this.frameEnd(); }); return cards; }; Game.prototype.upCards = function (cards) { if (!cards.length) return; this.packOutputs(function () { this.frameStart(); cards = cards.filter(function (card) { return card.up(); },this); this.frameEnd(); }); return cards; }; Game.prototype.upCardsAsyn = function (cards) { if (!cards.length) return; var upCards = []; this.frameStart(); return Callback.forEach(cards,function (card) { return card.upAsyn().callback(this,function (succ) { if (succ) upCards.push(card); }); }).callback(this,function () { this.frameEnd(); return upCards; }); }; Game.prototype.downCards = function (cards) { if (!cards.length) return; this.packOutputs(function () { this.frameStart(); cards = cards.filter(function (card) { card.down(); },this); this.frameEnd(); }); return cards; }; // Game.prototype.informPower = function () { // var cards = concat(this.turnPlayer.signis,this.turnPlayer.opponent.signis); // var powers = cards.map(function (card) { // return card.power; // }); // this.output({ // type: 'POWER', // content: { // cards: cards, // powers: powers // } // }); // }; Game.prototype.outputCardStates = function () { var cards = concat(this.turnPlayer.signis,this.turnPlayer.opponent.signis); var signiInfos = cards.map(function (card) { return { card: card, power: card.power, states: card.getStates() } },this); cards = [this.turnPlayer.lrig,this.turnPlayer.opponent.lrig]; var lrigInfos = cards.map(function (card) { return { card: card, states: card.getStates() } },this); var zones = concat(this.turnPlayer.signiZones,this.turnPlayer.opponent.signiZones); var zoneInfos = zones.map(function (zone) { return { zone: zone, states: zone.getStates() } },this); this.output({ type: 'CARD_STATES', content: { signiInfos: signiInfos, lrigInfos: lrigInfos, zoneInfos: zoneInfos } }); }; Game.prototype.addConstEffect = function (cfg,dontCompute) { var constEffect = new ConstEffect(this.constEffectManager,cfg); if (!dontCompute) { this.handleFrameEnd(); } return constEffect; }; Game.prototype.newEffect = function (eff) { return new Effect(this.effectManager,eff); }; Game.prototype.frame = function (thisp,func) { if (arguments.length !== 2) { debugger; } this.frameStart(); func.call(thisp); this.frameEnd(); }; Game.prototype.frameStart = function () { this._frameCount++; }; Game.prototype.frameEnd = function () { if (this._frameCount <= 0) { debugger; console.warn('game._frameCount <= 0'); this._frameCount = 0; return; } this._frameCount--; this.handleFrameEnd(); }; Game.prototype.handleFrameEnd = function () { if (this._frameCount) return; this._framedBeforeBlock = true; this.constEffectManager.compute(); this.triggeringEffects.forEach(function (effect,idx) { effect.trigger(this.triggeringEvents[idx]); },this); this.triggeringEffects.length = 0; this.triggeringEvents.length = 0; }; Game.prototype.pushTriggeringEffect = function (effect,event) { this.triggeringEffects.push(effect); this.triggeringEvents.push(event); }; Game.prototype.pushEffectSource = function (source) { this._sources.push(source || null); }; Game.prototype.popEffectSource = function (source) { return this._sources.pop(); }; Game.prototype.getEffectSource = function () { if (!this._sources.length) return null; return this._sources[this._sources.length-1]; }; Game.prototype.blockAsyn = function (source,thisp,func) { if (arguments.length !== 3) { if (arguments.length === 2) { source = null; thisp = arguments[0]; func = arguments[1]; } else { debugger; console.warn('game.blockAsyn() 参数个数不正确'); } } this.blockStart(source); return Callback.immediately().callback(thisp,func).callback(this,function (rtn) { return this.blockEndAsyn().callback(this,function () { return rtn; }); }); }; // Game.prototype.block = function (source,thisp,func) { // this.pushEffectSource(source); // func.call(thisp); // this.popEffectSource(); // }; Game.prototype.blockStart = function (source) { this.pushEffectSource(source); }; Game.prototype.blockEndAsyn = function () { if (this._sources.length <= 0) { debugger; console.warn('game._sources.length <= 0'); return Callback.immediately(); } this.popEffectSource(); return this.handleBlockEndAsyn(); }; Game.prototype.handleBlockEndAsyn = function () { if (this._sources.length) return Callback.immediately(); this.frameStart(); return this.turnPlayer.resetSignisAsyn().callback(this,function () { return this.turnPlayer.opponent.resetSignisAsyn(); }).callback(this,function () { return this.turnPlayer.resetAccesAsyn().callback(this,function () { return this.turnPlayer.opponent.resetAccesAsyn(); }); }).callback(this,function () { this.frameEnd(); return this.banishNonPositiveAsyn(); }).callback(this,function () { // 废弃【魅饰】和SIGNI下方的卡 this.trashCards(this.trashingCharms,{ isCharm: true }); this.trashCards(this.trashingCards); this.trashingCharms.length = 0; this.trashingCards.length = 0; return this.rebuildAsyn(); }).callback(this,function () { return this.effectManager.handleEffectsAsyn(); }); // return this.banishNonPositiveAsyn().callback(this,function () { // return this.rebuildAsyn().callback(this,function () { // return this.effectManager.handleEffectsAsyn(); // }); // }); }; Game.prototype.banishNonPositiveAsyn = function () { if (!this._framedBeforeBlock) return Callback.immediately(); this._framedBeforeBlock = false; var signis = concat(this.turnPlayer.signis,this.turnPlayer.opponent.signis).filter(function (signi) { return (signi.power <= 0); },this); if (!signis.length) Callback.immediately(); return Callback.immediately().callback(this,function () { this.pushEffectSource(null); if (this.trashWhenPowerBelowZero) { return this.trashCards(signis); } return this.banishCardsAsyn(signis,true); }).callback(this,function () { this.popEffectSource(); return this.banishNonPositiveAsyn(); }); }; Game.prototype.rebuildAsyn = function () { var players = [this.turnPlayer,this.turnPlayer.opponent].filter(function (player) { return !player.mainDeck.cards.length && player.trashZone.cards.length; },this); if (!players.length) return Callback.immediately(); return this.blockAsyn(this,function () { // console.log('game.rebuildAsyn()'); // 将废弃区的卡片放到牌组,洗切 this.frameStart(); return Callback.forEach(players,function (player) { var cards = player.trashZone.cards; return player.showCardsAsyn(cards,'CONFIRM_REFRESH_SELF').callback(this,function () { return player.opponent.showCardsAsyn(cards,'CONFIRM_REFRESH_OPPONENT'); }).callback(this,function () { player.rebuildCount++; this.moveCards(player.trashZone.cards,player.mainDeck); player.shuffle(); player.onRebuild.trigger({ rebuildCount: player.rebuildCount }); }); },this).callback(this,function () { this.frameEnd(); // 废弃一张生命护甲 var cards = []; players.forEach(function (player) { cards = cards.concat(player.lifeClothZone.getTopCards(1)); },this); this.trashCards(cards); }); // this.frame(this,function () { // players.forEach(function (player) { // player.rebuildCount++; // this.moveCards(player.trashZone.cards,player.mainDeck); // player.shuffle(); // player.onRebuild.trigger({ // rebuildCount: player.rebuildCount // }); // },this); // }); // // 废弃一张生命护甲 // var cards = []; // players.forEach(function (player) { // cards = cards.concat(player.lifeClothZone.getTopCards(1)); // },this); // this.trashCards(cards); }); }; Game.prototype.setAdd = function (source,obj,prop,value,isSet,arg) { if (!arg) arg = {}; if (!arg.forced) { if (obj.isEffectFiltered && obj.isEffectFiltered(source)) return; var card = null; if (obj.player && inArr(prop,Card.abilityProps)) { card = obj; } else if (isObj(value) && (value.constructor === Effect)) { card = value.source; if (card.isEffectFiltered(source)) return; } if (card) { if (card.canNotGainAbility || card.player.canNotGainAbility) return; if (card.canNotGainAbilityBySelfPlayer) { if (source.player === card.player) { return; } } } } var destroyTimming = [this.phase.onTurnEnd]; if (obj.type === 'SIGNI') { if (inArr(obj,obj.player.signis)) { destroyTimming.push(obj.onLeaveField); } } this.frameStart(); this.addConstEffect({ source: source, fixed: true, destroyTimming: destroyTimming, action: function (set,add) { var target = obj; if (obj.type === 'LRIG') { target = obj.player.lrig; } if (isSet) { set(target,prop,value,arg); } else { add(target,prop,value,arg); } } }); if (obj.triggerOnAffect) { obj.triggerOnAffect(source); } this.frameEnd(); }; Game.prototype.tillTurnEndSet = function (source,obj,prop,value,arg) { this.setAdd(source,obj,prop,value,true,arg); }; Game.prototype.tillTurnEndAdd = function (source,obj,prop,value,arg) { this.setAdd(source,obj,prop,value,false,arg); }; Game.prototype.getData = function (obj,key) { var hash = obj.gid + key; return this.dataObj[hash]; }; Game.prototype.setData = function (obj,key,value) { var hash = obj.gid + key; this.dataObj[hash] = value; }; Game.prototype.clearData = function () { this.dataObj = {}; }; Game.prototype.getOriginalValue = function (obj,prop) { return this.constEffectManager.getOriginalValue(obj,prop); }; Game.prototype.gameover = function (arg) { if (!arg) arg = {}; var win,surrender; if (arg.surrender === 'host') { surrender = true; win = false; } else if (arg.surrender === 'guest') { surrender = true; win = true; } else { surrender = false; win = (this.winner === this.hostPlayer); } if (!this.hostPlayer.lrig || !this.guestPlayer.lrig) { this.onGameover(null); return; } var replayObj = { win: win, surrender: surrender, selfLrig: this.hostPlayer.lrig.cid, opponentLrig: this.guestPlayer.lrig.cid, messagePacks: this.hostPlayer.messagePacks }; this.onGameover(replayObj); }; Game.prototype.getLiveMessagePacks = function () { return this.hostPlayer.messagePacks; }; Game.prototype.getSubsets = function (arr) { var subsets = []; for (var i = 0; i < (1 << arr.length); i++) { subsets.push(arr.filter(function (item,idx) { return idx & i; })); } return subsets; }; global.Game = Game;