Commit 65dd2b91 authored by 聖園ミカ's avatar 聖園ミカ 🐟

Merge branch 'master' of git.moenext.com:mycard/windbot

parents 56cbdb31 bb85bb0c
Pipeline #37663 passed with stages
in 2 minutes and 18 seconds
...@@ -8,7 +8,7 @@ on: ...@@ -8,7 +8,7 @@ on:
jobs: jobs:
build: build:
runs-on: windows-2019 runs-on: windows-2022
env: env:
Solution_Name: WindBot.sln Solution_Name: WindBot.sln
......
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<configuration> <configuration>
<startup> <startup>
<supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.0"/> <supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.8"/>
</startup> </startup>
</configuration> </configuration>
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
<AppDesignerFolder>Properties</AppDesignerFolder> <AppDesignerFolder>Properties</AppDesignerFolder>
<RootNamespace>BotWrapper</RootNamespace> <RootNamespace>BotWrapper</RootNamespace>
<AssemblyName>Bot</AssemblyName> <AssemblyName>Bot</AssemblyName>
<TargetFrameworkVersion>v4.0</TargetFrameworkVersion> <TargetFrameworkVersion>v4.8</TargetFrameworkVersion>
<FileAlignment>512</FileAlignment> <FileAlignment>512</FileAlignment>
<TargetFrameworkProfile /> <TargetFrameworkProfile />
</PropertyGroup> </PropertyGroup>
...@@ -22,6 +22,7 @@ ...@@ -22,6 +22,7 @@
<DefineConstants>DEBUG;TRACE</DefineConstants> <DefineConstants>DEBUG;TRACE</DefineConstants>
<ErrorReport>prompt</ErrorReport> <ErrorReport>prompt</ErrorReport>
<WarningLevel>4</WarningLevel> <WarningLevel>4</WarningLevel>
<Prefer32Bit>false</Prefer32Bit>
</PropertyGroup> </PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|x86'"> <PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|x86'">
<PlatformTarget>x86</PlatformTarget> <PlatformTarget>x86</PlatformTarget>
...@@ -31,6 +32,7 @@ ...@@ -31,6 +32,7 @@
<DefineConstants>TRACE</DefineConstants> <DefineConstants>TRACE</DefineConstants>
<ErrorReport>prompt</ErrorReport> <ErrorReport>prompt</ErrorReport>
<WarningLevel>4</WarningLevel> <WarningLevel>4</WarningLevel>
<Prefer32Bit>false</Prefer32Bit>
</PropertyGroup> </PropertyGroup>
<PropertyGroup> <PropertyGroup>
<StartupObject>BotWrapper.BotWrapper</StartupObject> <StartupObject>BotWrapper.BotWrapper</StartupObject>
......
...@@ -10,7 +10,7 @@ using System.Runtime.InteropServices; ...@@ -10,7 +10,7 @@ using System.Runtime.InteropServices;
[assembly: AssemblyConfiguration("")] [assembly: AssemblyConfiguration("")]
[assembly: AssemblyCompany("IceYGO")] [assembly: AssemblyCompany("IceYGO")]
[assembly: AssemblyProduct("WindBot")] [assembly: AssemblyProduct("WindBot")]
[assembly: AssemblyCopyright("Copyright © IceYGO 2017")] [assembly: AssemblyCopyright("Copyright © IceYGO 2015-2025")]
[assembly: AssemblyTrademark("")] [assembly: AssemblyTrademark("")]
[assembly: AssemblyCulture("")] [assembly: AssemblyCulture("")]
......
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<configuration> <configuration>
<startup><supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.0"/></startup></configuration> <startup><supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.8"/></startup></configuration>
...@@ -2186,7 +2186,7 @@ namespace WindBot.Game.AI.Decks ...@@ -2186,7 +2186,7 @@ namespace WindBot.Game.AI.Decks
public override int OnSelectOption(IList<int> options) public override int OnSelectOption(IList<int> options)
{ {
ClientCard currentSolvingChain = Duel.GetCurrentSolvingChainCard(); ChainInfo currentSolvingChain = Duel.GetCurrentSolvingChainInfo();
if (currentSolvingChain != null) if (currentSolvingChain != null)
{ {
// 1190=Add to Hand, 1152=Special Summon // 1190=Add to Hand, 1152=Special Summon
...@@ -2246,8 +2246,8 @@ namespace WindBot.Game.AI.Decks ...@@ -2246,8 +2246,8 @@ namespace WindBot.Game.AI.Decks
public override int OnSelectPlace(int cardId, int player, CardLocation location, int available) public override int OnSelectPlace(int cardId, int player, CardLocation location, int available)
{ {
ClientCard currentSovingChain = Duel.GetCurrentSolvingChainCard(); ChainInfo currentSovingChain = Duel.GetCurrentSolvingChainInfo();
if (currentSovingChain != null && currentSovingChain.Controller == 0 && currentSovingChain.IsCode(CardId.SprindTheIrondashDragon)) if (currentSovingChain != null && currentSovingChain.ActivatePlayer == 0 && currentSovingChain.IsCode(CardId.SprindTheIrondashDragon))
{ {
return SprindTheIrondashDragonMoveZone(available, null); return SprindTheIrondashDragonMoveZone(available, null);
} }
...@@ -2435,21 +2435,21 @@ namespace WindBot.Game.AI.Decks ...@@ -2435,21 +2435,21 @@ namespace WindBot.Game.AI.Decks
public override void OnChainSolved(int chainIndex) public override void OnChainSolved(int chainIndex)
{ {
ClientCard currentCard = Duel.GetCurrentSolvingChainCard(); ChainInfo currentCard = Duel.GetCurrentSolvingChainInfo();
if (currentCard != null) if (currentCard != null)
{ {
// if activation is negated, it can activate again. // if activation is negated, it can activate again.
if (currentCard.Controller == 0) if (currentCard.ActivatePlayer == 0)
{ {
List<int> activateCheck = new List<int> { CardId.NadirServant, CardId.FusionDeployment, CardId.BrandedFusion, CardId.BrandedInRed }; List<int> activateCheck = new List<int> { CardId.NadirServant, CardId.FusionDeployment, CardId.BrandedFusion, CardId.BrandedInRed };
if (currentCard.IsCode(activateCheck)) if (currentCard.IsCode(activateCheck))
{ {
activatedCardIdList.Add(currentCard.Id); activatedCardIdList.Add(currentCard.ActivateId);
} }
} }
if (!Duel.IsCurrentSolvingChainNegated()) if (!Duel.IsCurrentSolvingChainNegated())
{ {
if (currentCard.Controller == 1) if (currentCard.ActivatePlayer == 1)
{ {
if (currentCard.IsCode(_CardId.MaxxC)) if (currentCard.IsCode(_CardId.MaxxC))
enemyActivateMaxxC = true; enemyActivateMaxxC = true;
...@@ -2458,7 +2458,7 @@ namespace WindBot.Game.AI.Decks ...@@ -2458,7 +2458,7 @@ namespace WindBot.Game.AI.Decks
if (currentCard.IsCode(CardId.DimensionShifter)) if (currentCard.IsCode(CardId.DimensionShifter))
dimensionShifterCount = 2; dimensionShifterCount = 2;
} }
if (currentCard.Controller == 0 && currentCard.IsCode(CardId.NadirServant)) if (currentCard.ActivatePlayer == 0 && currentCard.IsCode(CardId.NadirServant))
{ {
nadirActivated = true; nadirActivated = true;
} }
......
...@@ -1153,8 +1153,8 @@ namespace WindBot.Game.AI.Decks ...@@ -1153,8 +1153,8 @@ namespace WindBot.Game.AI.Decks
public override void OnChainSolved(int chainIndex) public override void OnChainSolved(int chainIndex)
{ {
ClientCard currentCard = Duel.GetCurrentSolvingChainCard(); ChainInfo currentCard = Duel.GetCurrentSolvingChainInfo();
if (currentCard != null && !Duel.IsCurrentSolvingChainNegated() && currentCard.Controller == 1) if (currentCard != null && !Duel.IsCurrentSolvingChainNegated() && currentCard.ActivatePlayer == 1)
{ {
if (currentCard.IsCode(_CardId.MaxxC)) if (currentCard.IsCode(_CardId.MaxxC))
enemyActivateMaxxC = true; enemyActivateMaxxC = true;
...@@ -1166,7 +1166,7 @@ namespace WindBot.Game.AI.Decks ...@@ -1166,7 +1166,7 @@ namespace WindBot.Game.AI.Decks
{ {
for (int i = 0; i < 5; ++i) for (int i = 0; i < 5; ++i)
{ {
if (Enemy.SpellZone[i] == currentCard) if (Enemy.SpellZone[i] == currentCard.RelatedCard)
{ {
infiniteImpermanenceList.Add(4 - i); infiniteImpermanenceList.Add(4 - i);
break; break;
......
...@@ -771,8 +771,8 @@ namespace WindBot.Game.AI.Decks ...@@ -771,8 +771,8 @@ namespace WindBot.Game.AI.Decks
public override void OnChainSolved(int chainIndex) public override void OnChainSolved(int chainIndex)
{ {
ClientCard currentCard = Duel.GetCurrentSolvingChainCard(); ChainInfo currentCard = Duel.GetCurrentSolvingChainInfo();
if (currentCard != null && !Duel.IsCurrentSolvingChainNegated() && currentCard.Controller == 1) if (currentCard != null && !Duel.IsCurrentSolvingChainNegated() && currentCard.ActivatePlayer == 1)
{ {
if (currentCard.IsCode(_CardId.MaxxC)) if (currentCard.IsCode(_CardId.MaxxC))
enemyActivateMaxxC = true; enemyActivateMaxxC = true;
...@@ -782,7 +782,7 @@ namespace WindBot.Game.AI.Decks ...@@ -782,7 +782,7 @@ namespace WindBot.Game.AI.Decks
{ {
for (int i = 0; i < 5; ++i) for (int i = 0; i < 5; ++i)
{ {
if (Enemy.SpellZone[i] == currentCard) if (Enemy.SpellZone[i] == currentCard.RelatedCard)
{ {
infiniteImpermanenceList.Add(4 - i); infiniteImpermanenceList.Add(4 - i);
break; break;
......
...@@ -153,7 +153,7 @@ namespace WindBot.Game.AI.Decks ...@@ -153,7 +153,7 @@ namespace WindBot.Game.AI.Decks
{ {
// We should summon Horus the Black Flame Dragon LV6 if he can lvlup. // We should summon Horus the Black Flame Dragon LV6 if he can lvlup.
if (Enemy.GetMonsterCount() != 0 && !Util.IsAllEnemyBetterThanValue(2300 - 1, false)) if (Enemy.GetMonsterCount() != 0 && !Util.IsAllEnemyBetterThanValue(2300 - 1, false))
foreach (ClientCard card in Main.SummonableCards) foreach (ClientCard card in Duel.MainPhase.SummonableCards)
if (card.IsCode(11224103)) if (card.IsCode(11224103))
return false; return false;
......
...@@ -1646,17 +1646,17 @@ namespace WindBot.Game.AI.Decks ...@@ -1646,17 +1646,17 @@ namespace WindBot.Game.AI.Decks
public override void OnChainSolved(int chainIndex) public override void OnChainSolved(int chainIndex)
{ {
ClientCard currentCard = Duel.GetCurrentSolvingChainCard(); ChainInfo currentCard = Duel.GetCurrentSolvingChainInfo();
if (currentCard != null && !Duel.IsCurrentSolvingChainNegated()) if (currentCard != null && !Duel.IsCurrentSolvingChainNegated())
{ {
if (currentCard.Controller == 1) if (currentCard.ActivatePlayer == 1)
{ {
if (currentCard.IsCode(_CardId.MaxxC)) if (currentCard.IsCode(_CardId.MaxxC))
enemyActivateMaxxC = true; enemyActivateMaxxC = true;
if (currentCard.IsCode(CardId.DimensionShifter)) if (currentCard.IsCode(CardId.DimensionShifter))
dimensionShifterCount = 2; dimensionShifterCount = 2;
} }
if (currentCard.Controller == 0) if (currentCard.ActivatePlayer == 0)
{ {
if (currentCard.IsCode(CardId.LabrynthCooclock)) if (currentCard.IsCode(CardId.LabrynthCooclock))
cooclockAffected = true; cooclockAffected = true;
......
...@@ -69,10 +69,10 @@ namespace WindBot.Game.AI.Decks ...@@ -69,10 +69,10 @@ namespace WindBot.Game.AI.Decks
if (_lastDoubleSummon == Duel.Turn) if (_lastDoubleSummon == Duel.Turn)
return false; return false;
if (Main.SummonableCards.Count == 0) if (Duel.MainPhase.SummonableCards.Count == 0)
return false; return false;
if (Main.SummonableCards.Count == 1 && Main.SummonableCards[0].Level < 5) if (Duel.MainPhase.SummonableCards.Count == 1 && Duel.MainPhase.SummonableCards[0].Level < 5)
{ {
bool canTribute = false; bool canTribute = false;
foreach (ClientCard handCard in Bot.Hand) foreach (ClientCard handCard in Bot.Hand)
......
...@@ -64,6 +64,7 @@ namespace WindBot.Game.AI.Decks ...@@ -64,6 +64,7 @@ namespace WindBot.Game.AI.Decks
AddExecutor(ExecutorType.Activate, _CardId.GhostOgreAndSnowRabbit, GhostOgreAndSnowRabbitActivate); AddExecutor(ExecutorType.Activate, _CardId.GhostOgreAndSnowRabbit, GhostOgreAndSnowRabbitActivate);
AddExecutor(ExecutorType.Activate, _CardId.AshBlossom, AshBlossomActivate); AddExecutor(ExecutorType.Activate, _CardId.AshBlossom, AshBlossomActivate);
AddExecutor(ExecutorType.Activate, CardId.RyzealCross, RyzealCrossActivateCard);
AddExecutor(ExecutorType.Activate, _CardId.EvilswarmExcitonKnight, EvilswarmExcitonKnightActivate); AddExecutor(ExecutorType.Activate, _CardId.EvilswarmExcitonKnight, EvilswarmExcitonKnightActivate);
AddExecutor(ExecutorType.Activate, CardId.RyzealDeadnader, RyzealDeadnaderActivate); AddExecutor(ExecutorType.Activate, CardId.RyzealDeadnader, RyzealDeadnaderActivate);
AddExecutor(ExecutorType.Activate, CardId.RyzealDuodrive, RyzealDuodriveActivate); AddExecutor(ExecutorType.Activate, CardId.RyzealDuodrive, RyzealDuodriveActivate);
...@@ -82,7 +83,6 @@ namespace WindBot.Game.AI.Decks ...@@ -82,7 +83,6 @@ namespace WindBot.Game.AI.Decks
AddExecutor(ExecutorType.Activate, CardId.Bonfire, BonfireActivate); AddExecutor(ExecutorType.Activate, CardId.Bonfire, BonfireActivate);
AddExecutor(ExecutorType.Activate, CardId.DonnerDaggerFurHire, DonnerDaggerFurHireActivate); AddExecutor(ExecutorType.Activate, CardId.DonnerDaggerFurHire, DonnerDaggerFurHireActivate);
AddExecutor(ExecutorType.Activate, CardId.Number60DugaresTheTimeless, Number60DugaresTheTimelessActivate); AddExecutor(ExecutorType.Activate, CardId.Number60DugaresTheTimeless, Number60DugaresTheTimelessActivate);
AddExecutor(ExecutorType.Activate, CardId.RyzealCross, RyzealCrossActivateCard);
AddExecutor(ExecutorType.Activate, CardId.TripleTacticsTalent, TripleTacticsTalentActivate); AddExecutor(ExecutorType.Activate, CardId.TripleTacticsTalent, TripleTacticsTalentActivate);
AddExecutor(ExecutorType.Activate, CardId.Bonfire, BonfireActivateToSearchNecessary); AddExecutor(ExecutorType.Activate, CardId.Bonfire, BonfireActivateToSearchNecessary);
AddExecutor(ExecutorType.Activate, CardId.SeventhTachyon, SeventhTachyonActivate); AddExecutor(ExecutorType.Activate, CardId.SeventhTachyon, SeventhTachyonActivate);
...@@ -372,6 +372,26 @@ namespace WindBot.Game.AI.Decks ...@@ -372,6 +372,26 @@ namespace WindBot.Game.AI.Decks
return true; return true;
} }
public bool CheckCardShouldNegate(ChainInfo chainInfo)
{
if (chainInfo == null) return false;
ClientCard card = chainInfo.RelatedCard;
if (card == null) return false;
if (card.IsMonster() && card.HasSetcode(SetcodeTimeLord) && Duel.Phase == DuelPhase.Standby) return false;
if (NotToNegateIdList.Contains(card.Id)) return false;
if (card.HasSetcode(_Setcode.Danger) && card.Location == CardLocation.Hand) return false;
if (card.IsMonster() && chainInfo.HasLocation(CardLocation.MonsterZone) && chainInfo.HasPosition(CardPosition.Defence))
{
if (Enemy.MonsterZone.Any(c => CheckNumber41(c)) || Bot.MonsterZone.Any(c => CheckNumber41(c))) return false;
}
if (DefaultCheckWhetherCardIsNegated(card)) return false;
if (Duel.Player == 1 && card.IsCode(_CardId.MulcharmyPurulia, _CardId.MulcharmyFuwalos, _CardId.MulcharmyNyalus)) return false;
if (card.IsDisabled()) return false;
return true;
}
/// <summary> /// <summary>
/// Check whether bot is at advantage. /// Check whether bot is at advantage.
/// </summary> /// </summary>
...@@ -901,7 +921,7 @@ namespace WindBot.Game.AI.Decks ...@@ -901,7 +921,7 @@ namespace WindBot.Game.AI.Decks
public override IList<ClientCard> OnSelectCard(IList<ClientCard> cards, int min, int max, int hint, bool cancelable) public override IList<ClientCard> OnSelectCard(IList<ClientCard> cards, int min, int max, int hint, bool cancelable)
{ {
ClientCard currentSolvingChain = Duel.GetCurrentSolvingChainCard(); ChainInfo currentSolvingChain = Duel.GetCurrentSolvingChainInfo();
if (currentSolvingChain != null) if (currentSolvingChain != null)
{ {
if (botSolvingCross) if (botSolvingCross)
...@@ -933,7 +953,7 @@ namespace WindBot.Game.AI.Decks ...@@ -933,7 +953,7 @@ namespace WindBot.Game.AI.Decks
} }
} }
if (currentSolvingChain.Controller == 1 && currentSolvingChain.IsCode(_CardId.EvenlyMatched)) if (currentSolvingChain.ActivatePlayer == 1 && currentSolvingChain.IsCode(_CardId.EvenlyMatched))
{ {
Logger.DebugWriteLine("=== Evenly Matched activated."); Logger.DebugWriteLine("=== Evenly Matched activated.");
List<ClientCard> banishList = new List<ClientCard>(); List<ClientCard> banishList = new List<ClientCard>();
...@@ -975,7 +995,7 @@ namespace WindBot.Game.AI.Decks ...@@ -975,7 +995,7 @@ namespace WindBot.Game.AI.Decks
return Util.CheckSelectCount(banishList, cards, min, max); return Util.CheckSelectCount(banishList, cards, min, max);
} }
if (currentSolvingChain.Controller == 0) if (currentSolvingChain.ActivatePlayer == 0)
{ {
if (hint == HintMsg.AddToHand) if (hint == HintMsg.AddToHand)
{ {
...@@ -1681,31 +1701,31 @@ namespace WindBot.Game.AI.Decks ...@@ -1681,31 +1701,31 @@ namespace WindBot.Game.AI.Decks
public override void OnChainSolved(int chainIndex) public override void OnChainSolved(int chainIndex)
{ {
botSolvingCross = false; botSolvingCross = false;
ClientCard currentCard = Duel.GetCurrentSolvingChainCard(); ChainInfo currentChain = Duel.GetCurrentSolvingChainInfo();
if (currentCard != null) if (currentChain != null && !Duel.IsCurrentSolvingChainNegated())
{ {
if (!Duel.IsCurrentSolvingChainNegated()) if (!Duel.IsCurrentSolvingChainNegated())
{ {
if (currentCard.IsCode(_CardId.LockBird)) if (currentChain.IsCode(_CardId.LockBird))
lockBirdSolved = true; lockBirdSolved = true;
if (currentCard.IsCode(_CardId.DimensionShifter)) if (currentChain.IsCode(_CardId.DimensionShifter))
dimensionShifterCount = 2; dimensionShifterCount = 2;
if (currentCard.Controller == 1) if (currentChain.ActivatePlayer == 1)
{ {
if (currentCard.IsCode(_CardId.MaxxC)) if (currentChain.IsCode(_CardId.MaxxC))
enemyActivateMaxxC = true; enemyActivateMaxxC = true;
if (currentCard.IsCode(_CardId.MulcharmyPurulia)) if (currentChain.IsCode(_CardId.MulcharmyPurulia))
enemyActivatePurulia = true; enemyActivatePurulia = true;
if (currentCard.IsCode(_CardId.MulcharmyFuwalos)) if (currentChain.IsCode(_CardId.MulcharmyFuwalos))
enemyActivateFuwalos = true; enemyActivateFuwalos = true;
if (currentCard.IsCode(_CardId.MulcharmyNyalus)) if (currentChain.IsCode(_CardId.MulcharmyNyalus))
enemyActivateNyalus = true; enemyActivateNyalus = true;
} }
if (currentCard.Controller == 0) if (currentChain.ActivatePlayer == 0)
{ {
foreach (int checkId in CheckBotSolvedList) foreach (int checkId in CheckBotSolvedList)
{ {
if (currentCard.IsCode(checkId)) if (currentChain.IsCode(checkId))
{ {
botSolvedCardIdList.Add(checkId); botSolvedCardIdList.Add(checkId);
} }
...@@ -2758,10 +2778,10 @@ namespace WindBot.Game.AI.Decks ...@@ -2758,10 +2778,10 @@ namespace WindBot.Game.AI.Decks
// whether to negate by cross // whether to negate by cross
if (ActivateDescription == Util.GetStringId(CardId.RyzealCross, 3)) if (ActivateDescription == Util.GetStringId(CardId.RyzealCross, 3))
{ {
ClientCard currentSolvingChain = Duel.GetCurrentSolvingChainCard(); ChainInfo currentChainInfo = Duel.GetCurrentSolvingChainInfo();
if (currentSolvingChain != null && !Duel.IsCurrentSolvingChainNegated()) if (currentChainInfo != null && !Duel.IsCurrentSolvingChainNegated())
{ {
if (CheckCardShouldNegate(currentSolvingChain)) if (CheckCardShouldNegate(currentChainInfo))
{ {
Logger.DebugWriteLine("** cross negate"); Logger.DebugWriteLine("** cross negate");
activatedCardIdList.Add(CardId.RyzealCross + 2); activatedCardIdList.Add(CardId.RyzealCross + 2);
...@@ -2780,6 +2800,11 @@ namespace WindBot.Game.AI.Decks ...@@ -2780,6 +2800,11 @@ namespace WindBot.Game.AI.Decks
Bot.Banished.Any(c => c != null && c.IsFaceup() && c.HasSetcode(SetcodeRyzeal) && (c.IsCanRevive() || !c.HasType(CardType.Xyz)))); Bot.Banished.Any(c => c != null && c.IsFaceup() && c.HasSetcode(SetcodeRyzeal) && (c.IsCanRevive() || !c.HasType(CardType.Xyz))));
flag |= Bot.MonsterZone.Count(c => c != null && c.IsFaceup() && c.HasType(CardType.Xyz) && c.HasSetcode(SetcodeRyzeal) && (c.Overlays.Count() > 0 || canSetMaterial)) > 0; flag |= Bot.MonsterZone.Count(c => c != null && c.IsFaceup() && c.HasType(CardType.Xyz) && c.HasSetcode(SetcodeRyzeal) && (c.Overlays.Count() > 0 || canSetMaterial)) > 0;
if (Duel.MainPhase.SpecialSummonableCards.Any(c => c.IsCode(CardId.RyzealDuodrive)))
{
flag |= RyzealDuodriveSpSummonCheck();
}
return flag; return flag;
} }
......
...@@ -695,8 +695,8 @@ namespace WindBot.Game.AI.Decks ...@@ -695,8 +695,8 @@ namespace WindBot.Game.AI.Decks
public override void OnChainSolved(int chainIndex) public override void OnChainSolved(int chainIndex)
{ {
ClientCard currentCard = Duel.GetCurrentSolvingChainCard(); ChainInfo currentCard = Duel.GetCurrentSolvingChainInfo();
if (currentCard != null && !Duel.IsCurrentSolvingChainNegated() && currentCard.Controller == 1) if (currentCard != null && !Duel.IsCurrentSolvingChainNegated() && currentCard.ActivatePlayer == 1)
{ {
if (currentCard.IsCode(_CardId.MaxxC)) if (currentCard.IsCode(_CardId.MaxxC))
enemyActivateMaxxC = true; enemyActivateMaxxC = true;
...@@ -706,7 +706,7 @@ namespace WindBot.Game.AI.Decks ...@@ -706,7 +706,7 @@ namespace WindBot.Game.AI.Decks
{ {
for (int i = 0; i < 5; ++i) for (int i = 0; i < 5; ++i)
{ {
if (Enemy.SpellZone[i] == currentCard) if (Enemy.SpellZone[i] == currentCard.RelatedCard)
{ {
infiniteImpermanenceList.Add(4 - i); infiniteImpermanenceList.Add(4 - i);
break; break;
......
...@@ -201,8 +201,8 @@ namespace WindBot.Game.AI.Decks ...@@ -201,8 +201,8 @@ namespace WindBot.Game.AI.Decks
public override void OnChainSolved(int chainIndex) public override void OnChainSolved(int chainIndex)
{ {
ClientCard currentCard = Duel.GetCurrentSolvingChainCard(); ChainInfo currentCard = Duel.GetCurrentSolvingChainInfo();
if (currentCard != null && currentCard.Controller == 1) if (currentCard != null && currentCard.ActivatePlayer == 1)
{ {
if (Duel.IsCurrentSolvingChainNegated()) if (Duel.IsCurrentSolvingChainNegated())
{ {
...@@ -212,16 +212,16 @@ namespace WindBot.Game.AI.Decks ...@@ -212,16 +212,16 @@ namespace WindBot.Game.AI.Decks
if (Bot.MonsterZone.GetFirstMatchingCard(c => c.HasRace(CardRace.SpellCaster) && c.IsFaceup()) != null if (Bot.MonsterZone.GetFirstMatchingCard(c => c.HasRace(CardRace.SpellCaster) && c.IsFaceup()) != null
&& Bot.HasInSpellZone(CardId.MagicianRightHand, true)) && Bot.HasInSpellZone(CardId.MagicianRightHand, true))
{ {
Logger.DebugWriteLine("MagicianRightHand negate: " + currentCard.Name ?? "???"); Logger.DebugWriteLine("MagicianRightHand negate: " + currentCard.RelatedCard.Name ?? "???");
MagicianRightHand_used = true; MagicianRightHand_used = true;
} }
} }
if (!MagiciansLeftHand_used && currentCard.IsTrap() && currentCard.Controller == 1) if (!MagiciansLeftHand_used && currentCard.IsTrap() && currentCard.ActivatePlayer == 1)
{ {
if (Bot.MonsterZone.GetFirstMatchingCard(c => c.HasRace(CardRace.SpellCaster) && c.IsFaceup()) != null if (Bot.MonsterZone.GetFirstMatchingCard(c => c.HasRace(CardRace.SpellCaster) && c.IsFaceup()) != null
&& Bot.HasInSpellZone(CardId.MagiciansLeftHand, true)) && Bot.HasInSpellZone(CardId.MagiciansLeftHand, true))
{ {
Logger.DebugWriteLine("MagiciansLeftHand negate: " + currentCard.Name ?? "???"); Logger.DebugWriteLine("MagiciansLeftHand negate: " + currentCard.RelatedCard.Name ?? "???");
MagiciansLeftHand_used = true; MagiciansLeftHand_used = true;
} }
} }
...@@ -236,7 +236,7 @@ namespace WindBot.Game.AI.Decks ...@@ -236,7 +236,7 @@ namespace WindBot.Game.AI.Decks
{ {
for (int i = 0; i < 5; ++i) for (int i = 0; i < 5; ++i)
{ {
if (Enemy.SpellZone[i] == currentCard) if (Enemy.SpellZone[i] == currentCard.RelatedCard)
{ {
Impermanence_list.Add(4 - i); Impermanence_list.Add(4 - i);
break; break;
......
...@@ -16,9 +16,6 @@ namespace WindBot.Game.AI ...@@ -16,9 +16,6 @@ namespace WindBot.Game.AI
public GameAI AI { get; private set; } public GameAI AI { get; private set; }
public AIUtil Util { get; private set; } public AIUtil Util { get; private set; }
protected MainPhase Main { get; private set; }
protected BattlePhase Battle { get; private set; }
protected ExecutorType Type { get; private set; } protected ExecutorType Type { get; private set; }
protected ClientCard Card { get; private set; } protected ClientCard Card { get; private set; }
protected int ActivateDescription { get; private set; } protected int ActivateDescription { get; private set; }
...@@ -247,16 +244,6 @@ namespace WindBot.Game.AI ...@@ -247,16 +244,6 @@ namespace WindBot.Game.AI
return; return;
} }
public void SetMain(MainPhase main)
{
Main = main;
}
public void SetBattle(BattlePhase battle)
{
Battle = battle;
}
/// <summary> /// <summary>
/// Set global variables Type, Card, ActivateDescription for Executor /// Set global variables Type, Card, ActivateDescription for Executor
/// </summary> /// </summary>
......
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using YGOSharp.OCGWrapper;
using YGOSharp.OCGWrapper.Enums;
namespace WindBot.Game
{
public class ChainInfo
{
public ClientCard RelatedCard { get; private set; }
public int ActivatePlayer { get; private set; }
public int ActivateId { get; private set; }
public int ActivateController { get; private set; }
public int ActivatePosition { get; private set; }
public int ActivateSequence { get; private set; }
public CardLocation ActivateLocation { get; private set; }
public int ActivateLevel { get; private set; }
public int ActivateRank { get; private set; }
public int ActivateType { get; private set; }
public int ActivateRace { get; private set; }
public int ActivateAttack { get; private set; }
public int ActivateDefense { get; private set; }
public bool IsSpecialSummoned { get; private set; }
public int ActivateDescription { get; private set; }
public ChainInfo(ClientCard card)
: this(card, card.Controller, 0)
{
}
public ChainInfo(ClientCard card, int player, int desc)
{
RelatedCard = card;
ActivatePlayer = player;
ActivateId = card.Id;
ActivateController = card.Controller;
ActivatePosition = card.Position;
ActivateSequence = card.Sequence;
ActivateLocation = card.Location;
ActivateLevel = card.Level;
ActivateRank = card.Rank;
ActivateType = card.Type;
ActivateRace = card.Race;
ActivateAttack = card.Attack;
ActivateDefense = card.Defense;
ActivateAttack = card.Attack;
ActivateDefense = card.Defense;
IsSpecialSummoned = card.IsSpecialSummoned;
ActivateDescription = desc;
}
public bool HasPosition(CardPosition position)
{
return (ActivatePosition & (int)position) != 0;
}
public bool HasLocation(CardLocation location)
{
return ((int)ActivateLocation & (int)location) != 0;
}
public bool IsCode(int id)
{
return RelatedCard != null && RelatedCard.IsCode(id);
}
public bool IsCode(IList<int> ids)
{
return RelatedCard != null && RelatedCard.IsCode(ids);
}
public bool IsCode(params int[] ids)
{
return RelatedCard != null && RelatedCard.IsCode(ids);
}
public bool HasType(CardType type)
{
return RelatedCard != null && (RelatedCard.Type & (int)type) != 0;
}
public bool IsSpell()
{
return HasType(CardType.Spell);
}
public bool IsTrap()
{
return HasType(CardType.Trap);
}
}
}
...@@ -20,6 +20,7 @@ namespace WindBot.Game ...@@ -20,6 +20,7 @@ namespace WindBot.Game
public int LastChainPlayer { get; set; } public int LastChainPlayer { get; set; }
public CardLocation LastChainLocation { get; set; } public CardLocation LastChainLocation { get; set; }
public IList<ClientCard> CurrentChain { get; set; } public IList<ClientCard> CurrentChain { get; set; }
public IList<ChainInfo> CurrentChainInfo { get; set; }
public IList<ClientCard> ChainTargets { get; set; } public IList<ClientCard> ChainTargets { get; set; }
public IList<ClientCard> LastChainTargets { get; set; } public IList<ClientCard> LastChainTargets { get; set; }
public IList<ClientCard> ChainTargetOnly { get; set; } public IList<ClientCard> ChainTargetOnly { get; set; }
...@@ -37,6 +38,7 @@ namespace WindBot.Game ...@@ -37,6 +38,7 @@ namespace WindBot.Game
LastChainPlayer = -1; LastChainPlayer = -1;
LastChainLocation = 0; LastChainLocation = 0;
CurrentChain = new List<ClientCard>(); CurrentChain = new List<ClientCard>();
CurrentChainInfo = new List<ChainInfo>();
ChainTargets = new List<ClientCard>(); ChainTargets = new List<ClientCard>();
LastChainTargets = new List<ClientCard>(); LastChainTargets = new List<ClientCard>();
ChainTargetOnly = new List<ClientCard>(); ChainTargetOnly = new List<ClientCard>();
...@@ -45,6 +47,8 @@ namespace WindBot.Game ...@@ -45,6 +47,8 @@ namespace WindBot.Game
LastSummonedCards = new List<ClientCard>(); LastSummonedCards = new List<ClientCard>();
SolvingChainIndex = 0; SolvingChainIndex = 0;
NegatedChainIndexList = new List<int>(); NegatedChainIndexList = new List<int>();
MainPhase = new MainPhase();
BattlePhase = new BattlePhase();
} }
public ClientCard GetCard(int player, CardLocation loc, int seq) public ClientCard GetCard(int player, CardLocation loc, int seq)
...@@ -180,6 +184,12 @@ namespace WindBot.Game ...@@ -180,6 +184,12 @@ namespace WindBot.Game
return CurrentChain[SolvingChainIndex - 1]; return CurrentChain[SolvingChainIndex - 1];
} }
public ChainInfo GetCurrentSolvingChainInfo()
{
if (SolvingChainIndex == 0 || SolvingChainIndex > CurrentChainInfo.Count) return null;
return CurrentChainInfo[SolvingChainIndex - 1];
}
public bool IsCurrentSolvingChainNegated() public bool IsCurrentSolvingChainNegated()
{ {
return SolvingChainIndex > 0 && NegatedChainIndexList.Contains(SolvingChainIndex); return SolvingChainIndex > 0 && NegatedChainIndexList.Contains(SolvingChainIndex);
......
...@@ -193,7 +193,6 @@ namespace WindBot.Game ...@@ -193,7 +193,6 @@ namespace WindBot.Game
/// <returns>A new BattlePhaseAction containing the action to do.</returns> /// <returns>A new BattlePhaseAction containing the action to do.</returns>
public BattlePhaseAction OnSelectBattleCmd(BattlePhase battle) public BattlePhaseAction OnSelectBattleCmd(BattlePhase battle)
{ {
Executor.SetBattle(battle);
foreach (CardExecutor exec in Executor.Executors) foreach (CardExecutor exec in Executor.Executors)
{ {
if (exec.Type == ExecutorType.GoToMainPhase2 && battle.CanMainPhaseTwo && exec.Func()) // check if should enter main phase 2 directly if (exec.Type == ExecutorType.GoToMainPhase2 && battle.CanMainPhaseTwo && exec.Func()) // check if should enter main phase 2 directly
...@@ -440,7 +439,6 @@ namespace WindBot.Game ...@@ -440,7 +439,6 @@ namespace WindBot.Game
/// <returns>A new MainPhaseAction containing the action to do.</returns> /// <returns>A new MainPhaseAction containing the action to do.</returns>
public MainPhaseAction OnSelectIdleCmd(MainPhase main) public MainPhaseAction OnSelectIdleCmd(MainPhase main)
{ {
Executor.SetMain(main);
CheckSurrender(); CheckSurrender();
foreach (CardExecutor exec in Executor.Executors) foreach (CardExecutor exec in Executor.Executors)
{ {
......
...@@ -433,6 +433,7 @@ namespace WindBot.Game ...@@ -433,6 +433,7 @@ namespace WindBot.Game
// in case of ending duel in chain's solving // in case of ending duel in chain's solving
_duel.CurrentChain.Clear(); _duel.CurrentChain.Clear();
_duel.CurrentChainInfo.Clear();
_duel.ChainTargets.Clear(); _duel.ChainTargets.Clear();
_duel.ChainTargetOnly.Clear(); _duel.ChainTargetOnly.Clear();
_duel.SummoningCards.Clear(); _duel.SummoningCards.Clear();
...@@ -813,6 +814,8 @@ namespace WindBot.Game ...@@ -813,6 +814,8 @@ namespace WindBot.Game
if (card.Id == 0) if (card.Id == 0)
card.SetId(cardId); card.SetId(cardId);
int cc = GetLocalPlayer(packet.ReadByte()); int cc = GetLocalPlayer(packet.ReadByte());
packet.ReadInt16(); // trigger location + trigger sequence
int desc = packet.ReadInt32();
if (_debug) if (_debug)
if (card != null) Logger.WriteLine("(" + cc.ToString() + " 's " + (card.Name ?? "UnKnowCard") + " activate effect from " + (CardLocation)pcl + ")"); if (card != null) Logger.WriteLine("(" + cc.ToString() + " 's " + (card.Name ?? "UnKnowCard") + " activate effect from " + (CardLocation)pcl + ")");
_duel.LastChainLocation = (CardLocation)pcl; _duel.LastChainLocation = (CardLocation)pcl;
...@@ -821,6 +824,7 @@ namespace WindBot.Game ...@@ -821,6 +824,7 @@ namespace WindBot.Game
_duel.ChainTargetOnly.Clear(); _duel.ChainTargetOnly.Clear();
_duel.LastSummonPlayer = -1; _duel.LastSummonPlayer = -1;
_duel.CurrentChain.Add(card); _duel.CurrentChain.Add(card);
_duel.CurrentChainInfo.Add(new ChainInfo(card, cc, desc));
_duel.LastChainPlayer = cc; _duel.LastChainPlayer = cc;
} }
...@@ -855,6 +859,7 @@ namespace WindBot.Game ...@@ -855,6 +859,7 @@ namespace WindBot.Game
_duel.LastChainPlayer = -1; _duel.LastChainPlayer = -1;
_duel.LastChainLocation = 0; _duel.LastChainLocation = 0;
_duel.CurrentChain.Clear(); _duel.CurrentChain.Clear();
_duel.CurrentChainInfo.Clear();
_duel.ChainTargets.Clear(); _duel.ChainTargets.Clear();
_duel.LastChainTargets.Clear(); _duel.LastChainTargets.Clear();
_duel.ChainTargetOnly.Clear(); _duel.ChainTargetOnly.Clear();
......
...@@ -10,7 +10,7 @@ using System.Runtime.InteropServices; ...@@ -10,7 +10,7 @@ using System.Runtime.InteropServices;
[assembly: AssemblyConfiguration("")] [assembly: AssemblyConfiguration("")]
[assembly: AssemblyCompany("IceYGO")] [assembly: AssemblyCompany("IceYGO")]
[assembly: AssemblyProduct("WindBot")] [assembly: AssemblyProduct("WindBot")]
[assembly: AssemblyCopyright("Copyright © IceYGO 2015-2017")] [assembly: AssemblyCopyright("Copyright © IceYGO 2015-2025")]
[assembly: AssemblyTrademark("")] [assembly: AssemblyTrademark("")]
[assembly: AssemblyCulture("")] [assembly: AssemblyCulture("")]
......
# WindBot # WindBot
A C# bot for YGOPro, compatible with the [YGOSharp](https://github.com/IceYGO/ygosharp) and [SRVPro](https://github.com/moecube/srvpro) server. A C# bot for [YGOPro](https://github.com/Fluorohydride/ygopro), compatible with the [YGOSharp](https://github.com/IceYGO/ygosharp) and [SRVPro](https://github.com/mycard/srvpro) server.
### How to use: ### How to use:
...@@ -154,6 +154,11 @@ In this situation, it will be multi-threaded. This can be useful for servers, si ...@@ -154,6 +154,11 @@ In this situation, it will be multi-threaded. This can be useful for servers, si
The parameters are same as commandlines, but low cased. The parameters are same as commandlines, but low cased.
Note: Currently the server bind to all interfaces, so it requires elevated privileges to run. You can otherwise use the following command to add a URL ACL for your port (2399 for example), which allows all users to access it:
```
netsh http add urlacl url=http://+:2399/ user=Everyone
```
### Known issues ### Known issues
* If one chain includes two activation that use `AI.SelectCard`, the second one won't select correctly. * If one chain includes two activation that use `AI.SelectCard`, the second one won't select correctly.
......
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
<AppDesignerFolder>Properties</AppDesignerFolder> <AppDesignerFolder>Properties</AppDesignerFolder>
<RootNamespace>WindBot</RootNamespace> <RootNamespace>WindBot</RootNamespace>
<AssemblyName>WindBot</AssemblyName> <AssemblyName>WindBot</AssemblyName>
<TargetFrameworkVersion>v4.0</TargetFrameworkVersion> <TargetFrameworkVersion>v4.8</TargetFrameworkVersion>
<FileAlignment>512</FileAlignment> <FileAlignment>512</FileAlignment>
<TargetFrameworkProfile /> <TargetFrameworkProfile />
</PropertyGroup> </PropertyGroup>
...@@ -21,6 +21,7 @@ ...@@ -21,6 +21,7 @@
<PlatformTarget>x86</PlatformTarget> <PlatformTarget>x86</PlatformTarget>
<ErrorReport>prompt</ErrorReport> <ErrorReport>prompt</ErrorReport>
<CodeAnalysisRuleSet>MinimumRecommendedRules.ruleset</CodeAnalysisRuleSet> <CodeAnalysisRuleSet>MinimumRecommendedRules.ruleset</CodeAnalysisRuleSet>
<Prefer32Bit>false</Prefer32Bit>
</PropertyGroup> </PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Release|x86'"> <PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Release|x86'">
<OutputPath>bin\Release\</OutputPath> <OutputPath>bin\Release\</OutputPath>
...@@ -30,9 +31,10 @@ ...@@ -30,9 +31,10 @@
<PlatformTarget>x86</PlatformTarget> <PlatformTarget>x86</PlatformTarget>
<ErrorReport>prompt</ErrorReport> <ErrorReport>prompt</ErrorReport>
<CodeAnalysisRuleSet>MinimumRecommendedRules.ruleset</CodeAnalysisRuleSet> <CodeAnalysisRuleSet>MinimumRecommendedRules.ruleset</CodeAnalysisRuleSet>
<Prefer32Bit>false</Prefer32Bit>
</PropertyGroup> </PropertyGroup>
<PropertyGroup> <PropertyGroup>
<StartupObject /> <StartupObject>WindBot.Program</StartupObject>
</PropertyGroup> </PropertyGroup>
<PropertyGroup> <PropertyGroup>
<ApplicationIcon>WindBot.ico</ApplicationIcon> <ApplicationIcon>WindBot.ico</ApplicationIcon>
...@@ -40,6 +42,12 @@ ...@@ -40,6 +42,12 @@
<PropertyGroup> <PropertyGroup>
<AutoGenerateBindingRedirects>true</AutoGenerateBindingRedirects> <AutoGenerateBindingRedirects>true</AutoGenerateBindingRedirects>
</PropertyGroup> </PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|AnyCPU'">
<Prefer32Bit>false</Prefer32Bit>
</PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Release|AnyCPU'">
<Prefer32Bit>false</Prefer32Bit>
</PropertyGroup>
<ItemGroup> <ItemGroup>
<Reference Include="Mono.Data.Sqlite"> <Reference Include="Mono.Data.Sqlite">
<HintPath>.\Mono.Data.Sqlite.dll</HintPath> <HintPath>.\Mono.Data.Sqlite.dll</HintPath>
...@@ -137,6 +145,7 @@ ...@@ -137,6 +145,7 @@
<Compile Include="Game\AI\ExecutorType.cs" /> <Compile Include="Game\AI\ExecutorType.cs" />
<Compile Include="Game\BattlePhase.cs" /> <Compile Include="Game\BattlePhase.cs" />
<Compile Include="Game\BattlePhaseAction.cs" /> <Compile Include="Game\BattlePhaseAction.cs" />
<Compile Include="Game\ChainInfo.cs" />
<Compile Include="Game\ClientCard.cs" /> <Compile Include="Game\ClientCard.cs" />
<Compile Include="Game\ClientField.cs" /> <Compile Include="Game\ClientField.cs" />
<Compile Include="Game\Deck.cs" /> <Compile Include="Game\Deck.cs" />
......
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