武将:职业天赋接口
This commit is contained in:
@@ -1,9 +1,9 @@
|
||||
import { Application, BackendSession, ChannelService, HandlerService, } from 'pinus';
|
||||
import { handleCost, addItems, unlockFigure, getCoinObject, getGoldObject } from '../../../services/role/rewardService';
|
||||
import { calPlayerCeAndSave, calAllHeroCe } from '../../../services/playerCeService';
|
||||
import { resResult, deepCopy, reduceCe } from '../../../pubUtils/util';
|
||||
import { resResult, deepCopy, reduceCe, parseGoodStr } from '../../../pubUtils/util';
|
||||
import { STATUS } from '../../../consts/statusCode';
|
||||
import { HeroModel, Connect, HeroSkin, HeroUpdate, EPlace } from '../../../db/Hero';
|
||||
import { HeroModel, Connect, HeroSkin, HeroUpdate, EPlace, Talent } from '../../../db/Hero';
|
||||
import { CURRENCY_BY_TYPE, CURRENCY_TYPE, CONSUME_TYPE, HERO_GROW_MAX, HERO_SYSTEM_TYPE, ABI_STAGE, DEBUG_MAGIC_WORD, HERO_INITIAL_QUALITY, REDIS_KEY, TASK_TYPE, ITEM_CHANGE_REASON } from '../../../consts';
|
||||
import { RoleModel } from '../../../db/Role';
|
||||
import { ItemModel } from '../../../db/Item';
|
||||
@@ -16,10 +16,10 @@ import { PvpDefenseModel } from '../../../db/PvpDefense';
|
||||
import { checkTask, checkTaskInHeroQUalityUp, checkTaskInHeroStarUp, checkTaskInHeroTrain, checkTaskInHeroWakeUp } from '../../../services/task/taskService';
|
||||
import { isNumber, pick } from 'underscore';
|
||||
import { updateEplaces } from '../../../services/equipService';
|
||||
import { addConsumeToHero } from '../../../services/roleService';
|
||||
import { addConsumeToHero, checkUnlockTalentCondition, initSkinTalent, updateSkinTalent } from '../../../services/roleService';
|
||||
import { JewelModel, jewelUpdate } from '../../../db/Jewel';
|
||||
import { CalHeroCe } from '../../../domain/roleField/calCe';
|
||||
import { REBORN } from '../../../pubUtils/dicParam';
|
||||
import { HERO, REBORN } from '../../../pubUtils/dicParam';
|
||||
import { createHero, createHeroes } from '../../../services/role/createHero';
|
||||
import { CheckMeterial } from '../../../services/role/checkMaterial';
|
||||
|
||||
@@ -685,6 +685,100 @@ export class HeroHandler {
|
||||
return resResult(STATUS.SUCCESS, { curHero: { ...curHero, ce: reduceCe(curHero.ce) }, curJewels, goods });
|
||||
}
|
||||
|
||||
// 解锁天赋
|
||||
public async unlockTalent(msg: { hid: number, id: number }, session: BackendSession) {
|
||||
let roleId = session.get('roleId');
|
||||
let sid = session.get('sid');
|
||||
let { hid, id } = msg;
|
||||
|
||||
let hero = await HeroModel.findByHidAndRole(hid, roleId);
|
||||
if(!hero) return resResult(STATUS.HERO_NOT_FIND);
|
||||
|
||||
let dicHeroTalent = gameData.heroTalent.get(id);
|
||||
if(!dicHeroTalent) return resResult(STATUS.WRONG_PARMS);
|
||||
|
||||
let skins = hero.skins||[];
|
||||
let curSkin = skins.find(cur => cur.enable);
|
||||
if(!curSkin) return resResult(STATUS.HERO_SKIN_NOT_FIND);
|
||||
|
||||
let talent = curSkin.talent.find(cur => cur.id == id);
|
||||
if(talent) return resResult(STATUS.TALENT_HAS_UNLOCKED);
|
||||
|
||||
let usedTalentPoint = curSkin.usedTalentPoint;
|
||||
if(!checkUnlockTalentCondition(hid, id, curSkin.talent, usedTalentPoint)) {
|
||||
return resResult(STATUS.TALENT_CONDITION_NOT_FIT);
|
||||
}
|
||||
|
||||
let totalTalentPoint = gameData.talentPointOfJob.get(hero.job);
|
||||
let needTalentPoint = dicHeroTalent.level.find(cur => cur.lv == 1)?.cost||0;
|
||||
if(totalTalentPoint - usedTalentPoint - needTalentPoint < 0) {
|
||||
return resResult(STATUS.TALENT_POINT_NOT_ENOUGH);
|
||||
}
|
||||
|
||||
let { newSkins, newTalents } = updateSkinTalent(skins, new Talent(id), usedTalentPoint + needTalentPoint);
|
||||
|
||||
hero = await calPlayerCeAndSave(HERO_SYSTEM_TYPE.TALENT, sid, roleId, hero, { skins: newSkins });
|
||||
return resResult(STATUS.SUCCESS, { curHero: {...pick(hero, ['hid', 'skins', 'skinId']) }});
|
||||
|
||||
}
|
||||
|
||||
// 升级天赋
|
||||
public async upgradeTalent(msg: { hid: number, id: number }, session: BackendSession) {
|
||||
let roleId = session.get('roleId');
|
||||
let sid = session.get('sid');
|
||||
let { hid, id } = msg;
|
||||
|
||||
let hero = await HeroModel.findByHidAndRole(hid, roleId);
|
||||
if(!hero) return resResult(STATUS.HERO_NOT_FIND);
|
||||
|
||||
let skins = hero.skins||[];
|
||||
let curSkin = skins.find(cur => cur.enable);
|
||||
if(!curSkin) return resResult(STATUS.HERO_SKIN_NOT_FIND);
|
||||
|
||||
let talent = curSkin.talent.find(cur => cur.id == id);
|
||||
if(!talent) return resResult(STATUS.TALENT_NOT_UNLOCKED);
|
||||
|
||||
let dicHeroTalent = gameData.heroTalent.get(id);
|
||||
if(!dicHeroTalent) return resResult(STATUS.WRONG_PARMS);
|
||||
|
||||
let usedTalentPoint = curSkin.usedTalentPoint;
|
||||
let totalTalentPoint = gameData.talentPointOfJob.get(hero.job);
|
||||
let nextLv = dicHeroTalent.level.find(cur => cur.lv == talent.level + 1);
|
||||
if(!nextLv) return resResult(STATUS.TALENT_LEVEL_MAX);
|
||||
let needTalentPoint = nextLv.cost||0;
|
||||
if(totalTalentPoint - usedTalentPoint - needTalentPoint < 0) {
|
||||
return resResult(STATUS.TALENT_POINT_NOT_ENOUGH);
|
||||
}
|
||||
|
||||
let { newSkins, newTalents } = updateSkinTalent(skins, new Talent(id, talent.level + 1), usedTalentPoint + needTalentPoint);
|
||||
|
||||
hero = await calPlayerCeAndSave(HERO_SYSTEM_TYPE.TALENT, sid, roleId, hero, { skins: newSkins });
|
||||
return resResult(STATUS.SUCCESS, { curHero: {...pick(hero, ['hid', 'skins', 'skinId']) }});
|
||||
}
|
||||
|
||||
// 洗点
|
||||
public async resetTalent(msg: { hid: number }, session: BackendSession) {
|
||||
let roleId = session.get('roleId');
|
||||
let sid = session.get('sid');
|
||||
let { hid } = msg;
|
||||
|
||||
let hero = await HeroModel.findByHidAndRole(hid, roleId);
|
||||
if(!hero) return resResult(STATUS.HERO_NOT_FIND);
|
||||
|
||||
let skins = hero.skins||[];
|
||||
let curSkin = skins.find(cur => cur.enable);
|
||||
if(!curSkin) return resResult(STATUS.HERO_SKIN_NOT_FIND);
|
||||
|
||||
let costItem = parseGoodStr(HERO.HERO_TALENTPOINT_RESET);
|
||||
let consumeResult = await handleCost(roleId, sid, costItem, ITEM_CHANGE_REASON.RESET_TALENT);
|
||||
if(!consumeResult) return resResult(STATUS.ROLE_MATERIAL_NOT_ENOUGH);
|
||||
|
||||
|
||||
let newSkins = initSkinTalent(skins);
|
||||
hero = await calPlayerCeAndSave(HERO_SYSTEM_TYPE.TALENT, sid, roleId, hero, { skins: newSkins });
|
||||
return resResult(STATUS.SUCCESS, { curHero: {...pick(hero, ['hid', 'skins', 'skinId']) }});
|
||||
}
|
||||
|
||||
// ! debug接口 一键全武将
|
||||
public async debugGetAllHeroes(msg: { magicWord: string }, session: BackendSession) {
|
||||
let roleId: string = session.get('roleId');
|
||||
|
||||
@@ -127,8 +127,6 @@ export class CreateHeroes extends UpdateHeroes {
|
||||
private resultHeroes: HeroType[] = [];
|
||||
private heroNum = 0;
|
||||
// 推送信息
|
||||
private taskPushMessage: TaskListReturn[] = [];
|
||||
private activityTaskPushMessage = [];
|
||||
private figureInfos: { heads: Figure[], frames: Figure[], spines: Figure[] }[] = [];
|
||||
private skinPushMessages: { heros: {skins: HeroSkin[], hid: number}[], skins: {id: number, hid: number, inc: number, reason: number }[]} = { heros: [], skins: [] };
|
||||
|
||||
@@ -138,9 +136,9 @@ export class CreateHeroes extends UpdateHeroes {
|
||||
let skinInfos = skin.map(cur => ({ id: cur.id, hid, inc: 1, reason: ITEM_CHANGE_REASON.GET_HERO_UNLOCK_SKIN}))
|
||||
this.skinPushMessages.skins.push(...skinInfos);
|
||||
if(skin) allSkins.push(...skin);
|
||||
let skins: { id: number, skin: string, enable: boolean, skinId: number }[] = [];
|
||||
let skins: HeroSkin[] = [];
|
||||
for(let skin of allSkins) {
|
||||
skins.push({ id: skin.id, skin: skin._id, enable: skin.id == initSkinInfo.id, skinId: skin.skinId });
|
||||
skins.push(new HeroSkin(skin.id, skin.skinId, skin._id, skin.id == initSkinInfo.id));
|
||||
}
|
||||
return skins
|
||||
}
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
import { DEFAULT_HEROES, DEFAULT_HERO_LV, FIGURE_UNLOCK_CONDITION, HERO_SYSTEM_TYPE, LINEUP_NUM } from "../../consts";
|
||||
import { HeroModel, HeroUpdate } from "../../db/Hero";
|
||||
import { HeroModel, HeroSkin, HeroUpdate } from "../../db/Hero";
|
||||
import { RoleModel, RoleUpdate } from "../../db/Role";
|
||||
import { SkinModel, SkinUpdate } from "../../db/Skin";
|
||||
import { TopHero } from "../../domain/dbGeneral";
|
||||
@@ -29,7 +29,7 @@ export function getInitRoleInfo() {
|
||||
initSkins.push(skinInfo);
|
||||
// 武将
|
||||
let hero = new HeroModel();
|
||||
let heroInfo = {...hero.toJSON(), hid, star, quality, hName, job, skins: [{ id: initialSkin, skin: skinInfo._id, enable: true, skinId: skinInfo.skinId }], skinId: skinInfo.skinId, lv: DEFAULT_HERO_LV, exp: getHeroExpByLv(DEFAULT_HERO_LV - 1) || 0 };
|
||||
let heroInfo = {...hero.toJSON(), hid, star, quality, hName, job, skins: [new HeroSkin(initialSkin, skinInfo.skinId, skinInfo._id, true)], skinId: skinInfo.skinId, lv: DEFAULT_HERO_LV, exp: getHeroExpByLv(DEFAULT_HERO_LV - 1) || 0 };
|
||||
let calHeroCe = new CalHeroCe(hid, heroInfo);
|
||||
let heroAttr = calHeroCe.cal(HERO_SYSTEM_TYPE.INIT);
|
||||
let ce = calHeroCe.getCalculatedCe(roleAttr);
|
||||
|
||||
@@ -367,7 +367,7 @@ export async function addSkin(roleId: string, roleName: string, sid: string, ski
|
||||
if (hero) { // 有武将的,将皮肤链接到武将上
|
||||
let curSkin = hero.skins.find(cur => cur.id == skinId);
|
||||
if (!curSkin) {
|
||||
hero.skins.push({ id: skinId, skin: skin._id, enable, skinId: skin.skinId });
|
||||
hero.skins.push(new HeroSkin(skin.id, skin.skinId, skin._id, enable ));
|
||||
await HeroModel.updateHeroInfo(roleId, hero.hid, hero);
|
||||
}
|
||||
return hero;
|
||||
|
||||
@@ -1,14 +1,14 @@
|
||||
import { ChannelUser } from './../domain/ChannelUser';
|
||||
import { Channel, pinus } from 'pinus';
|
||||
import { getRandValueByMinMax, getRandEelm, decodeIdCntArrayStr } from '../pubUtils/util';
|
||||
import { DEFAULT_HEROES, ROLE_SELECT, TERAPH_RANDOM } from "../consts";
|
||||
import { DEFAULT_HEROES, ROLE_SELECT, TALENT_RELATION_TYPE, TERAPH_RANDOM } from "../consts";
|
||||
import { DicTeraph } from '../pubUtils/dictionary/DicTeraph';
|
||||
import { Teraph, RoleModel, RoleType, RoleUpdate } from '../db/Role';
|
||||
import { SCHOOL } from '../pubUtils/dicParam';
|
||||
import { gameData } from '../pubUtils/data';
|
||||
import { gameData, getHeroInitTalent } from '../pubUtils/data';
|
||||
import { SchoolModel } from '../db/School';
|
||||
import { SclResultInter, SclPosInter, RewardInter, ItemInter } from '../pubUtils/interface';
|
||||
import { HeroUpdate } from '../db/Hero';
|
||||
import { HeroSkin, HeroUpdate, Talent } from '../db/Hero';
|
||||
import { SkinUpdate } from '../db/Skin';
|
||||
import { Figure } from '../domain/dbGeneral';
|
||||
import { pick } from 'underscore';
|
||||
@@ -172,4 +172,83 @@ export function addConsumeToHero(oldConsume: Reward[] = [], consumes: ItemInter[
|
||||
newConsume.push({ id, count });
|
||||
}
|
||||
return newConsume;
|
||||
}
|
||||
|
||||
export function checkUnlockTalentCondition(hid: number, id: number, talents: Talent[], usedTalentPoint: number) {
|
||||
let dicHero = gameData.hero.get(hid);
|
||||
let dicHeroTalent = gameData.heroTalent.get(id);
|
||||
if(!dicHeroTalent) return false;
|
||||
|
||||
if(dicHero.talentId != dicHeroTalent.talentId) return false;
|
||||
|
||||
// 累计消耗n点天赋点
|
||||
if(dicHeroTalent.preTotalPoint > usedTalentPoint) return false;
|
||||
// 互斥的天赋没有被解锁
|
||||
for(let { type, ids } of dicHeroTalent.relation) {
|
||||
if(type == TALENT_RELATION_TYPE.CONFLICT) {
|
||||
let hasTalent = talents.find(talent => ids.indexOf(talent.id) != -1);
|
||||
if(hasTalent) return false;
|
||||
}
|
||||
}
|
||||
// 前置节点
|
||||
let orFlag = false; // 只要其中一个满足就可以
|
||||
let maxPrecost = 0; // 前置节点最多的消耗
|
||||
for(let arr of dicHeroTalent.prepositionId) {
|
||||
let andFlag = true; // arr内所有节点都需要满足
|
||||
for(let id of arr) {
|
||||
let curTalent = talents.find(cur => cur.id == id);
|
||||
if(curTalent) {
|
||||
let dic = gameData.heroTalent.get(id);
|
||||
if(dic) {
|
||||
let allCost = 0;
|
||||
for(let { lv, cost } of dic.level) {
|
||||
if(curTalent.level >= lv) allCost += cost;
|
||||
}
|
||||
if(allCost > maxPrecost) maxPrecost = allCost;
|
||||
}
|
||||
} else {
|
||||
andFlag = false;
|
||||
}
|
||||
}
|
||||
if(andFlag) orFlag = true;
|
||||
}
|
||||
if(!orFlag) return false;
|
||||
// 在前置节点中消耗至少n点天赋点(在有多个前置节点时,取最高的1个)
|
||||
if(dicHeroTalent.preSinglePoint > maxPrecost) return false;
|
||||
return true;
|
||||
}
|
||||
|
||||
export function updateSkinTalent(skins: HeroSkin[], newTalent: Talent, usedTalentPoint: number) {
|
||||
let newSkins: HeroSkin[] = [];
|
||||
let newTalents: Talent[] = [];
|
||||
for(let skin of skins) {
|
||||
if(skin.enable) {
|
||||
let hasNewTalent = false;
|
||||
for(let talent of skin.talent) {
|
||||
if(talent.id == newTalent.id) {
|
||||
newTalents.push(newTalent);
|
||||
hasNewTalent = true;
|
||||
} else {
|
||||
newTalents.push(talent);
|
||||
}
|
||||
}
|
||||
if(!hasNewTalent) newTalents.push(newTalent);
|
||||
newSkins.push({...skin, talent: newTalents, usedTalentPoint })
|
||||
} else {
|
||||
newSkins.push(skin);
|
||||
}
|
||||
}
|
||||
return { newSkins, newTalents };
|
||||
}
|
||||
|
||||
export function initSkinTalent(skins: HeroSkin[]) {
|
||||
let newSkins: HeroSkin[] = [];
|
||||
for(let skin of skins) {
|
||||
if(skin.enable) {
|
||||
newSkins.push({ ...skin, talent: getHeroInitTalent(skin.skinId), usedTalentPoint: 0 });
|
||||
} else {
|
||||
newSkins.push(skin);
|
||||
}
|
||||
}
|
||||
return newSkins
|
||||
}
|
||||
@@ -30,7 +30,8 @@ export enum HERO_SYSTEM_TYPE {
|
||||
JEWEL_RESET_RANDSE = 27, // 天晶石洗练
|
||||
JEWEL_QUENCH = 28, // 天晶石淬炼
|
||||
REBIRTH = 29, // 武将重生
|
||||
};
|
||||
TALENT = 30, // 天赋
|
||||
};
|
||||
|
||||
// 武将上限
|
||||
export const HERO_GROW_MAX = {
|
||||
|
||||
@@ -433,6 +433,7 @@ export const FILENAME = {
|
||||
DIC_HERO_QUALITY_UP: 'dic_zyz_hero_quality_up',
|
||||
DIC_HERO_STAR: 'dic_zyz_hero_star',
|
||||
DIC_HERO_WAKE: 'dic_zyz_hero_wake',
|
||||
DIC_HERO_TALENT: 'dic_zyz_hero_talent',
|
||||
DIC_HERO_SKILL: 'dic_zyz_heroskill',
|
||||
DIC_JOB: 'dic_zyz_job',
|
||||
DIC_KING_EXP: 'dic_zyz_kingexp',
|
||||
@@ -1011,6 +1012,7 @@ export enum ITEM_CHANGE_REASON {
|
||||
ACT_GUILD_PAY_REWARD = 144, // 活动-军团人数奖励
|
||||
RECEIVE_CHAPTER_BOX = 145, // 领取主线章节宝箱
|
||||
JEWEL_INHERIT = 146, // 天晶继承
|
||||
RESET_TALENT = 147, // 洗点
|
||||
}
|
||||
|
||||
export enum TA_EVENT {
|
||||
@@ -1090,4 +1092,10 @@ export enum LOG_TYPE {
|
||||
export enum CE_CHANGE_REASON {
|
||||
HERO = 'hero', // 武将
|
||||
|
||||
}
|
||||
|
||||
export enum TALENT_RELATION_TYPE {
|
||||
NORMAL = 1, // 正常
|
||||
CONFLICT = 2, // 冲突
|
||||
REPLACE = 3, // 替换
|
||||
}
|
||||
@@ -288,6 +288,11 @@ export const STATUS = {
|
||||
ROLE_SHORT_HERO_CONECTION: { code: 30308, simStr: '未拥有羁绊武将' },
|
||||
HERO_FAVOUR_LEVEL_REACH_MAXT: { code: 30309, simStr: '武将好感等级以达到最大' },
|
||||
HERO_JOB_REACH_MAX_STAGE: { code: 30310, simStr: '武将已达到最大的职业阶级' },
|
||||
TALENT_HAS_UNLOCKED: { code: 30311, simStr: '该天赋已经解锁' },
|
||||
TALENT_CONDITION_NOT_FIT: { code: 30312, simStr: '天赋解锁条件未达成' },
|
||||
TALENT_POINT_NOT_ENOUGH: { code: 30313, simStr: '天赋点不足' },
|
||||
TALENT_NOT_UNLOCKED: { code: 30314, simStr: '该天赋未解锁' },
|
||||
TALENT_LEVEL_MAX: { code: 30315, simStr: '该天赋等级已达最高' },
|
||||
|
||||
// 装备养成 30400-30499
|
||||
ROLE_EQUIP_PLACE_NOT_ENOUGH: { code: 30400, simStr: '装备栏未装备或无可强化' },
|
||||
|
||||
@@ -7,6 +7,7 @@ import { reduceCe } from '../pubUtils/util';
|
||||
import Skin from './Skin';
|
||||
import { SearchHeroParam } from '../domain/backEndField/search';
|
||||
import { Reward } from '../domain/battleField/pvp';
|
||||
import { getHeroInitTalent } from '../pubUtils/data';
|
||||
|
||||
type CeAttrUpdate = Partial<CeAttrData>;
|
||||
export class CeAttrData {
|
||||
@@ -61,6 +62,18 @@ export class Connect {
|
||||
level: number;
|
||||
}
|
||||
|
||||
export class Talent {
|
||||
@prop({ required: true })
|
||||
id: number; // 天赋表id
|
||||
@prop({ required: true })
|
||||
level: number; // 激活等级
|
||||
|
||||
constructor(id: number, level = 1) {
|
||||
this.id = id;
|
||||
this.level = level;
|
||||
}
|
||||
}
|
||||
|
||||
export class HeroSkin {
|
||||
@prop({ required: true })
|
||||
id: number;
|
||||
@@ -70,6 +83,19 @@ export class HeroSkin {
|
||||
skin: Ref<Skin>;
|
||||
@prop({ required: true })
|
||||
enable: boolean;
|
||||
@prop({ required: true, type: Talent, _id: false })
|
||||
talent: Talent[]; // 天赋树
|
||||
@prop({ required: true })
|
||||
usedTalentPoint: number; // 已使用的天赋点数
|
||||
|
||||
constructor(id: number, skinId: number, skin: string, enable: boolean) {
|
||||
this.id = id;
|
||||
this.skinId = skinId;
|
||||
this.skin = skin;
|
||||
this.enable = enable;
|
||||
this.talent = getHeroInitTalent(skinId);
|
||||
this.usedTalentPoint = 0;
|
||||
}
|
||||
}
|
||||
|
||||
export class Stone {
|
||||
|
||||
File diff suppressed because one or more lines are too long
@@ -140,7 +140,6 @@ export class CalHeroCe {
|
||||
this.getSingleAttrObj(id).updateAttr({ inc: { fixUp: attr } });
|
||||
}
|
||||
}
|
||||
addSeidEffect.bind(this, dicJob.seid);
|
||||
}
|
||||
|
||||
|
||||
|
||||
@@ -5,7 +5,7 @@ import { dicEvent, dicEventList, loadEvent } from "./dictionary/DicEvent";
|
||||
import { dicExpedition, DicExpedition, loadExpedition } from "./dictionary/DicExpedition";
|
||||
import { dicExpeditionPoint, loadExpeditionPoint } from "./dictionary/DicExpeditionPoint";
|
||||
import { dicHeroSkill, loadHeroSkill } from "./dictionary/DicHeroSkill";
|
||||
import { dicJob, jobClassAndgrades, jobClassMaxGrades, loadJob } from "./dictionary/DicJob";
|
||||
import { dicJob, jobClassAndgrades, jobClassMaxGrades, loadJob, talentPointOfJob } from "./dictionary/DicJob";
|
||||
import { dicKingExp, maxPlayerLv, loadKingExp } from "./dictionary/DicKingExp";
|
||||
import { dicCharExp, loadCharExp } from "./dictionary/DicCharExp";
|
||||
import { dicQuestion, loadQuestion } from "./dictionary/DicQuestion";
|
||||
@@ -101,6 +101,8 @@ import { dicEquipQualityExtra, loadEquipQualityExtra } from './dictionary/DicEqu
|
||||
import { dicEquipSuit, dicEquipSuitByJobClass, loadEquipSuit } from "./dictionary/DicEquipSuit";
|
||||
import { dicJewelCondition, loadJewelCondition } from './dictionary/DicJewelCondition';
|
||||
import { dicMainStarBox, dicMainStarBoxByChapter, loadMainStarBox } from './dictionary/DicMainStarBox';
|
||||
import { dicHeroTalent, initTalents, loadHeroTalent } from './dictionary/DicHeroTalent';
|
||||
import { Talent } from "../db/Hero";
|
||||
|
||||
export const gameData = {
|
||||
daily: dicDaily,
|
||||
@@ -252,6 +254,9 @@ export const gameData = {
|
||||
heroIdByWar: dicHeroIdByWar,
|
||||
mainBox: dicMainStarBox,
|
||||
mainBoxByChapter: dicMainStarBoxByChapter,
|
||||
heroTalent: dicHeroTalent,
|
||||
initTalents: initTalents,
|
||||
talentPointOfJob: talentPointOfJob,
|
||||
};
|
||||
|
||||
// 在此提供一些原先在gamedata中提供的方法,以便更方便获取gameData数据
|
||||
@@ -869,6 +874,14 @@ export function getRandEffectByGroupAndLevel(group: number, level: number) {
|
||||
return gameData.randomEffectPool.get(id);
|
||||
}
|
||||
|
||||
// 根据皮肤获得他的初始天赋id
|
||||
export function getHeroInitTalent(skinId: number) {
|
||||
let dicHero = gameData.hero.get(skinId);
|
||||
if(!dicHero) return [];
|
||||
let ids = gameData.initTalents.get(dicHero.talentId)||[];
|
||||
return ids.map(id => (new Talent(id)));
|
||||
}
|
||||
|
||||
// 初始加载
|
||||
function initDatas() {
|
||||
parseDicParam();
|
||||
@@ -1035,6 +1048,7 @@ function loadDatas() {
|
||||
loadStone();
|
||||
loadJewelCondition();
|
||||
loadMainStarBox();
|
||||
loadHeroTalent();
|
||||
}
|
||||
|
||||
// 重载dicParam
|
||||
|
||||
@@ -86,8 +86,8 @@ export const EXTERIOR = {
|
||||
EXTERIOR_APPEARANCE: 11419, // 默认形象id
|
||||
};
|
||||
export const HTTPMANAGE = {
|
||||
ADDRESSTYPE: 1, // 1:开发服(dev),2:正式服(official),3:测试服(alpha),4:版号服(isbn) 5: 37测试服(sq1)
|
||||
SERVERTYPE: 2, // ADDRESSTYPE下服务器的筛选:1:正式服(official),2:开发服(develop) 3. 测试服(alpha)
|
||||
ADDRESSTYPE: 3, // 1:开发服(dev),2:正式服(official),3:测试服(alpha),4:版号服(isbn) 5: 37测试服(sq1)
|
||||
SERVERTYPE: 3, // ADDRESSTYPE下服务器的筛选:1:正式服(official),2:开发服(develop) 3. 测试服(alpha)
|
||||
};
|
||||
export const CHAT_SYSTEM = {
|
||||
RECENT_GROUP_MSGS_CNT: 10, // 群消息获取条数
|
||||
@@ -262,9 +262,15 @@ export const VIP = {
|
||||
VIP_TOWER_TASK_CAN_SEND_AT_ONCE_WITHOUT_VIP: 0, // 镇念没有月卡是否可以被一键悬赏
|
||||
VIP_TOWER_TASK_CAN_SEND_AT_ONCE_WITH_VIP: 1, // 镇念塔有月卡是否可以被一键悬赏
|
||||
VIP_TOWER_HUNG_UP_RATIO: 1.2, // 镇念塔挂机奖励加成倍率,在dic_zyz_tower的rewardOfcollect基础上直接乘以这个倍数
|
||||
VIP_REGRET_CNT_WITHOUT_VIP: 10, // 没有月卡时候的悔棋次数
|
||||
VIP_REGRET_CNT_WITH_VIP: 11, // 有月卡时候的悔棋次数
|
||||
VIP_REGRET_CNT_WITHOUT_VIP: 0, // 没有月卡时候的悔棋次数
|
||||
VIP_REGRET_CNT_WITH_VIP: 1, // 有月卡时候的悔棋次数
|
||||
VIP_PVP_CHALLENGE_COUNTS_ADD: 2, // pvp上限增加,如果有月卡,在PVP.PVP_CHALLENGE_COUNTS的基础上增加X次
|
||||
VIP_DAILY_TIMERS_PER_DAY_ADD: 2, // 每日任务免费次数增加,如果有月卡,在dic_daily的timesPerDay的基础上增加多少次
|
||||
VIP_DUNGEON_CONST_FREE_ADD: 2, // 秘境免费次数增加,如果有月卡,在DUNGEON_CONST.DUNGEON_CONST_FREE的基础上加多少次
|
||||
};
|
||||
export const PUBLIC_NOTICE_PERIOD = {
|
||||
PUBLIC_NOTICE_PERIOD_TIME: 24, // 活动公示期时间
|
||||
};
|
||||
export const HERO = {
|
||||
HERO_TALENTPOINT_RESET: '31002&100', // 天赋点重置花费
|
||||
};
|
||||
|
||||
@@ -33,10 +33,12 @@ export interface DicHero {
|
||||
readonly recruit: boolean;
|
||||
// 武将图标id
|
||||
readonly face_id: string;
|
||||
// 天赋树id
|
||||
readonly talentId: number;
|
||||
}
|
||||
|
||||
type KeysEnum<T> = { [P in keyof Required<T>]: true };
|
||||
const DicHeroKeys: KeysEnum<DicHero> = {heroId: true, name: true, quality: true, camp: true, jobClass: true, jobid: true, skill: true, pieceId: true, initialStars: true, pieceCount: true, baseAbilityArr: true, baseAbilityUpArr: true, initialSkin: true, recruit: true, face_id: true};
|
||||
const DicHeroKeys: KeysEnum<DicHero> = {heroId: true, name: true, quality: true, camp: true, jobClass: true, jobid: true, skill: true, pieceId: true, initialStars: true, pieceCount: true, baseAbilityArr: true, baseAbilityUpArr: true, initialSkin: true, recruit: true, face_id: true, talentId: true};
|
||||
export const dicHero = new Map<number, DicHero>();
|
||||
export function loadHero() {
|
||||
dicHero.clear();
|
||||
|
||||
81
shared/pubUtils/dictionary/DicHeroTalent.ts
Normal file
81
shared/pubUtils/dictionary/DicHeroTalent.ts
Normal file
@@ -0,0 +1,81 @@
|
||||
// 武将升星表
|
||||
import { decodeArrayListStr, parseNumberList, readFileAndParse } from '../util'
|
||||
import { FILENAME } from '../../consts';
|
||||
const _ = require('lodash');
|
||||
|
||||
export interface DicHeroTalent {
|
||||
// 唯一id
|
||||
readonly id: number;
|
||||
// 天赋组id
|
||||
readonly talentId: number;
|
||||
// 节点层数
|
||||
readonly nodeSort: number;
|
||||
// 和其他节点关系
|
||||
readonly relation: [{ type: number, ids?: number[] }];
|
||||
// 节点等级数
|
||||
readonly level: [{ lv: number, seid: number, cost: number }];
|
||||
// 前置节点
|
||||
readonly prepositionId: number[][];
|
||||
// 需要满足在天赋树中累计消耗n点天赋点
|
||||
readonly preTotalPoint: number;
|
||||
// 在前置节点中消耗至少n点天赋点(在有多个前置节点时,取最高的1个)
|
||||
readonly preSinglePoint: number;
|
||||
}
|
||||
|
||||
|
||||
type KeysEnum<T> = { [P in keyof Required<T>]: true };
|
||||
const DicHeroTalentKeys: KeysEnum<DicHeroTalent> = {id: true, talentId: true, nodeSort: true, relation: true, level: true, prepositionId: true, preTotalPoint: true, preSinglePoint: true};
|
||||
|
||||
export const dicHeroTalent = new Map<number, DicHeroTalent>();
|
||||
export const initTalents = new Map<number, number[]>();
|
||||
export function loadHeroTalent() {
|
||||
dicHeroTalent.clear();
|
||||
|
||||
let arr = readFileAndParse(FILENAME.DIC_HERO_TALENT);
|
||||
|
||||
arr.forEach(o => {
|
||||
o.relation = parseRelation(o.type, o.relationId);
|
||||
o.level = parseLevel(o.levelCount, o.levelSeid, o.levelConsume);
|
||||
o.prepositionId = parsePrePositionId(o.prepositionId);
|
||||
dicHeroTalent.set(o.id, _.pick(o, Object.keys(DicHeroTalentKeys)));
|
||||
if(o.prepositionId.length == 0 && o.levelConsume == '&') {
|
||||
if(!initTalents.has(o.talentId)) {
|
||||
initTalents.set(o.talentId, []);
|
||||
}
|
||||
initTalents.get(o.talentId).push(o.id);
|
||||
}
|
||||
});
|
||||
arr = undefined;
|
||||
}
|
||||
|
||||
function parseRelation(type: string, relationId: string) {
|
||||
let relation: {type: number, ids?: number[]}[] = [];
|
||||
let types = parseNumberList(type);
|
||||
let relations = decodeArrayListStr(relationId);
|
||||
for(let i = 0; i < types.length; i++) {
|
||||
if(relations[i]) {
|
||||
relation.push({
|
||||
type: types[i],
|
||||
ids: relations[i].map(id => parseInt(id))
|
||||
});
|
||||
}
|
||||
}
|
||||
return relation;
|
||||
}
|
||||
|
||||
function parseLevel(levelCount: number, levelSeid: string, levelConsume: string) {
|
||||
let levelSeids = parseNumberList(levelSeid);
|
||||
let levelConsumes = parseNumberList(levelConsume);
|
||||
let level: { lv: number, seid: number, cost: number }[] = [];
|
||||
for(let i = 0; i < levelCount; i++) {
|
||||
level.push({ lv: i + 1, seid: levelSeids[i]||0, cost: levelConsumes[i]||0});
|
||||
}
|
||||
return level;
|
||||
}
|
||||
|
||||
function parsePrePositionId(str: string) {
|
||||
let arr = decodeArrayListStr(str);
|
||||
return arr.map(arr1 => {
|
||||
return arr1.map(str => parseInt(str));
|
||||
});
|
||||
}
|
||||
@@ -17,8 +17,8 @@ export interface DicJob {
|
||||
readonly job_class: number;
|
||||
// 职业类别
|
||||
readonly type: number;
|
||||
// 特性
|
||||
readonly seid: Array<number>;
|
||||
// // 特性
|
||||
// readonly seid: Array<number>;
|
||||
// 训练消耗
|
||||
readonly trainingConsume: Array<RewardInter>;
|
||||
// 升阶消耗
|
||||
@@ -27,15 +27,17 @@ export interface DicJob {
|
||||
readonly ceAttr: Map<number, {id: number, attr: number}>;
|
||||
// 一共有多少阶升级
|
||||
readonly maxStage: number;
|
||||
|
||||
// 到这一阶能获得多少天赋点
|
||||
readonly talentPoint: number;
|
||||
}
|
||||
|
||||
type KeysEnum<T> = { [P in keyof Required<T>]: true };
|
||||
const DicJobKeys: KeysEnum<DicJob> = {jobid: true, name: true, grade: true, unlockLevel: true, job_class: true, type: true, seid: true,trainingConsume: true, upGradeConsume: true, ceAttr: true, maxStage: true};
|
||||
const DicJobKeys: KeysEnum<DicJob> = {jobid: true, name: true, grade: true, unlockLevel: true, job_class: true, type: true, trainingConsume: true, upGradeConsume: true, ceAttr: true, maxStage: true, talentPoint: true};
|
||||
|
||||
export const dicJob = new Map<number, DicJob>();
|
||||
export const jobClassMaxGrades = new Map<number, {grade:number, jobid:number}>();
|
||||
export const jobClassAndgrades = new Map<string, {jobid:number, unlockLevel:number}>();
|
||||
export const talentPointOfJob = new Map<number, number>();
|
||||
|
||||
export function loadJob() {
|
||||
dicJob.clear();
|
||||
@@ -43,6 +45,7 @@ export function loadJob() {
|
||||
jobClassAndgrades.clear();
|
||||
|
||||
let arr = readFileAndParse(FILENAME.DIC_JOB);
|
||||
let jobByJobClass = new Map<number, number[]>(); // jobClass => jobid[]
|
||||
|
||||
arr.forEach(o => {
|
||||
if(o.isPlayer) {
|
||||
@@ -56,9 +59,25 @@ export function loadJob() {
|
||||
if (!jobClass || jobClass.grade < o.grade) {
|
||||
jobClassMaxGrades.set(o.job_class, {grade: o.grade,jobid: o.jobid});
|
||||
}
|
||||
jobClassAndgrades.set(o.job_class+'_'+o.grade,{unlockLevel:o.unlockLevel, jobid:o.jobid});
|
||||
jobClassAndgrades.set(o.job_class+'_'+o.grade,{unlockLevel:o.unlockLevel, jobid:o.jobid});
|
||||
if(!jobByJobClass.has(o.job_class)) {
|
||||
jobByJobClass.set(o.job_class, []);
|
||||
}
|
||||
jobByJobClass.get(o.job_class).push(o.jobid);
|
||||
}
|
||||
});
|
||||
for(let [_, { job_class, talentPoint }] of dicJob) {
|
||||
let jobs = jobByJobClass.get(job_class)||[];
|
||||
for(let jobid of jobs) {
|
||||
if(!talentPointOfJob.has(jobid)) {
|
||||
talentPointOfJob.set(jobid, talentPoint);
|
||||
} else {
|
||||
talentPointOfJob.set(jobid, talentPointOfJob.get(jobid) + talentPoint);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
jobByJobClass = undefined;
|
||||
arr = undefined;
|
||||
}
|
||||
|
||||
|
||||
@@ -2,10 +2,10 @@
|
||||
* 体力系统
|
||||
*/
|
||||
|
||||
import { HERO_SYSTEM_TYPE, ABI_TYPE, HERO_CE_RATIO, LINEUP_NUM } from '../consts';
|
||||
import { HERO_SYSTEM_TYPE, ABI_TYPE, HERO_CE_RATIO, LINEUP_NUM, TALENT_RELATION_TYPE } from '../consts';
|
||||
|
||||
import { cal, deepCopy, getAllAttrStage, reduceCe } from './util';
|
||||
import { HeroModel, HeroType, HeroUpdate, CeAttrData, EPlace, Stone } from '../db/Hero';
|
||||
import { HeroModel, HeroType, HeroUpdate, CeAttrData, EPlace, Stone, Talent } from '../db/Hero';
|
||||
import { RoleModel, RoleType, RoleUpdate, CeAttrDataRole } from '../db/Role';
|
||||
import { AttributeCal } from '../domain/roleField/attribute';
|
||||
import { ABI_STAGE, SEID_TYPE } from '../consts';
|
||||
@@ -20,6 +20,7 @@ import { GuildModel } from '../db/Guild';
|
||||
import { DicJob } from './dictionary/DicJob';
|
||||
import { saveCeChangeLog } from './logUtil';
|
||||
import { JewelType } from '../db/Jewel';
|
||||
import { type } from 'os';
|
||||
|
||||
// 修改并下发战力
|
||||
export async function calPlayerCeAndSave(type: number, roleId: string, originHero: HeroType, update: HeroUpdate, args?: Array<number>, params?: any) {
|
||||
@@ -161,7 +162,7 @@ export async function calPlayerCe(hero: HeroType, update: HeroUpdate, type: numb
|
||||
heroAttrs = calHeroTrainIncAttr(hero, update);
|
||||
break;
|
||||
case HERO_SYSTEM_TYPE.STAGEUP:
|
||||
heroAttrs = calHeroJobStageUpIncAttr(hero, update, addSeidList, removeSeidList);
|
||||
// heroAttrs = calHeroJobStageUpIncAttr(hero, update, addSeidList, removeSeidList);
|
||||
break;
|
||||
case HERO_SYSTEM_TYPE.SKIN:
|
||||
heroAttrs = calHeroWearSkinIncAttr(hero, update, addSeidList, removeSeidList);
|
||||
@@ -197,6 +198,9 @@ export async function calPlayerCe(hero: HeroType, update: HeroUpdate, type: numb
|
||||
case HERO_SYSTEM_TYPE.SCROLL:
|
||||
heroAttrs = calHeroCeScrollIncAttr(hero, update);
|
||||
break;
|
||||
case HERO_SYSTEM_TYPE.TALENT:
|
||||
heroAttrs = calHeroTalent(hero, update, addSeidList, removeSeidList);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
@@ -460,29 +464,29 @@ export function calHeroTrainIncAttr(originHero: HeroType, update: HeroUpdate) {
|
||||
* @param {number[]} addSeidList 用于更新被动
|
||||
* @param {number[]} removeSeidList 用于更新被动
|
||||
*/
|
||||
export function calHeroJobStageUpIncAttr(originHero: HeroType, update: HeroUpdate, addSeidList: Array<number>, removeSeidList: Array<number>) {
|
||||
let { job: oldJob, attr: heroAttrs } = originHero;
|
||||
let { job = oldJob } = update;
|
||||
// export function calHeroJobStageUpIncAttr(originHero: HeroType, update: HeroUpdate, addSeidList: Array<number>, removeSeidList: Array<number>) {
|
||||
// let { job: oldJob, attr: heroAttrs } = originHero;
|
||||
// let { job = oldJob } = update;
|
||||
|
||||
let lastJob = gameData.job.get(oldJob);
|
||||
let currentJob = gameData.job.get(job);
|
||||
let lastSeids = lastJob?.seid||new Array<number>();
|
||||
let curSeids = currentJob?.seid||new Array<number>();
|
||||
// let lastJob = gameData.job.get(oldJob);
|
||||
// let currentJob = gameData.job.get(job);
|
||||
// let lastSeids = lastJob?.seid||new Array<number>();
|
||||
// let curSeids = currentJob?.seid||new Array<number>();
|
||||
|
||||
for (let seid of curSeids) {
|
||||
let index = findIndex(lastSeids, seid);
|
||||
if (index < 0) {
|
||||
addSeidList.push(seid, 0);
|
||||
}
|
||||
}
|
||||
for (let seid of lastSeids) {
|
||||
let index = findIndex(curSeids, seid);
|
||||
if (index < 0) {
|
||||
removeSeidList.push(seid, 0);
|
||||
}
|
||||
}
|
||||
return heroAttrs;
|
||||
}
|
||||
// for (let seid of curSeids) {
|
||||
// let index = findIndex(lastSeids, seid);
|
||||
// if (index < 0) {
|
||||
// addSeidList.push(seid, 0);
|
||||
// }
|
||||
// }
|
||||
// for (let seid of lastSeids) {
|
||||
// let index = findIndex(curSeids, seid);
|
||||
// if (index < 0) {
|
||||
// removeSeidList.push(seid, 0);
|
||||
// }
|
||||
// }
|
||||
// return heroAttrs;
|
||||
// }
|
||||
|
||||
/**
|
||||
* 初始计算兵种属性
|
||||
@@ -515,7 +519,7 @@ export function calHeroJobAttr(originHero: HeroType, hero: HeroUpdate, addSeidLi
|
||||
};
|
||||
|
||||
originHero.attr = heroAttrs;
|
||||
heroAttrs = calHeroJobStageUpIncAttr(originHero, hero, addSeidList, removeSeidList);
|
||||
// heroAttrs = calHeroJobStageUpIncAttr(originHero, hero, addSeidList, removeSeidList);
|
||||
return heroAttrs;
|
||||
}
|
||||
|
||||
@@ -552,6 +556,8 @@ export function calHeroWearSkinIncAttr(originHero: HeroType, update: HeroUpdate,
|
||||
calEquipStrengthIncAttr(originHero, update, eplaceIds);
|
||||
calEquipQualityIncAttr(originHero, update, eplaceIds);
|
||||
calEquipStarIncAttr(originHero, update, eplaceIds, addSeidList, removeSeidList);
|
||||
// 天赋点
|
||||
calHeroTalent(originHero, update, addSeidList, removeSeidList);
|
||||
|
||||
return heroAttrs;
|
||||
}
|
||||
@@ -990,6 +996,43 @@ function calHeroCeScrollIncAttr(originHero: HeroType, update: HeroUpdate) {
|
||||
return heroAttrs;
|
||||
}
|
||||
|
||||
function calHeroTalent(originHero: HeroType, update: HeroUpdate, addSeidList: number[], removeSeidList: number[]) {
|
||||
let { attr: heroAttrs, skins: oldSkins } = originHero;
|
||||
let { skins } = update;
|
||||
let oldSkin = oldSkins.find(cur => cur.enable);
|
||||
let newSkin = skins.find(cur => cur.enable);
|
||||
let oldSeids = getTalentSeid(oldSkin.talent);
|
||||
let newSeids = getTalentSeid(newSkin.talent);
|
||||
for(let seid of newSeids) addSeidList.push(seid);
|
||||
for(let seid of oldSeids) removeSeidList.push(seid);
|
||||
return heroAttrs
|
||||
}
|
||||
|
||||
function getTalentSeid(talent: Talent[]) {
|
||||
let seids = new Map<number, number[]>(); // id, seids
|
||||
for(let { id, level } of talent) {
|
||||
let dicHeroTalent = gameData.heroTalent.get(id);
|
||||
for(let { type, ids} of dicHeroTalent.relation) {
|
||||
if(type == TALENT_RELATION_TYPE.REPLACE) {
|
||||
for(let id of ids) {
|
||||
seids.delete(id);
|
||||
}
|
||||
}
|
||||
}
|
||||
for(let { lv, seid } of dicHeroTalent.level) {
|
||||
if(level >= lv) {
|
||||
if(!seids.has(id)) seids.set(id, []);
|
||||
seids.get(id).push(seid);
|
||||
}
|
||||
}
|
||||
}
|
||||
let result: number[] = [];
|
||||
for(let [_, ids] of seids) {
|
||||
result.push(...ids);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
/**
|
||||
* 升爵位
|
||||
* @param role 角色数据
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
394
shared/resource/jsons/dic_zyz_hero_talent.json
Normal file
394
shared/resource/jsons/dic_zyz_hero_talent.json
Normal file
@@ -0,0 +1,394 @@
|
||||
[
|
||||
{
|
||||
"id": 1,
|
||||
"talentId": 1,
|
||||
"nodeSort": 1,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "援护初始",
|
||||
"levelCount": 1,
|
||||
"levelSeid": "31&",
|
||||
"levelConsume": "&",
|
||||
"prepositionId": "&",
|
||||
"preTotalPoint": 0,
|
||||
"preSinglePoint": 0
|
||||
},
|
||||
{
|
||||
"id": 2,
|
||||
"talentId": 1,
|
||||
"nodeSort": 2,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "攻击初始",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "50201&50301&50211&50311&50221&50321",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "1&",
|
||||
"preTotalPoint": 0,
|
||||
"preSinglePoint": 0
|
||||
},
|
||||
{
|
||||
"id": 3,
|
||||
"talentId": 1,
|
||||
"nodeSort": 2,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "物防初始",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "401&402&501&502&402",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "1&",
|
||||
"preTotalPoint": 0,
|
||||
"preSinglePoint": 0
|
||||
},
|
||||
{
|
||||
"id": 4,
|
||||
"talentId": 1,
|
||||
"nodeSort": 2,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "策防初始",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "401&402&501&502&402",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "1&",
|
||||
"preTotalPoint": 0,
|
||||
"preSinglePoint": 0
|
||||
},
|
||||
{
|
||||
"id": 5,
|
||||
"talentId": 1,
|
||||
"nodeSort": 3,
|
||||
"type": "2&3",
|
||||
"relationId": "6&7&13&14|1&",
|
||||
"name": "高级援护",
|
||||
"levelCount": 3,
|
||||
"levelSeid": "1001&1002&1011",
|
||||
"levelConsume": "5&5&5",
|
||||
"prepositionId": "2&",
|
||||
"preTotalPoint": 40,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 6,
|
||||
"talentId": 1,
|
||||
"nodeSort": 3,
|
||||
"type": "2&3",
|
||||
"relationId": "5&7&12&14|1&",
|
||||
"name": "减伤光环",
|
||||
"levelCount": 3,
|
||||
"levelSeid": "11001&11002&11003",
|
||||
"levelConsume": "5&5&5",
|
||||
"prepositionId": "3&",
|
||||
"preTotalPoint": 40,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 7,
|
||||
"talentId": 1,
|
||||
"nodeSort": 3,
|
||||
"type": "2&3",
|
||||
"relationId": "5&6&12&13|1&",
|
||||
"name": "抗伤光环",
|
||||
"levelCount": 3,
|
||||
"levelSeid": "16001&16002&16003",
|
||||
"levelConsume": "5&5&5",
|
||||
"prepositionId": "4&",
|
||||
"preTotalPoint": 40,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 8,
|
||||
"talentId": 1,
|
||||
"nodeSort": 4,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "格挡",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "1101&1102&1201&1202&1301",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "5|6|7",
|
||||
"preTotalPoint": 50,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 9,
|
||||
"talentId": 1,
|
||||
"nodeSort": 5,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "反击伤害",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "23611&23621&23631&23641&23651",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "8&",
|
||||
"preTotalPoint": 60,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 10,
|
||||
"talentId": 1,
|
||||
"nodeSort": 5,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "物理减伤",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "11006&11007&11008&11009&11010",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "8&",
|
||||
"preTotalPoint": 60,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 11,
|
||||
"talentId": 1,
|
||||
"nodeSort": 5,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "策略减伤",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "16006&16007&16008&16009&16010",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "8&",
|
||||
"preTotalPoint": 60,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 12,
|
||||
"talentId": 1,
|
||||
"nodeSort": 6,
|
||||
"type": "2&3",
|
||||
"relationId": "6&7&13&14|5&",
|
||||
"name": "援护强化",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "1201&1202&1301&1302&1402",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "9&",
|
||||
"preTotalPoint": 70,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 13,
|
||||
"talentId": 1,
|
||||
"nodeSort": 6,
|
||||
"type": "2&3",
|
||||
"relationId": "5&7&12&14|6&",
|
||||
"name": "减伤强化",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "2800&2801&2802&2803&2804",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "10&",
|
||||
"preTotalPoint": 70,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 14,
|
||||
"talentId": 1,
|
||||
"nodeSort": 6,
|
||||
"type": "2&3",
|
||||
"relationId": "5&6&12&13|7&",
|
||||
"name": "抗伤强化",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "2805&2806&2807&2808&2809",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "11&",
|
||||
"preTotalPoint": 70,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 15,
|
||||
"talentId": 2,
|
||||
"nodeSort": 1,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "援护初始",
|
||||
"levelCount": 1,
|
||||
"levelSeid": "31&",
|
||||
"levelConsume": "&",
|
||||
"prepositionId": "&",
|
||||
"preTotalPoint": 0,
|
||||
"preSinglePoint": 0
|
||||
},
|
||||
{
|
||||
"id": 16,
|
||||
"talentId": 2,
|
||||
"nodeSort": 2,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "攻击初始",
|
||||
"levelCount": 4,
|
||||
"levelSeid": "50201&50301&50211&50311&50221",
|
||||
"levelConsume": "1&2&3&4",
|
||||
"prepositionId": "15&",
|
||||
"preTotalPoint": 0,
|
||||
"preSinglePoint": 0
|
||||
},
|
||||
{
|
||||
"id": 17,
|
||||
"talentId": 2,
|
||||
"nodeSort": 2,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "物防初始",
|
||||
"levelCount": 4,
|
||||
"levelSeid": "401&402&501&502",
|
||||
"levelConsume": "1&2&3&4",
|
||||
"prepositionId": "15&",
|
||||
"preTotalPoint": 0,
|
||||
"preSinglePoint": 0
|
||||
},
|
||||
{
|
||||
"id": 18,
|
||||
"talentId": 2,
|
||||
"nodeSort": 2,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "策防初始",
|
||||
"levelCount": 4,
|
||||
"levelSeid": "401&402&501&402",
|
||||
"levelConsume": "1&2&3&4",
|
||||
"prepositionId": "15&",
|
||||
"preTotalPoint": 0,
|
||||
"preSinglePoint": 0
|
||||
},
|
||||
{
|
||||
"id": 19,
|
||||
"talentId": 2,
|
||||
"nodeSort": 3,
|
||||
"type": "2&3",
|
||||
"relationId": "20&21&27&28|15&",
|
||||
"name": "高级援护",
|
||||
"levelCount": 3,
|
||||
"levelSeid": "1001&1002&1011",
|
||||
"levelConsume": "5&5&5",
|
||||
"prepositionId": "16&",
|
||||
"preTotalPoint": 30,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 20,
|
||||
"talentId": 2,
|
||||
"nodeSort": 3,
|
||||
"type": "2&3",
|
||||
"relationId": "19&21&26&28|15&",
|
||||
"name": "减伤光环",
|
||||
"levelCount": 3,
|
||||
"levelSeid": "11001&11002&11003",
|
||||
"levelConsume": "5&5&5",
|
||||
"prepositionId": "17&",
|
||||
"preTotalPoint": 30,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 21,
|
||||
"talentId": 2,
|
||||
"nodeSort": 3,
|
||||
"type": "2&3",
|
||||
"relationId": "19&20&26&27|15&",
|
||||
"name": "抗伤光环",
|
||||
"levelCount": 3,
|
||||
"levelSeid": "16001&16002&16003",
|
||||
"levelConsume": "5&5&5",
|
||||
"prepositionId": "18&",
|
||||
"preTotalPoint": 30,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 22,
|
||||
"talentId": 2,
|
||||
"nodeSort": 4,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "格挡",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "1101&1102&1201&1202&1301",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "19|20|21",
|
||||
"preTotalPoint": 40,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 23,
|
||||
"talentId": 2,
|
||||
"nodeSort": 5,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "反击伤害",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "23611&23621&23631&23641&23651",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "22&",
|
||||
"preTotalPoint": 50,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 24,
|
||||
"talentId": 2,
|
||||
"nodeSort": 5,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "物理减伤",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "11006&11007&11008&11009&11010",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "22&",
|
||||
"preTotalPoint": 50,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 25,
|
||||
"talentId": 2,
|
||||
"nodeSort": 5,
|
||||
"type": "1&",
|
||||
"relationId": "&",
|
||||
"name": "策略减伤",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "16006&16007&16008&16009&16010",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "22&",
|
||||
"preTotalPoint": 50,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 26,
|
||||
"talentId": 2,
|
||||
"nodeSort": 6,
|
||||
"type": "2&3",
|
||||
"relationId": "20&21&27&28|23&",
|
||||
"name": "援护强化",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "1201&1202&1301&1302&1402",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "23&",
|
||||
"preTotalPoint": 60,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 27,
|
||||
"talentId": 2,
|
||||
"nodeSort": 6,
|
||||
"type": "2&3",
|
||||
"relationId": "19&21&26&28|24&",
|
||||
"name": "减伤强化",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "2800&2801&2802&2803&2804",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "24&",
|
||||
"preTotalPoint": 60,
|
||||
"preSinglePoint": 10
|
||||
},
|
||||
{
|
||||
"id": 28,
|
||||
"talentId": 2,
|
||||
"nodeSort": 6,
|
||||
"type": "2&3",
|
||||
"relationId": "19&20&26&27|25&",
|
||||
"name": "抗伤强化",
|
||||
"levelCount": 5,
|
||||
"levelSeid": "2805&2806&2807&2808&2809",
|
||||
"levelConsume": "1&2&3&4&5",
|
||||
"prepositionId": "25&",
|
||||
"preTotalPoint": 60,
|
||||
"preSinglePoint": 10
|
||||
}
|
||||
]
|
||||
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user