js: 七圣召唤七日历练: 支持使用指定的队伍 (#874)

* js: 七圣召唤七日历练: 支持使用指定的队伍

* js: 七圣召唤七日历练: 细微改进脚本格式
This commit is contained in:
Patrick-Ze
2025-05-19 23:34:44 +08:00
committed by GitHub
parent 8e17c5c207
commit 9b9464a337
3 changed files with 350 additions and 319 deletions

View File

@@ -1,220 +1,242 @@
(async function () { // 切换到指定的队伍
async function switchCardTeam(Name) {
let captureRegion = captureGameRegion();
let teamName = captureRegion.find(RecognitionObject.ocr(1305, 793, 206, 46));
log.info("当前队伍名称: {text}", teamName.text);
if (teamName.text != Name) {
click(1312, 812); //点击队伍名称的糟糕UI
await sleep(1000);
// 存储挑战玩家信息 moveMouseTo(100, 200);
let textArray = []; leftButtonDown();
let skipNum = 0; // 不能一次移动太多,否则会丢拖动
//检查挑战结果 await checkChallengeResults(); for (let i = 1; i <= 9; i++) {
async function checkChallengeResults() { await sleep(50);
const region1 = RecognitionObject.ocr(785, 890, 340, 82);// 对话区域 moveMouseTo(200 * i, 200);
let capture = captureGameRegion();
let res1 = capture.find(region1);
if (res1.isEmpty()){
await sleep(1000);
click(960, 540);
await sleep(500);
click(1860,50 );//避免失败卡死
await sleep(1000);
click(1600,260 );
await sleep(1000);
click(1180,756 );
await sleep(6000);
click(754,915 );//退出挑战
await sleep(4000);
await autoConversation();
await sleep(1000);
return;
}
else{
await sleep(1000);
click(754,915 );//退出挑战
await sleep(4000);
await autoConversation();
await sleep(1000);
return;
} }
await sleep(200);
leftButtonUp();
await sleep(1000);
captureRegion = captureGameRegion();
for (let i = 0; i < 4; i++) {
let x = 135 + 463 * i;
let res = captureRegion.find(RecognitionObject.ocr(x, 762, 230, 46));
if (res.text == Name) {
log.info("切换至队伍: {text}", res.text);
res.click();
await sleep(500);
click(1164, 1016); // 选择
await sleep(4000); // 等待"出战牌组"的强制延时框消失
break;
}
}
}
}
(async function () {
// 存储挑战玩家信息
let textArray = [];
let skipNum = 0;
//检查挑战结果 await checkChallengeResults();
async function checkChallengeResults() {
const region1 = RecognitionObject.ocr(785, 890, 340, 82); // 对话区域
let capture = captureGameRegion();
let res1 = capture.find(region1);
if (res1.isEmpty()) {
await sleep(1000);
click(960, 540);
await sleep(500);
click(1860, 50); //避免失败卡死
await sleep(1000);
click(1600, 260);
await sleep(1000);
click(1180, 756);
await sleep(6000);
click(754, 915); //退出挑战
await sleep(4000);
await autoConversation();
await sleep(1000);
return;
} else {
await sleep(1000);
click(754, 915); //退出挑战
await sleep(4000);
await autoConversation();
await sleep(1000);
return;
}
} }
//自动对话,直到出现选项框 await autoConversation(); //自动对话,直到出现选项框 await autoConversation();
async function autoConversation() { async function autoConversation() {
await sleep(2000);//点击后等待一段时间避免误判 await sleep(2000); //点击后等待一段时间避免误判
const region1 = RecognitionObject.ocr(785, 890, 340, 82);// 对话区域 const region1 = RecognitionObject.ocr(785, 890, 340, 82); // 对话区域
const region2 = RecognitionObject.ocr(1250, 400, 660, 440);// 选项区域 const region2 = RecognitionObject.ocr(1250, 400, 660, 440); // 选项区域
let talkTime = 0; let talkTime = 0;
//最多10次对话 //最多10次对话
while (talkTime < 20) { while (talkTime < 20) {
let capture = captureGameRegion(); let capture = captureGameRegion();
let res1 = capture.find(region1); let res1 = capture.find(region1);
let res2 = capture.find(region2); let res2 = capture.find(region2);
if (!res1.isEmpty() && res2.isEmpty()){ if (!res1.isEmpty() && res2.isEmpty()) {
talkTime++; talkTime++;
keyPress("VK_SPACE"); keyPress("VK_SPACE");
await sleep(500); await sleep(500);
keyPress("VK_SPACE"); keyPress("VK_SPACE");
await sleep(500); await sleep(500);
} } else if (!res1.isEmpty() && !res2.isEmpty()) {
else if(!res1.isEmpty() && !res2.isEmpty()){ keyPress("F");
keyPress("F"); await sleep(1000);
await sleep(1000); log.info("已选择谈话内容");
log.info("已选择谈话内容"); return;
return; } else if (res1.isEmpty() && !res2.isEmpty()) {
} log.info("谈话完成");
else if(res1.isEmpty() && !res2.isEmpty()){ await sleep(1000);
log.info("谈话完成"); return;
await sleep(1000);
return;
} }
talkTime++; talkTime++;
await sleep(1500); await sleep(1500);
} }
throw new Error('对话时间超时'); throw new Error("对话时间超时");
} }
//检测传送结束 //检测传送结束
async function tpEndDetection() { async function tpEndDetection() {
const region = RecognitionObject.ocr(1690, 230, 75, 350);// 队伍名称区域 const region = RecognitionObject.ocr(1690, 230, 75, 350); // 队伍名称区域
let tpTime = 0; let tpTime = 0;
await sleep(500);//点击传送后等待一段时间避免误判 await sleep(500); //点击传送后等待一段时间避免误判
//最多30秒传送时间 //最多30秒传送时间
while (tpTime < 300) { while (tpTime < 300) {
let capture = captureGameRegion(); let capture = captureGameRegion();
let res = capture.find(region); let res = capture.find(region);
if (!res.isEmpty()){ if (!res.isEmpty()) {
log.info("传送完成"); log.info("传送完成");
await sleep(1200);//传送结束后有僵直 await sleep(1200); //传送结束后有僵直
return; return;
} }
tpTime++; tpTime++;
await sleep(100); await sleep(100);
} }
throw new Error('传送时间超时'); throw new Error("传送时间超时");
} }
// 打开地图,查看玩家位置,并前往相应位置 // 打开地图,查看玩家位置,并前往相应位置
const cardPlayerRo = RecognitionObject.TemplateMatch( file.ReadImageMatSync("assets/cardPlayer.png")); const cardPlayerRo = RecognitionObject.TemplateMatch(file.ReadImageMatSync("assets/cardPlayer.png"));
const detectCardPlayer = async () => { const detectCardPlayer = async () => {
// 定义要检测的6个点位及对应的处理函数 // 定义要检测的6个点位及对应的处理函数
let i =0; let i = 0;
let findNum = 0; let findNum = 0;
const checkPoints = [ const checkPoints = [
{ x: 640, y: 750, action: async () => await gotoTable1() }, // 点位1 { x: 640, y: 750, action: async () => await gotoTable1() }, // 点位1
{ x: 810, y: 790, action: async () => await gotoTable2() }, // 点位2 { x: 810, y: 790, action: async () => await gotoTable2() }, // 点位2
{ x: 810, y: 600, action: async () => await gotoTable3() }, // 点位3 { x: 810, y: 600, action: async () => await gotoTable3() }, // 点位3
{ x: 610, y: 360, action: async () => await gotoTable4() }, // 点位4 { x: 610, y: 360, action: async () => await gotoTable4() }, // 点位4
{ x: 700, y: 5, action: async () => await gotoTable5() }, // 点位5 { x: 700, y: 5, action: async () => await gotoTable5() }, // 点位5
{ x: 290, y: 530, action: async () => await gotoTable6() } // 点位6 { x: 290, y: 530, action: async () => await gotoTable6() }, // 点位6
]; ];
keyPress("M");
await sleep(1200);
click(48, 441); //放大地图
await sleep(300);
click(48, 441); //放大地图
await sleep(300);
click(48, 441); //放大地图
await sleep(300);
keyPress("M"); //地图拖动到指定位置
await sleep(1200); moveMouseTo(200, 200);
click(48, 441);//放大地图 leftButtonDown();
await sleep(300); await sleep(500);
click(48, 441);//放大地图 moveMouseTo(170, 288);
await sleep(300); await sleep(500);
click(48, 441);//放大地图 moveMouseTo(104, 1000);
await sleep(300); await sleep(500);
leftButtonUp();
//地图拖动到指定位置 await sleep(500);
moveMouseTo(200,200 );
leftButtonDown(); await sleep(500);
moveMouseTo(170,288 );await sleep(500);
moveMouseTo(104,1000 );await sleep(500);
leftButtonUp();await sleep(500);
// 获取游戏区域截图 // 获取游戏区域截图
const captureRegion = captureGameRegion(); const captureRegion = captureGameRegion();
for (const point of checkPoints) { for (const point of checkPoints) {
i++; i++;
// 遍历所有检测点位 // 遍历所有检测点位
const cropRegion = captureRegion.DeriveCrop( const cropRegion = captureRegion.DeriveCrop(point.x, point.y, 160, 160);
point.x ,
point.y ,
160,
160
);
// 在裁剪区域中查找卡片 // 在裁剪区域中查找卡片
const result = cropRegion.Find(cardPlayerRo); const result = cropRegion.Find(cardPlayerRo);
// 如果找到卡片 // 如果找到卡片
if (!result.IsEmpty()) { if (!result.IsEmpty()) {
findNum++; findNum++;
if (findNum - skipNum == 1) { if (findNum - skipNum == 1) {
log.info(`在点位${i}找到玩家,执行对应操作`); log.info(`在点位${i}找到玩家,执行对应操作`);
await sleep(1000); await sleep(1000);
keyPress("ESCAPE"); keyPress("ESCAPE");
await sleep(1500); await sleep(1500);
await point.action(); // 调用该点位对应的函数 await point.action(); // 调用该点位对应的函数
return true; // 返回true表示已找到并处理 return true; // 返回true表示已找到并处理
} }
} }
} }
// 所有点位都未找到 // 所有点位都未找到
log.info("未在任何检测点找到玩家"); log.info("未在任何检测点找到玩家");
textArray.length = 0; textArray.length = 0;
return false; return false;
} };
//获取挑战对象名称 //获取挑战对象名称
async function captureAndStoreTexts() { async function captureAndStoreTexts() {
// 清空数组 // 清空数组
textArray = []; textArray = [];
// 四个固定位置坐标 // 四个固定位置坐标
const positions = [ const positions = [
{x: 450, y: 620}, { x: 450, y: 620 },
{x: 760, y: 620}, { x: 760, y: 620 },
{x: 1070, y: 620}, { x: 1070, y: 620 },
{x: 1380, y: 620} { x: 1380, y: 620 },
]; ];
// 截取区域大小 // 截取区域大小
const width = 240; const width = 240;
const height = 100; const height = 100;
await sleep(500); await sleep(500);
keyPress("F6") keyPress("F6");
await sleep(1000); await sleep(1000);
click(300, 370);//点击七日历练 click(300, 370); //点击七日历练
await sleep(1000); await sleep(1000);
// 获取游戏区域截图 // 获取游戏区域截图
const captureRegion = captureGameRegion(); const captureRegion = captureGameRegion();
// 遍历四个位置进行OCR识别 // 遍历四个位置进行OCR识别
for (const pos of positions) { for (const pos of positions) {
// 创建OCR识别区域 // 创建OCR识别区域
const ocrRo = RecognitionObject.ocr(pos.x, pos.y, width, height);//挑战者名字区域 const ocrRo = RecognitionObject.ocr(pos.x, pos.y, width, height); //挑战者名字区域
const ocrRo2 = RecognitionObject.ocr(pos.x, pos.y+100, width, height);//挑战是否完成 const ocrRo2 = RecognitionObject.ocr(pos.x, pos.y + 100, width, height); //挑战是否完成
// 在指定区域进行OCR识别 // 在指定区域进行OCR识别
const result = captureRegion.find(ocrRo); const result = captureRegion.find(ocrRo);
const result2 = captureRegion.find(ocrRo2); const result2 = captureRegion.find(ocrRo2);
if (!result.isEmpty() && result.text) { if (!result.isEmpty() && result.text) {
// 存储识别结果和对应位置 // 存储识别结果和对应位置
if(result2.text == "追踪"){ if (result2.text == "追踪") {
log.info(`识别到文本: ${result.text} 位置: (${pos.x}, ${pos.y})`); log.info(`识别到文本: ${result.text} 位置: (${pos.x}, ${pos.y})`);
textArray.push({ textArray.push({
text: result.text.trim(), text: result.text.trim(),
x: pos.x + width / 2, // 点击中心位置 x: pos.x + width / 2, // 点击中心位置
y: pos.y + height / 2 y: pos.y + height / 2,
});} });
}
} else { } else {
log.warn(`位置 (${pos.x}, ${pos.y}) 未识别到文本`); log.warn(`位置 (${pos.x}, ${pos.y}) 未识别到文本`);
} }
} }
log.info(`剩余挑战人数:${textArray.length}`); log.info(`剩余挑战人数:${textArray.length}`);
keyPress("ESCAPE"); keyPress("ESCAPE");
await sleep(1000); await sleep(1000);
} }
//检查是否有对应的挑战对手 //检查是否有对应的挑战对手
@@ -230,38 +252,38 @@ async function searchAndClickTexts() {
const searchY = 440; const searchY = 440;
const searchWidth = 150; const searchWidth = 150;
const searchHeight = 195; const searchHeight = 195;
// 获取游戏区域截图 // 获取游戏区域截图
const captureRegion = captureGameRegion(); const captureRegion = captureGameRegion();
// 在限定区域内进行OCR识别 // 在限定区域内进行OCR识别
const ocrRo = RecognitionObject.ocr(searchX, searchY, searchWidth, searchHeight); const ocrRo = RecognitionObject.ocr(searchX, searchY, searchWidth, searchHeight);
const results = captureRegion.findMulti(ocrRo); const results = captureRegion.findMulti(ocrRo);
// 遍历OCR结果 // 遍历OCR结果
for (let i = 0; i < results.count; i++) { for (let i = 0; i < results.count; i++) {
const res = results[i]; const res = results[i];
const resText = res.text.trim(); const resText = res.text.trim();
// 在存储的文本数组中查找匹配项 // 在存储的文本数组中查找匹配项
const index = textArray.findIndex(item => item.text === resText); const index = textArray.findIndex((item) => item.text === resText);
if (index !== -1) { if (index !== -1) {
// 找到匹配项,点击对应位置 // 找到匹配项,点击对应位置
log.info(`找到匹配文本: ${resText}`); log.info(`找到匹配文本: ${resText}`);
skipNum = 0; skipNum = 0;
// 点击存储的位置 // 点击存储的位置
await keyMouseScript.runFile(`assets/ALT点击.json`); await keyMouseScript.runFile(`assets/ALT点击.json`);
await sleep(500); await sleep(500);
res.click(); res.click();
await sleep(500); await sleep(500);
await keyMouseScript.runFile(`assets/ALT释放.json`); await keyMouseScript.runFile(`assets/ALT释放.json`);
await Playcards(); await Playcards();
// 从数组中移除已处理的文本 // 从数组中移除已处理的文本
textArray.splice(index, 1); textArray.splice(index, 1);
return true; return true;
} }
} }
@@ -271,207 +293,209 @@ async function searchAndClickTexts() {
} }
//函数:打开地图前往猫尾酒馆 //函数:打开地图前往猫尾酒馆
async function gotoTavern() { async function gotoTavern() {
const tavernRo = RecognitionObject.TemplateMatch(file.ReadImageMatSync("assets/tavern.png")); const tavernRo = RecognitionObject.TemplateMatch(file.ReadImageMatSync("assets/tavern.png"));
await genshin.returnMainUi(); await genshin.returnMainUi();
await sleep(1000); await sleep(1000);
keyPress("m"); keyPress("m");
await sleep(1500); await sleep(1500);
click(1841, 1015);//地图选择 click(1841, 1015); //地图选择
await sleep(1000); await sleep(1000);
click(1460, 140);//蒙德 click(1460, 140); //蒙德
await sleep(1200); await sleep(1200);
click(48, 441);//放大地图 click(48, 441); //放大地图
await sleep(400); await sleep(400);
click(48, 441);//放大地图 click(48, 441); //放大地图
await sleep(400); await sleep(400);
click(48, 441);//放大地图 click(48, 441); //放大地图
await sleep(400); await sleep(400);
click(48, 441);//放大地图 click(48, 441); //放大地图
await sleep(400); await sleep(400);
click(48, 441);//放大地图 click(48, 441); //放大地图
await sleep(400); await sleep(400);
click(1000, 645);//猫尾酒馆
await sleep(600); click(1000, 645); //猫尾酒馆
let tavern = captureGameRegion().find(tavernRo); await sleep(600);
if (tavern.isExist()) { let tavern = captureGameRegion().find(tavernRo);
if (tavern.isExist()) {
tavern.click(); tavern.click();
await sleep(500); await sleep(500);
} } else {
else{ throw new Error("未能找到猫尾酒馆");
throw new Error('未能找到猫尾酒馆');
}
click(1707, 1010);//确认传送
await sleep(1000);
await tpEndDetection();
} }
click(1707, 1010); //确认传送
await sleep(1000);
await tpEndDetection();
}
//函数:对话和打牌 //函数:对话和打牌
async function Playcards() { async function Playcards() {
await sleep(800);//略微俯视,避免名字出现在选项框附近,导致错误点击 await sleep(800); //略微俯视,避免名字出现在选项框附近,导致错误点击
moveMouseBy(0, 1030); moveMouseBy(0, 1030);
await sleep(1000); await sleep(1000);
await autoConversation(); await autoConversation();
log.info("对话完成"); log.info("对话完成");
await sleep(1500); await sleep(1500);
click(1610,900 );//点击挑战 if (settings.partyName != undefined) {
await sleep(8000); await switchCardTeam(settings.partyName);
await dispatcher.runTask(new SoloTask("AutoGeniusInvokation"));
await sleep(3000);
await checkChallengeResults();
await sleep(1000);
} }
click(1610, 900); //点击挑战
await sleep(8000);
await dispatcher.runTask(new SoloTask("AutoGeniusInvokation"));
await sleep(3000);
await checkChallengeResults();
await sleep(1000);
}
//前往一号桌 //前往一号桌
async function gotoTable1() { async function gotoTable1() {
log.info(`前往1号桌`); log.info(`前往1号桌`);
keyDown("d"); keyDown("d");
await sleep(1500); await sleep(1500);
keyUp("d"); keyUp("d");
keyDown("w"); keyDown("w");
await sleep(400); await sleep(400);
keyUp("w"); keyUp("w");
keyDown("d"); keyDown("d");
keyDown("w"); keyDown("w");
await sleep(1200); await sleep(1200);
keyUp("d"); keyUp("d");
keyUp("w"); keyUp("w");
await sleep(700); await sleep(700);
} }
//前往二号桌 //前往二号桌
async function gotoTable2() { async function gotoTable2() {
log.info(`前往2号桌`); log.info(`前往2号桌`);
keyDown("d"); keyDown("d");
await sleep(1500); await sleep(1500);
keyUp("d"); keyUp("d");
keyDown("w"); keyDown("w");
await sleep(400); await sleep(400);
keyUp("w"); keyUp("w");
keyDown("d"); keyDown("d");
keyDown("w"); keyDown("w");
await sleep(1200); await sleep(1200);
keyUp("d"); keyUp("d");
keyUp("w"); keyUp("w");
keyDown("s"); keyDown("s");
await sleep(700); await sleep(700);
keyUp("s"); keyUp("s");
await sleep(700); await sleep(700);
} }
//前往三号桌 //前往三号桌
async function gotoTable3() { async function gotoTable3() {
log.info(`前往3号桌`); log.info(`前往3号桌`);
keyDown("w"); keyDown("w");
await sleep(2000); await sleep(2000);
keyUp("w"); keyUp("w");
keyDown("d"); keyDown("d");
await sleep(5000); await sleep(5000);
keyUp("d"); keyUp("d");
keyDown("a"); keyDown("a");
await sleep(1500); await sleep(1500);
keyUp("a"); keyUp("a");
await sleep(700); await sleep(700);
} }
//前往四号桌 //前往四号桌
async function gotoTable4() { async function gotoTable4() {
log.info(`前往4号桌`); log.info(`前往4号桌`);
keyDown("w"); keyDown("w");
await sleep(2000); await sleep(2000);
keyUp("w"); keyUp("w");
keyDown("d"); keyDown("d");
await sleep(5000); await sleep(5000);
keyUp("d"); keyUp("d");
keyDown("a"); keyDown("a");
await sleep(1500); await sleep(1500);
keyUp("a"); keyUp("a");
keyDown("d"); keyDown("d");
await sleep(200); await sleep(200);
keyUp("d"); keyUp("d");
keyDown("w"); keyDown("w");
await sleep(2000); await sleep(2000);
keyUp("w"); keyUp("w");
await sleep(700); await sleep(700);
} }
//前往一号包间 //前往一号包间
async function gotoTable5() { async function gotoTable5() {
log.info(`前往1号包间`); log.info(`前往1号包间`);
keyDown("w"); keyDown("w");
await sleep(2500); await sleep(2500);
keyUp("w"); keyUp("w");
keyDown("d"); keyDown("d");
await sleep(200); await sleep(200);
keyUp("d"); keyUp("d");
await sleep(500); await sleep(500);
keyPress("ESCAPE"); keyPress("ESCAPE");
await sleep(1500); await sleep(1500);
keyPress("ESCAPE"); keyPress("ESCAPE");
await sleep(1500); await sleep(1500);
keyDown("w"); keyDown("w");
await sleep(5900); await sleep(5900);
keyUp("w"); keyUp("w");
await sleep(700); await sleep(700);
} }
//前往二号包间 //前往二号包间
async function gotoTable6() { async function gotoTable6() {
log.info(`前往2号包间`); log.info(`前往2号包间`);
await sleep(1500); await sleep(1500);
keyDown("d"); keyDown("d");
await sleep(1500); await sleep(1500);
keyUp("d"); keyUp("d");
keyDown("w"); keyDown("w");
keyDown("d"); keyDown("d");
await sleep(4000); await sleep(4000);
keyUp("d"); keyUp("d");
keyUp("w"); keyUp("w");
keyDown("a"); keyDown("a");
await sleep(1500); await sleep(1500);
keyUp("a"); keyUp("a");
keyDown("w"); keyDown("w");
await sleep(3000); await sleep(3000);
keyPress("VK_SPACE"); keyPress("VK_SPACE");
await sleep(1000); await sleep(1000);
keyUp("w"); keyUp("w");
keyDown("s"); keyDown("s");
await sleep(1000); await sleep(1000);
keyPress("VK_SPACE"); keyPress("VK_SPACE");
await sleep(700); await sleep(700);
keyUp("s"); keyUp("s");
await sleep(500); await sleep(500);
}
}
async function openMap() {
await gotoTavern();
keyDown("w");
await sleep(2000);
keyUp("w");
keyDown("d");
await sleep(5000);
keyUp("d");
await sleep(700);
}
async function openMap() {
await gotoTavern();
keyDown("w");
await sleep(2000);
keyUp("w");
keyDown("d");
await sleep(5000);
keyUp("d");
await sleep(700);
}
//主流程 //主流程
log.info(`前往猫尾酒馆`); log.info(`前往猫尾酒馆`);
await gotoTavern(); await gotoTavern();
await captureAndStoreTexts(); await captureAndStoreTexts();
if (textArray.length != 0){ if (textArray.length != 0) {
await detectCardPlayer(); await detectCardPlayer();
await searchAndClickTexts(); await searchAndClickTexts();
}
for (let i = 0; i < 20; i++) {
//循环兜底,避免角色未到达指定位置
if (textArray.length === 0) break;
await gotoTavern();
await detectCardPlayer();
await searchAndClickTexts();
} }
for (let i = 0;i < 20; i++) {//循环兜底,避免角色未到达指定位置
if (textArray.length === 0) break;
await gotoTavern();
await detectCardPlayer();
await searchAndClickTexts();
}
await genshin.returnMainUi(); await genshin.returnMainUi();
await sleep(500); await sleep(500);
keyPress("F6") keyPress("F6");
await sleep(1000); await sleep(1000);
click(300, 370);//点击七日历练 click(300, 370); //点击七日历练
await sleep(1000); await sleep(1000);
log.info(`打牌结束`); log.info(`打牌结束`);
})(); })();

View File

@@ -1,7 +1,7 @@
{ {
"manifest_version": 1, "manifest_version": 1,
"name": "打牌一条龙", "name": "打牌一条龙",
"version": "1.81", "version": "1.9",
"description": "只靠一个牌组的话,胜率还是太低了,所以有没有既简单又强势的卡组推荐下呢?", "description": "只靠一个牌组的话,胜率还是太低了,所以有没有既简单又强势的卡组推荐下呢?",
"authors": [ "authors": [
{ {

View File

@@ -0,0 +1,7 @@
[
{
"name": "partyName",
"type": "input-text",
"label": "设置要使用的七圣召唤队伍名称"
}
]