slot0 = class("Chapter", import(".BaseVO")) slot0.SelectFleet = 1 slot0.CustomFleet = 2 slot0.CHAPTER_STATE = { i18n("level_chapter_state_high_risk"), i18n("level_chapter_state_risk"), i18n("level_chapter_state_low_risk"), i18n("level_chapter_state_safety") } function slot0.Ctor(slot0, slot1) slot0.configId = slot1.id slot0.id = slot0.configId slot0.progress = defaultValue(slot1.progress, 0) slot0.defeatCount = slot1.defeat_count or 0 slot0.passCount = slot1.pass_count or 0 slot0.todayDefeatCount = slot1.today_defeat_count or 0 slot0.expireTime = slot1.active_time slot0.awardIndex = slot1.index or 0 slot0.theme = ChapterTheme.New(slot0:getConfig("theme")) slot0.achieves = {} for slot6 = 1, 3 do if slot0:getConfig("star_require_" .. slot6) > 0 then table.insert(slot0.achieves, { type = slot7, config = slot0:getConfig("num_" .. slot6), count = ({ defaultValue(slot1.kill_boss_count, 0), defaultValue(slot1.kill_enemy_count, 0), defaultValue(slot1.take_box_count, 0) })[slot6] }) end end slot0.winConditions = {} for slot7, slot8 in pairs(slot0:getConfig("win_condition")) do table.insert(slot0.winConditions, { type = slot8[1], param = slot8[2] }) end slot0.loseConditions = {} for slot8, slot9 in pairs(slot0:getConfig("lose_condition")) do table.insert(slot0.loseConditions, { type = slot9[1], param = slot9[2] }) end slot0.dropShipIdList = {} slot0.eliteFleetList = { {}, {}, {} } slot0.eliteCommanderList = { {}, {}, {} } slot0.pathFinder = nil slot0.active = false slot0.dueTime = nil slot0.cells = {} slot0.fleets = {} slot0.findex = 0 slot0.champions = {} slot0.roundIndex = 0 slot0.subAutoAttack = 0 slot0.barriers = 0 slot0.loopFlag = 0 slot0.cellAttachments = {} slot0.buff_list = {} if slot1.buff_list then for slot8, slot9 in ipairs(slot1.buff_list) do slot0.buff_list[slot8] = slot9 end end slot0.extraFlagList = {} slot0.wallAssets = {} if slot0:getConfig("wall_prefab") and #slot0:getConfig("wall_prefab") > 0 then slot8 = "wall_prefab" for slot8, slot9 in ipairs(slot0:getConfig(slot8)) do slot0.wallAssets[slot9[1] .. "_" .. slot9[2]] = slot9 end end end function slot0.BuildEliteFleetList(slot0) slot1 = { {}, {}, {} } slot2 = { {}, {}, {} } for slot6, slot7 in ipairs(slot0 or {}) do slot8 = {} for slot12, slot13 in ipairs(slot7.main_id) do slot8[#slot8 + 1] = slot13 end slot1[slot6] = slot8 for slot13, slot14 in ipairs(slot7.commanders) do if getProxy(CommanderProxy):getCommanderById(slot14.id) and Commander.canEquipToFleetList(slot2, slot6, slot14.pos, slot15) then -- Nothing end end slot2[slot6] = { [slot16] = slot15 } end return slot1, slot2 end function slot0.__index(slot0, slot1) if slot1 == "fleet" then return slot0.fleets[slot0.findex] end return rawget(slot0, slot1) or uv0[slot1] end function slot0.getFleetById(slot0, slot1) return _.detect(slot0.fleets, function (slot0) return slot0.id == uv0 end) end function slot0.getFleetByShipVO(slot0, slot1) slot3 = nil for slot7, slot8 in ipairs(slot0.fleets) do if slot8:getShip(slot1.id) then slot3 = slot8 break end end return slot3 end function slot0.getMaxCount(slot0) if #slot0:getConfig("risk_levels") == 0 then return 0 end return slot1[1][1] end function slot0.hasMitigation(slot0) if not LOCK_MITIGATION then return slot0:getConfig("mitigation_level") > 0 else return false end end function slot0.getRemainPassCount(slot0) return math.max(slot0:getMaxCount() - slot0.passCount, 0) end function slot0.getRiskLevel(slot0) slot1 = slot0:getRemainPassCount() for slot6, slot7 in ipairs(slot0:getConfig("risk_levels")) do if slot1 <= slot7[1] and slot7[2] <= slot1 then return slot6 end end end function slot0.getMitigationRate(slot0) return math.min(slot0.passCount, slot0:getMaxCount()) * (LOCK_MITIGATION and 0 or slot0:getConfig("mitigation_rate")) end function slot0.getRepressInfo(slot0) return { repressMax = slot0:getMaxCount(), repressCount = slot0.passCount, repressReduce = slot0:getMitigationRate(), repressLevel = LOCK_MITIGATION and 0 or slot0:getRemainPassCount() > 0 and 0 or slot0:getConfig("mitigation_level") or 0, repressEnemyHpRant = 1 - slot0:getStageCell(slot0.fleet.line.row, slot0.fleet.line.column).data / 10000 } end function slot0.getChapterState(slot0) return uv0.CHAPTER_STATE[slot0:getRiskLevel()] end function slot0.getPlayType(slot0) return slot0:getConfig("model") end function slot0.bindConfigTable(slot0) return pg.chapter_template end function slot0.isTypeDefence(slot0) return slot0:getPlayType() == ChapterConst.TypeDefence end function slot0.getConfig(slot0, slot1) if slot0:isLoop() then if pg.chapter_template_loop[slot0.id][slot1] ~= nil then return slot2[slot1] end if (slot1 == "air_dominance" or slot1 == "best_air_dominance") and slot2.air_dominance_loop_rate ~= nil then return math.floor(slot0:getConfigTable()[slot1] * slot2.air_dominance_loop_rate * 0.01) end end return slot0:getConfigTable()[slot1] end function slot0.update(slot0, slot1) slot0.active = true slot0.dueTime = slot1.time slot0.loopFlag = slot1.loop_flag slot4 = slot0:getConfig("float_items") slot5 = slot0:getConfig("grids") slot0.cells = {} function slot6(slot0) slot1 = ChapterCell.Line2Name(slot0.pos.row, slot0.pos.column) if slot0.item_type == ChapterConst.AttachStory and slot0.item_data == ChapterConst.StoryTrigger then if uv0.cellAttachments[slot1] then warning("Multi Cell Attachemnts in one cell " .. slot0.pos.row .. " " .. slot0.pos.column) end uv0.cellAttachments[slot1] = ChapterCell.New(slot0) slot0 = { item_id = 0, item_data = 0, item_flag = 0, pos = { row = slot0.pos.row, column = slot0.pos.column }, item_type = ChapterConst.AttachNone } end if not uv0.cells[slot1] or uv0.cells[slot1].attachment == ChapterConst.AttachNone then if ChapterCell.New(slot0).attachment == ChapterConst.AttachOni_Target or slot2.attachment == ChapterConst.AttachOni then slot2.attachment = ChapterConst.AttachNone end if _.detect(uv1, function (slot0) return slot0[1] == uv0.row and slot0[2] == uv0.column end) then slot2.item = slot3[3] slot2.itemOffset = Vector2(slot3[4], slot3[5]) end uv0.cells[slot1] = slot2 return slot2 end end _.each(slot1.cell_list, function (slot0) uv0(slot0) end) _.each(slot5, function (slot0) (uv0.cells[ChapterCell.Line2Name(slot0[1], slot0[2])] or uv1({ pos = { row = slot0[1], column = slot0[2] }, item_type = ChapterConst.AttachNone })):SetWalkable(slot0[3]) end) slot0.indexMax = Vector2(-ChapterConst.MaxRow, -ChapterConst.MaxColumn) slot0.indexMin = Vector2(ChapterConst.MaxRow, ChapterConst.MaxColumn) _.each(slot5, function (slot0) uv0.indexMin.x = math.min(uv0.indexMin.x, slot0[1]) uv0.indexMin.y = math.min(uv0.indexMin.y, slot0[2]) uv0.indexMax.x = math.max(uv0.indexMax.x, slot0[1]) uv0.indexMax.y = math.max(uv0.indexMax.y, slot0[2]) end) _.each(slot1.cell_flag_list or {}, function (slot0) if uv0.cells[ChapterCell.Line2Name(slot0.pos.row, slot0.pos.column)] then slot2:updateFlagList(slot0) end end) slot0.buff_list = {} if slot1.buff_list then for slot10, slot11 in ipairs(slot1.buff_list) do slot0.buff_list[slot10] = slot11 end end slot0.pathFinder = OrientedPathFinding.New({}, ChapterConst.MaxRow, ChapterConst.MaxColumn) slot7 = slot0:getNpcShipByType() slot0.operationBuffList = {} for slot11, slot12 in ipairs(slot1.operation_buff) do slot0.operationBuffList[#slot0.operationBuffList + 1] = slot12 end slot0.fleets = {} for slot11, slot12 in ipairs(slot1.group_list) do slot13 = ChapterFleet.New() slot13:setup(slot0) slot13:updateNpcShipList(slot7) slot13:update(slot12) slot0.fleets[slot11] = slot13 end slot0.fleets = _.sort(slot0.fleets, function (slot0, slot1) return slot0.id < slot1.id end) if slot1.escort_list then for slot11, slot12 in ipairs(slot1.escort_list) do slot0.fleets[#slot0.fleets + 1] = ChapterTransportFleet.New(slot12, #slot0.fleets + 1) end end slot0.findex = 0 slot0.findex = slot0:getNextValidIndex() if slot0.findex == 0 then slot0.findex = 1 end slot0.champions = {} if slot1.ai_list then for slot11, slot12 in ipairs(slot1.ai_list) do if slot12.item_flag ~= 1 then slot0:mergeChampionImmediately(ChapterChampionPackage.New(slot12)) end end end slot0.roundIndex = slot1.round slot0.subAutoAttack = slot1.is_submarine_auto_attack slot0.modelCount = slot1.model_act_count slot0.airDominanceStatus = nil slot0.extraFlagList = {} for slot11, slot12 in ipairs(slot1.extra_flag_list) do table.insert(slot0.extraFlagList, slot12) end slot0.defeatEnemies = slot1.kill_count or 0 slot0.BaseHP = slot1.chapter_hp or 0 slot0.orignalShipCount = slot1.init_ship_count or 0 slot0.combo = slot1.continuous_kill_count or 0 slot0.scoreHistory = slot0.scoreHistory or {} for slot11 = ys.Battle.BattleConst.BattleScore.D, ys.Battle.BattleConst.BattleScore.S do slot0.scoreHistory[slot11] = 0 end if slot1.battle_statistics then for slot11, slot12 in ipairs(slot1.battle_statistics) do slot0.scoreHistory[slot12.id] = slot12.count end end slot8 = {} if slot1.chapter_strategy_list then for slot12, slot13 in ipairs(slot1.chapter_strategy_list) do slot8[slot13.id] = slot13.count end end slot0.strategies = slot8 slot0.duties = {} if #slot1.fleet_duties > 0 then _.each(slot1.fleet_duties, function (slot0) uv0.duties[slot0.key] = slot0.value end) end slot0.activateAmbush = not slot0:isLoop() and slot0:GetWillActiveAmbush() end function slot0.retreat(slot0, slot1) if slot1 then slot0.todayDefeatCount = slot0.todayDefeatCount + 1 slot0:updateTodayDefeatCount() end slot0.active = false slot0.dueTime = nil slot0.cells = {} slot0.fleets = {} slot0.findex = 1 slot0.champions = {} slot0.cellAttachments = {} slot0.round = 0 slot0.airDominanceStatus = nil end function slot0.clearSubChapter(slot0) slot0.expireTime = nil slot0.awardIndex = nil end function slot0.existLoop(slot0) return pg.chapter_template_loop[slot0.id] ~= nil end function slot0.canActivateLoop(slot0) return slot0.progress == 100 end function slot0.isLoop(slot0) return slot0.loopFlag == 1 end function slot0.getRound(slot0) return slot0.roundIndex % 2 end function slot0.getRoundNum(slot0) return math.floor(slot0.roundIndex / 2) end function slot0.IncreaseRound(slot0) slot0.roundIndex = slot0.roundIndex + 1 end function slot0.existMoveLimit(slot0) return slot0:getConfig("is_limit_move") == 1 or slot0:existOni() or slot0:isPlayingWithBombEnemy() end function slot0.getChapterCell(slot0, slot1, slot2) return Clone(slot0.cells[ChapterCell.Line2Name(slot1, slot2)]) end function slot0.GetRawChapterCell(slot0, slot1, slot2) return slot0.cells[ChapterCell.Line2Name(slot1, slot2)] end function slot0.findChapterCell(slot0, slot1, slot2) for slot6, slot7 in pairs(slot0.cells) do if slot7.attachment == slot1 and (not slot2 or slot7.attachmentId == slot2) then return Clone(slot7) end end return nil end function slot0.findChapterCells(slot0, slot1, slot2) slot3 = {} for slot7, slot8 in pairs(slot0.cells) do if slot8.attachment == slot1 and (not slot2 or slot8.attachmentId == slot2) then table.insert(slot3, Clone(slot8)) end end return slot3 end function slot0.mergeChapterCell(slot0, slot1) slot4 = slot0.cells[ChapterCell.Line2Name(slot1.row, slot1.column)] == nil or slot3.attachment ~= slot1.attachment or slot3.attachmentId ~= slot1.attachmentId if slot3 then slot3.attachment = slot1.attachment slot3.attachmentId = slot1.attachmentId slot3.flag = slot1.flag slot3.data = slot1.data slot3.rival = slot1.rival slot1 = slot3 end if slot4 and ChapterConst.NeedMarkAsLurk(slot1) then slot1.trait = ChapterConst.TraitLurk end if slot1.attachment == ChapterConst.AttachBoss and slot0:getChampionIndex(slot1.row, slot1.column) then table.remove(slot0.champions, slot5) end slot0:updateChapterCell(slot1) end function slot0.updateChapterCell(slot0, slot1) slot0.cells[ChapterCell.Line2Name(slot1.row, slot1.column)] = Clone(slot1) end function slot0.clearChapterCell(slot0, slot1, slot2) slot4 = slot0.cells[ChapterCell.Line2Name(slot1, slot2)] slot4.attachment = ChapterConst.AttachNone slot4.attachmentId = 0 slot4.flag = ChapterConst.CellFlagActive slot4.data = 0 slot4.trait = ChapterConst.TraitNone end function slot0.GetChapterCellAttachemnts(slot0) return slot0.cellAttachments end function slot0.GetRawChapterAttachemnt(slot0, slot1, slot2) return slot0.cellAttachments[ChapterCell.Line2Name(slot1, slot2)] end function slot0.getShip(slot0, slot1) slot2 = nil for slot6 = 1, #slot0.fleets do if slot0.fleets[slot6]:getShip(slot1) then break end end return slot2 end function slot0.getShips(slot0) _.each(slot0.fleets, function (slot0) _.each(slot0:getShips(true), function (slot0) table.insert(uv0, Clone(slot0)) end) end) return {} end function slot0.getFirstValidIndex(slot0) slot1 = 0 for slot5, slot6 in ipairs(slot0.fleets) do if slot6:isValid() then slot1 = slot5 break end end return slot1 end function slot0.getNextValidIndex(slot0) slot1 = 0 for slot5 = slot0.findex + 1, #slot0.fleets do if slot0.fleets[slot5]:getFleetType() == FleetType.Normal and slot0.fleets[slot5]:isValid() then slot1 = slot5 break end end if slot1 == 0 then for slot5 = 1, slot0.findex - 1 do if slot0.fleets[slot5]:getFleetType() == FleetType.Normal and slot0.fleets[slot5]:isValid() then slot1 = slot5 break end end end return slot1 end function slot0.shipInWartime(slot0, slot1) return _.any(slot0.fleets, function (slot0) return slot0.ships[uv0] ~= nil end) end function slot0.existAmbush(slot0) return slot0:getConfig("is_ambush") == 1 or slot0:getConfig("is_air_attack") == 1 end function slot0.GetWillActiveAmbush(slot0) if not slot0:existAmbush() then return false end slot1 = slot0:getConfig("avoid_require") return not _.any(slot0.fleets, function (slot0) return slot0:getFleetType() == FleetType.Normal and uv0 <= slot0:getInvestSums(true) end) end function slot0.getAmbushRate(slot0, slot1, slot2) slot4 = slot0:getConfig("investigation_ratio") slot7 = _.detect(slot0:getConfig("ambush_ratio_extra"), function (slot0) return #slot0 == 1 end) slot6 = (_.detect(slot0:getConfig("ambush_ratio_extra"), function (slot0) return slot0[1] == uv0.row and slot0[2] == uv0.column end) and slot6[3] / 10000 or 0) + (slot7 and slot7[1] / 10000 or 0) if slot6 == 0 then slot8 = 0.05 + slot4 / (slot4 + slot1:getInvestSums()) / 4 * math.max(slot1.step - 1, 0) + slot6 - slot1:getEquipAmbushRateReduce() end return math.clamp(slot8, 0, 1) end function slot0.getAmbushDodge(slot0, slot1) slot2 = slot1.line slot3 = slot1:getDodgeSums() if (_.detect(slot0:getConfig("ambush_ratio_extra"), function (slot0) return slot0[1] == uv0.row and slot0[2] == uv0.column end) and slot6[3] / 10000 or 0) == 0 then slot5 = slot3 / (slot3 + slot0:getConfig("avoid_ratio")) + slot1:getEquipDodgeRateUp() end return math.clamp(slot5, 0, 1) end function slot0.isValid(slot0) if slot0:getPlayType() == ChapterConst.TypeMainSub then return slot0.active or slot0.expireTime and pg.TimeMgr.GetInstance():GetServerTime() < slot0.expireTime end return true end function slot0.inWartime(slot0) return slot0.dueTime and pg.TimeMgr.GetInstance():GetServerTime() < slot0.dueTime end function slot0.inActTime(slot0) if slot0:getConfig("act_id") == 0 then return true end slot2 = slot1 and getProxy(ActivityProxy):getActivityById(slot1) return slot2 and not slot2:isEnd() end function slot0.getRemainTime(slot0) return slot0.dueTime and math.max(slot0.dueTime - pg.TimeMgr.GetInstance():GetServerTime() - 1, 0) or 0 end function slot0.getStartTime(slot0) return math.max(slot0.dueTime - slot0:getConfig("time"), 0) end function slot0.isUnlock(slot0) if slot0:getConfig("pre_chapter") == 0 then return true else return getProxy(ChapterProxy):getChapterById(slot1):isClear() end end function slot0.isClear(slot0) if slot0:getPlayType() == ChapterConst.TypeMainSub then return true else return slot0.progress >= 100 end end function slot0.ifNeedHide(slot0) slot1 = false if pg.chapter_setting and table.contains(pg.chapter_setting.all, slot0.id) and pg.chapter_setting[slot0.id].hide == 1 then slot1 = true end if slot1 and slot0:isClear() then return true end end function slot0.existAchieve(slot0) return #slot0.achieves > 0 end function slot0.isAllAchieve(slot0) for slot5, slot6 in ipairs(slot0.achieves) do if ChapterConst.IsAchieved(slot6) then slot1 = 0 + 1 end end return slot1 == #slot0.achieves end function slot0.findPath(slot0, slot1, slot2, slot3) slot4 = {} for slot8 = 0, ChapterConst.MaxRow - 1 do slot4[slot8] = slot4[slot8] or {} for slot12 = 0, ChapterConst.MaxColumn - 1 do slot4[slot8][slot12] = slot4[slot8][slot12] or {} slot13 = PathFinding.PrioForbidden slot14 = ChapterConst.ForbiddenAll if slot0.cells[ChapterCell.Line2Name(slot8, slot12)] and slot16:IsWalkable() then slot13 = PathFinding.PrioNormal if slot0:considerAsObstacle(slot1, slot16.row, slot16.column) then slot13 = PathFinding.PrioObstacle end slot14 = (slot1 ~= ChapterConst.SubjectPlayer or slot16.forbiddenDirections) and ChapterConst.ForbiddenNone end slot4[slot8][slot12].forbiddens = slot14 slot4[slot8][slot12].priority = slot13 end end if slot1 == ChapterConst.SubjectPlayer then for slot9, slot10 in ipairs(slot0:getCoastalGunArea()) do slot4[slot10.row][slot10.column].priority = math.max(slot4[slot10.row][slot10.column].priority, PathFinding.PrioObstacle) end end if slot4[slot3.row] and slot4[slot3.row][slot3.column] then slot5.priority = slot0:considerAsStayPoint(slot1, slot3.row, slot3.column) and PathFinding.PrioNormal or PathFinding.PrioObstacle end slot0.pathFinder.cells = slot4 return slot0.pathFinder:Find(slot2, slot3) end function slot0.FindBossPath(slot0, slot1, slot2, slot3) slot4 = {} for slot8 = 0, ChapterConst.MaxRow - 1 do slot4[slot8] = slot4[slot8] or {} for slot12 = 0, ChapterConst.MaxColumn - 1 do slot4[slot8][slot12] = slot4[slot8][slot12] or {} slot13 = PathFinding.PrioForbidden slot14 = ChapterConst.ForbiddenAll slot15 = nil if slot0.cells[ChapterCell.Line2Name(slot8, slot12)] and slot17:IsWalkable() then slot13 = PathFinding.PrioNormal if slot0:considerAsObstacle(slot1, slot17.row, slot17.column) then slot13 = PathFinding.PrioObstacle end slot18, slot19 = slot0:existEnemy(slot1, slot17.row, slot17.column) if slot18 then slot13 = PathFinding.PrioNormal slot15 = slot19 ~= ChapterConst.AttachBoss end slot14 = (slot1 ~= ChapterConst.SubjectPlayer or slot17.forbiddenDirections) and ChapterConst.ForbiddenNone end slot4[slot8][slot12].forbiddens = slot14 slot4[slot8][slot12].priority = slot13 slot4[slot8][slot12].isEnemy = slot15 end end if slot1 == ChapterConst.SubjectPlayer then for slot9, slot10 in ipairs(slot0:getCoastalGunArea()) do slot4[slot10.row][slot10.column].priority = math.max(slot4[slot10.row][slot10.column].priority, PathFinding.PrioObstacle) end end if slot4[slot3.row] and slot4[slot3.row][slot3.column] then slot5.priority = slot0:considerAsStayPoint(slot1, slot3.row, slot3.column) and PathFinding.PrioNormal or PathFinding.PrioObstacle end return OrientedWeightPathFinding.StaticFind(slot4, ChapterConst.MaxRow, ChapterConst.MaxColumn, slot2, slot3) end function slot0.getWaveCount(slot0) for slot5, slot6 in pairs(slot0.cells) do if slot6.attachment == ChapterConst.AttachEnemy and underscore.detect(slot0:getConfig("grids"), function (slot0) if slot0[1] == uv0.row and slot0[2] == uv0.column and (slot0[4] == ChapterConst.AttachElite or slot0[4] == ChapterConst.AttachEnemy) then return true end return false end) then slot1 = 0 + 1 end end slot3 = slot0:getConfig("elite_refresh") for slot8, slot9 in pairs(slot0:getConfig("enemy_refresh")) do if slot8 <= #slot3 then slot4 = 0 + slot9 + slot3[slot8] end if slot1 <= slot4 then return slot8 end end return 1 end function slot0.bossRefreshed(slot0) for slot4, slot5 in pairs(slot0.cells) do if slot5.attachment == ChapterConst.AttachBoss then return true end end return false end function slot0.getFleetAmmo(slot0, slot1) if slot1:getFleetType() == FleetType.Normal then slot2 = slot1:getShipAmmo() + slot0:getConfig("ammo_total") elseif slot3 == FleetType.Submarine then slot2 = slot2 + slot0:getConfig("ammo_submarine") end return slot2, slot1.restAmmo end function slot0.GetActiveStrategies(slot0) table.insert(_.filter(slot0.fleet:getStrategies(), function (slot0) return pg.strategy_data_template[slot0.id] and slot1.type ~= ChapterConst.StgTypeBindFleetPassive end), 1, { id = slot0.fleet:getFormationStg() }) if slot0:GetSubmarineFleet() then table.insert(slot1, 3, { id = ChapterConst.StrategyHuntingRange }) table.insert(slot1, 4, { id = ChapterConst.StrategySubAutoAttack }) table.insert(slot1, 5, { id = ChapterConst.StrategySubTeleport }) end if #slot0.strategies > 0 then for slot7, slot8 in pairs(slot0.strategies) do table.insert(slot1, { id = slot7, count = slot8 }) end end return slot1 end function slot0.getFleetStgs(slot0, slot1) if slot0:getPlayType() == ChapterConst.TypeLagacy then slot2 = _.filter(slot1:getStrategies(), function (slot0) return pg.strategy_data_template[slot0.id].id == ChapterConst.StrategyRepair or slot1.id == ChapterConst.StrategyExchange end) end return slot2 end function slot0.getFleetStates(slot0, slot1) slot3, slot4 = slot0:getFleetAmmo(slot1) if ChapterConst.AmmoRich <= slot4 then table.insert({}, ChapterConst.StrategyAmmoRich) elseif slot4 <= ChapterConst.AmmoPoor then table.insert(slot2, ChapterConst.StrategyAmmoPoor) end function slot10(slot0) return slot0.id end table.insertto(slot2, _.map(_.filter(slot1:getStrategies(), function (slot0) return pg.strategy_data_template[slot0.id] and slot1.type == ChapterConst.StgTypeBindFleetPassive and slot0.count > 0 end), slot10)) table.insertto(slot2, slot1.stgIds) for slot10, slot11 in ipairs(slot0:getConfig("chapter_buff")) do table.insert(slot2, slot11) end if OPEN_AIR_DOMINANCE and slot0:getConfig("air_dominance") > 0 then table.insert(slot2, slot0:getAirDominanceStg()) end for slot10, slot11 in ipairs(slot0:getExtraFlags()) do table.insert(slot2, ChapterConst.Status2Stg[slot11]) end if slot0:getOperationBuffDescStg() then table.insert(slot2, slot7) end return slot2 end function slot0.GetShowingStartegies(slot0) slot2 = slot0:getFleetStates(slot0.fleet) if slot0:getPlayType() == ChapterConst.TypeDOALink and slot0:GetBuffOfLinkAct() then table.insert(slot2, pg.gameset.doa_fever_strategy.description[table.indexof(pg.gameset.doa_fever_buff.description, slot3)]) end return _.filter(slot2, function (slot0) return pg.strategy_data_template[slot0].icon ~= "" end) end function slot0.getAirDominanceStg(slot0) slot1, slot2 = slot0:getAirDominanceValue() return ChapterConst.AirDominance[slot2].StgId end function slot0.getAirDominanceValue(slot0) for slot6, slot7 in ipairs(slot0.fleets) do if slot7:isValid() then slot1 = 0 + slot7:getFleetAirDominanceValue() slot2 = 0 + slot7:getAntiAircraftSums() end end return slot1, calcAirDominanceStatus(slot1, slot0:getConfig("air_dominance"), slot2), slot0.airDominanceStatus end function slot0.getOperationBuffDescStg(slot0) for slot4, slot5 in ipairs(slot0.operationBuffList) do if pg.benefit_buff_template[slot5].benefit_type == Chapter.OPERATION_BUFF_TYPE_DESC then return slot5 end end end function slot0.setAirDominanceStatus(slot0, slot1) slot0.airDominanceStatus = slot1 end function slot0.updateExtraFlags(slot0, slot1, slot2) slot3 = false for slot7, slot8 in ipairs(slot2) do for slot12, slot13 in ipairs(slot0.extraFlagList) do if slot13 == slot8 then table.remove(slot0.extraFlagList, slot12) slot3 = true break end end end for slot7, slot8 in ipairs(slot1) do if not table.contains(slot0.extraFlagList, slot8) then table.insert(slot0.extraFlagList, slot8) slot3 = true end end return slot3 end function slot0.getExtraFlags(slot0) return slot0.extraFlagList end function slot0.updateShipStg(slot0, slot1, slot2, slot3) slot0.fleet:updateShipStg(slot1, slot2, slot3) end function slot0.UpdateBuffList(slot0, slot1) if not slot1 then return end for slot5, slot6 in ipairs(slot1) do if not _.include(slot0.buff_list, slot6) then table.insert(slot0.buff_list, slot6) end end end function slot0.getFleetBattleBuffs(slot0, slot1) slot2 = slot0.buff_list and Clone(slot0.buff_list) or {} _.each(slot0:getFleetStates(slot1), function (slot0) if pg.strategy_data_template[slot0].buff_id ~= 0 then table.insert(uv0, slot1) end end) table.insertto(slot2, slot0:GetFleetAttachmentConfig("attach_buff", slot1.line.row, slot1.line.column) or {}) _.each(slot0:GetWeather(), function (slot0) if type(pg.weather_data_template[slot0].effect_args) == "table" and slot1.buff and slot1.buff > 0 then table.insert(uv0, slot1.buff) end end) return slot2, slot0:buildBattleBuffList(slot1) end function slot0.GetFleetAttachmentConfig(slot0, slot1, slot2, slot3) if not slot0.cellAttachments[ChapterCell.Line2Name(slot2 or slot0.fleet.line.row, slot3 or slot0.fleet.line.column)] then return end return uv0.GetEventTemplateByKey(slot1, slot5.attachmentId) end function slot0.GetCellEventByKey(slot0, slot1, slot2, slot3) if not slot0.cells[ChapterCell.Line2Name(slot2 or slot0.fleet.line.row, slot3 or slot0.fleet.line.column)] then return end return uv0.GetEventTemplateByKey(slot1, slot5.attachmentId) end function slot0.GetEventTemplateByKey(slot0, slot1) if not pg.map_event_template[slot1] then return end slot3 = nil for slot7, slot8 in ipairs(slot2.effect) do if slot8[1] == slot0 then for slot12 = 2, #slot8 do table.insert(slot3 or {}, slot8[slot12]) end end end return slot3 end function slot0.buildBattleBuffList(slot0, slot1) slot2 = {} slot3, slot4 = slot0:triggerSkill(slot1, FleetSkill.TypeBattleBuff) if slot3 and #slot3 > 0 then slot5 = {} for slot9, slot10 in ipairs(slot3) do slot5[slot12] = slot5[slot1:findCommanderBySkillId(slot4[slot9].id)] or {} table.insert(slot5[slot12], slot10) end for slot9, slot10 in pairs(slot5) do table.insert(slot2, { slot9, slot10 }) end end for slot9, slot10 in pairs(slot1:getCommanders()) do for slot15, slot16 in ipairs(slot10:getTalents()) do if #slot16:getBuffsAddition() > 0 then slot18 = nil for slot22, slot23 in ipairs(slot2) do if slot23[1] == slot10 then slot18 = slot23[2] break end end if not slot18 then table.insert(slot2, { slot10, {} }) end for slot22, slot23 in ipairs(slot17) do table.insert(slot18, slot23) end end end end return slot2 end function slot0.updateShipHp(slot0, slot1, slot2) slot0.fleet:updateShipHp(slot1, slot2) end function slot0.updateFleetShipHp(slot0, slot1, slot2) for slot6, slot7 in ipairs(slot0.fleets) do slot7:updateShipHp(slot1, slot2) if slot7.id ~= slot0.fleet.id then slot7:clearShipHpChange() end end end function slot0.DealDMG2Fleets(slot0, slot1) for slot5, slot6 in ipairs(slot0.fleets) do slot6:DealDMG2Ships(slot1) end end function slot0.clearEliterFleetByIndex(slot0, slot1) if slot1 > #slot0.eliteFleetList then return end slot0.eliteFleetList[slot1] = {} end function slot0.wrapEliteFleet(slot0, slot1) slot2 = {} slot3 = _.flatten(slot0:getEliteFleetList()[slot1]) for slot7, slot8 in pairs(slot0:getEliteFleetCommanders()[slot1]) do table.insert(slot2, { pos = slot7, id = slot8 }) end return Fleet.New({ id = 1, ship_list = slot3, commanders = slot2 }) end function slot0.setEliteCommanders(slot0, slot1) slot0.eliteCommanderList = slot1 end function slot0.getEliteFleetCommanders(slot0) return slot0.eliteCommanderList end function slot0.updateCommander(slot0, slot1, slot2, slot3) slot0.eliteCommanderList[slot1][slot2] = slot3 end function slot0.getEliteFleetList(slot0) slot0:EliteShipTypeFilter() return slot0.eliteFleetList end function slot0.setEliteFleetList(slot0, slot1) slot0.eliteFleetList = slot1 end function slot0.IsEliteFleetLegal(slot0) slot1 = 0 slot2 = 0 slot4 = 0 slot5, slot6 = nil for slot10 = 1, #slot0.eliteFleetList do slot11, slot12 = slot0:singleEliteFleetVertify(slot10) if not slot11 then if not slot12 then if slot10 >= 3 then slot3 = 0 + 1 else slot1 = slot1 + 1 end else slot5 = slot12 slot6 = slot10 end elseif slot10 >= 3 then slot4 = slot4 + 1 else slot2 = slot2 + 1 end end if slot1 == 2 then return false, i18n("elite_disable_no_fleet") elseif slot2 == 0 then return false, slot5 elseif slot3 + slot4 < slot0:getConfig("submarine_num") then return false, slot5 end slot7, slot8 = slot0:IsPropertyLimitationSatisfy() for slot13, slot14 in ipairs(slot7) do slot9 = 1 * slot14 end if slot9 ~= 1 then return false, i18n("elite_disable_property_unsatisfied") end return true, slot6 end function slot0.IsPropertyLimitationSatisfy(slot0) slot1 = getProxy(BayProxy):getRawData() slot3 = { [slot8[1]] = 0 } for slot7, slot8 in ipairs(slot0:getConfig("property_limitation")) do -- Nothing end slot4 = 0 for slot8 = 1, 2 do if slot0:singleEliteFleetVertify(slot8) then slot9 = { [slot17] = 0 } slot11 = 0 for slot15, slot16 in ipairs(slot2) do slot17, slot18, slot19, ({})[slot17] = unpack(slot16) if string.sub(slot17, 1, 5) == "fleet" then -- Nothing end end for slot16, slot17 in ipairs(slot0.eliteFleetList[slot8]) do slot4 = slot4 + 1 slot19 = intProperties(slot1[slot17]:getProperties()) for slot23, slot24 in pairs(slot3) do if string.sub(slot23, 1, 5) == "fleet" then if slot23 == "fleet_totle_level" then slot9[slot23] = slot9[slot23] + slot18.level end elseif slot23 == "level" then slot3[slot23] = slot24 + slot18.level else slot3[slot23] = slot24 + slot19[slot23] end end end for slot16, slot17 in pairs(slot9) do if slot16 == "fleet_totle_level" and slot10[slot16] < slot17 then slot3[slot16] = slot3[slot16] + 1 end end end end slot5 = {} for slot9, slot10 in ipairs(slot2) do slot11, slot12, slot13, slot14 = unpack(slot10) if slot11 == "level" and slot4 > 0 then slot3[slot11] = math.ceil(slot3[slot11] / slot4) end slot5[slot9] = AttributeType.EliteConditionCompare(slot12, slot3[slot11], slot13) and 1 or 0 end return slot5, slot3 end function slot0.EliteShipTypeFilter(slot0) function slot1(slot0, slot1, slot2) for slot7, slot8 in ipairs(slot1) do if slot8 == 0 then slot3 = 0 + 1 end end slot4 = {} for slot8, slot9 in ipairs(Clone(slot1)) do if type(slot9) == "string" then for slot14, slot15 in ipairs(Clone(ShipType.BundleList[slot9])) do slot4[#slot4 + 1] = slot15 end end end for slot8, slot9 in ipairs(slot4) do slot1[#slot1 + 1] = slot9 end for slot8, slot9 in pairs(slot2) do if table.contains(slot1, slot8:getShipType()) then slot11 = nil for slot15, slot16 in ipairs(slot1) do if slot16 == slot10 then slot11 = slot15 break end end table.remove(slot1, slot11) elseif slot3 - 1 < 0 then slot11 = nil for slot15, slot16 in ipairs(slot0) do if slot16 == slot9 then slot11 = slot15 break end end table.remove(slot0, slot11) end end end slot2 = getProxy(BayProxy):getRawData() for slot6, slot7 in ipairs(slot0.eliteFleetList) do for slot11 = #slot7, 1, -1 do if slot2[slot7[slot11]] == nil then table.remove(slot7, slot11) end end end slot6 = "limitation" for slot6, slot7 in ipairs(slot0:getConfig(slot6)) do slot10 = {} slot11 = {} for slot15, slot16 in ipairs(slot0.eliteFleetList[slot6]) do if slot2[slot16]:getTeamType() == TeamType.Main then -- Nothing elseif slot18 == TeamType.Vanguard then slot10[slot17] = slot16 elseif slot18 == TeamType.Submarine then slot11[slot17] = slot16 end end slot1(slot8, slot7[1], { [slot17] = slot16 }) slot1(slot8, slot7[2], slot10) slot1(slot8, { 0, 0, 0 }, slot11) end end function slot0.singleEliteFleetVertify(slot0, slot1) slot2 = getProxy(BayProxy):getRawData() slot4 = slot0:getConfig("limitation") if #slot0.eliteFleetList[slot1] == 0 then return false else slot5 = 0 slot6 = 0 slot7 = {} for slot11, slot12 in ipairs(slot3) do if slot2[slot12]:getFlag("inEvent") then return false, i18n("elite_disable_ship_escort") end if slot13:getTeamType() == TeamType.Main then slot5 = slot5 + 1 elseif slot14 == TeamType.Vanguard then slot6 = slot6 + 1 end slot7[#slot7 + 1] = slot13:getShipType() end if slot1 >= 3 then -- Nothing elseif slot5 * slot6 == 0 and slot5 + slot6 ~= 0 then return false else slot8 = 1 for slot12, slot13 in ipairs(slot4[slot1]) do slot15 = 0 for slot19, slot20 in ipairs(slot13) do if type(slot20) == "number" then slot14 = 0 + slot20 if slot20 ~= 0 and table.contains(slot7, slot20) then slot15 = 1 end elseif type(slot20) == "string" then slot14 = slot14 + 1 slot21 = Clone(ShipType.BundleList[slot20]) if _.any(slot7, function (slot0) return table.contains(uv0, slot0) end) then slot15 = 1 end end end if slot14 == 0 then slot15 = 1 end slot8 = slot8 * slot15 end if slot8 == 0 then return false, i18n("elite_disable_formation_unsatisfied") end end end return true end function slot0.getDragExtend(slot0) slot1 = slot0.theme slot3 = 99999999 slot4 = 0 slot5 = 0 for slot9, slot10 in pairs(slot0.cells) do if slot10.row < 99999999 then slot2 = slot10.row end if slot4 < slot10.row then slot4 = slot10.row end if slot10.column < slot3 then slot3 = slot10.column end if slot5 < slot10.column then slot5 = slot10.column end end slot6 = (slot5 + slot3) * 0.5 slot7 = (slot4 + slot2) * 0.5 slot8 = slot1.cellSize + slot1.cellSpace return math.max((slot6 - slot3 + 1) * slot8.x, 0), math.max((slot5 - slot6 + 1) * slot8.x, 0), math.max((slot7 - slot2 + 1) * slot8.y, 0), math.max((slot4 - slot7 + 1) * slot8.y, 0) end function slot0.getPoisonArea(slot0, slot1) slot3 = slot0.theme.cellSize + slot0.theme.cellSpace slot4 = math.floor(slot3.x * slot1) slot5 = math.floor(slot3.y * slot1) for slot9, slot10 in pairs(slot0.cells) do if slot10:checkHadFlag(ChapterConst.FlagPoison) then -- Nothing end end return { [slot9] = { x = math.floor(slot4 * (slot10.column - slot0.indexMin.y)), y = math.floor(slot5 * (slot10.row - slot0.indexMin.x)), w = slot4, h = slot5 } } end function slot0.selectFleets(slot0, slot1, slot2) slot3 = Clone(slot1) or {} for slot7 = #slot3, 1, -1 do if not slot2[slot3[slot7]] or not slot8:isUnlock() or slot8:isLegalToFight() ~= true then table.remove(slot3, slot7) end end slot6 = slot0:getConfig("submarine_num") for slot10 = #({ [FleetType.Normal] = _.filter(slot3, function (slot0) return uv0[slot0]:getFleetType() == FleetType.Normal end), [FleetType.Submarine] = _.filter(slot3, function (slot0) return uv0[slot0]:getFleetType() == FleetType.Submarine end) })[FleetType.Normal], slot0:getConfig("group_num") + 1, -1 do table.remove(slot4[FleetType.Normal], slot10) end for slot10 = #slot4[FleetType.Submarine], slot6 + 1, -1 do table.remove(slot4[FleetType.Submarine], slot10) end for slot10, slot11 in pairs(slot4) do if #slot11 == 0 then slot12 = 0 if slot10 == FleetType.Normal then slot12 = slot5 elseif slot10 == FleetType.Submarine then slot12 = slot6 end for slot16, slot17 in pairs(slot2) do if slot12 <= #slot11 then break end if slot17 and slot17:getFleetType() == slot10 and slot17:isUnlock() and slot17:isLegalToFight() == true then table.insert(slot11, slot16) end end end end slot3 = {} for slot10, slot11 in pairs(slot4) do for slot15, slot16 in ipairs(slot11) do table.insert(slot3, slot16) end end table.sort(slot3, function (slot0, slot1) return slot0 < slot1 end) return slot3 end function slot0.getInEliteShipIDs(slot0) slot1 = {} for slot5, slot6 in ipairs(slot0.eliteFleetList) do for slot10, slot11 in ipairs(slot6) do slot1[#slot1 + 1] = slot11 end end return slot1 end function slot0.activeAlways(slot0) if getProxy(ChapterProxy):getMapById(slot0:getConfig("map")):isActivity() and type(pg.activity_template[slot0:getConfig("act_id")].config_client) == "table" then return table.contains(slot3.config_client.prevs or {}, slot0.id) end return false end function slot0.getPrevChapterName(slot0) slot1 = "" if slot0:getConfig("pre_chapter") ~= 0 then slot1 = slot0:bindConfigTable()[slot2].chapter_name end return slot1 end function slot0.getMaxColumnByRow(slot0, slot1) for slot6, slot7 in pairs(slot0.cells) do if slot7.row == slot1 then slot2 = math.max(-1, slot7.column) end end return slot2 end function slot0.withNpc(slot0) _.any(slot0.fleets, function (slot0) if slot0.npcShip then uv0 = _.detect(slot0:getShips(true), function (slot0) return slot0.id == uv0.npcShip.id end) end return uv0 ~= nil end) return Clone(nil) end function slot0.getDefeatStory(slot0, slot1) if #slot0:getConfig("defeat_story_count") <= 0 then return nil end slot4 = slot3 while slot4 > 0 do if slot2[slot4] <= slot1 then break end slot4 = slot4 - 1 end return slot0:getConfig("defeat_story")[slot4] end function slot0.getFleet(slot0, slot1, slot2, slot3) return _.detect(slot0.fleets, function (slot0) return slot0.line.row == uv0 and slot0.line.column == uv1 and (not uv2 or slot0:getFleetType() == uv2) and slot0:isValid() end) or _.detect(slot0.fleets, function (slot0) return slot0.line.row == uv0 and slot0.line.column == uv1 and (not uv2 or slot0:getFleetType() == uv2) end) end function slot0.getFleetIndex(slot0, slot1, slot2, slot3) if slot0:getFleet(slot1, slot2, slot3) then return table.indexof(slot0.fleets, slot4) end end function slot0.getOni(slot0) return _.detect(slot0.champions, function (slot0) return slot0.attachment == ChapterConst.AttachOni end) end function slot0.getChampion(slot0, slot1, slot2) return _.detect(slot0.champions, function (slot0) return slot0.row == uv0 and slot0.column == uv1 end) end function slot0.getChampionIndex(slot0, slot1, slot2) if slot0:getChampion(slot1, slot2) then return table.indexof(slot0.champions, slot3) end end function slot0.getChampionVisibility(slot0, slot1, slot2, slot3) return slot1.flag == ChapterConst.CellFlagActive end function slot0.mergeChampion(slot0, slot1) if slot0:getChampionIndex(slot1.row, slot1.column) then slot0.champions[slot2] = slot1 return true else slot1.trait = ChapterConst.TraitLurk table.insert(slot0.champions, slot1) return false end end function slot0.mergeChampionImmediately(slot0, slot1) if slot0:getChampionIndex(slot1.row, slot1.column) then slot0.champions[slot2] = slot1 return true else table.insert(slot0.champions, slot1) return false end end function slot0.RemoveChampion(slot0, slot1) if table.indexof(slot0.champions, slot1) then table.remove(slot0.champions, slot2) end end function slot0.considerAsObstacle(slot0, slot1, slot2, slot3) if not slot0:getChapterCell(slot2, slot3) or not slot4:IsWalkable(slot1) then return true end if slot0:existBarrier(slot2, slot3) then return true end if slot1 == ChapterConst.SubjectPlayer then if slot4.flag == ChapterConst.CellFlagActive then if slot4.attachment == ChapterConst.AttachEnemy or slot4.attachment == ChapterConst.AttachAmbush or slot4.attachment == ChapterConst.AttachElite or slot4.attachment == ChapterConst.AttachBoss or slot4.attachment == ChapterConst.AttachAreaBoss or slot4.attachment == ChapterConst.AttachBomb_Enemy then return true end if slot4.attachment == ChapterConst.AttachBox and pg.box_data_template[slot4.attachmentId].type == ChapterConst.BoxTorpedo then return true end if slot4.attachment == ChapterConst.AttachStory then return true end end if slot0:existChampion(slot2, slot3) then return true end elseif slot1 == ChapterConst.SubjectChampion and slot0:existFleet(FleetType.Normal, slot2, slot3) then return true end return false end function slot0.considerAsStayPoint(slot0, slot1, slot2, slot3) if not slot0:getChapterCell(slot2, slot3) or not slot4:IsWalkable() then return false end if slot0:existBarrier(slot2, slot3) then return false end if slot1 == ChapterConst.SubjectPlayer then if slot4.flag == ChapterConst.CellFlagActive and slot4.attachment == ChapterConst.AttachStory then return true end if slot4.attachment == ChapterConst.AttachLandbase and pg.land_based_template[slot4.attachmentId] and pg.land_based_template[slot4.attachmentId].type == ChapterConst.LBHarbor then return false end if slot0:existFleet(FleetType.Normal, slot2, slot3) then return false end if slot0:existOni(slot2, slot3) then return false end if slot0:existBombEnemy(slot2, slot3) then return false end elseif slot1 == ChapterConst.SubjectChampion then if slot4.flag ~= ChapterConst.CellFlagDisabled and slot4.attachment ~= ChapterConst.AttachNone then return false end if slot0:getChampion(slot2, slot3) and slot5.flag ~= ChapterConst.CellFlagDisabled then return false end end return true end function slot0.existAny(slot0, slot1, slot2) if slot0:getChapterCell(slot1, slot2).attachment ~= ChapterConst.AttachNone and slot3.flag == ChapterConst.CellFlagActive then return true end if slot0:existFleet(nil, slot1, slot2) then return true end if slot0:existChampion(slot1, slot2) then return true end end function slot0.existBarrier(slot0, slot1, slot2) if slot0:getChapterCell(slot1, slot2).attachment == ChapterConst.AttachBox and slot3.flag == ChapterConst.CellFlagActive and pg.box_data_template[slot3.attachmentId].type == ChapterConst.BoxBarrier then return true end if slot3.attachment == ChapterConst.AttachStory and slot3.flag == ChapterConst.CellFlagTriggerActive and pg.map_event_template[slot3.attachmentId].type == ChapterConst.StoryObstacle then return true end if slot0:getChampion(slot1, slot2) and slot4.flag ~= ChapterConst.CellFlagDisabled and pg.expedition_data_template[slot4.attachmentId] and slot5.type == 97 then return true end return false end function slot0.existEnemy(slot0, slot1, slot2, slot3) if slot1 == ChapterConst.SubjectPlayer then if slot0:getChapterCell(slot2, slot3) and slot4.flag == ChapterConst.CellFlagActive and (slot4.attachment == ChapterConst.AttachAmbush or slot4.attachment == ChapterConst.AttachEnemy or slot4.attachment == ChapterConst.AttachElite or slot4.attachment == ChapterConst.AttachBoss or slot4.attachment == ChapterConst.AttachAreaBoss or slot4.attachment == ChapterConst.AttachBomb_Enemy) then return true, slot4.attachment end if slot0:existChampion(slot2, slot3) then return true, ChapterConst.AttachChampion end elseif slot1 == ChapterConst.SubjectChampion and (slot0:existFleet(FleetType.Normal, slot2, slot3) or slot0:existFleet(FleetType.Transport, slot2, slot3)) then return true end end function slot0.existFleet(slot0, slot1, slot2, slot3) if _.any(slot0.fleets, function (slot0) return slot0.line.row == uv0 and slot0.line.column == uv1 and (not uv2 or slot0:getFleetType() == uv2) and slot0:isValid() end) then return true end end function slot0.existChampion(slot0, slot1, slot2) if _.any(slot0.champions, function (slot0) return slot0.flag == ChapterConst.CellFlagActive and slot0.row == uv0 and slot0.column == uv1 and uv2:getChampionVisibility(slot0) end) then return true end end function slot0.existAlly(slot0, slot1) return _.any(slot0.fleets, function (slot0) return slot0.id ~= uv0.id and slot0.line.row == uv0.line.row and slot0.line.column == uv0.line.column and slot0:isValid() end) end function slot0.existOni(slot0, slot1, slot2) return _.any(slot0.champions, function (slot0) return slot0.attachment == ChapterConst.AttachOni and slot0.flag == ChapterConst.CellFlagActive and (not uv0 or uv0 == slot0.row) and (not uv1 or uv1 == slot0.column) end) end function slot0.existBombEnemy(slot0, slot1, slot2) if slot1 and slot2 then return slot0:getChapterCell(slot1, slot2).attachment == ChapterConst.AttachBomb_Enemy and slot3.flag == ChapterConst.CellFlagActive end for slot6, slot7 in pairs(slot0.cells) do if slot7.attachment == ChapterConst.AttachBomb_Enemy and slot7.flag == ChapterConst.CellFlagActive and (not slot1 or slot1 == slot7.row) and (not slot2 or slot2 == slot7.column) then return true end end return false end function slot0.isPlayingWithBombEnemy(slot0) for slot4, slot5 in pairs(slot0.cells) do if slot5.attachment == ChapterConst.AttachBomb_Enemy then return true end end return false end function slot0.existCoastalGunNoMatterLiveOrDead(slot0) for slot4, slot5 in pairs(slot0.cells) do if slot5.attachment == ChapterConst.AttachLandbase and pg.land_based_template[slot5.attachmentId].type == ChapterConst.LBCoastalGun then return true end end return false end slot1 = { { 1, 0 }, { -1, 0 }, { 0, 1 }, { 0, -1 } } function slot0.calcWalkableCells(slot0, slot1, slot2, slot3, slot4) slot5 = {} for slot9 = 0, ChapterConst.MaxRow - 1 do if not slot5[slot9] then slot5[slot9] = {} end for slot13 = 0, ChapterConst.MaxColumn - 1 do slot5[slot9][slot13] = slot0.cells[ChapterCell.Line2Name(slot9, slot13)] and slot15:IsWalkable() end end slot6 = {} if slot1 == ChapterConst.SubjectPlayer then for slot11, slot12 in ipairs(slot0:getCoastalGunArea()) do slot6[slot12.row .. "_" .. slot12.column] = true end end if not slot0:GetRawChapterCell(slot2, slot3) then return {} end slot9 = { { step = 0, row = slot2, column = slot3, forbiddens = slot8.forbiddenDirections } } slot10 = {} while #slot9 > 0 do table.insert(slot10, table.remove(slot9, 1)) _.each(uv0, function (slot0) slot1 = { row = uv0.row + slot0[1], column = uv0.column + slot0[2], step = uv0.step + 1 } if not uv1:GetRawChapterCell(slot1.row, slot1.column) then return end slot1.forbiddens = slot2.forbiddenDirections if slot1.step <= uv2 and not OrientedPathFinding.IsDirectionForbidden(uv0, slot0[1], slot0[2]) and not (_.any(uv3, function (slot0) return slot0.row == uv0.row and slot0.column == uv0.column end) or _.any(uv4, function (slot0) return slot0.row == uv0.row and slot0.column == uv0.column end)) and uv5[slot1.row][slot1.column] then table.insert(uv6, slot1) if not uv1:existEnemy(uv7, slot1.row, slot1.column) and not uv1:existBarrier(slot1.row, slot1.column) and not uv8[slot1.row .. "_" .. slot1.column] then table.insert(uv3, slot1) end end end) end return _.filter(slot7, function (slot0) return slot0.row == uv0 and slot0.column == uv1 or uv2:considerAsStayPoint(uv3, slot0.row, slot0.column) end) end function slot0.calcAreaCells(slot0, slot1, slot2, slot3, slot4) slot5 = {} for slot9 = 0, ChapterConst.MaxRow - 1 do if not slot5[slot9] then slot5[slot9] = {} end for slot13 = 0, ChapterConst.MaxColumn - 1 do slot5[slot9][slot13] = slot0.cells[ChapterCell.Line2Name(slot9, slot13)] and slot15:IsWalkable() end end slot6 = {} slot7 = { { step = 0, row = slot1, column = slot2 } } slot8 = {} while #slot7 > 0 do table.insert(slot8, table.remove(slot7, 1)) _.each(uv0, function (slot0) if ({ row = uv0.row + slot0[1], column = uv0.column + slot0[2], step = uv0.step + 1 }).row >= 0 and slot1.row < ChapterConst.MaxRow and slot1.column >= 0 and slot1.column < ChapterConst.MaxColumn and slot1.step <= uv1 and not (_.any(uv2, function (slot0) return slot0.row == uv0.row and slot0.column == uv0.column end) or _.any(uv3, function (slot0) return slot0.row == uv0.row and slot0.column == uv0.column end)) then table.insert(uv2, slot1) if uv4[slot1.row][slot1.column] and uv5 <= slot1.step then table.insert(uv6, slot1) end end end) end return slot6 end function slot0.calcSquareBarrierCells(slot0, slot1, slot2, slot3) slot4 = {} for slot8 = -slot3, slot3 do for slot12 = -slot3, slot3 do if slot0:getChapterCell(slot1 + slot8, slot2 + slot12) and slot15:IsWalkable() and (slot0:existBarrier(slot13, slot14) or not slot0:existAny(slot13, slot14)) then table.insert(slot4, { row = slot13, column = slot14 }) end end end return slot4 end function slot0.checkAnyInteractive(slot0) slot1 = slot0.fleet.line slot2 = slot0:getChapterCell(slot1.row, slot1.column) slot3 = false if slot0.fleet:getFleetType() == FleetType.Normal then if slot0:existEnemy(ChapterConst.SubjectPlayer, slot2.row, slot2.column) then if slot0:getRound() == ChapterConst.RoundPlayer then slot3 = true end elseif slot2.attachment == ChapterConst.AttachAmbush or slot2.attachment == ChapterConst.AttachBox then if slot2.flag ~= ChapterConst.CellFlagDisabled then slot3 = true end elseif slot2.attachment == ChapterConst.AttachStory then slot3 = slot2.flag == ChapterConst.CellFlagActive elseif slot2.attachment == ChapterConst.AttachSupply and slot2.attachmentId > 0 then slot4, slot5 = slot0:getFleetAmmo(slot0.fleet) if slot5 < slot4 then slot3 = true end elseif slot2.attachment == ChapterConst.AttachBox and slot2.flag ~= ChapterConst.CellFlagDisabled then slot3 = true end end return slot3 end function slot0.calcAttachmenArea(slot0, slot1) slot2 = slot0.theme for slot10, slot11 in pairs(slot0.cells) do if slot11.attachment == slot1 then slot3 = math.min(ChapterConst.MaxRow + 1, slot11.row) slot4 = math.max(-1, slot11.row) slot5 = math.min(ChapterConst.MaxColumn + 1, slot11.column) slot6 = math.max(-1, slot11.column) end end slot8 = Vector2(slot6 - slot5 + 1, slot4 - slot3 + 1) slot8.x = slot8.x * (slot2.cellSize.x + slot2.cellSpace.x) - slot2.cellSpace.x slot8.y = slot8.y * (slot2.cellSize.y + slot2.cellSpace.y) - slot2.cellSpace.y return slot3, slot4, slot5, slot6, Vector2((slot5 - slot6) * 0.5 * (slot2.cellSize.x + slot2.cellSpace.x), (slot4 - slot3) * 0.5 * (slot2.cellSize.y + slot2.cellSpace.y)), slot8 end function slot0.getQuadCellPic(slot0, slot1) slot2 = nil if slot1.trait == ChapterConst.TraitLurk then -- Nothing elseif (slot1.attachment == ChapterConst.AttachEnemy or slot1.attachment == ChapterConst.AttachElite or slot1.attachment == ChapterConst.AttachAmbush or slot1.attachment == ChapterConst.AttachBoss or slot1.attachment == ChapterConst.AttachAreaBoss or slot1.attachment == ChapterConst.AttachBomb_Enemy) and slot1.flag == ChapterConst.CellFlagActive then slot2 = "cell_enemy" elseif slot1.attachment == ChapterConst.AttachBox and slot1.flag == ChapterConst.CellFlagActive then if pg.box_data_template[slot1.attachmentId].type == ChapterConst.BoxDrop or slot3.type == ChapterConst.BoxStrategy or slot3.type == ChapterConst.BoxSupply or slot3.type == ChapterConst.BoxEnemy then slot2 = "cell_box" elseif slot3.type == ChapterConst.BoxTorpedo then slot2 = "cell_enemy" elseif slot3.type == ChapterConst.BoxBarrier then slot2 = "cell_green" end elseif slot1.attachment == ChapterConst.AttachStory then if slot1.flag == ChapterConst.CellFlagTriggerActive then slot2 = pg.map_event_template[slot1.attachmentId].grid_color and #slot3 > 0 and slot3 or nil end elseif slot1.attachment == ChapterConst.AttachSupply and slot1.attachmentId > 0 then slot2 = "cell_box" elseif slot1.attachment == ChapterConst.AttachTransport_Target then slot2 = "cell_box" elseif slot1.attachment == ChapterConst.AttachLandbase and pg.land_based_template[slot1.attachmentId] and (slot3.type == ChapterConst.LBHarbor or slot3.type == ChapterConst.LBDock) then slot2 = "cell_box" end return slot2 end function slot0.getMapShip(slot0, slot1) slot2 = nil if slot1:isValid() and not _.detect(slot1:getShips(false), function (slot0) return slot0.isNpc and slot0.hpRant > 0 end) then if slot1:getFleetType() == FleetType.Normal then slot2 = slot1:getShipsByTeam(TeamType.Main, false)[1] elseif slot3 == FleetType.Submarine then slot2 = slot1:getShipsByTeam(TeamType.Submarine, false)[1] end end return slot2 end function slot0.getTorpedoShip(slot0, slot1) slot2 = nil if slot1:getFleetType() == FleetType.Submarine then slot2 = slot1:getShipsByTeam(TeamType.Submarine, false)[1] elseif slot1:getFleetType() == FleetType.Normal then slot2 = _.detect(slot1:getShipsByTeam(TeamType.Vanguard, false), function (slot0) return slot0:getShipType() == ShipType.QuZhu end) end return slot2 end function slot0.getCVship(slot0, slot1) slot2 = nil if slot1:getFleetType() == FleetType.Normal then slot2 = _.detect(slot1:getShipsByTeam(TeamType.Main, false), function (slot0) return table.contains(ShipType.BundleList[ShipType.BundleAircraftCarrier], slot0:getShipType()) end) end return slot2 end function slot0.getBBship(slot0, slot1) slot2 = nil if slot1:getFleetType() == FleetType.Normal then slot2 = _.detect(slot1:getShipsByTeam(TeamType.Main, false), function (slot0) return table.contains(ShipType.BundleList[ShipType.BundleBattleShip], slot0:getShipType()) end) end return slot2 end function slot0.GetSubmarineFleet(slot0) for slot4, slot5 in pairs(slot0.fleets) do if slot5:getFleetType() == FleetType.Submarine and slot5:isValid() then return slot5, slot4 end end end function slot0.getStageCell(slot0, slot1, slot2) if slot0:getChampion(slot1, slot2) and slot3.flag ~= ChapterConst.CellFlagDisabled then return slot3 end if slot0:getChapterCell(slot1, slot2) and slot4.flag ~= ChapterConst.CellFlagDisabled then return slot4 end end function slot0.getStageId(slot0, slot1, slot2) if slot0:getChampion(slot1, slot2) and slot3.flag ~= ChapterConst.CellFlagDisabled then return slot3.id end if slot0:getChapterCell(slot1, slot2) and slot4.flag ~= ChapterConst.CellFlagDisabled then return slot4.attachmentId end end function slot0.getStageExtraAwards(slot0) if slot0:getPlayType() == ChapterConst.TypeMainSub then slot1 = _.filter(pg.expedition_data_by_map.all, function (slot0) return type(pg.expedition_data_by_map[slot0].drop_by_map_display) == "table" and #slot1 > 0 end) if pg.expedition_data_by_map[slot1[math.min(#slot1, slot0.awardIndex)]] then return slot2.drop_by_map_display[table.indexof(slot1, slot0:getConfig("map"))] end end end function slot0.GetExtraCostRate(slot0) slot2 = {} for slot6, slot7 in ipairs(slot0.operationBuffList) do slot8 = pg.benefit_buff_template[slot7] slot2[#slot2 + 1] = slot8 if slot8.benefit_type == uv0.OPERATION_BUFF_TYPE_COST then slot1 = 1 + slot8.benefit_effect * 0.01 end end return math.max(1, slot1), slot2 end function slot0.getFleetCost(slot0, slot1) if slot0:getPlayType() == ChapterConst.TypeExtra then return { gold = 0, oil = 0 }, { gold = 0, oil = 0 } end return slot1:getCost(slot0:GetExtraCostRate()) end function slot0.GetWinConditions(slot0) return slot0.winConditions end function slot0.GetLoseConditions(slot0) return slot0.loseConditions end function slot0.CanQuickPlay(slot0) return pg.chapter_setting[slot0.id] and slot1.expedite > 0 end function slot0.GetQuickPlayFlag(slot0) return PlayerPrefs.GetInt("chapter_quickPlay_flag_" .. slot0.id, 0) == 1 end function slot0.writeDrops(slot0, slot1) _.each(slot1, function (slot0) if slot0.dropType == DROP_TYPE_SHIP and not table.contains(uv0.dropShipIdList, slot0.id) then table.insert(uv0.dropShipIdList, slot0.id) end end) end function slot0.UpdateDropShipList(slot0, slot1) for slot5, slot6 in ipairs(slot1) do if not table.contains(slot0.dropShipIdList, slot6) then table.insert(slot0.dropShipIdList, slot6) end end end function slot0.GetDropShipList(slot0) return slot0.dropShipIdList end function slot0.writeBack(slot0, slot1, slot2) for slot8, slot9 in pairs(slot0.fleet.ships) do function (slot0) if uv0.statistics[slot0.id] then slot0.hpRant = slot1.bp end end(slot9) end slot3.restAmmo = math.max(slot3.restAmmo - 1, 0) _.each(_.filter(slot3:getStrategies(), function (slot0) return pg.strategy_data_template[slot0.id] and slot1.type == ChapterConst.StgTypeBindFleetPassive and slot0.count > 0 end), function (slot0) uv0:consumeOneStrategy(slot0.id) end) if slot2.statistics.submarineAid then if _.detect(slot0.fleets, function (slot0) return slot0:getFleetType() == FleetType.Submarine and slot0:isValid() end) and not slot6:inHuntingRange(slot3.line.row, slot3.line.column) then slot6:consumeOneStrategy(ChapterConst.StrategyCallSubOutofRange) end if slot6 then for slot10, slot11 in pairs(slot6.ships) do slot4(slot11) end slot6.restAmmo = math.max(slot6.restAmmo - 1, 0) end end slot0:UpdateComboHistory(slot2.statistics._battleScore) if slot1 then slot6 = nil if _.detect(slot0.champions, function (slot0) return slot0.id == uv0.stageId and slot0.row == uv1.line.row and slot0.column == uv1.line.column and slot0.flag ~= ChapterConst.CellFlagDisabled end) then slot7:Iter() slot6 = slot7.attachment else slot8 = slot0:getChapterCell(slot3.line.row, slot3.line.column) slot8.flag = ChapterConst.CellFlagDisabled slot0:updateChapterCell(slot8) slot6 = slot8.attachment end if slot6 == ChapterConst.AttachEnemy or slot6 == ChapterConst.AttachElite or slot6 == ChapterConst.AttachChampion then if (not slot7 or slot7.flag == ChapterConst.CellFlagDisabled) and _.detect(slot0.achieves, function (slot0) return slot0.type == ChapterConst.AchieveType2 end) then slot8.count = slot8.count + 1 end elseif slot6 == ChapterConst.AttachBoss and _.detect(slot0.achieves, function (slot0) return slot0.type == ChapterConst.AchieveType1 end) then slot8.count = slot8.count + 1 end if slot0:CheckChapterWin() then pg.TrackerMgr.GetInstance():Tracking(TRACKING_KILL_BOSS) end if not slot7 or slot7.flag == ChapterConst.CellFlagDisabled then slot0:RemoveChampion(slot7) slot3.defeatEnemies = slot3.defeatEnemies + 1 slot0.defeatEnemies = slot0.defeatEnemies + 1 end if slot0:getPlayType() == ChapterConst.TypeMainSub and slot6 == ChapterConst.AttachBoss and slot2.statistics._battleScore == ys.Battle.BattleConst.BattleScore.S then slot11 = getProxy(ChapterProxy) slot11.subProgress = math.max(slot11.subProgress, table.indexof(_.filter(pg.expedition_data_by_map.all, function (slot0) return type(pg.expedition_data_by_map[slot0].drop_by_map_display) == "table" and #slot1 > 0 end), slot0:getConfig("map")) + 1) end getProxy(ChapterProxy):RecordLastDefeatedEnemy(slot0.id, { score = slot2.statistics._battleScore, line = { row = slot3.line.row, column = slot3.line.column }, type = slot6 }) end end function slot0.UpdateProgressAfterSkipBattle(slot0) slot2 = slot0.fleet.line slot3, slot4 = nil if slot0:existChampion(slot2.row, slot2.column) then slot4 = slot0:getChampion(slot2.row, slot2.column) slot4:Iter() slot3 = slot4.attachment else slot4 = slot0:getChapterCell(slot1.line.row, slot1.line.column) slot4.flag = ChapterConst.CellFlagDisabled slot0:updateChapterCell(slot4) slot3 = slot4.attachment end if slot3 == ChapterConst.AttachEnemy or slot3 == ChapterConst.AttachElite or slot3 == ChapterConst.AttachChampion and slot4.flag == ChapterConst.CellFlagDisabled then if _.detect(slot0.achieves, function (slot0) return slot0.type == ChapterConst.AchieveType2 end) then slot5.count = slot5.count + 1 end elseif slot3 == ChapterConst.AttachBoss and _.detect(slot0.achieves, function (slot0) return slot0.type == ChapterConst.AchieveType1 end) then slot5.count = slot5.count + 1 end if slot0:CheckChapterWin() then pg.TrackerMgr.GetInstance():Tracking(TRACKING_KILL_BOSS) end if slot3 ~= ChapterConst.AttachChampion or slot4.flag == ChapterConst.CellFlagDisabled then if slot3 == ChapterConst.AttachChampion then slot0:RemoveChampion(slot4) end slot1.defeatEnemies = slot1.defeatEnemies + 1 slot0.defeatEnemies = slot0.defeatEnemies + 1 end if slot0:getPlayType() == ChapterConst.TypeMainSub and slot3 == ChapterConst.AttachBoss then slot8 = getProxy(ChapterProxy) slot8.subProgress = math.max(slot8.subProgress, table.indexof(_.filter(pg.expedition_data_by_map.all, function (slot0) return type(pg.expedition_data_by_map[slot0].drop_by_map_display) == "table" and #slot1 > 0 end), slot0:getConfig("map")) + 1) end getProxy(ChapterProxy):RecordLastDefeatedEnemy(slot0.id, { score = ys.Battle.BattleConst.BattleScore.S, line = { row = slot1.line.row, column = slot1.line.column }, type = slot3 }) end function slot0.UpdateProgressOnRetreat(slot0) _.each(slot0.achieves, function (slot0) if slot0.type == ChapterConst.AchieveType3 then if _.all(_.values(uv0.cells), function (slot0) if slot0.attachment == ChapterConst.AttachEnemy or slot0.attachment == ChapterConst.AttachElite or slot0.attachment == ChapterConst.AttachBoss or slot0.attachment == ChapterConst.AttachBox and pg.box_data_template[slot0.attachmentId].type == ChapterConst.BoxEnemy then return slot0.flag == ChapterConst.CellFlagDisabled end return true end) and _.all(uv0.champions, function (slot0) return slot0.flag == ChapterConst.CellFlagDisabled end) then slot0.count = slot0.count + 1 end elseif slot0.type == ChapterConst.AchieveType4 then if uv0.orignalShipCount <= slot0.config then slot0.count = slot0.count + 1 end elseif slot0.type == ChapterConst.AchieveType5 then if not _.any(uv0:getShips(), function (slot0) return slot0:getShipType() == uv0.config end) then slot0.count = slot0.count + 1 end elseif slot0.type == ChapterConst.AchieveType6 then slot0.count = math.max((uv0.scoreHistory[0] or 0) + (uv0.scoreHistory[1] or 0) <= 0 and uv0.combo or 0, slot0.count or 0) end end) if slot0.progress == 100 then slot0.passCount = slot0.passCount + 1 end slot0.progress = math.min(slot0.progress + slot0:getConfig("progress_boss"), 100) if slot0.progress < 100 and slot2 >= 100 then getProxy(ChapterProxy):RecordJustClearChapters(slot0.id, true) end slot0.defeatCount = slot0.defeatCount + 1 if getProxy(ChapterProxy):getMapById(slot0:getConfig("map")):getMapType() == Map.ELITE then pg.TrackerMgr.GetInstance():Tracking(TRACKING_HARD_CHAPTER, slot0.id) elseif slot4 == Map.SCENARIO then if slot0.progress == 100 and slot0.passCount == 0 then pg.TrackerMgr.GetInstance():Tracking(TRACKING_HIGHEST_CHAPTER, slot0.id) end if slot0.defeatCount == 1 then if slot0.id == 304 then pg.TrackerMgr.GetInstance():Tracking(TRACKING_FIRST_PASS_3_4) elseif slot0.id == 404 then pg.TrackerMgr.GetInstance():Tracking(TRACKING_FIRST_PASS_4_4) elseif slot0.id == 504 then pg.TrackerMgr.GetInstance():Tracking(TRACKING_FIRST_PASS_5_4) elseif slot0.id == 604 then pg.TrackerMgr.GetInstance():Tracking(TRACKING_FIRST_PASS_6_4) elseif slot0.id == 1204 then pg.TrackerMgr.GetInstance():Tracking(TRACKING_FIRST_PASS_12_4) elseif slot0.id == 1301 then pg.TrackerMgr.GetInstance():Tracking(TRACKING_FIRST_PASS_13_1) elseif slot0.id == 1302 then pg.TrackerMgr.GetInstance():Tracking(TRACKING_FIRST_PASS_13_2) elseif slot0.id == 1303 then pg.TrackerMgr.GetInstance():Tracking(TRACKING_FIRST_PASS_13_3) elseif slot0.id == 1304 then pg.TrackerMgr.GetInstance():Tracking(TRACKING_FIRST_PASS_13_4) end end end end function slot0.UpdateComboHistory(slot0, slot1) getProxy(ChapterProxy):RecordComboHistory(slot0.id, { scoreHistory = Clone(slot0.scoreHistory), combo = Clone(slot0.combo) }) slot0.scoreHistory = slot0.scoreHistory or {} slot0.scoreHistory[slot1] = (slot0.scoreHistory[slot1] or 0) + 1 if slot1 <= ys.Battle.BattleConst.BattleScore.C then slot0.combo = 0 else slot0.combo = (slot0.combo or 0) + 1 end end function slot0.CheckChapterWin(slot0) slot3 = ChapterConst.ReasonVictory for slot7, slot8 in pairs(slot0:GetWinConditions()) do if slot8.type == 1 then _.each(slot0:findChapterCells(ChapterConst.AttachBoss), function (slot0) if slot0 and slot0.flag == ChapterConst.CellFlagDisabled then uv0 = uv0 + 1 end end) slot2 = false or slot8.param <= 0 elseif slot8.type == 2 then slot2 = slot2 or slot8.param <= slot0:GetDefeatCount() elseif slot8.type == 3 then slot2 = slot2 or slot0:CheckTransportState() == 1 elseif slot8.type == 4 then slot2 = slot2 or slot8.param < slot0:getRoundNum() elseif slot8.type == 5 then slot9 = slot8.param slot2 = slot2 or not (_.any(slot0.champions, function (slot0) for slot5, slot6 in pairs(slot0.idList) do slot1 = slot0.attachmentId == uv0 or slot6 == uv0 end return slot1 and slot0.flag ~= ChapterConst.CellFlagDisabled end) or _.any(slot0.cells, function (slot0) return slot0.attachmentId == uv0 and slot0.flag ~= ChapterConst.CellFlagDisabled end)) elseif slot8.type == 6 then slot9 = slot8.param slot2 = slot2 or _.any(slot0.fleets, function (slot0) return slot0:getFleetType() == FleetType.Normal and slot0:isValid() and slot0.line.row == uv0[1] and slot0.line.column == uv0[2] end) end if slot2 then break end end return slot2, slot3 end function slot0.CheckChapterLose(slot0) slot3 = ChapterConst.ReasonDefeat for slot7, slot8 in pairs(slot0:GetLoseConditions()) do if slot8.type == 1 then slot2 = false or not _.any(slot0.fleets, function (slot0) return slot0:getFleetType() == FleetType.Normal and slot0:isValid() end) elseif slot8.type == 2 then slot3 = (slot2 or slot0.BaseHP <= 0) and ChapterConst.ReasonDefeatDefense end if slot2 then break end end if slot0:getPlayType() == ChapterConst.TypeTransport then slot2 = slot2 or slot0:CheckTransportState() == -1 end return slot2, slot3 end function slot0.CheckChapterWillWin(slot0) if slot0:existOni() then return true elseif slot0:isPlayingWithBombEnemy() then return true end slot1, slot2 = slot0:CheckChapterWin() if slot1 then return true end end function slot0.triggerSkill(slot0, slot1, slot2) slot3 = _.filter(slot1:findSkills(slot2), function (slot0) return _.any(slot0:GetTriggers(), function (slot0) return slot0[1] == FleetSkill.TriggerInSubTeam and slot0[2] == 1 end) == (uv0:getFleetType() == FleetType.Submarine) and _.all(slot0:GetTriggers(), function (slot0) return uv0:triggerCheck(uv1, uv2, slot0) end) end) return _.reduce(slot3, nil, function (slot0, slot1) if slot1:GetType() == FleetSkill.TypeMoveSpeed or slot2 == FleetSkill.TypeHuntingLv or slot2 == FleetSkill.TypeTorpedoPowerUp then return (slot0 or 0) + slot1:GetArgs()[1] elseif slot2 == FleetSkill.TypeAmbushDodge or slot2 == FleetSkill.TypeAirStrikeDodge then return math.max(slot0 or 0, slot3[1]) elseif slot2 == FleetSkill.TypeAttack or slot2 == FleetSkill.TypeStrategy then slot0 = slot0 or {} table.insert(slot0, slot3) return slot0 elseif slot2 == FleetSkill.TypeBattleBuff then slot0 = slot0 or {} table.insert(slot0, slot3[1]) return slot0 end end), slot3 end function slot0.triggerCheck(slot0, slot1, slot2, slot3) if slot3[1] == FleetSkill.TriggerDDCount then return slot3[2] <= #_.filter(slot1:getShipsByTeam(TeamType.Vanguard, false), function (slot0) return slot0:getShipType() == ShipType.QuZhu end) and slot6 <= slot3[3] elseif slot4 == FleetSkill.TriggerDDHead then return #slot1:getShipsByTeam(TeamType.Vanguard, false) > 0 and slot5[1]:getShipType() == ShipType.QuZhu elseif slot4 == FleetSkill.TriggerVanCount then return slot3[2] <= #slot1:getShipsByTeam(TeamType.Vanguard, false) and #slot5 <= slot3[3] elseif slot4 == FleetSkill.TriggerShipCount then return slot3[3] <= #_.filter(slot1:getShips(false), function (slot0) return table.contains(uv0[2], slot0:getShipType()) end) and #slot5 <= slot3[4] elseif slot4 == FleetSkill.TriggerAroundEnemy then slot5 = { row = slot1.line.row, column = slot1.line.column } return _.any(_.values(slot0.cells), function (slot0) slot1 = not uv0:existOni(slot0.row, slot0.column) and not uv0:existBombEnemy(slot0.row, slot0.column) and (uv0:existChampion(slot0.row, slot0.column) and uv0:getChampion(slot0.row, slot0.column):getConfig("type") or uv0:existEnemy(ChapterConst.SubjectPlayer, slot0.row, slot0.column) and pg.expedition_data_template[slot0.attachmentId].type or nil) return ManhattonDist(uv1, { row = slot0.row, column = slot0.column }) <= uv2[2] and (type(uv2[3]) == "number" and uv2[3] == slot1 or type(uv2[3]) == "table" and table.contains(uv2[3], slot1)) end) elseif slot4 == FleetSkill.TriggerNekoPos then for slot9, slot10 in pairs(slot1:getCommanders()) do if slot1:findCommanderBySkillId(slot2.id).id == slot10.id and slot9 == slot3[2] then return true end end elseif slot4 == FleetSkill.TriggerAroundLand then slot5 = { row = slot1.line.row, column = slot1.line.column } return _.any(_.values(slot0.cells), function (slot0) return not slot0:IsWalkable() and ManhattonDist(uv0, { row = slot0.row, column = slot0.column }) <= uv1[2] end) elseif slot4 == FleetSkill.TriggerAroundCombatAlly then slot5 = { row = slot1.line.row, column = slot1.line.column } return _.any(slot0.fleets, function (slot0) return uv0.id ~= slot0.id and slot0:getFleetType() == FleetType.Normal and uv1:existEnemy(ChapterConst.SubjectPlayer, slot0.line.row, slot0.line.column) and ManhattonDist(uv2, { row = slot0.line.row, column = slot0.line.column }) <= uv3[2] end) elseif slot4 == FleetSkill.TriggerInSubTeam then return true end end slot2 = { { 1, 0 }, { -1, 0 }, { 0, 1 }, { 0, -1 } } function slot0.checkOniState(slot0) slot1 = slot0:getOni() if _.all(uv0, function (slot0) slot1 = { uv0.row + slot0[1], uv0.column + slot0[2] } if uv1:existFleet(FleetType.Normal, slot1[1], slot1[2]) then return true end if not uv1:getChapterCell(slot1[1], slot1[2]) or not slot2:IsWalkable() then return true end if uv1:existBarrier(slot2.row, slot2.column) then return true end end) then return 1 end if _.any(slot0:getOniChapterInfo().escape_grids, function (slot0) return slot0[1] == uv0.row and slot0[2] == uv0.column end) then return 2 end end function slot0.onOniEnter(slot0) for slot4, slot5 in pairs(slot0.cells) do slot5.attachment = ChapterConst.AttachNone slot5.attachmentId = nil slot5.flag = nil slot5.data = nil end slot0.champions = {} slot0.modelCount = slot0:getOniChapterInfo().special_item slot0.roundIndex = 0 end function slot0.onBombEnemyEnter(slot0) for slot4, slot5 in pairs(slot0.cells) do slot5.attachment = ChapterConst.AttachNone slot5.attachmentId = nil slot5.flag = nil slot5.data = nil end slot0.champions = {} slot0.modelCount = 0 slot0.roundIndex = 0 end function slot0.clearSubmarineFleet(slot0) for slot4 = #slot0.fleets, 1, -1 do if slot0.fleets[slot4]:getFleetType() == FleetType.Submarine then table.remove(slot0.fleets, slot4) end end end function slot0.getOniChapterInfo(slot0) return pg.chapter_capture[slot0.id] end function slot0.getBombChapterInfo(slot0) return pg.chapter_boom[slot0.id] end function slot0.getSpAppearStory(slot0) if slot0:existOni() then for slot4, slot5 in ipairs(slot0.champions) do if slot5.trait == ChapterConst.TraitLurk and slot5.attachment == ChapterConst.AttachOni and slot5:getConfig("appear_story") and #slot6 > 0 then return slot6 end end elseif slot0:isPlayingWithBombEnemy() then for slot4, slot5 in pairs(slot0.cells) do if slot5.attachment == ChapterConst.AttachBomb_Enemy and slot5.trait == ChapterConst.TraitLurk and pg.specialunit_template[slot5.attachmentId].appear_story and #slot6.appear_story > 0 then return slot6.appear_story end end end end function slot0.getSpAppearGuide(slot0) if slot0:existOni() then for slot4, slot5 in ipairs(slot0.champions) do if slot5.trait == ChapterConst.TraitLurk and slot5.attachment == ChapterConst.AttachOni and slot5:getConfig("appear_guide") and #slot6 > 0 then return slot6 end end elseif slot0:isPlayingWithBombEnemy() then for slot4, slot5 in pairs(slot0.cells) do if slot5.attachment == ChapterConst.AttachBomb_Enemy and slot5.trait == ChapterConst.TraitLurk and pg.specialunit_template[slot5.attachmentId].appear_guide and #slot6.appear_guide > 0 then return slot6.appear_guide end end end end function slot0.CheckTransportState(slot0) if not _.detect(slot0.fleets, function (slot0) return slot0:getFleetType() == FleetType.Transport end) then return -1 end slot2 = slot0:findChapterCell(ChapterConst.AttachTransport_Target) if not slot1:isValid() then return -1 elseif slot1.line.row == slot2.row and slot1.line.column == slot2.column and not slot0:existEnemy(ChapterConst.SubjectPlayer, slot2.row, slot2.column) then return 1 else return 0 end end function slot0.getCoastalGunArea(slot0) slot1 = {} for slot5, slot6 in pairs(slot0.cells) do if slot6.attachment == ChapterConst.AttachLandbase and slot6.flag ~= ChapterConst.CellFlagDisabled and pg.land_based_template[slot6.attachmentId].type == ChapterConst.LBCoastalGun then slot8 = slot7.function_args slot9 = { math.abs(slot8[1]), math.abs(slot8[2]) } slot10 = { Mathf.Sign(slot8[1]), Mathf.Sign(slot8[2]) } for slot15 = 1, math.max(slot9[1], slot9[2]) do table.insert(slot1, { row = slot6.row + math.min(slot9[1], slot15) * slot10[1], column = slot6.column + math.min(slot9[2], slot15) * slot10[2] }) end end end return slot1 end function slot0.GetAntiAirGunArea(slot0) slot1 = {} slot2 = {} for slot6, slot7 in pairs(slot0.cells) do if slot7.attachment == ChapterConst.AttachLandbase and slot7.flag ~= ChapterConst.CellFlagDisabled and pg.land_based_template[slot7.attachmentId].type == ChapterConst.LBAntiAir then slot13 = {} if math.abs(slot8.function_args[1]) > 0 then -- Nothing end while next({ [function (slot0, slot1) return ChapterConst.MaxColumn * slot0 + slot1 end(slot7.row, slot7.column)] = slot7 }) do slot14 = next(slot12) slot12[slot14] = nil if math.abs(slot12[slot14].row - slot7.row) <= slot10 and math.abs(slot15.column - slot7.column) <= slot10 then slot13[slot14] = slot15 for slot19 = 1, #uv0 do if not slot13[slot11(slot15.row + uv0[slot19][1], slot15.column + uv0[slot19][2])] then slot12[slot22] = { row = slot20, column = slot21 } end end end end for slot17, slot18 in pairs(slot13) do slot2[slot17] = slot18 end end end for slot6, slot7 in pairs(slot2) do table.insert(slot1, slot7) end return slot1 end function slot0.getNpcShipByType(slot0, slot1) slot2 = {} slot3 = getProxy(TaskProxy) slot8 = "npc_data" for slot8, slot9 in ipairs(slot0:getConfig(slot8)) do slot10 = pg.npc_squad_template[slot9] if not slot1 or slot1 == slot10.type and function (slot0) if slot0 == 0 then return true end return uv0:getTaskVO(slot0) and not slot1:isFinish() end(slot10.task_id) then for slot14, slot15 in ipairs({ "vanguard_list", "main_list" }) do for slot19, slot20 in ipairs(slot10[slot15]) do table.insert(slot2, NpcShip.New({ id = slot20[1], configId = slot20[1], level = slot20[2] })) end end end end return slot2 end function slot0.GetDefeatCount(slot0) return slot0.defeatEnemies end function slot0.ExistDivingChampion(slot0) return _.any(slot0.champions, function (slot0) return slot0.flag == ChapterConst.CellFlagDiving end) end function slot0.IsSkipPrecombat(slot0) return slot0:isLoop() and getProxy(ChapterProxy):GetSkipPrecombat() end function slot0.CanActivateAutoFight(slot0) return pg.chapter_template_loop[slot0.id] and slot1.fightauto == 1 and slot0:isLoop() and AutoBotCommand.autoBotSatisfied() and not slot0:existOni() and not slot0:existBombEnemy() end function slot0.IsAutoFight(slot0) return slot0:CanActivateAutoFight() and getProxy(ChapterProxy):GetChapterAutoFlag(slot0.id) == 1 end function slot0.getTodayDefeatCount(slot0) return getProxy(DailyLevelProxy):getChapterDefeatCount(slot0.configId) end function slot0.isTriesLimit(slot0) return slot0:getConfig("count") and slot1 > 0 end function slot0.updateTodayDefeatCount(slot0) getProxy(DailyLevelProxy):updateChapterDefeatCount(slot0.configId) end function slot0.enoughTimes2Start(slot0) if slot0:isTriesLimit() then return slot0:getTodayDefeatCount() < slot0:getConfig("count") else return true end end function slot0.GetDailyBonusQuota(slot0) for slot6, slot7 in ipairs(slot0:getConfig("boss_expedition_id")) do slot2 = math.max(0, pg.expedition_activity_template[slot7] and slot8.bonus_time or 0) end if pg.chapter_defense[slot0.id] then slot2 = math.max(slot2, slot3.bonus_time or 0) end return not getProxy(ChapterProxy):getMapById(slot0:getConfig("map")):isRemaster() and slot2 > 0 and math.max(slot2 - slot0.todayDefeatCount, 0) > 0 end slot0.OPERATION_BUFF_TYPE_COST = "more_oil" slot0.OPERATION_BUFF_TYPE_REWARD = "extra_drop" slot0.OPERATION_BUFF_TYPE_EXP = "chapter_up" slot0.OPERATION_BUFF_TYPE_DESC = "desc" function slot0.GetSPOperationItemCacheKey(slot0) return "specialOPItem_" .. slot0 end function slot0.GetSPBuffByItem(slot0) for slot4, slot5 in pairs(pg.benefit_buff_template) do if tonumber(slot5.benefit_condition) == slot0 then return slot5.id end end end function slot0.GetOperationDesc(slot0) slot1 = "" for slot5, slot6 in ipairs(slot0.operationBuffList) do if pg.benefit_buff_template[slot6].benefit_type == uv0.OPERATION_BUFF_TYPE_DESC then slot1 = slot7.desc break end end return slot1 end function slot0.GetOperationBuffList(slot0) return slot0.operationBuffList end function slot0.GetAllEnemies(slot0, slot1) for slot6, slot7 in pairs(slot0.cells) do if (slot7.attachment == ChapterConst.AttachEnemy or slot7.attachment == ChapterConst.AttachElite or slot7.attachment == ChapterConst.AttachBoss or slot7.attachment == ChapterConst.AttachBox and pg.box_data_template[slot7.attachmentId].type == ChapterConst.BoxEnemy) and (slot1 or slot7.flag ~= ChapterConst.CellFlagDisabled) then table.insert({}, slot7) end end for slot6, slot7 in pairs(slot0.champions) do if slot1 or slot7.flag ~= ChapterConst.CellFlagDisabled then table.insert(slot2, slot7) end end return slot2 end function slot0.GetFleetofDuty(slot0, slot1) slot2 = nil for slot6, slot7 in ipairs(slot0.fleets) do if slot7:isValid() and slot7:getFleetType() == FleetType.Normal then if (slot0.duties[slot7.id] or 0) == ChapterFleet.DUTY_KILLALL or slot1 and slot8 == ChapterFleet.DUTY_KILLBOSS or not slot1 and slot8 == ChapterFleet.DUTY_CLEANPATH then return slot7 end slot2 = slot7 end end return slot2 end function slot0.GetBuffOfLinkAct(slot0) if slot0:getPlayType() == ChapterConst.TypeDOALink then slot1 = pg.gameset.doa_fever_buff.description return _.detect(slot0.buff_list, function (slot0) return table.contains(uv0, slot0) end) end end function slot0.GetAttachmentStories(slot0) for slot7, slot8 in pairs(slot0.cellAttachments) do if uv0.GetEventTemplateByKey("mult_story", slot8.attachmentId) then slot3 = nil or slot9[1] if slot0.cells[slot7] and slot10.flag == ChapterConst.CellFlagDisabled then slot2 = 0 + 1 end end end return slot3, slot2 end function slot0.GetWeather(slot0, slot1, slot2) return slot0.cells[ChapterCell.Line2Name(slot1 or slot0.fleet.line.row, slot2 or slot0.fleet.line.column)] and slot4:GetWeatherFlagList() or {} end function slot0.GetChapterLastFleetCacheKey(slot0) return "lastFleetIndex_" .. (slot0 or 0) end function slot0.SaveChapterLastFleetCache(slot0, slot1) if not slot0 or not slot1 then return end for slot7, slot8 in ipairs(slot1) do slot2 = 0 + bit.lshift(slot8, 8 * (slot7 - 1)) end PlayerPrefs.SetInt(uv0.GetChapterLastFleetCacheKey(slot0), slot2) PlayerPrefs.Save() end function slot0.GetChapterLastFleetCache(slot0) if not slot0 then return {} end slot1 = PlayerPrefs.GetInt(uv0.GetChapterLastFleetCacheKey(slot0), 0) slot2 = {} slot3 = 255 slot4 = 8 while slot1 > 0 do table.insert(slot2, bit.band(slot1, slot3)) slot1 = bit.rshift(slot1, slot4) end return slot2 end return slot0