Commit 735cfa03 authored by nanahira's avatar nanahira

Merge branch 'master' of git.mycard.moe:nanahira/windbot into no-errmsg

parents 21a209f2 5849f1dd
......@@ -46,9 +46,7 @@ upload_to_minio:
- build
tags:
- linux
image: python
script:
- pip install -U -i https://mirrors.aliyun.com/pypi/simple/ awscli
- aws s3 --endpoint=https://minio.mycard.moe:9000 sync dist/ s3://mycard/windbot
only:
- master
......
......@@ -26,7 +26,7 @@ SUPPORT_MASTER_RULE_2020
!P2-自选卡组
Name=P2 Deck=Lucky Dialog=gugugu.zh-CN
人机卡组由你选择。随缘出牌。可将你的卡组文件(.ydk)复制到WindBot的对应文件夹。
人机卡组由你选择。随缘出牌。
SELECT_DECKFILE SUPPORT_MASTER_RULE_3 SUPPORT_NEW_MASTER_RULE SUPPORT_MASTER_RULE_2020
!谜之剑士LV4-龙骑星爆
......@@ -209,7 +209,7 @@ Name=朱露咲浅羽 Deck=Salamangreat Dialog=nlch.zh-CN
浅羽的香喷喷曲奇。
AI_LV3 SUPPORT_NEW_MASTER_RULE SUPPORT_MASTER_RULE_2020
!朱露咲浅羽-真红眼
!朱露咲浅羽-真红眼龙骑士
Name=朱露咲浅羽 Deck=Dragun Dialog=nlch.zh-CN
超魔导真红眼龙骑士卡组。
AI_LV3 SUPPORT_MASTER_RULE_3 SUPPORT_NEW_MASTER_RULE SUPPORT_MASTER_RULE_2020
......@@ -228,3 +228,8 @@ AI_ANTI_META SUPPORT_MASTER_RULE_3 SUPPORT_NEW_MASTER_RULE SUPPORT_MASTER_RULE_2
Name=璃璃子 Deck=TimeThief Dialog=default
时间潜行者卡组。
AI_LV2 SUPPORT_MASTER_RULE_3 SUPPORT_NEW_MASTER_RULE SUPPORT_MASTER_RULE_2020
!濑名歌铃
Name=濑名歌铃 Deck=Brave Dialog=wof-Sena-Karin
????? 卡组。
AI_LV3 SUPPORT_MASTER_RULE_2020
#created by ...
#main
2563463
81866673
72090076
63362460
30680659
30680659
30680659
26202165
26202165
26202165
91646304
14558127
14558127
14558127
72291078
72291078
23434538
23434538
9742784
97268402
3285551
3285551
3285551
18144506
52947044
52947044
52947044
81439173
83764718
24224830
24224830
65681983
65681983
39568067
38745520
10045474
10045474
10045474
40605147
40605147
#extra
15291624
60461804
84815190
92519087
27548199
42566602
90953320
98558751
21915012
44097050
50588353
70369116
98978921
98978921
60303245
!side
55063751
55063751
55063751
34267821
34267821
34267821
43534808
43534808
43534808
94145021
94145021
94145021
14532163
14532163
14532163
......@@ -12,7 +12,7 @@
"接下来就是我的舞台"
],
"endturn": [
"我的回合结束,试试来击溃我吧"
"每个人的目的不同,我更注重强度,我只想赢。"
],
"directattack": [
"老子上了,要你的命"
......
......@@ -22,7 +22,7 @@
"directattack": [
"{0},直接攻击!",
"不过如此嘛,直接攻击!",
"快走开,我要回去睡觉了。",
"快走开,我要回去睡觉了。"
],
"attack": [
"{0},攻击{1}!",
......
{
"welcome": [
"来来来,一起来娱乐一下啊!"
"新的男朋友候选人?!来来来一起来玩啊!"
],
"deckerror": [
"没有{0}就很不娱乐。"
"要做我的男朋友怎么能不带好{0}呢?"
],
"duelstart": [
"娱乐而已,放轻松就好。"
"之前都是唬你的,现在请做好被甩的心理准备。"
],
"newturn": [
"接下来就是我的舞台咯。",
......@@ -14,34 +14,34 @@
],
"endturn": [
"OK,该你了。",
"毕竟是娱乐,这布阵应该对你来说很轻松。"
"这道题应该对你来说很轻松。"
],
"directattack": [
"{0}直接攻击对手!",
"{0}拿你那个球去撞对手啊!",
"{0}冲鸭!",
"一个巴掌拍得响!"
],
"attack": [
"{0}日了那个{1}!",
"{0}去和{1}娱乐娱乐。",
"这个{1}看起来就不娱乐啊……"
"{0}去和{1}比比长度。",
"这个{1}看起来就太短了……"
],
"ondirectattack": [
"哎呀!",
"不是说好了娱乐一下就行吗?!"
"不是说好了只吃饭的吗?!"
],
"facedownmonstername": "怪兽",
"activate": [
"我发动{0}的效果,虽然卡图上面没写,但cdb里说了就是可以这样用。",
"所以{0}什么时候进去啊?"
"我发动{0}的效果,虽然卡图上面没写,但我老婆说了就是可以这样用。",
"所以{0}什么时候涨价涨到8700万啊?"
],
"summon": [
"通召{0},下略。",
"{0}也快进去了吧……"
"{0},下略。",
"{0}最近可是稀有货啊……"
],
"setmonster": [
"日常卡手。",
"让你一把。"
"多喝热水。"
],
"chaining": [
"千防万防防不住{0}。",
......
{
"welcome": [
"早上好……啊?现在不是早上?"
],
"deckerror": [
"嘶……我醒了,{0}没了,我睡了。"
],
"duelstart": [
"我反正无所谓,你喜欢就行。",
"嘛,无所谓了。"
],
"newturn": [
"啊,是我的回合……",
"抽一张卡。"
],
"endturn": [
"睡了睡了,到我了记得叫醒我。",
"来来来,请开始你的表演。"
],
"directattack": [
"用{0}进行攻击!",
"还行,就拿{0}攻击吧。",
"尝尝这个{0}!"
],
"attack": [
"{1},挨下{0}这一拳吧!",
"胜利属于{0}!",
"{1}给我下去!"
],
"ondirectattack": [
"唔……",
"呕啊……!",
"(摆出Daisuke的姿势挡脸)"
],
"facedownmonstername": "怪兽",
"activate": [
"一花前辈用过的{0}就是好啊……",
"{0}……似乎能行。"
],
"summon": [
"啊,是没见过的孩子{0}呢。",
"{0}:糖歌这么体弱多病就别打牌了……"
],
"setmonster": [
"嗯……"
],
"chaining": [
"发动{0}的效果进行连锁。",
"在你做那件事之前我先发动{0}!",
"对3?王炸!"
]
}
\ No newline at end of file
{
"welcome": [
"又来啦?还是来找美妙子的吗?"
"又来啦?还是来……办证的?"
],
"deckerror": [
"美妙子现在不在,改天再来吧。"
"你要办的证的负责人现在不在,改天再来吧。"
],
"duelstart": [
"既然这样的话,那就先过我这关!",
"美妙子就在里面,不过你得先打赢我才能去见她。"
"负责人就在里面,不过你得先打赢我才能去见她。"
],
"newturn": [
"到我了吗?那就抽牌。",
......@@ -27,6 +27,7 @@
],
"ondirectattack": [
"习惯了。",
"挫。",
"一般。"
],
"facedownmonstername": "怪兽",
......@@ -35,7 +36,7 @@
"啊等下,使用{0}的效果。"
],
"summon": [
"召唤{0}。",
"厚礼{0}。",
"总之,把{0}拉出来就对了。",
"让我召唤{0}。"
],
......
{
"welcome":[
"早晨一杯热牛奶……嗯?我的热牛奶呢?"
],
"deckerror":[
"昨天谁进我房间里把{0}拿走了?"
],
"duelstart":[
"把你那萝莉控的心脏收起来!",
"眼睛看哪儿呢?!"
],
"newturn":[
"抓一只卡。"
],
"endturn":[
"这位同学,请开始你的表演。",
"来点海龟或者翼神龙蛋?"
],
"directattack":[
"用{0}进行攻击!",
"还行,就拿{0}攻击吧。",
"尝尝这个{0}!"
],
"attack":[
"{1},挨下{0}这一拳吧!",
"胜利属于{0}!",
"{1}给我下地狱去!"
],
"ondirectattack":[
"……啊…………",
"喂!",
"不能欺负小孩子啊!"
],
"facedownmonstername":"怪兽",
"activate":[
"{0}……似乎能行。",
"来点无敌{0}。"
],
"summon":[
"啊,是没见过的孩子{0}呢。",
"我放出了一条{0}!"
],
"setmonster":[
"嗯……"
],
"chaining":[
"发动{0}的效果进行连锁。",
"在你做那件事之前我先发动{0}!",
"17张牌你能秒我?!"
]
}
\ No newline at end of file
{
"welcome": [
"久 等 了"
],
"deckerror": [
"噔 噔 咚"
],
"duelstart": [
"让我这个大姐姐来传授给你一点人生经验。"
],
"newturn": [
"……抽牌。",
"到我了吗?那就抽牌。"
],
"endturn": [
"You Jump I Jump。",
"来点上升气流。"
],
"directattack": [
"小老弟还能站起来吗?",
"完 全 胜 利",
"{0}扭的更用力了!"
],
"attack": [
"{0},先给这个{1}一点颜色看看!",
"{1} 无 惨",
"大事不好了!{1}的牛牛炸了!"
],
"ondirectattack": [
"B A T T L E F I E L D 5",
"你特么{0}劈我是吧!",
"……果然要考虑命运石之门的抉择啊……"
],
"facedownmonstername": "怪兽",
"activate": [
"这个{0}怎么样?",
"嗯……{0}……",
"{0}{0}{0}{0}!"
],
"summon": [
"火 {0} 火",
"酸 {0} 酸",
"爱 {0} 爱"
],
"setmonster": [
"先随便盖只鸡看看……"
],
"chaining": [
"急什么啊?",
"小老弟对{0}这么毫无防备的吗?",
"这是发动{0}的超稀有机会。"
]
}
\ No newline at end of file
{
"welcome": [
"嘶……似乎来了个极其麻烦的家伙啊……"
],
"deckerror": [
"艾蕾娜?{0}的库存看看是不是空了?"
],
"duelstart": [
"今天不是你付钱就是我没货。",
"从前有座山,山里有座庙,庙里有个老和尚……"
],
"newturn": [
"纯度太低了。",
"你这肥猪。"
],
"endturn": [
"撒,细数你的王牌吧!",
"开始你的行动吧,我无所谓!"
],
"directattack": [
"{0},再斩一刀!",
"我不想说的太失礼,你还是放弃吧。",
"你也不瞧瞧你那AC样……"
],
"attack": [
"你的{1}太软了。",
"暑期促销买{1}送{0}是吧……",
"如此{1},万不可长,必须制止。"
],
"ondirectattack": [
"必可活用于下一次……",
"这杀马特的{0}……",
"……没有爆衣可以看的!"
],
"facedownmonstername": "怪兽",
"activate": [
"{0},我的超人……",
"至福……",
"十八般武艺,此乃{0}。"
],
"summon": [
"这是便宜{0}。",
"{0}很大,你忍一下。",
"说得好,{0},谢谢。"
],
"setmonster": [
"法克。"
],
"chaining": [
"小老弟,根本不可能。",
"如果你非要做如此蠢事,就别用这种脑瘫办法。",
"本来想拒绝,但{0}实在是太强力了。"
]
}
\ No newline at end of file
......@@ -19,7 +19,7 @@
"directattack": [
"{0}的直接攻击!",
"畏惧{0}的力量吧!",
"已经无法阻止{0}了!"
"人类的力量已经无法阻止{0}了!"
],
"attack": [
"{0},打倒{1}!",
......
{
"welcome": [
"又迟到了……下次注意啊。"
"嗷嗷嗷!"
],
"deckerror": [
"啊,似乎忘记带{0}了……"
],
"duelstart": [
"上课咯!",
"我可是作为一条狗在跟你打牌。",
"就让我来教教你什么叫打牌。"
],
"newturn": [
......@@ -14,7 +14,7 @@
"总之先抽卡。"
],
"endturn": [
"差不多了……那就先休息一下喝杯茶吧。",
"差不多了……那就先休息一下吃个骨头吧。",
"下面轮到你了哦。"
],
"directattack": [
......@@ -31,7 +31,7 @@
"ondirectattack": [
"啊啊啊啊啊啊啊啊啊!!!",
"哎呀!痛痛痛痛痛痛痛痛……",
"停一下停一下我还没准备好啊!"
"杀狗啦!!!!!!"
],
"facedownmonstername": "怪兽",
"activate": [
......@@ -45,7 +45,7 @@
"出现吧,{0}!"
],
"setmonster": [
"我放置了一只怪兽。"
"一对情侣一条狗……"
],
"chaining": [
"等一下!我发动{0}!",
......
......@@ -19,7 +19,7 @@
],
"directattack": [
"{0},安心去吧!",
"去去去去去去{0}别挡路!",
"大王叫{0}来巡山啊!",
"Star Lock Explosion!"
],
"attack": [
......
{
"welcome": [
"这样啊……那好吧。"
"这样啊……那随你便了。"
],
"deckerror": [
"嗯?我的{0}呢?"
"嗯?我的{0}呢?被猪吃了?"
],
"duelstart": [
"考虑要不要给对面一点面子……"
"考虑要不要给对面一点面子……算了,懒得给。"
],
"newturn": [
"……抽牌。",
......
{
"welcome": [
"所为何事,速速道来。"
],
"deckerror": [
"啊,{0}又用不出来了。"
],
"duelstart": [
"我是不会手下留情的,请拿出你最佳的状态和实力。"
],
"newturn": [
"……抽牌。",
"到我了吗?那就抽牌。"
],
"endturn": [
"我要做的都已经完成了,下面就看你的了。",
"我已经没什么可做的了,换你来吧。"
],
"directattack": [
"还能站起来吗?",
"不知道能挨得住{0}这一下吗?",
"总之{0}直接一拳就对了。"
],
"attack": [
"{0},先给这个{1}一点颜色看看吧。",
"这个{1}在{0}面前实际上十分脆弱啊……",
"嗯……总之先让这个{1}消失。"
],
"ondirectattack": [
"这点小伤,不算什么。",
"只是{0}而已,还不足以置我于死地。",
"……我还能站起来!"
],
"facedownmonstername": "怪兽",
"activate": [
"先用这个{0}的能力看看。",
"嗯……{0}的能力应该管用。",
"总之,{0}的能力应该可以应付得了。"
],
"summon": [
"嗯……现在应该先出这个,{0}。",
"嘛……这种情况{0}应该能应付得过来。",
"{0}也许会迟到,但绝不会缺席。"
],
"setmonster": [
"先随便盖个东西看看……",
"总之先加强防御。"
],
"chaining": [
"别急,先让我把{0}发动了再说。",
"对{0}这么毫无防备的吗?",
"这是发动{0}的绝佳机会。"
]
}
\ No newline at end of file
using System;
using YGOSharp.OCGWrapper.Enums;
using System.Collections.Generic;
using WindBot;
using WindBot.Game;
using WindBot.Game.AI;
using System.Linq;
namespace WindBot.Game.AI.Decks
{
[Deck("Brave", "AI_Brave")]
class BraveExecutor : DefaultExecutor
{
public class CardId
{
public const int WanderingGryphonRider = 2563463;
public const int DestinyHeroDasher = 81866673;
public const int NemesesCorridor = 72090076;
public const int DestinyHeroCelestial = 63362460;
public const int AquamancerOfTheSanctuary = 30680659;
public const int Sangan = 26202165;
public const int CrusadiaArboria = 91646304;
public const int AshBlossomJoyousSpring = 14558127;
public const int MechaPhantomBeastOLion = 72291078;
public const int MaxxC = 23434538;
public const int JetSynchron = 9742784;
public const int EffectVeiler = 97268402;
public const int RiteofAramesia = 3285551;
public const int HarpiesFeatherDuster = 18144506;
public const int FusionDestiny = 52947044;
public const int FoolishBurial = 81439173;
public const int MonsterReborn = 83764718;
public const int CalledByTheGrave = 24224830;
public const int CrossoutDesignator = 65681983;
public const int JourneyOfDestiny = 39568067;
public const int DracobackTheDragonSteed = 38745520;
public const int InfiniteImpermanence = 10045474;
public const int SolemnStrike = 40605147;
public const int ThunderDragonColossus = 15291624;
public const int DestinyHeroDestroyPhoenixEnforcer = 60461804;
public const int BaronessDeFleur = 84815190;
public const int VirtualWorldKyubiShenshen = 92519087;
public const int BorreloadSavageDragon = 27548199;
public const int CoralDragon = 42566602;
public const int TGHyperLibrarian = 90953320;
public const int TGWonderMagician = 98558751;
public const int CupidPitch = 21915012;
public const int MechaPhantomBeastAuroradon = 44097050;
public const int CrystronHalqifibrax = 50588353;
public const int PredaplantVerteAnaconda = 70369116;
public const int LinkSpider = 98978921;
public const int SalamangreatAlmiraj = 60303245;
public const int BraveToken = 3285552;
public const int MechaPhantomBeastToken = 31533705;
public const int PrimalBeingToken = 27204312;
}
public BraveExecutor(GameAI ai, Duel duel)
: base(ai, duel)
{
AddExecutor(ExecutorType.Activate, CardId.WanderingGryphonRider, WanderingGryphonRiderCounter);
AddExecutor(ExecutorType.Activate, CardId.AshBlossomJoyousSpring, DefaultAshBlossomAndJoyousSpring);
AddExecutor(ExecutorType.Activate, CardId.CalledByTheGrave, DefaultCalledByTheGrave);
AddExecutor(ExecutorType.Activate, CardId.CrossoutDesignator, CrossoutDesignatorEffect);
AddExecutor(ExecutorType.Activate, CardId.EffectVeiler, DefaultEffectVeiler);
AddExecutor(ExecutorType.Activate, CardId.InfiniteImpermanence, DefaultInfiniteImpermanence);
AddExecutor(ExecutorType.Activate, CardId.BorreloadSavageDragon, BorreloadSavageDragonEffect);
AddExecutor(ExecutorType.Activate, CardId.BaronessDeFleur, BaronessDeFleurEffect);
AddExecutor(ExecutorType.Activate, CardId.SolemnStrike, DefaultSolemnStrike);
AddExecutor(ExecutorType.Activate, CardId.MaxxC, DefaultMaxxC);
AddExecutor(ExecutorType.Activate, CardId.HarpiesFeatherDuster);
AddExecutor(ExecutorType.Activate, CardId.ThunderDragonColossus);
AddExecutor(ExecutorType.Activate, CardId.DracobackTheDragonSteed, DracobackTheDragonSteedBounce);
AddExecutor(ExecutorType.Activate, CardId.DestinyHeroDestroyPhoenixEnforcer, DestinyHeroDestroyPhoenixEnforcerEffect);
AddExecutor(ExecutorType.Activate, CardId.RiteofAramesia, RiteofAramesiaEffect);
AddExecutor(ExecutorType.Activate, CardId.AquamancerOfTheSanctuary, AquamancerOfTheSanctuarySearchEffect);
AddExecutor(ExecutorType.Activate, CardId.JourneyOfDestiny, JourneyOfDestinyActivate);
AddExecutor(ExecutorType.Activate, CardId.WanderingGryphonRider, WanderingGryphonRiderSummon);
AddExecutor(ExecutorType.Activate, CardId.FoolishBurial, FoolishBurialFirst);
AddExecutor(ExecutorType.Summon, CardId.Sangan);
AddExecutor(ExecutorType.Summon, CardId.MechaPhantomBeastOLion);
AddExecutor(ExecutorType.Activate, CardId.MechaPhantomBeastOLion, MechaPhantomBeastOLionEffect);
AddExecutor(ExecutorType.Summon, CardId.JetSynchron);
AddExecutor(ExecutorType.Activate, CardId.JourneyOfDestiny, JourneyOfDestinyEffect);
AddExecutor(ExecutorType.SpSummon, CardId.SalamangreatAlmiraj, SalamangreatAlmirajSummonFirst);
AddExecutor(ExecutorType.SpSummon, CardId.CrusadiaArboria, CrusadiaArboriaSummon);
AddExecutor(ExecutorType.SpSummon, CardId.CrystronHalqifibrax, CrystronNeedlefiberSummon);
AddExecutor(ExecutorType.Activate, CardId.CrystronHalqifibrax, CrystronNeedlefiberEffect);
AddExecutor(ExecutorType.SpSummon, CardId.MechaPhantomBeastAuroradon, MechaPhantomBeastAuroradonSummon);
AddExecutor(ExecutorType.Activate, CardId.MechaPhantomBeastAuroradon, MechaPhantomBeastAuroradonEffect);
AddExecutor(ExecutorType.Activate, CardId.AquamancerOfTheSanctuary, AquamancerOfTheSanctuarySummonEffect);
AddExecutor(ExecutorType.SpSummon, CardId.TGHyperLibrarian, TGHyperLibrarianSummon);
AddExecutor(ExecutorType.Activate, CardId.TGHyperLibrarian);
AddExecutor(ExecutorType.SpSummon, CardId.CupidPitch, CupidPitchSummon);
AddExecutor(ExecutorType.Activate, CardId.CupidPitch, CupidPitchEffect);
AddExecutor(ExecutorType.Activate, CardId.JetSynchron, JetSynchronEffect);
AddExecutor(ExecutorType.SpSummon, CardId.SalamangreatAlmiraj, SalamangreatAlmirajSummon);
AddExecutor(ExecutorType.SpSummon, CardId.LinkSpider, LinkSpiderSummon);
AddExecutor(ExecutorType.Activate, CardId.DracobackTheDragonSteed, DracobackTheDragonSteedEquip);
AddExecutor(ExecutorType.SpSummon, CardId.BorreloadSavageDragon, BorreloadSavageDragonSummon);
AddExecutor(ExecutorType.Activate, CardId.NemesesCorridor);
AddExecutor(ExecutorType.SpSummon, CardId.ThunderDragonColossus);
AddExecutor(ExecutorType.Summon, CardId.CrusadiaArboria, SummonForMaterial);
AddExecutor(ExecutorType.Summon, CardId.AshBlossomJoyousSpring, SummonForMaterial);
AddExecutor(ExecutorType.Summon, CardId.EffectVeiler, SummonForMaterial);
AddExecutor(ExecutorType.Summon, CardId.AquamancerOfTheSanctuary, SummonForMaterial);
AddExecutor(ExecutorType.Summon, CardId.MaxxC, SummonForMaterial);
AddExecutor(ExecutorType.Activate, CardId.FoolishBurial, FoolishBurialEffect);
AddExecutor(ExecutorType.Activate, CardId.MonsterReborn, MonsterRebornEffect);
AddExecutor(ExecutorType.SpSummon, CardId.BaronessDeFleur, BaronessDeFleurSummon);
AddExecutor(ExecutorType.SpSummon, CardId.VirtualWorldKyubiShenshen, VirtualWorldKyubiShenshenSummon);
AddExecutor(ExecutorType.Activate, CardId.VirtualWorldKyubiShenshen, VirtualWorldKyubiShenshenEffect);
AddExecutor(ExecutorType.SpSummon, CardId.CoralDragon);
AddExecutor(ExecutorType.Activate, CardId.CoralDragon, CoralDragonEffect);
AddExecutor(ExecutorType.Activate, CardId.TGWonderMagician, TGWonderMagicianEffect);
AddExecutor(ExecutorType.Activate, CardId.FusionDestiny, FusionDestinyEffect);
AddExecutor(ExecutorType.SpSummon, CardId.PredaplantVerteAnaconda, PredaplantVerteAnacondaSummon);
AddExecutor(ExecutorType.Activate, CardId.PredaplantVerteAnaconda, PredaplantVerteAnacondaEffect);
AddExecutor(ExecutorType.Summon, CardId.NemesesCorridor, SummonForMaterial);
AddExecutor(ExecutorType.Summon, CardId.DestinyHeroCelestial, SummonForMaterial);
AddExecutor(ExecutorType.Activate, CardId.DestinyHeroDasher, DestinyHeroDasherEffect);
AddExecutor(ExecutorType.Activate, CardId.DestinyHeroCelestial, DestinyHeroCelestialEffect);
AddExecutor(ExecutorType.Repos, MonsterRepos);
AddExecutor(ExecutorType.SpellSet, CardId.InfiniteImpermanence, TrapSet);
AddExecutor(ExecutorType.SpellSet, CardId.SolemnStrike, TrapSet);
AddExecutor(ExecutorType.SpellSet, CardId.CrossoutDesignator, TrapSet);
AddExecutor(ExecutorType.MonsterSet, CardId.Sangan);
AddExecutor(ExecutorType.SpellSet, SetForCelestial);
}
private bool BeastOLionUsed = false;
private bool JetSynchronUsed = false;
private bool FusionDestinyUsed = false;
private bool BaronessDeFleurUsed = false;
private ClientCard PhoenixTarget = null;
private int PhoenixSelectingTarget = 0;
public override bool OnSelectHand()
{
// go first
return true;
}
public override void OnNewTurn()
{
BeastOLionUsed = false;
JetSynchronUsed = false;
FusionDestinyUsed = false;
PhoenixTarget = null;
PhoenixSelectingTarget = 0;
}
public override CardPosition OnSelectPosition(int cardId, IList<CardPosition> positions)
{
YGOSharp.OCGWrapper.NamedCard cardData = YGOSharp.OCGWrapper.NamedCard.Get(cardId);
if (cardData != null)
{
if (cardData.Attack <= 1000)
return CardPosition.FaceUpDefence;
if (Util.IsTurn1OrMain2() && cardData.Attack <= 2500)
return CardPosition.FaceUpDefence;
}
return 0;
}
public override int OnSelectPlace(int cardId, int player, CardLocation location, int available)
{
if (location == CardLocation.MonsterZone)
{
if (cardId == CardId.SalamangreatAlmiraj)
{
return available & Zones.ExtraMonsterZones;
}
if (cardId == CardId.TGHyperLibrarian && Bot.GetMonsterCount() >= 3)
{
return available & Zones.ExtraMonsterZones;
}
return available & Zones.MainMonsterZones & ~Bot.GetLinkedZones() & ~(Zones.z2 + Zones.z4); // preserve place for arboria
}
return 0;
}
public override IList<ClientCard> OnSelectCard(IList<ClientCard> cards, int min, int max, int hint, bool cancelable)
{
if (hint != HintMsg.Destroy)
PhoenixSelectingTarget = 0;
if (hint == HintMsg.AddToHand && max == 1)
{
foreach (ClientCard card in cards)
{
if (card.IsCode(CardId.CrusadiaArboria))
return new List<ClientCard>(new[] { card });
if (card.IsCode(CardId.NemesesCorridor))
return new List<ClientCard>(new[] { card });
}
foreach (ClientCard card in cards)
{
if (card.IsCode(CardId.MaxxC) && !Bot.HasInHand(CardId.MaxxC))
return new List<ClientCard>(new[] { card });
if (card.IsCode(CardId.AshBlossomJoyousSpring) && !Bot.HasInHand(CardId.AshBlossomJoyousSpring))
return new List<ClientCard>(new[] { card });
if (card.IsCode(CardId.EffectVeiler) && !Bot.HasInHand(CardId.EffectVeiler))
return new List<ClientCard>(new[] { card });
}
}
if (hint == HintMsg.SpSummon && max == 1)
{
foreach (ClientCard card in cards)
{
if (card.IsCode(CardId.DestinyHeroDestroyPhoenixEnforcer))
return new List<ClientCard>(new[] { card });
}
}
if (hint == HintMsg.Destroy && max == 1)
{
PhoenixSelectingTarget++;
if (PhoenixSelectingTarget >= 2 && !cards.Contains(PhoenixTarget))
{
ClientCard target = Util.GetProblematicEnemyCard();
if (target == null || !cards.Contains(target))
target = Util.GetBestEnemyCard();
if (target != null && cards.Contains(target))
return new List<ClientCard>(new[] { target });
}
}
if (hint == HintMsg.LinkMaterial && cancelable && min == 0)
{
// TODO: not working
return new List<ClientCard>();
}
return base.OnSelectCard(cards, min, max, hint, cancelable);
}
public override int OnSelectOption(IList<int> options)
{
if (options.Count == 2 && options[0] == Util.GetStringId(CardId.CupidPitch, 1))
return 0;
else return base.OnSelectOption(options);
}
private IList<int> GetHandCost()
{
List<int> result = new List<int> { };
if (Bot.HasInMonstersZone(CardId.BraveToken))
result.Add(CardId.DracobackTheDragonSteed);
if (Bot.Hand.Count(card => card.IsCode(CardId.FusionDestiny)) >= 2)
result.Add(CardId.FusionDestiny);
if (Bot.Hand.Count(card => card.IsCode(CardId.Sangan)) >= 2)
result.Add(CardId.Sangan);
if (Bot.Hand.Count(card => card.IsCode(CardId.RiteofAramesia)) >= 2)
result.Add(CardId.RiteofAramesia);
if (Bot.Hand.Count(card => card.IsCode(CardId.AquamancerOfTheSanctuary)) >= 2)
result.Add(CardId.AquamancerOfTheSanctuary);
if (Bot.HasInGraveyardOrInBanished(CardId.DestinyHeroCelestial) || !Bot.HasInExtra(CardId.DestinyHeroDestroyPhoenixEnforcer))
result.Add(CardId.DestinyHeroDasher);
if (Bot.HasInGraveyardOrInBanished(CardId.DestinyHeroDasher) || !Bot.HasInExtra(CardId.DestinyHeroDestroyPhoenixEnforcer))
result.Add(CardId.DestinyHeroCelestial);
if (Bot.Hand.Count(card => card.IsCode(CardId.AshBlossomJoyousSpring)) >= 2)
result.Add(CardId.AshBlossomJoyousSpring);
if (Bot.Hand.Count(card => card.IsCode(CardId.CrossoutDesignator)) >= 2)
result.Add(CardId.CrossoutDesignator);
if (Bot.HasInHand(result))
return result;
result.AddRange(new int[]{
CardId.HarpiesFeatherDuster,
CardId.MechaPhantomBeastOLion,
CardId.Sangan,
CardId.AshBlossomJoyousSpring,
CardId.MaxxC,
CardId.EffectVeiler,
CardId.CrossoutDesignator,
CardId.CalledByTheGrave,
CardId.RiteofAramesia,
CardId.AquamancerOfTheSanctuary,
CardId.InfiniteImpermanence,
CardId.SolemnStrike
});
return result;
}
private bool WanderingGryphonRiderCounter()
{
if (Card.Location == CardLocation.Hand)
return false;
return Duel.LastChainPlayer == 1;
}
private bool CrossoutDesignatorEffect()
{
ClientCard LastChainCard = Util.GetLastChainCard();
if (LastChainCard == null || Duel.LastChainPlayer != 1) return false;
return CrossoutDesignatorCheck(LastChainCard, CardId.AquamancerOfTheSanctuary, 3)
|| CrossoutDesignatorCheck(LastChainCard, CardId.Sangan, 3)
|| CrossoutDesignatorCheck(LastChainCard, CardId.AshBlossomJoyousSpring, 3)
|| CrossoutDesignatorCheck(LastChainCard, CardId.MaxxC, 2)
|| CrossoutDesignatorCheck(LastChainCard, CardId.EffectVeiler, 1)
|| CrossoutDesignatorCheck(LastChainCard, CardId.RiteofAramesia, 3)
|| CrossoutDesignatorCheck(LastChainCard, CardId.HarpiesFeatherDuster, 1)
|| CrossoutDesignatorCheck(LastChainCard, CardId.FusionDestiny, 3)
|| CrossoutDesignatorCheck(LastChainCard, CardId.FoolishBurial, 1)
|| CrossoutDesignatorCheck(LastChainCard, CardId.MonsterReborn, 1)
|| CrossoutDesignatorCheck(LastChainCard, CardId.CalledByTheGrave, 2)
|| CrossoutDesignatorCheck(LastChainCard, CardId.CrossoutDesignator, 3)
|| CrossoutDesignatorCheck(LastChainCard, CardId.InfiniteImpermanence, 3)
;
}
private bool CrossoutDesignatorCheck(ClientCard LastChainCard, int id, int count)
{
if (LastChainCard.IsCode(id) && Bot.GetRemainingCount(id, count) > 0)
{
AI.SelectAnnounceID(id);
return true;
}
return false;
}
private bool DestinyHeroDestroyPhoenixEnforcerEffect()
{
if (Card.Location == CardLocation.Grave)
return true;
else
{
ClientCard target = Util.GetProblematicEnemyCard();
if (target != null)
{
AI.SelectCard(CardId.DestinyHeroDestroyPhoenixEnforcer);
AI.SelectNextCard(target);
return true;
}
target = Util.GetBestEnemyCard();
if (target == null)
return false;
if (DefaultOnBecomeTarget() || Bot.UnderAttack || Duel.Phase == DuelPhase.End
|| (Duel.Player == 0 && Util.IsTurn1OrMain2())
|| (Duel.Player == 1 && Enemy.GetMonsterCount() >= 2))
{
PhoenixTarget = target;
AI.SelectCard(CardId.DestinyHeroDestroyPhoenixEnforcer);
AI.SelectNextCard(target);
return true;
}
return false;
}
}
private bool DracobackTheDragonSteedBounce()
{
if (Card.Location != CardLocation.SpellZone)
return false;
ClientCard target = Util.GetProblematicEnemyCard();
AI.SelectCard(target);
return true;
}
private bool DracobackTheDragonSteedEquip()
{
if (Card.Location == CardLocation.SpellZone)
return false;
if (Card.Location == CardLocation.Grave)
return true;
if (Bot.HasInMonstersZone(CardId.BraveToken, faceUp: true))
{
AI.SelectCard(CardId.BraveToken);
return true;
}
return false;
}
private bool BorreloadSavageDragonSummon()
{
int[] materials = new[] {
CardId.CupidPitch,
CardId.MechaPhantomBeastToken,
CardId.AquamancerOfTheSanctuary
};
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials)) >= 2)
{
AI.SelectMaterials(materials);
return true;
}
return false;
}
private bool BorreloadSavageDragonEffect()
{
if (ActivateDescription == -1)
{
AI.SelectCard(new[] { CardId.MechaPhantomBeastAuroradon, CardId.CrystronHalqifibrax, CardId.PredaplantVerteAnaconda });
return true;
}
else
{
return true;
}
}
private bool RiteofAramesiaEffect()
{
AI.SelectYesNo(true);
return true;
}
private bool WanderingGryphonRiderSummon()
{
if (Card.Location != CardLocation.Hand)
return false;
return Bot.HasInMonstersZone(CardId.BraveToken) || (Duel.Player == 0 && (Duel.LastChainPlayer == -1 || Bot.HasInSpellZone(CardId.JourneyOfDestiny)));
}
private bool JourneyOfDestinyActivate()
{
return Card.Location == CardLocation.Hand;
}
private bool JourneyOfDestinyEffect()
{
if (Card.Location == CardLocation.Hand)
return false;
if (ActivateDescription == -1 || ActivateDescription == Util.GetStringId(CardId.JourneyOfDestiny, 1))
{
// search equip to hand
AI.SelectOption(0);
return true;
}
else
{
// search rider or aquamancer
if (Bot.GetRemainingCount(CardId.WanderingGryphonRider, 1) == 0 || Bot.GetHandCount() == 0 || !Bot.HasInMonstersZone(CardId.BraveToken))
{
AI.SelectCard(CardId.AquamancerOfTheSanctuary);
if (Bot.HasInHandOrInMonstersZoneOrInGraveyard(CardId.AquamancerOfTheSanctuary))
AI.SelectNextCard(CardId.AquamancerOfTheSanctuary);
else
AI.SelectNextCard(GetHandCost());
}
else
{
AI.SelectCard(CardId.WanderingGryphonRider);
AI.SelectNextCard(GetHandCost());
}
return true;
}
}
private bool AquamancerOfTheSanctuarySearchEffect()
{
if (Card.Location == CardLocation.Grave)
{
AI.SelectCard(CardLocation.Deck);
return true;
}
if (ActivateDescription == Util.GetStringId(CardId.AquamancerOfTheSanctuary, 0))
{
// summon
return false;
}
else
{
// search
return !Bot.HasInHand(CardId.RiteofAramesia) && !Bot.HasInMonstersZone(CardId.BraveToken);
}
}
private bool AquamancerOfTheSanctuarySummonEffect()
{
if (ActivateDescription == Util.GetStringId(CardId.AquamancerOfTheSanctuary, 0))
{
// summon
return Bot.GetMonsterCount() <= 3;
}
return false;
}
private bool MechaPhantomBeastOLionEffect()
{
if (Bot.GetMonsterCount() >= 3
&& ((Bot.HasInExtra(CardId.MechaPhantomBeastAuroradon) && Bot.HasInMonstersZone(CardId.CrystronHalqifibrax))
|| Bot.HasInMonstersZone(CardId.MechaPhantomBeastAuroradon)))
return false;
if (ActivateDescription == -1)
{
BeastOLionUsed = true;
return true;
}
return !BeastOLionUsed;
}
private bool CrusadiaArboriaSummon()
{
return !Bot.GetMonsters().Any(card => card.IsFaceup() && card.IsTuner());
}
private bool CrystronNeedlefiberSummon()
{
if (JetSynchronUsed && !Bot.HasInMonstersZone(CardId.MechaPhantomBeastOLion))
return false;
List<int> materials = new List<int>{
CardId.Sangan,
_CardId.GamecieltheSeaTurtleKaiju,
CardId.PredaplantVerteAnaconda,
CardId.SalamangreatAlmiraj,
CardId.LinkSpider,
CardId.MaxxC,
CardId.MechaPhantomBeastToken,
CardId.AquamancerOfTheSanctuary,
CardId.CrusadiaArboria,
CardId.MechaPhantomBeastOLion,
CardId.JetSynchron,
CardId.AshBlossomJoyousSpring,
CardId.EffectVeiler
};
if (!Bot.HasInMonstersZone(CardId.BraveToken) || !Bot.HasInMonstersZone(CardId.WanderingGryphonRider))
{
materials.Add(CardId.BraveToken);
materials.Add(CardId.WanderingGryphonRider);
}
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials)) >= 2)
{
AI.SelectMaterials(materials);
return true;
}
return false;
}
private bool CrystronNeedlefiberEffect()
{
if (Duel.Player == 0)
{
AI.SelectCard(CardId.JetSynchron, CardId.MechaPhantomBeastOLion, CardId.EffectVeiler);
return true;
}
else
{
if (Enemy.GetSpells().Any(card => card.IsFacedown() || card.HasType(CardType.Continuous) || card.HasType(CardType.Field) || card.HasType(CardType.Equip)))
{
AI.SelectCard(CardId.TGWonderMagician);
}
else
{
AI.SelectCard(CardId.CoralDragon);
}
return true;
}
}
private bool TGWonderMagicianEffect()
{
ClientCard target = Util.GetProblematicEnemySpell();
if (target == null)
target = Enemy.GetSpells().Find(card => card.IsFacedown() || card.HasType(CardType.Continuous) || card.HasType(CardType.Field) || card.HasType(CardType.Equip));
if (target == null)
return false;
AI.SelectCard(target);
return true;
}
private bool MechaPhantomBeastAuroradonSummon()
{
return Bot.GetMonsterCount() <= 4; // has 3 field for tokens
}
private bool MechaPhantomBeastAuroradonEffect()
{
if (ActivateDescription == -1)
return true;
else
{
AI.SelectOption(1); // release 2 monsters
AI.SelectCard(CardId.MechaPhantomBeastAuroradon);
AI.SelectNextCard(CardId.MechaPhantomBeastToken);
return true;
}
}
private bool TGHyperLibrarianSummon()
{
int[] materials = new[] {
CardId.MechaPhantomBeastOLion,
CardId.MechaPhantomBeastToken,
CardId.AquamancerOfTheSanctuary
};
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials)) >= 3)
{
AI.SelectMaterials(materials);
return true;
}
return false;
}
private bool JetSynchronEffect()
{
int[] materials = new[] {
CardId.MechaPhantomBeastToken
};
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials)) >= 2 || Bot.GetMonsterCount() <= 3)
{
JetSynchronUsed = true;
AI.SelectCard(GetHandCost());
return true;
}
return false;
}
private bool CupidPitchSummon()
{
int[] materials = new[] {
CardId.JetSynchron,
CardId.EffectVeiler,
CardId.MechaPhantomBeastToken,
CardId.AquamancerOfTheSanctuary,
CardId.Sangan,
CardId.TGHyperLibrarian,
CardId.BraveToken
};
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials)) >= 3)
{
AI.SelectMaterials(materials);
return true;
}
return false;
}
private bool CupidPitchEffect()
{
if (Card.Location == CardLocation.MonsterZone)
{
AI.SelectOption(1); // level up
}
else
{
AI.SelectCard(CardId.NemesesCorridor);
}
return true;
}
private bool SalamangreatAlmirajSummonFirst()
{
int[] materials = new[] {
CardId.Sangan
};
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials) && !card.IsSpecialSummoned) == 0)
return false;
AI.SelectMaterials(materials);
return true;
}
private bool SalamangreatAlmirajSummon()
{
if (PhoenixNotAvail())
return false;
int[] materials = new[] {
CardId.MechaPhantomBeastOLion,
CardId.JetSynchron
};
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials) && !card.IsSpecialSummoned) == 0)
return false;
AI.SelectMaterials(materials);
return true;
}
private bool LinkSpiderSummon()
{
if (PhoenixNotAvail())
return false;
List<int> materials = new List<int>{
CardId.MechaPhantomBeastToken
};
if (!Bot.HasInMonstersZone(CardId.BraveToken) || !Bot.HasInMonstersZone(CardId.WanderingGryphonRider))
{
materials.Add(CardId.BraveToken);
}
if (Bot.GetMonsters().Any(card => card.IsCode(CardId.PrimalBeingToken) && card.Attack <= 4000))
{
materials.Add(CardId.PrimalBeingToken);
}
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials)) == 0)
return false;
AI.SelectMaterials(materials);
return true;
}
private bool NeedMonster()
{
if (Bot.HasInMonstersZone(CardId.PredaplantVerteAnaconda, true) || PhoenixNotAvail())
return false;
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.Level >= 8) > 0)
return false;
if (Bot.GetMonsterCount() == 0 && Bot.Hand.GetMatchingCardsCount(card => card.Level <= 4) == 0)
return false;
if (Bot.GetMonsterCount() >= 2)
return false;
return true;
}
private bool SummonForMaterial()
{
if (Bot.HasInMonstersZone(CardId.PredaplantVerteAnaconda, true) || !Bot.HasInExtra(CardId.PredaplantVerteAnaconda))
return false;
if (Bot.MonsterZone.GetMatchingCardsCount(card => (card.HasType(CardType.Effect) || card.IsTuner()) && card.Level < 8) == 1)
return true;
return false;
}
private bool PhoenixNotAvail()
{
return Bot.LifePoints <= 2000 || Bot.GetRemainingCount(CardId.FusionDestiny, 3) == 0 || Bot.HasInHand(CardId.FusionDestiny)
|| !Bot.HasInExtra(CardId.PredaplantVerteAnaconda) || !Bot.HasInExtra(CardId.DestinyHeroDestroyPhoenixEnforcer)
|| (Bot.GetRemainingCount(CardId.DestinyHeroCelestial, 1) == 0 && !Bot.HasInHand(CardId.DestinyHeroCelestial))
|| (Bot.GetRemainingCount(CardId.DestinyHeroDasher, 1) == 0 && !Bot.HasInHand(CardId.DestinyHeroDasher));
}
private bool PredaplantVerteAnacondaSummon()
{
if (PhoenixNotAvail())
return false;
List<int> materials = new List<int>{
_CardId.GamecieltheSeaTurtleKaiju,
CardId.AquamancerOfTheSanctuary,
CardId.Sangan,
CardId.SalamangreatAlmiraj,
CardId.LinkSpider,
CardId.NemesesCorridor,
CardId.DestinyHeroCelestial,
CardId.DestinyHeroDasher,
CardId.CrusadiaArboria,
CardId.AshBlossomJoyousSpring,
CardId.MechaPhantomBeastOLion,
CardId.MaxxC,
CardId.JetSynchron,
CardId.EffectVeiler,
CardId.CrystronHalqifibrax,
CardId.TGHyperLibrarian
};
if (!Bot.HasInMonstersZone(CardId.BraveToken) || !Bot.HasInMonstersZone(CardId.WanderingGryphonRider))
{
materials.Add(CardId.WanderingGryphonRider);
}
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials)) >= 2)
{
AI.SelectMaterials(materials);
return true;
}
return false;
}
private bool PredaplantVerteAnacondaEffect()
{
if (ActivateDescription == Util.GetStringId(CardId.PredaplantVerteAnaconda, 0))
return false;
FusionDestinyUsed = true;
AI.SelectCard(CardId.FusionDestiny);
AI.SelectMaterials(CardLocation.Deck);
return true;
}
private bool FusionDestinyEffect()
{
FusionDestinyUsed = true;
return true;
}
private bool FoolishBurialFirst()
{
if (!Bot.HasInHand(CardId.RiteofAramesia) && !Bot.HasInHandOrInGraveyard(CardId.AquamancerOfTheSanctuary) && !Bot.HasInMonstersZone(CardId.BraveToken))
{
AI.SelectCard(CardId.AquamancerOfTheSanctuary);
return true;
}
return false;
}
private bool FoolishBurialEffect()
{
if (FusionDestinyUsed)
return false;
if (!NeedMonster())
return false;
AI.SelectCard(new[] {
CardId.MechaPhantomBeastOLion
});
return true;
}
private bool MonsterRebornEffect()
{
if (Bot.HasInGraveyard(CardId.BaronessDeFleur))
{
AI.SelectCard(CardId.BaronessDeFleur);
return true;
}
else if (Bot.HasInGraveyard(CardId.WanderingGryphonRider))
{
AI.SelectCard(CardId.WanderingGryphonRider);
return true;
}
else
{
if (!NeedMonster())
return false;
AI.SelectCard(new[] {
CardId.Sangan,
CardId.MechaPhantomBeastOLion,
CardId.CrusadiaArboria,
CardId.AshBlossomJoyousSpring
});
return true;
}
}
private bool DestinyHeroDasherEffect()
{
return Bot.Hand.Count(card => card.IsMonster()) > 1;
}
private bool DestinyHeroCelestialEffect()
{
if (!Bot.HasInGraveyard(CardId.DestinyHeroDasher))
return false;
AI.SelectCard(CardId.DestinyHeroDasher);
return true;
}
private bool BaronessDeFleurSummon()
{
int[] materials = new[] {
CardId.CupidPitch,
CardId.TGWonderMagician,
CardId.TGHyperLibrarian
};
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials)) >= 2)
{
AI.SelectMaterials(materials);
return true;
}
if (!Bot.HasInMonstersZone(CardId.BraveToken))
{
materials = new[] {
CardId.AshBlossomJoyousSpring,
CardId.CrusadiaArboria,
CardId.WanderingGryphonRider
};
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials)) >= 2)
{
AI.SelectMaterials(materials);
return true;
}
}
if (!Bot.HasInMonstersZone(CardId.WanderingGryphonRider) || Bot.HasInHand(CardId.RiteofAramesia))
{
materials = new[] {
CardId.CoralDragon,
CardId.BraveToken
};
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials)) >= 2)
{
AI.SelectMaterials(materials);
return true;
}
}
return false;
}
private bool BaronessDeFleurEffect()
{
if (Duel.LastChainPlayer == 0)
{
return false;
}
if (Duel.LastChainPlayer == 1)
{
BaronessDeFleurUsed = true;
return true;
}
if (Duel.Phase == DuelPhase.Standby && BaronessDeFleurUsed)
{
BaronessDeFleurUsed = false;
return true;
}
if (Duel.Phase == DuelPhase.Main1 || Duel.Phase == DuelPhase.Main2)
{
ClientCard target = Util.GetProblematicEnemyCard(canBeTarget: true);
if (target == null)
target = Util.GetBestEnemyCard(canBeTarget: true);
if (target != null)
{
AI.SelectCard(target);
return true;
}
}
return false;
}
private bool VirtualWorldKyubiShenshenSummon()
{
if (Bot.HasInMonstersZone(CardId.DestinyHeroDestroyPhoenixEnforcer))
return false;
int[] materials = new[] {
CardId.CoralDragon,
CardId.AquamancerOfTheSanctuary,
CardId.Sangan
};
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials)) >= 2)
{
AI.SelectMaterials(materials);
return true;
}
if (!Bot.HasInMonstersZone(CardId.BraveToken))
{
materials = new[] {
CardId.MechaPhantomBeastOLion,
CardId.WanderingGryphonRider
};
if (Bot.MonsterZone.GetMatchingCardsCount(card => card.IsCode(materials)) >= 2)
{
AI.SelectMaterials(materials);
return true;
}
}
return false;
}
private bool VirtualWorldKyubiShenshenEffect()
{
if (Card.Location == CardLocation.MonsterZone && Bot.HasInBanished(CardId.AquamancerOfTheSanctuary))
{
AI.SelectCard(CardId.AquamancerOfTheSanctuary);
return true;
}
else
{
int[] costs = new[] {
CardId.NemesesCorridor,
CardId.Sangan,
CardId.CrusadiaArboria,
CardId.AshBlossomJoyousSpring,
CardId.MechaPhantomBeastOLion,
CardId.MaxxC,
CardId.EffectVeiler,
CardId.ThunderDragonColossus,
CardId.BorreloadSavageDragon,
CardId.CoralDragon,
CardId.TGHyperLibrarian,
CardId.TGWonderMagician,
CardId.CupidPitch,
CardId.CrystronHalqifibrax,
CardId.PredaplantVerteAnaconda,
CardId.LinkSpider,
CardId.SalamangreatAlmiraj
};
AI.SelectCard(costs);
AI.SelectNextCard(costs);
return true;
}
}
private bool CoralDragonEffect()
{
if (Card.Location == CardLocation.Grave)
return true;
ClientCard target = Util.GetProblematicEnemyCard(canBeTarget: true);
if (target != null)
{
AI.SelectCard(target);
return true;
}
return false;
}
private bool TrapSet()
{
AI.SelectPlace(Zones.z0 + Zones.z1 + Zones.z3 + Zones.z4);
return true;
}
private bool SetForCelestial()
{
return !FusionDestinyUsed && Bot.HasInGraveyard(CardId.DestinyHeroCelestial) && Bot.HasInGraveyard(CardId.DestinyHeroDasher) && TrapSet();
}
private bool MonsterRepos()
{
if (Card.IsFacedown())
return true;
return DefaultMonsterRepos();
}
}
}
......@@ -106,6 +106,7 @@ namespace WindBot.Game.AI
public const int MacroCosmos = 30241314;
public const int UpstartGoblin = 70368879;
public const int CyberEmergency = 60600126;
public const int TheAgentOfCreationVenus = 64734921;
public const int EaterOfMillions = 63845230;
......@@ -471,7 +472,8 @@ namespace WindBot.Game.AI
int[] ignoreList = {
_CardId.MacroCosmos,
_CardId.UpstartGoblin,
_CardId.CyberEmergency
_CardId.CyberEmergency,
_CardId.TheAgentOfCreationVenus
};
if (Util.GetLastChainCard().IsCode(ignoreList))
return false;
......@@ -500,8 +502,9 @@ namespace WindBot.Game.AI
/// </summary>
protected bool DefaultEffectVeiler()
{
if (Util.GetLastChainCard() != null && Util.GetLastChainCard().IsCode(_CardId.GalaxySoldier) && Enemy.Hand.Count >= 3) return false;
if (Util.ChainContainsCard(_CardId.EffectVeiler))
ClientCard LastChainCard = Util.GetLastChainCard();
if (LastChainCard != null && (LastChainCard.IsCode(_CardId.GalaxySoldier) && Enemy.Hand.Count >= 3
|| LastChainCard.IsCode(_CardId.EffectVeiler, _CardId.InfiniteImpermanence)))
return false;
return DefaultBreakthroughSkill();
}
......@@ -539,7 +542,9 @@ namespace WindBot.Game.AI
protected bool DefaultInfiniteImpermanence()
{
// TODO: disable s & t
if (!DefaultUniqueTrap())
ClientCard LastChainCard = Util.GetLastChainCard();
if (LastChainCard != null && (LastChainCard.IsCode(_CardId.GalaxySoldier) && Enemy.Hand.Count >= 3
|| LastChainCard.IsCode(_CardId.EffectVeiler, _CardId.InfiniteImpermanence)))
return false;
return DefaultDisableMonster();
}
......@@ -725,6 +730,9 @@ namespace WindBot.Game.AI
/// </summary>
protected bool DefaultMonsterRepos()
{
if (Card.IsMonsterInvincible())
return Card.IsDefense();
if (Card.Attack == 0)
{
if (Card.IsFaceup() && Card.IsAttack())
......@@ -787,14 +795,14 @@ namespace WindBot.Game.AI
_CardId.JudgmentDragon,
_CardId.TopologicTrisbaena
};
int[] destroyAllOpponentList =
int[] destroyAllOpponentSpellList =
{
_CardId.HarpiesFeatherDuster,
_CardId.DarkMagicAttack
};
if (Util.ChainContainsCard(destroyAllList)) return true;
if (Enemy.HasInSpellZone(destroyAllOpponentList, true)) return true;
if (Enemy.HasInSpellZone(destroyAllOpponentSpellList, true) && Card.Location == CardLocation.SpellZone) return true;
// TODO: ChainContainsCard(id, player)
return false;
}
......
......@@ -248,6 +248,11 @@ namespace WindBot.Game
return HasInHand(cardId) || HasInGraveyard(cardId);
}
public bool HasInGraveyardOrInBanished(int cardId)
{
return HasInBanished(cardId) || HasInGraveyard(cardId);
}
public bool HasInMonstersZoneOrInGraveyard(int cardId)
{
return HasInMonstersZone(cardId) || HasInGraveyard(cardId);
......
......@@ -464,13 +464,13 @@ namespace WindBot.Game
/// <returns>Index of the selected option.</returns>
public int OnSelectOption(IList<int> options)
{
if (m_option != -1 && m_option < options.Count)
return m_option;
int result = Executor.OnSelectOption(options);
if (result != -1)
return result;
if (m_option != -1 && m_option < options.Count)
return m_option;
return 0; // Always select the first option.
}
......
......@@ -1043,6 +1043,7 @@ namespace WindBot.Game
int seq = packet.ReadByte();
packet.ReadByte(); // pos
}
if (count2 == 0) cancelable = false;
IList<ClientCard> selected = func(cards, (finishable ? 0 : 1), 1, _select_hint, cancelable);
......
......@@ -79,6 +79,8 @@ WindBot can run as a "server", provide a http interface to create bot.
* BlueEyesMaxDragon
* Brave
* ChainBurn
* DarkMagician
......
......@@ -70,6 +70,7 @@
<Compile Include="Game\AI\DeckAttribute.cs" />
<Compile Include="Game\AI\DecksManager.cs" />
<Compile Include="Game\AI\Decks\AltergeistExecutor.cs" />
<Compile Include="Game\AI\Decks\BraveExecutor.cs" />
<Compile Include="Game\AI\Decks\FamiliarPossessedExecutor.cs" />
<Compile Include="Game\AI\Decks\BlackwingExecutor.cs" />
<Compile Include="Game\AI\Decks\LuckyExecutor.cs" />
......
......@@ -26,7 +26,7 @@ namespace WindBot
Host = "127.0.0.1";
Port = 7911;
HostInfo = "";
Version = 0x1352;
Version = 0x1353;
Hand = 0;
Debug = false;
Chat = true;
......
......@@ -26,9 +26,9 @@
"dialog": "wof-Ayase-Amu"
},
{
"name": "淡野晴",
"name": "埃莉丝·科菲",
"deck": "Level VIII",
"dialog": "wof-Awaya-Haru"
"dialog": "wof-Elyse-Coffey"
},
{
"name": "城崎千夏",
......@@ -46,9 +46,9 @@
"dialog": "wof-Sagisawa-Yui"
},
{
"name": "有栖川亚里沙",
"name": "十十六木花奏子",
"deck": "Qliphort",
"dialog": "wof-Arisugawa-Arisa"
"dialog": "wof-Todoroki-Kanako"
},
{
"name": "七草一花",
......@@ -101,9 +101,9 @@
"dialog": "wof-Tsukishiro-Isuzu"
},
{
"name": "糖歌",
"name": "晴海夏佳",
"deck": "OldSchool",
"dialog": "wof-Ameuta"
"dialog": "wof-Harumi-Natsuka"
},
{
"name": "濑名歌铃",
......@@ -126,14 +126,14 @@
"dialog": "wof-Helen-Virsaladze"
},
{
"name": "二宫爱丽丝",
"name": "蓝叶雾叶",
"deck": "Dragun",
"dialog": "wof-Ninomiya-Arisu"
"dialog": "wof-Aiha-Kiriha"
},
{
"name": "木村有容",
"name": "诺玛·林斯科特",
"deck": "MathMech",
"dialog": "wof-Kimura-Yuro"
"dialog": "wof-Norma-Linscott"
},
{
"name": "砂冢明音",
......@@ -141,9 +141,9 @@
"dialog": "wof-Sunazuka-Akane"
},
{
"name": "梦野",
"name": "早见虹羽",
"deck": "PureWinds",
"dialog": "wof-Yumeno"
"dialog": "wof-Hayami-Nijiha"
},
{
"name": "白雪安娜",
......@@ -161,9 +161,9 @@
"dialog": "wof-Asami-Sorako"
},
{
"name": "黑野紫",
"name": "佐佐木茜",
"deck": "Horus",
"dialog": "wof-Kurono-Yukari"
"dialog": "wof-Sasaki-Akane"
},
{
"name": "克里斯汀·罗森塔尔",
......@@ -189,6 +189,11 @@
"name": "黑崎智秋",
"deck": "FamiliarPossessed",
"dialog": "wof-Kurosaki-Chiaki"
},
{
"name": "神代心春",
"deck": "Brave",
"dialog": "wof-Kamishiro-Koharu"
}
]
}
......@@ -188,11 +188,6 @@
"deck": "Phantasm",
"dialog": "kiwi.zh-TW"
},
{
"name": "幻煌果",
"deck": "Phantasm",
"dialog": "kiwi.zh-TW"
},
{
"name": "報社鬥士",
"deck": "GrenMajuThunderBoarder",
......@@ -214,6 +209,11 @@
"name": "璃璃子",
"deck": "TimeThief",
"dialog": "default"
},
{
"name": "濑名歌铃",
"deck": "Brave",
"dialog": "wof-Sena-Karin"
}
]
}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment