HearthBuddy BotManager
MainWindow
private void button_0_Click(object sender, RoutedEventArgs e)
{
Configuration.Instance.SaveAll();
this.button_0.IsEnabled = false;
this.comboBox_1.IsEnabled = false;
this.comboBox_0.IsEnabled = false;
this.checkBox_0.IsEnabled = false;
this.listBox_0.IsEnabled = false;
try
{
this.bool_0 = false;
if (BotManager.IsRunning)
{
BotManager.Stop();
}
else
{
BotManager.Start();
}
}
finally
{
this.bool_0 = true;
}
this.stopwatch_0.Restart();
}
BotManager
public static bool Start()
{
object obj = BotManager.object_0;
bool result;
lock (obj)
{
if (BotManager.IsRunning)
{
BotManager.ilog_0.ErrorFormat("[Start] The BotThread is already running. Please use BotManager.Stop first.", Array.Empty<object>());
result = false;
}
else if (!TritonHs.Initialized)
{
BotManager.ilog_0.ErrorFormat("[Start] TritonHs is not initialized yet.", Array.Empty<object>());
result = false;
}
else if (BotManager.CurrentBot == null)
{
BotManager.ilog_0.ErrorFormat("[Start] There is no bot to run. Please assign a bot first.", Array.Empty<object>());
result = false;
}
else
{
BotManager.ilog_0.InfoFormat("[Start] Now creating the BotThread.", Array.Empty<object>());
BotManager.bool_3 = false;
BotManager.BotThread = new Thread(new ThreadStart(BotManager.smethod_4));
BotManager.BotThread.Start();
result = true;
}
}
return result;
}
private static void smethod_4()
{
object obj = BotManager.object_0;
lock (obj)
{
Thread.Sleep();
}
TritonHs.Memory.DisableCache();
TritonHs.Memory.ClearCache();
TritonHs.Memory.Executor.FrameDropWaitTime = 15000u;
TritonHs.Memory.Executor.ExecuteWaitTime = ;
try
{
TritonHs.smethod_2(true);
}
catch
{
}
try
{
BotManager.smethod_0(BotManager.CurrentBot);
goto IL_FA;
}
catch
{
BotManager.autoResetEvent_0.Set();
goto IL_FA;
}
IL_87:
try
{
if (BotManager.MsBeforeNextTick != )
{
Thread.Sleep(BotManager.MsBeforeNextTick);
BotManager.MsBeforeNextTick = ;
}
BotManager.smethod_1(BotManager.CurrentBot);
BotManager.bool_3 = false;
if (BotManager.MsBetweenTicks != )
{
Thread.Sleep(BotManager.MsBetweenTicks);
}
}
catch (InjectionDesyncException)
{
BotManager.ilog_0.DebugFormat("[BotThreadFunction] An InjectionDesyncException was detected.", Array.Empty<object>());
BotManager.bool_3 = true;
TritonHs.InvokeEvent(BotManager.eventHandler_1, new object[]
{
null,
new ClientFrozenEventArgs()
});
}
catch
{
}
IL_FA:
if (!BotManager.autoResetEvent_0.WaitOne())
{
goto IL_87;
}
try
{
BotManager.smethod_2(BotManager.CurrentBot);
}
catch
{
}
BotManager.BotThread = null;
if (BotManager.bool_3)
{
TritonHs.InvokeEvent(BotManager.eventHandler_1, new object[]
{
null,
new ClientFrozenEventArgs()
});
return;
}
try
{
TritonHs.smethod_2(true);
}
catch
{
}
}
重点在ibot_1.Start();,这里启动后,进入DefaultBot的流程
internal static void smethod_0(IBot ibot_1)
{
try
{
object obj = BotManager.object_0;
lock (obj)
{
if (BotManager.bool_0)
{
return;
}
BotManager.bool_0 = true;
}
BotManager.smethod_3(ibot_1, BotManager.botEvent_0);
try
{
using (TritonHs.AcquireFrame())
{
using (TritonHs.Memory.TemporaryCacheState(false))
{
TritonHs.Memory.ClearCache();
ibot_1.Start();
}
}
}
catch (Exception exception)
{
BotManager.ilog_0.Error("[Start] Exception during execution:", exception);
throw;
}
BotManager.smethod_3(ibot_1, BotManager.botEvent_1);
}
finally
{
object obj = BotManager.object_0;
lock (obj)
{
BotManager.bool_0 = false;
}
}
}
MainWindow
private void method_11(IBot ibot_0)
{
base.Dispatcher.BeginInvoke(new Action(this.method_35), Array.Empty<object>());
if (this.bool_0)
{
base.Dispatcher.BeginInvoke(new Action(this.method_36), Array.Empty<object>());
}
}
DefaultBot
namespace Triton.Bot.Logic.Bots.DefaultBot
{
// Token: 0x02000256 RID: 598
public class DefaultBot : IRunnable, IAuthored, IBase, IBot, IConfigurable
public void Start()
{
this.coroutine_0 = new Coroutine(new Func<Task>(this.method_53));
this.bool_2 = false;
GameEventManager.Instance.Start();
PluginManager.Start();
RoutineManager.Start();
ProcessHookManager.Enable();
GameEventManager.NewGame += this.method_18;
GameEventManager.GameOver += this.method_19;
GameEventManager.MulliganConfirm += this.method_16;
InactivePlayerKicker inactivePlayerKicker = InactivePlayerKicker.Get();
if (inactivePlayerKicker != null)
{
inactivePlayerKicker.SetShouldCheckForInactivity(false);
}
}
DefaultBot.Tick方法中,以下三行代码
GameEventManager.Instance.Tick();
PluginManager.Tick();
RoutineManager.Tick();
public void Tick()
{
if (this.coroutine_0.IsFinished)
{
DefaultBot.ilog_0.DebugFormat("The bot coroutine has finished in a state of {0}", this.coroutine_0.Status);
BotManager.Stop();
return;
}
if (!TritonHs.IsClientInUsableState(true))
{
BotManager.MsBeforeNextTick += ;
this.method_17();
return;
}
if (!this.bool_2 && ChatMgr.Get().FriendListFrame != null)
{
DefaultBot.ilog_0.ErrorFormat("[Tick] FriendListFrame != null.", Array.Empty<object>());
Client.LeftClickAtDialog(BnetBar.Get().m_friendButton.m_OnlineCountText.Transform.Position);
BotManager.MsBeforeNextTick += ;
this.method_17();
return;
}
bool flag;
if (TritonHs.HandleDialog(new ShouldAcceptFriendlyChallengeDelegate(this.ShouldAcceptFriendlyChallenge), out flag))
{
if (flag)
{
BotManager.Stop();
}
BotManager.MsBeforeNextTick += ;
this.method_17();
return;
}
GameEventManager.Instance.Tick();
PluginManager.Tick();
RoutineManager.Tick();
try
{
this.coroutine_0.Resume();
}
catch
{
this.method_17();
throw;
}
}
RoutineManager.Tick
RoutineManager.CurrentRoutine就是silver fish
public static void Tick()
{
RoutineManager.Tick(RoutineManager.CurrentRoutine);
}
调试的时候,routine.Tick();会自动反编译,加载出一个DefaultRoutine.dll【这里的调试必须是F11】
public static void Tick(IRoutine routine)
{
RoutineManager.smethod_0(routine, RoutineManager.routineEvent_2);
try
{
routine.Tick();
}
catch (Exception exception)
{
RoutineManager.ilog_0.Error("Exception during routine Tick.", exception);
}
RoutineManager.smethod_0(routine, RoutineManager.routineEvent_3);
}
namespace Buddy.Coroutines
{
// Token: 0x0200009E RID: 158
public sealed class Coroutine : IDisposable
public void Resume()
{
if (this.IsDisposed)
{
throw new ObjectDisposedException(base.GetType().FullName);
}
if (this.IsFinished)
{
throw new InvalidOperationException("This coroutine has finished execution and cannot be resumed.");
}
if (Coroutine.coroutine_0 == this)
{
throw new InvalidOperationException("A coroutine cannot resume itself");
}
this.method_0(false);
}
private void method_0(bool bool_2)
{
SynchronizationContext synchronizationContext = SynchronizationContext.Current;
Coroutine coroutine = Coroutine.coroutine_0;
try
{
Coroutine.coroutine_0 = this;
SynchronizationContext.SetSynchronizationContext(null);
Action action = this.action_0;
this.action_0 = null;
action();
if (!bool_2)
{
this.Ticks++;
}
this.method_2(bool_2);
}
finally
{
Coroutine.coroutine_0 = coroutine;
SynchronizationContext.SetSynchronizationContext(synchronizationContext);
}
}
[CompilerGenerated]
private sealed class Class67
public void method_0()
{
Task<object> task;
try
{
task = this.func_0();
}
catch (Exception innerException)
{
throw this.coroutine_0.method_1(new CoroutineUnhandledException("Exception was thrown by root coroutine task producer", innerException));
}
if (task == null)
{
throw this.coroutine_0.method_1(new CoroutineBehaviorException("The root coroutine task producer returned null"));
}
this.coroutine_0.task_0 = task;
}
上面的default routine加载后,直接搜索getHoldList
定位到出错的地方
ERROR Logger (null) - [Tick] Exception during execution:
Buddy.Coroutines.CoroutineUnhandledException: Exception was thrown by coroutine ---> System.NullReferenceException: Object reference not set to an instance of an object.
at HREngine.Bots.Mulligan.getHoldList(MulliganData mulliganData, Behavior behave) in
Routines\DefaultRoutine\Silverfish\ai\Mulligan.cs:line 307
at HREngine.Bots.DefaultRoutine.d__41.MoveNext() in
public async Task<bool> Logic(string type, object context)
{
bool flag = type == "mulligan";
bool result;
if (flag)
{
await this.MulliganLogic(context as MulliganData);
result = true;
}
else if (type == "emote")
{
await this.EmoteLogic(context as EmoteData);
result = true;
}
else if (type == "our_turn")
{
await this.OurTurnLogic();
result = true;
}
else if (type == "opponent_turn")
{
await this.OpponentTurnLogic();
result = true;
}
else if (type == "our_turn_combat")
{
await this.OurTurnCombatLogic();
result = true;
}
else if (type == "opponent_turn_combat")
{
await this.OpponentTurnCombatLogic();
result = true;
}
else if (type == "arena_draft")
{
await this.ArenaDraftLogic(context as ArenaDraftData);
result = true;
}
else if (type == "handle_quests")
{
await this.HandleQuestsLogic(context as QuestData);
result = true;
}
else
{
result = false;
}
return result;
}
在DefaultRoutine加载后,可以直接添加监视
CardDB.Instance.cardIdstringToEnum("ULD_194") ,ULD_194 ,HREngine.Bots.CardDB.cardIDEnum
- CardDB.Instance.getCardDataFromID(HREngine.Bots.CardDB.cardIDEnum.ULD_194) ,{HREngine.Bots.CardDB.Card} ,HREngine.Bots.CardDB.Card
cost是0x3E8=1000,name是unknown
HearthBuddy BotManager的更多相关文章
- HearthBuddy Plugin编写遇到的问题
错误1 赋值问题 貌似编译器版本有点低,无法识别C#的高级语法糖 属性的初始值,必须是public bool IsEnabled { get{return true;} } 不能写成public bo ...
- HearthBuddy的plugin加载
// Hearthbuddy.Windows.MainWindow // Token: 0x060001FF RID: 511 RVA: 0x0008951C File Offset: 0x00087 ...
- HearthBuddy修改系统时间
将以下代码保存在.bat文件,然后用管理员权限运行 pushd "%~dp0" #下面修改时间,根据操作系统的语言不同,会有不同的格式,比如2019-10-26date 10/26 ...
- HearthBuddy的class276以及class247
使用de4dot-cex反编译原版的hearthbuddy得到的 链接: https://pan.baidu.com/s/1hT79LpIjbyvODsjnkSe_5A 提取码: iemx class ...
- HearthBuddy Ai调试实战1-->出牌的时候,少召唤了图腾就结束回合
期望通过ai的调试,来搞明白出牌的逻辑. 55是投火无面者63是恐狼前锋34是风怒36是自动漩涡打击装置13是空灵召唤者, "LocStringZhCn": "<b ...
- HearthBuddy 日志模块
// Triton.Common.LogUtilities.CustomLogger // Token: 0x04000BD8 RID: 3032 private Level level_0 = Le ...
- HearthBuddy炉石兄弟 如何调试ai
Sepefeets's update to botmaker's Silverfish AI This AI is a Custom Class for Hearthranger and Hearth ...
- HearthBuddy CSharpCodeProvider 如何编译cs文件
源码 源码1 namespace Hearthbuddy.Windows{ // Token: 0x02000041 RID: 65 public class MainWindow : W ...
- HearthBuddy DONOTDELETE.bin
namespace Hearthbuddy{ // Token: 0x02000022 RID: 34 public class App : System.Windows.Applicat ...
随机推荐
- input type=color 设置颜色
在设置背景色的时候,使用html5 type=color 标签,但是初始值一直都是黑色的,背景如果没有设置的时候,应该是白色,比如文本图元,所以需要设置一个初始的颜色值, 注意: value不实用,怎 ...
- sql 将英文句子中的单词首字母转换为大写
create function dbo.pTitleCase(@StrIn nvarchar(max))returns nvarchar(max)as begin; declare @StrOut n ...
- ORACLE和SQL查询库数据量
ORACLE根据账号查询每张表数据量: select t.table_name,t.num_rows from user_tables t ORDER BY NUM_ROWS DESC; SQL SE ...
- phpstorm 习惯设置
phpstorm 习惯设置 1. 字体:Source Code Pro 大小:14 链接: https://pan.baidu.com/s/1HLpbduBHFvbq1a10QV4uCg 提取码: y ...
- 这些JVM命令配置参数你知道吗?
JVM是多数开发人员视为理所当然的Java功能和性能背后的重负荷机器.然而,我们很少有人能理解JVM是如何进行工作的—像任务分配和垃圾收集.转动线程.打开和关闭文件.中断和/或JIT编译Java字节码 ...
- Django项目实战—用户头像上传
1 将文件保存到服务器本地 upload.html <!DOCTYPE html> <html lang="en"> <head> <me ...
- 使用 visual studio 2019 社区版打开touchgfx工程注意项
@2019-09-23 [环境] touchgfx designer 4.10.0 visual studio 2019 社区版 [问题] #error 1 使用 visual studio 2019 ...
- PAT Advanced 1153 Decode Registration Card of PAT (25 分)
A registration card number of PAT consists of 4 parts: the 1st letter represents the test level, nam ...
- find 命令和sed命令练习
1.查找/var目录下属主为root,且属组为mail的所有文件 find /var/ -user root -group mail 2.查找/var目录下不属于root.lp.gdm的所有文件 fi ...
- 软件测试能满足测试的sql
作为一个软件测试工程师,我们在测试过程中往往需要对数据库数据进行操作,但是我们的操作大多以查询居多,有时会涉及到新增,修改,删除等操作,所以我们其实并不需要对数据库的操作有特别深入的了解,以下是我在工 ...