Advertisement
Guest User

Untitled

a guest
Oct 7th, 2018
171
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
text 23.02 KB | None | 0 0
  1. // This script mines ore from asteroids.
  2. // before running this script, make sure to prepare as follows:
  3. // +enter bookmark for mining site and bookmark for station in the configuration section below.
  4. // +in the Overview create a preset which includes asteroids and rats and enter the name of that preset in the configuration section below at 'OverviewPreset'. The bot will make sure this preset is loaded when it needs to use the overview.
  5. // +set Overview to sort by distance with the nearest entry at the top.
  6. // +in the Inventory select the 'List' view.
  7. // +set the UI language to english.
  8. // +use a ship with an ore hold.
  9. // +put some light drones into your ships' drone bay. The bot will make use of them to attack rats when HP are too low (configurable) or it gets jammed.
  10. // +enable the info panel 'System info'. The bot will use the button in there to access bookmarks and asteroid belts.
  11. // +arrange windows to not occlude modules or info panels.
  12. // +in the ship UI, disable "Display Passive Modules" and disable "Display Empty Slots" and enable "Display Module Tooltips". The bot uses the module tooltips to automatically identify the properties of the modules.
  13. //
  14. // for optional features (such as warp to safe on hostile in local) see the configuration section below.
  15.  
  16. using BotSharp.ToScript.Extension;
  17. using Parse = Sanderling.Parse;
  18.  
  19. // begin of configuration section ->
  20.  
  21. // The bot uses the bookmarks from the menu which is opened from the button in the 'System info' panel.
  22.  
  23. // Bookmarks of places to mine. Add additional bookmarks separated by comma.
  24. string[] SetMiningSiteBookmark = new[] {
  25. "ore",
  26. };
  27.  
  28. // Bookmark of location where ore should be unloaded.
  29. string UnloadBookmark = "station";
  30.  
  31. // Name of the container to unload to as shown in inventory.
  32. string UnloadDestContainerName = "Item Hangar";
  33.  
  34. // when this is set to true, the bot will try to unload when undocked.
  35. bool UnloadInSpace = false;
  36.  
  37. // Bookmark of place to retreat to to prevent ship loss.
  38. string RetreatBookmark = UnloadBookmark;
  39.  
  40. // The bot loads this preset to the active tab.
  41. string OverviewPreset = null;
  42.  
  43. var ActivateHardener = true; // activate shield hardener.
  44.  
  45. // bot will start fighting (and stop mining) when hitpoints are lower.
  46. var DefenseEnterHitpointThresholdPercent = 85;
  47. var DefenseExitHitpointThresholdPercent = 90;
  48.  
  49. var EmergencyWarpOutHitpointPercent = 60;
  50.  
  51. var FightAllRats = false; // when this is set to true, the bot will attack rats independent of shield hp.
  52.  
  53. var EnterOffloadOreHoldFillPercent = 95; // percentage of ore hold fill level at which to enter the offload process.
  54.  
  55. var RetreatOnNeutralOrHostileInLocal = false; // warp to RetreatBookmark when a neutral or hostile is visible in local.
  56.  
  57. bool returnDronesToBayOnRetreat = false; // when set to true, bot will attempt to dock back the drones before retreating
  58.  
  59. // <- end of configuration section
  60.  
  61.  
  62. Func<object> BotStopActivity = () => null;
  63.  
  64. Func<object> NextActivity = MainStep;
  65.  
  66. for(;;)
  67. {
  68. MemoryUpdate();
  69.  
  70. Host.Log(
  71. "ore hold fill: " + OreHoldFillPercent + "%" +
  72. ", mining range: " + MiningRange +
  73. ", mining modules (inactive): " + SetModuleMiner?.Length + "(" + SetModuleMinerInactive?.Length + ")" +
  74. ", shield.hp: " + ShieldHpPercent + "%" +
  75. ", retreat: " + RetreatReason +
  76. ", JLA: " + JammedLastAge +
  77. ", overview.rats: " + ListRatOverviewEntry?.Length +
  78. ", overview.roids: " + ListAsteroidOverviewEntry?.Length +
  79. ", offload count: " + OffloadCount +
  80. ", nextAct: " + NextActivity?.Method?.Name);
  81.  
  82. CloseModalUIElement();
  83.  
  84. if(0 < RetreatReason?.Length && !(Measurement?.IsDocked ?? false))
  85. {
  86. if (returnDronesToBayOnRetreat) {
  87. DroneEnsureInBay();
  88. }
  89.  
  90. if (!returnDronesToBayOnRetreat || (returnDronesToBayOnRetreat && 0 == DronesInSpaceCount)) {
  91. InitiateDockToOrWarpToBookmark(RetreatBookmark);
  92. }
  93. continue;
  94. }
  95.  
  96. NextActivity = NextActivity?.Invoke() as Func<object>;
  97.  
  98. if(BotStopActivity == NextActivity)
  99. break;
  100.  
  101. if(null == NextActivity)
  102. NextActivity = MainStep;
  103.  
  104. Host.Delay(1111);
  105. }
  106.  
  107. // seconds since ship was jammed.
  108. long? JammedLastAge => Jammed ? 0 : (Host.GetTimeContinuousMilli() - JammedLastTime) / 1000;
  109.  
  110. int? ShieldHpPercent => ShipUi?.HitpointsAndEnergy?.Shield / 10;
  111.  
  112. bool DefenseExit =>
  113. (Measurement?.IsDocked ?? false) ||
  114. !(0 < ListRatOverviewEntry?.Length) ||
  115. (DefenseExitHitpointThresholdPercent < ShieldHpPercent && !(JammedLastAge < 40) &&
  116. !(FightAllRats && 0 < ListRatOverviewEntry?.Length));
  117.  
  118. bool DefenseEnter =>
  119. !DefenseExit ||
  120. !(DefenseEnterHitpointThresholdPercent < ShieldHpPercent) || JammedLastAge < 10;
  121.  
  122. bool OreHoldFilledForOffload => Math.Max(0, Math.Min(100, EnterOffloadOreHoldFillPercent)) <= OreHoldFillPercent;
  123.  
  124. Int64? JammedLastTime = null;
  125. string RetreatReasonTemporary = null;
  126. string RetreatReasonPermanent = null;
  127. string RetreatReason => RetreatReasonPermanent ?? RetreatReasonTemporary;
  128. int? LastCheckOreHoldFillPercent = null;
  129.  
  130. int OffloadCount = 0;
  131.  
  132. Func<object> MainStep()
  133. {
  134. if(Measurement?.IsDocked ?? false)
  135. {
  136. InInventoryUnloadItems();
  137.  
  138. if (0 < RetreatReasonPermanent?.Length)
  139. return BotStopActivity;
  140.  
  141. if (0 < RetreatReason?.Length)
  142. return MainStep;
  143.  
  144. Undock();
  145. }
  146.  
  147. if(DefenseEnter)
  148. {
  149. Host.Log("enter defense.");
  150. return DefenseStep;
  151. }
  152.  
  153. EnsureOverviewTypeSelectionLoaded();
  154.  
  155. EnsureWindowInventoryOpenOreHold();
  156.  
  157. if(ReadyForManeuver)
  158. {
  159. DroneEnsureInBay();
  160.  
  161. if(OreHoldFilledForOffload && 0 == DronesInSpaceCount)
  162. {
  163. if(ReadyForManeuver)
  164. InitiateDockToOrWarpToBookmark(UnloadBookmark);
  165.  
  166. if (UnloadInSpace)
  167. {
  168. Host.Delay(4444);
  169. InInventoryUnloadItems();
  170. }
  171.  
  172. return MainStep;
  173. }
  174.  
  175. if(!(0 < ListAsteroidOverviewEntry?.Length))
  176. InitiateWarpToRandomMiningSite();
  177. }
  178.  
  179. ModuleMeasureAllTooltip();
  180.  
  181. if(ActivateHardener)
  182. ActivateHardenerExecute();
  183.  
  184. return InBeltMineStep;
  185. }
  186.  
  187. int RandomInt() => new Random((int)Host.GetTimeContinuousMilli()).Next();
  188.  
  189. T RandomElement<T>(IEnumerable<T> sequence)
  190. {
  191. var array = (sequence as T[]) ?? sequence?.ToArray();
  192.  
  193. if (!(0 < array?.Length))
  194. return default(T);
  195.  
  196. return array[RandomInt() % array.Length];
  197. }
  198.  
  199. void CloseModalUIElement()
  200. {
  201. var ButtonClose =
  202. ModalUIElement?.ButtonText?.FirstOrDefault(button => (button?.Text).RegexMatchSuccessIgnoreCase("close|no|ok"));
  203.  
  204. Sanderling.MouseClickLeft(ButtonClose);
  205. }
  206.  
  207. void DroneLaunch()
  208. {
  209. Host.Log("launch drones.");
  210. Sanderling.MouseClickRight(DronesInBayListEntry);
  211. Sanderling.MouseClickLeft(Menu?.FirstOrDefault()?.EntryFirstMatchingRegexPattern("launch", RegexOptions.IgnoreCase));
  212. }
  213.  
  214. void DroneEnsureInBay()
  215. {
  216. if(0 == DronesInSpaceCount)
  217. return;
  218.  
  219. DroneReturnToBay();
  220.  
  221. Host.Delay(4444);
  222. }
  223.  
  224. void DroneReturnToBay()
  225. {
  226. Host.Log("return drones to bay.");
  227. Sanderling.MouseClickRight(DronesInSpaceListEntry);
  228. Sanderling.MouseClickLeft(Menu?.FirstOrDefault()?.EntryFirstMatchingRegexPattern("return.*bay", RegexOptions.IgnoreCase));
  229. }
  230.  
  231. Func<object> DefenseStep()
  232. {
  233. if(DefenseExit)
  234. {
  235. Host.Log("exit defense.");
  236. return null;
  237. }
  238.  
  239. if (!(0 < DronesInSpaceCount))
  240. DroneLaunch();
  241.  
  242. EnsureOverviewTypeSelectionLoaded();
  243.  
  244. var SetRatName =
  245. ListRatOverviewEntry?.Select(entry => Regex.Split(entry?.Name ?? "", @"\s+")?.FirstOrDefault())
  246. ?.Distinct()
  247. ?.ToArray();
  248.  
  249. var SetRatTarget = Measurement?.Target?.Where(target =>
  250. SetRatName?.Any(ratName => target?.TextRow?.Any(row => row.RegexMatchSuccessIgnoreCase(ratName)) ?? false) ?? false);
  251.  
  252. var RatTargetNext = SetRatTarget?.OrderBy(target => target?.DistanceMax ?? int.MaxValue)?.FirstOrDefault();
  253.  
  254. if(null == RatTargetNext)
  255. {
  256. Host.Log("no rat targeted.");
  257. Sanderling.MouseClickRight(ListRatOverviewEntry?.FirstOrDefault());
  258. Sanderling.MouseClickLeft(MenuEntryLockTarget);
  259. }
  260. else
  261. {
  262. Host.Log("rat targeted. sending drones.");
  263. Sanderling.MouseClickLeft(RatTargetNext);
  264. Sanderling.MouseClickRight(DronesInSpaceListEntry);
  265. Sanderling.MouseClickLeft(Menu?.FirstOrDefault()?.EntryFirstMatchingRegexPattern("engage", RegexOptions.IgnoreCase));
  266. }
  267.  
  268. return DefenseStep;
  269. }
  270.  
  271. Func<object> InBeltMineStep()
  272. {
  273. if (DefenseEnter)
  274. {
  275. Host.Log("enter defense.");
  276. return DefenseStep;
  277. }
  278.  
  279. EnsureWindowInventoryOpenOreHold();
  280.  
  281. EnsureOverviewTypeSelectionLoaded();
  282.  
  283. if(OreHoldFilledForOffload)
  284. return null;
  285.  
  286. var moduleMinerInactive = SetModuleMinerInactive?.FirstOrDefault();
  287.  
  288. if (null == moduleMinerInactive)
  289. {
  290. Host.Delay(7777);
  291. return InBeltMineStep;
  292. }
  293.  
  294. var setTargetAsteroidInRange =
  295. SetTargetAsteroid?.Where(target => target?.DistanceMax <= MiningRange)?.ToArray();
  296.  
  297. var setTargetAsteroidInRangeNotAssigned =
  298. setTargetAsteroidInRange?.Where(target => !(0 < target?.Assigned?.Length))?.ToArray();
  299.  
  300. Host.Log("targeted asteroids in range (without assignment): " + setTargetAsteroidInRange?.Length + " (" + setTargetAsteroidInRangeNotAssigned?.Length + ")");
  301.  
  302. if(0 < setTargetAsteroidInRangeNotAssigned?.Length)
  303. {
  304. var targetAsteroidInputFocus =
  305. setTargetAsteroidInRangeNotAssigned?.FirstOrDefault(target => target?.IsSelected ?? false);
  306.  
  307. if(null == targetAsteroidInputFocus)
  308. Sanderling.MouseClickLeft(setTargetAsteroidInRangeNotAssigned?.FirstOrDefault());
  309.  
  310. ModuleToggle(moduleMinerInactive);
  311.  
  312. return InBeltMineStep;
  313. }
  314.  
  315. var asteroidOverviewEntryNext = ListAsteroidOverviewEntry?.FirstOrDefault();
  316. var asteroidOverviewEntryNextNotTargeted = ListAsteroidOverviewEntry?.FirstOrDefault(entry => !((entry?.MeTargeted ?? false) || (entry?.MeTargeting ?? false)));
  317.  
  318. Host.Log("next asteroid: (" + asteroidOverviewEntryNext?.Name + " , distance: " + asteroidOverviewEntryNext?.DistanceMax + ")" +
  319. ", next asteroid not targeted: (" + asteroidOverviewEntryNext?.Name + " , distance: " + asteroidOverviewEntryNext?.DistanceMax + ")");
  320.  
  321. if(null == asteroidOverviewEntryNext)
  322. {
  323. Host.Log("no asteroid available");
  324. return null;
  325. }
  326.  
  327. if(null == asteroidOverviewEntryNextNotTargeted)
  328. {
  329. Host.Log("all asteroids targeted");
  330. return null;
  331. }
  332.  
  333. if (!(asteroidOverviewEntryNextNotTargeted.DistanceMax < MiningRange))
  334. {
  335. if(!(1111 < asteroidOverviewEntryNext?.DistanceMin))
  336. {
  337. Host.Log("distance between asteroids too large");
  338. return null;
  339. }
  340.  
  341. Host.Log("out of range, approaching");
  342. ClickMenuEntryOnMenuRoot(asteroidOverviewEntryNext, "approach");
  343. }
  344. else
  345. {
  346. Host.Log("initiate lock asteroid");
  347. ClickMenuEntryOnMenuRoot(asteroidOverviewEntryNextNotTargeted, "^lock");
  348. }
  349.  
  350. return InBeltMineStep;
  351. }
  352.  
  353.  
  354. Sanderling.Parse.IMemoryMeasurement Measurement =>
  355. Sanderling?.MemoryMeasurementParsed?.Value;
  356.  
  357. IWindow ModalUIElement =>
  358. Measurement?.EnumerateReferencedUIElementTransitive()?.OfType<IWindow>()?.Where(window => window?.isModal ?? false)
  359. ?.OrderByDescending(window => window?.InTreeIndex ?? int.MinValue)
  360. ?.FirstOrDefault();
  361.  
  362. IEnumerable<Parse.IMenu> Menu => Measurement?.Menu;
  363.  
  364. Parse.IShipUi ShipUi => Measurement?.ShipUi;
  365.  
  366. bool Jammed => ShipUi?.EWarElement?.Any(EwarElement => (EwarElement?.EWarType).RegexMatchSuccess("electronic")) ?? false;
  367.  
  368. Sanderling.Interface.MemoryStruct.IMenuEntry MenuEntryLockTarget =>
  369. Menu?.FirstOrDefault()?.Entry?.FirstOrDefault(entry => entry.Text.RegexMatchSuccessIgnoreCase("^lock"));
  370.  
  371. Sanderling.Parse.IWindowOverview WindowOverview =>
  372. Measurement?.WindowOverview?.FirstOrDefault();
  373.  
  374. Sanderling.Parse.IWindowInventory WindowInventory =>
  375. Measurement?.WindowInventory?.FirstOrDefault();
  376.  
  377. IWindowDroneView WindowDrones =>
  378. Measurement?.WindowDroneView?.FirstOrDefault();
  379.  
  380. ITreeViewEntry InventoryActiveShipOreHold =>
  381. WindowInventory?.ActiveShipEntry?.TreeEntryFromCargoSpaceType(ShipCargoSpaceTypeEnum.OreHold);
  382.  
  383. IInventoryCapacityGauge OreHoldCapacityMilli =>
  384. (InventoryActiveShipOreHold?.IsSelected ?? false) ? WindowInventory?.SelectedRightInventoryCapacityMilli : null;
  385.  
  386. int? OreHoldFillPercent => (int?)((OreHoldCapacityMilli?.Used * 100) / OreHoldCapacityMilli?.Max);
  387.  
  388. Tab OverviewPresetTabActive =>
  389. WindowOverview?.PresetTab
  390. ?.OrderByDescending(tab => tab?.LabelColorOpacityMilli ?? 0)
  391. ?.FirstOrDefault();
  392.  
  393. string OverviewTypeSelectionName =>
  394. WindowOverview?.Caption?.RegexMatchIfSuccess(@"\(([^\)]*)\)")?.Groups?[1]?.Value;
  395.  
  396. Parse.IOverviewEntry[] ListRatOverviewEntry => WindowOverview?.ListView?.Entry?.Where(entry =>
  397. (entry?.MainIconIsRed ?? false) && (entry?.IsAttackingMe ?? false))
  398. ?.OrderBy(entry => entry?.DistanceMax ?? int.MaxValue)
  399. ?.ToArray();
  400.  
  401. Parse.IOverviewEntry[] ListAsteroidOverviewEntry =>
  402. WindowOverview?.ListView?.Entry
  403. ?.Where(entry => null != OreTypeFromAsteroidName(entry?.Name))
  404. ?.OrderBy(entry => entry.DistanceMax ?? int.MaxValue)
  405. ?.ToArray();
  406.  
  407.  
  408. DroneViewEntryGroup DronesInBayListEntry =>
  409. WindowDrones?.ListView?.Entry?.OfType<DroneViewEntryGroup>()?.FirstOrDefault(Entry => null != Entry?.Caption?.Text?.RegexMatchIfSuccess(@"Drones in bay", RegexOptions.IgnoreCase));
  410.  
  411. DroneViewEntryGroup DronesInSpaceListEntry =>
  412. WindowDrones?.ListView?.Entry?.OfType<DroneViewEntryGroup>()?.FirstOrDefault(Entry => null != Entry?.Caption?.Text?.RegexMatchIfSuccess(@"Drones in Local Space", RegexOptions.IgnoreCase));
  413.  
  414. int? DronesInSpaceCount => DronesInSpaceListEntry?.Caption?.Text?.AsDroneLabel()?.Status?.TryParseInt();
  415.  
  416. bool ReadyForManeuverNot =>
  417. Measurement?.ShipUi?.Indication?.LabelText?.Any(indicationLabel =>
  418. (indicationLabel?.Text).RegexMatchSuccessIgnoreCase("warp|docking")) ?? false;
  419.  
  420. bool ReadyForManeuver => !ReadyForManeuverNot && !(Measurement?.IsDocked ?? true);
  421.  
  422. Sanderling.Parse.IShipUiTarget[] SetTargetAsteroid =>
  423. Measurement?.Target?.Where(target =>
  424. target?.TextRow?.Any(textRow => textRow.RegexMatchSuccessIgnoreCase("asteroid")) ?? false)?.ToArray();
  425.  
  426. Sanderling.Interface.MemoryStruct.IListEntry WindowInventoryItem =>
  427. WindowInventory?.SelectedRightInventory?.ListView?.Entry?.FirstOrDefault();
  428.  
  429. Sanderling.Accumulation.IShipUiModule[] SetModuleMiner =>
  430. Sanderling.MemoryMeasurementAccu?.Value?.ShipUiModule?.Where(module => module?.TooltipLast?.Value?.IsMiner ?? false)?.ToArray();
  431.  
  432. Sanderling.Accumulation.IShipUiModule[] SetModuleMinerInactive =>
  433. SetModuleMiner?.Where(module => !(module?.RampActive ?? false))?.ToArray();
  434.  
  435. int? MiningRange => SetModuleMiner?.Select(module =>
  436. module?.TooltipLast?.Value?.RangeOptimal ?? module?.TooltipLast?.Value?.RangeMax ?? module?.TooltipLast?.Value?.RangeWithin ?? 0)?.DefaultIfEmpty(0)?.Min();;
  437.  
  438. WindowChatChannel chatLocal =>
  439. Sanderling.MemoryMeasurementParsed?.Value?.WindowChatChannel
  440. ?.FirstOrDefault(windowChat => windowChat?.Caption?.RegexMatchSuccessIgnoreCase("local") ?? false);
  441.  
  442. // assuming that own character is always visible in local
  443. bool hostileOrNeutralsInLocal => 1 != chatLocal?.ParticipantView?.Entry?.Count(IsNeutralOrEnemy);
  444.  
  445. // extract the ore type from the name as seen in overview. "Asteroid (Plagioclase)"
  446. string OreTypeFromAsteroidName(string AsteroidName) =>
  447. AsteroidName.ValueFromRegexMatchGroupAtIndex(@"Asteroid \(([^\)]+)", 0);
  448.  
  449. void ClickMenuEntryOnMenuRoot(IUIElement MenuRoot, string MenuEntryRegexPattern)
  450. {
  451. Sanderling.MouseClickRight(MenuRoot);
  452.  
  453. var Menu = Measurement?.Menu?.FirstOrDefault();
  454.  
  455. var MenuEntry = Menu?.EntryFirstMatchingRegexPattern(MenuEntryRegexPattern, RegexOptions.IgnoreCase);
  456.  
  457. Sanderling.MouseClickLeft(MenuEntry);
  458. }
  459.  
  460. void EnsureWindowInventoryOpen()
  461. {
  462. if (null != WindowInventory)
  463. return;
  464.  
  465. Host.Log("open Inventory.");
  466. Sanderling.MouseClickLeft(Measurement?.Neocom?.InventoryButton);
  467. }
  468.  
  469. void EnsureWindowInventoryOpenOreHold()
  470. {
  471. EnsureWindowInventoryOpen();
  472.  
  473. var inventoryActiveShip = WindowInventory?.ActiveShipEntry;
  474.  
  475. if(InventoryActiveShipOreHold == null && !(inventoryActiveShip?.IsExpanded ?? false))
  476. Sanderling.MouseClickLeft(inventoryActiveShip?.ExpandToggleButton);
  477.  
  478. if(!(InventoryActiveShipOreHold?.IsSelected ?? false))
  479. Sanderling.MouseClickLeft(InventoryActiveShipOreHold);
  480. }
  481.  
  482. // sample label text: Intensive Reprocessing Array <color=#66FFFFFF>1,123 m</color>
  483. string InventoryContainerLabelRegexPatternFromContainerName(string containerName) =>
  484. @"^\s*" + Regex.Escape(containerName) + @"\s*($|\<)";
  485.  
  486. void InInventoryUnloadItems() => InInventoryUnloadItemsTo(UnloadDestContainerName);
  487.  
  488. void InInventoryUnloadItemsTo(string DestinationContainerName)
  489. {
  490. Host.Log("unload items to '" + DestinationContainerName + "'.");
  491.  
  492. EnsureWindowInventoryOpenOreHold();
  493.  
  494. for (;;)
  495. {
  496. var oreHoldListItem = WindowInventory?.SelectedRightInventory?.ListView?.Entry?.ToArray();
  497.  
  498. var oreHoldItem = oreHoldListItem?.FirstOrDefault();
  499.  
  500. if(null == oreHoldItem)
  501. break; // 0 items in OreHold
  502.  
  503. if(1 < oreHoldListItem?.Length)
  504. ClickMenuEntryOnMenuRoot(oreHoldItem, @"select\s*all");
  505.  
  506. var DestinationContainerLabelRegexPattern =
  507. InventoryContainerLabelRegexPatternFromContainerName(DestinationContainerName);
  508.  
  509. var DestinationContainer =
  510. WindowInventory?.LeftTreeListEntry?.SelectMany(entry => new[] { entry }.Concat(entry.EnumerateChildNodeTransitive()))
  511. ?.FirstOrDefault(entry => entry?.Text?.RegexMatchSuccessIgnoreCase(DestinationContainerLabelRegexPattern) ?? false);
  512.  
  513. if (null == DestinationContainer)
  514. Host.Log("error: Inventory entry labeled '" + DestinationContainerName + "' not found");
  515.  
  516. Sanderling.MouseDragAndDrop(oreHoldItem, DestinationContainer);
  517. }
  518. }
  519.  
  520. bool InitiateWarpToRandomMiningSite() =>
  521. InitiateDockToOrWarpToBookmark(RandomElement(SetMiningSiteBookmark));
  522.  
  523. bool InitiateDockToOrWarpToBookmark(string bookmarkOrFolder)
  524. {
  525. Host.Log("dock to or warp to bookmark or random bookmark in folder: '" + bookmarkOrFolder + "'");
  526.  
  527. var listSurroundingsButton = Measurement?.InfoPanelCurrentSystem?.ListSurroundingsButton;
  528.  
  529. Sanderling.MouseClickRight(listSurroundingsButton);
  530.  
  531. var bookmarkMenuEntry = Measurement?.Menu?.FirstOrDefault()?.EntryFirstMatchingRegexPattern("^" + bookmarkOrFolder + "$", RegexOptions.IgnoreCase);
  532.  
  533. if(null == bookmarkMenuEntry)
  534. {
  535. Host.Log("menu entry not found for bookmark or folder: '" + bookmarkOrFolder + "'");
  536. return true;
  537. }
  538.  
  539. var currentLevelMenuEntry = bookmarkMenuEntry;
  540.  
  541. for (var menuLevel = 1; ; ++menuLevel)
  542. {
  543. Sanderling.MouseClickLeft(currentLevelMenuEntry);
  544.  
  545. var menu = Measurement?.Menu?.ElementAtOrDefault(menuLevel);
  546. var dockMenuEntry = menu?.EntryFirstMatchingRegexPattern("dock", RegexOptions.IgnoreCase);
  547. var warpMenuEntry = menu?.EntryFirstMatchingRegexPattern(@"warp.*within\s*0", RegexOptions.IgnoreCase);
  548. var approachEntry = menu?.EntryFirstMatchingRegexPattern(@"approach", RegexOptions.IgnoreCase);
  549.  
  550. var maneuverMenuEntry = dockMenuEntry ?? warpMenuEntry;
  551.  
  552. if (null != maneuverMenuEntry)
  553. {
  554. Host.Log("initiating '" + maneuverMenuEntry.Text + "' on entry '" + currentLevelMenuEntry?.Text + "'");
  555. Sanderling.MouseClickLeft(maneuverMenuEntry);
  556. return false;
  557. }
  558.  
  559. if (null != approachEntry)
  560. {
  561. Host.Log("found menu entry '" + approachEntry.Text + "'. Assuming we are already there.");
  562. return false;
  563. }
  564.  
  565. var setBookmarkOrFolderMenuEntry =
  566. menu?.Entry; // assume that each entry on the current menu level is a bookmark or a bookmark folder.
  567.  
  568. var nextLevelMenuEntry = RandomElement(setBookmarkOrFolderMenuEntry);
  569.  
  570. if(null == nextLevelMenuEntry)
  571. {
  572. Host.Log("no suitable menu entry found");
  573. return true;
  574. }
  575.  
  576. currentLevelMenuEntry = nextLevelMenuEntry;
  577. }
  578. }
  579.  
  580. void Undock()
  581. {
  582. while(Measurement?.IsDocked ?? true)
  583. {
  584. Sanderling.MouseClickLeft(Measurement?.WindowStation?.FirstOrDefault()?.UndockButton);
  585. Host.Log("waiting for undocking to complete.");
  586. Host.Delay(8000);
  587. }
  588.  
  589. Host.Delay(4444);
  590. Sanderling.InvalidateMeasurement();
  591. }
  592.  
  593. void ModuleMeasureAllTooltip()
  594. {
  595. Host.Log("measure tooltips of all modules.");
  596.  
  597. for (;;)
  598. {
  599. var NextModule = Sanderling.MemoryMeasurementAccu?.Value?.ShipUiModule?.FirstOrDefault(m => null == m?.TooltipLast);
  600.  
  601. if(null == NextModule)
  602. break;
  603.  
  604. Host.Log("measure module.");
  605. // take multiple measurements of module tooltip to reduce risk to keep bad read tooltip.
  606. Sanderling.MouseMove(NextModule);
  607. Sanderling.WaitForMeasurement();
  608. Sanderling.MouseMove(NextModule);
  609. }
  610. }
  611.  
  612. void ActivateHardenerExecute()
  613. {
  614. var SubsetModuleHardener =
  615. Sanderling.MemoryMeasurementAccu?.Value?.ShipUiModule
  616. ?.Where(module => module?.TooltipLast?.Value?.IsHardener ?? false);
  617.  
  618. var SubsetModuleToToggle =
  619. SubsetModuleHardener
  620. ?.Where(module => !(module?.RampActive ?? false));
  621.  
  622. foreach (var Module in SubsetModuleToToggle.EmptyIfNull())
  623. ModuleToggle(Module);
  624. }
  625.  
  626. void ModuleToggle(Sanderling.Accumulation.IShipUiModule Module)
  627. {
  628. var ToggleKey = Module?.TooltipLast?.Value?.ToggleKey;
  629.  
  630. Host.Log("toggle module using " + (null == ToggleKey ? "mouse" : Module?.TooltipLast?.Value?.ToggleKeyTextLabel?.Text));
  631.  
  632. if(null == ToggleKey)
  633. Sanderling.MouseClickLeft(Module);
  634. else
  635. Sanderling.KeyboardPressCombined(ToggleKey);
  636. }
  637.  
  638. void EnsureOverviewTypeSelectionLoaded()
  639. {
  640. if(null == OverviewPresetTabActive || null == WindowOverview || null == OverviewPreset)
  641. return;
  642.  
  643. if(string.Equals(OverviewTypeSelectionName, OverviewPreset, StringComparison.OrdinalIgnoreCase))
  644. return;
  645.  
  646. Host.Log("loading preset '" + OverviewPreset + "' to overview (current selection is '" + OverviewTypeSelectionName + "').");
  647. Sanderling.MouseClickRight(OverviewPresetTabActive);
  648. Sanderling.MouseClickLeft(Menu?.FirstOrDefault()?.EntryFirstMatchingRegexPattern("load.*preset", RegexOptions.IgnoreCase));
  649. var PresetMenuEntry = Menu?.ElementAtOrDefault(1)?.EntryFirstMatchingRegexPattern(@"^\s*" + Regex.Escape(OverviewPreset) + @"\s*$", RegexOptions.IgnoreCase);
  650.  
  651. if(null == PresetMenuEntry)
  652. {
  653. Host.Log("error: menu entry '" + OverviewPreset + "' not found");
  654. return;
  655. }
  656.  
  657. Sanderling.MouseClickLeft(PresetMenuEntry);
  658. }
  659.  
  660. void MemoryUpdate()
  661. {
  662. RetreatUpdate();
  663. JammedLastTimeUpdate();
  664. OffloadCountUpdate();
  665. }
  666.  
  667. void JammedLastTimeUpdate()
  668. {
  669. if(Jammed)
  670. JammedLastTime = Host.GetTimeContinuousMilli();
  671. }
  672.  
  673. bool MeasurementEmergencyWarpOutEnter =>
  674. !(Measurement?.IsDocked ?? false) && !(EmergencyWarpOutHitpointPercent < ShieldHpPercent);
  675.  
  676. void RetreatUpdate()
  677. {
  678. RetreatReasonTemporary = (RetreatOnNeutralOrHostileInLocal && hostileOrNeutralsInLocal) ? "hostile or neutral in local" : null;
  679.  
  680. if (!MeasurementEmergencyWarpOutEnter)
  681. return;
  682.  
  683. // measure multiple times to avoid being scared off by noise from a single measurement.
  684. Sanderling.InvalidateMeasurement();
  685.  
  686. if (!MeasurementEmergencyWarpOutEnter)
  687. return;
  688.  
  689. RetreatReasonPermanent = "shield hp";
  690. }
  691.  
  692. void OffloadCountUpdate()
  693. {
  694. var OreHoldFillPercentSynced = OreHoldFillPercent;
  695.  
  696. if(!OreHoldFillPercentSynced.HasValue)
  697. return;
  698.  
  699. if(0 == OreHoldFillPercentSynced && OreHoldFillPercentSynced < LastCheckOreHoldFillPercent)
  700. ++OffloadCount;
  701.  
  702. LastCheckOreHoldFillPercent = OreHoldFillPercentSynced;
  703. }
  704.  
  705. bool IsNeutralOrEnemy(IChatParticipantEntry participantEntry) =>
  706. !(participantEntry?.FlagIcon?.Any(flagIcon =>
  707. new[] { "good standing", "excellent standing", "Pilot is in your (fleet|corporation)", }
  708. .Any(goodStandingText =>
  709. flagIcon?.HintText?.RegexMatchSuccessIgnoreCase(goodStandingText) ?? false)) ?? false);
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement