private string Replace(string prefix, string s, WoWUnit theunit) { string fnname = "FTWCore.Replace"; MyTimer.Start(fnname); if (theunit != null && theunit.Guid == StyxWoW.Me.Guid) { if (StyxWoW.Me.HasAura("Eclipse (Solar)")) { FTWProps._eclipsedirection = -1; } else if (StyxWoW.Me.HasAura("Eclipse (Lunar)")) { FTWProps._eclipsedirection = 1; } } // Replace aura checks with parenthese (Target.HasMyAura("Bad Aura") > 3) string findstring = string.Format(@"{0}\.(?<action>.*)\(""(?<aura>.*)""\)", prefix); Regex rg = new Regex(findstring); Match match = rg.Match(s); Dictionary <String, Object> d = new Dictionary <String, Object>(); while (match.Success) { if (!d.ContainsKey(match.Value)) { string action = match.Groups["action"].Value; string aura = match.Groups["aura"].Value; int value = 0; if (false) { } else if (action == "Weapon1HasAura") { value = (theunit != null && FTWCoreItems.WeaponEnchant(1) == aura) ? 1 : 0; } else if (action == "Weapon2HasAura") { value = (theunit != null && FTWCoreItems.WeaponEnchant(2) == aura) ? 1 : 0; } else if (action == "AuraExpiring") { value = (theunit != null) ? theunit.AuraExpiring(aura) : 0; } else if (action == "CanCast") { value = (theunit != null && SpellManager.CanCast(aura, theunit)) ? 1 : 0; } else if (action == "ClearBehindMe") { value = (theunit != null && StyxWoW.Me.ClearBehindMe(aura) ? 1 : 0); } else if (action == "HasAura") { value = (theunit != null && theunit.HasAura(aura)) ? 1 : 0; } else if (action == "HasItem") { value = StyxWoW.Me.HasItem(aura) ? 1 : 0; } else if (action == "HasMyAura") { value = (theunit != null && theunit.HasMyAura(aura)) ? 1 : 0; } else if (action == "HasSpell") { value = (StyxWoW.Me.HasSpell(aura)) ? 1 : 0; } else if (action == "HasTotem") { value = FTWCoreStatus.HasTotem(aura) ? 1 : 0; } else if (action == "IsCasting") { value = (theunit != null && theunit.CurrentSpellName() == aura) ? 1 : 0; } else if (action == "ItemOnCooldown") { value = FTWCoreItems.ItemOnCooldown(aura) ? 1 : 0; } else if (action == "MyAuraExpiring") { value = (theunit != null) ? theunit.MyAuraExpiring(aura) : 0; } else if (action == "MyStackCount") { value = (theunit != null) ? theunit.MyStackCount(aura) : 0; } else if (action == "OnCooldown") { value = FTWCoreStatus.OnCooldown(aura) ? 1 : 0; } else if (action == "PartyWithAura") { value = FTWCoreUnits.GetPartyWithAura(aura); } else if (action == "PartyWithHealth") { value = FTWCoreUnits.GetPartyWithHealth(aura); } else if (action == "StackCount") { value = (theunit != null) ? theunit.StackCount(aura) : 0; } else if (action == "NumItems") { value = (theunit != null) ? FTWCoreItems.NumItems(aura) : 0; } else { throw new Exception(string.Format("Unknown action {0}!", action)); } d.Add(match.Value, 0); s = s.Replace(match.Value, value.ToString()); } match = match.NextMatch(); } // Replace simple properties for (int i = 0; i < FTWProps.PropertyNames.Count(); i++) { string propname = FTWProps.PropertyNames[i].Trim(); if (propname.Length > 0) { string search = prefix + "." + propname; if (s.Contains(search)) { s = s.Replace(search, GetProperty(theunit, propname).ToString()); } } } // Stop the timer MyTimer.Stop(fnname); return(s); }
public static bool On_Cast(string firstpart, string spellname, WoWUnit target, bool castnow, bool castonce, bool debugspell = false) { string fnname = "FTWCore.On_Cast"; MyTimer.Start(fnname); LocalPlayer Me = StyxWoW.Me; WoWSpell spell = null; bool retval = true; Color ds = Color.Magenta; if (target != null && !target.IsFriendly && target.IsDead) { if (debugspell) { FTWLogger.log(ds, "Target is hostile and dead"); } MyTimer.Stop(fnname); return(false); } if (FTWCoreStatus.OnCooldown(spellname, debugspell)) { MyTimer.Stop(fnname); return(false); } if (target != null && !target.IsFriendly && !FTWUtils.MovementDisabled()) { FTWCoreMovement.Face(target); } // Clear out expired CastOnce entries (older than CastOnceTime seconds) foreach (KeyValuePair <String, DateTime> kvp in FTWProps.CastOnce.ToList()) { if (DateTime.Now > kvp.Value) { FTWProps.CastOnce.Remove(kvp.Key); } } // Leave early if CastOnce contains unexpired entry for this unit if (FTWProps.CastOnce.ContainsKey(spellname + target.Guid.ToString())) { if (debugspell) { FTWLogger.log(ds, "Already cast {0} on {1}", spellname, target.Name); } MyTimer.Stop(fnname); return(false); } if (SpellManager.GlobalCooldown) { if (debugspell) { FTWLogger.log(ds, "..."); } MyTimer.Stop(fnname); return(false); } try { if (FTWProps.fakecooldowns.ContainsKey(spellname)) { spell = WoWSpell.FromId(FTWProps.fakecooldowns[spellname].SpellID); } else if (SpellManager.Spells.ContainsKey(spellname)) { spell = SpellManager.Spells[spellname]; } else { FTWLogger.debug(Color.Gray, "Unknown spell {0}", spellname); MyTimer.Stop(fnname); return(false); } } catch (Exception ex) { FTWLogger.log(Color.Pink, "Error when getting spell {0}: {1}", spellname, ex.Message); MyTimer.Stop(fnname); return(false); } if (spell == null) { FTWLogger.debug(Color.Gray, "Unknown spell '{0}'", spellname); MyTimer.Stop(fnname); return(false); } // Check whether or not you know the spell according to WoW if (false) // leave commented out for now { List <string> known = Lua.GetReturnValues(String.Format("return IsSpellKnown({0})", spell.Id), "SpellKnown.lua"); int knownspell = Int32.Parse(known[0]); if (knownspell == 0) { FTWLogger.log(Color.Gray, "GetSpellInfo says you don't know spell {0} ({1})", spellname, known[0]); MyTimer.Stop(fnname); return(false); } } if (false) { // Check power requirements List <string> values = Lua.GetReturnValues(String.Format("return GetSpellInfo({0})", spell.Id), "SpellInfo.lua"); //string[] vars = {"name", "rank", "icon", "cost", "isFunnel", "powerType", "castTime", "minRange", "maxRange"}; int powerType = Int32.Parse(values[5]); int powerCost = Int32.Parse(values[3]); string[] powertypes = { "mana", "rage", "focus", "energy", "happiness", "runes", "runic power", "soul shards", "eclipse", "holy power", "alternate power", "dark force", "chi", "shadow orbs", "burning embers", "demonic fury" }; if (StyxWoW.Me.UnitPower(powerType) < powerCost) { FTWLogger.log(Color.Orange, "NOT ENOUGH {0} - Requires: {1}, but I have: {2}", powertypes[powerType], powerCost, StyxWoW.Me.UnitPower(powerType)); MyTimer.Stop(fnname); return(false); } } if (!spell.IsSelfOnlySpell) { if (target == null) { FTWLogger.log(Color.Orange, "Can't cast {0} on null target!", spell.Name); if (debugspell) { FTWLogger.log(ds, "Null target"); } MyTimer.Stop(fnname); return(false); } double dist = target.Distance; // target.DistanceCalc(); if ((spell.MinRange > 0 || spell.MaxRange > 0) && !(dist >= spell.MinRange && dist <= spell.MaxRange)) { FTWLogger.log(Color.Orange, "Can't cast spell {0} {1} on {2} health {3:0.0} dist {4:0.0} minrange {5} maxrange {6}", spellname, spell.Id, target.SafeName(), target.HealthPercent, target.DistanceCalc(), spell.MinRange, spell.MaxRange); if (debugspell) { FTWLogger.log(ds, "Out of range"); } MyTimer.Stop(fnname); return(false); } if (!target.InLineOfSight) { FTWLogger.log(Color.Orange, "Can't cast spell {0} {1} on {2} health {3:0.0} dist {4:0.0} (not in line of sight)", spellname, spell.Id, target.SafeName(), target.HealthPercent, target.DistanceCalc()); if (debugspell) { FTWLogger.log(ds, "Not in line of sight. Attempts = {0}.", FTWProps.not_in_los_attempts); } MyTimer.Stop(fnname); FTWProps.not_in_los_attempts += 1; Navigator.MoveTo(target.Location); Thread.Sleep(2000); return(false); } if (spellname == "Death Pact" && Me.GotAlivePet) { Me.Pet.Target(); for (int l = 0; l < 2; l++) { Thread.Sleep(10); } } if (FTWProps.CastOver.Contains(FTWProps.lastspellcastname)) { retval = true; } else if (spell.IsSelfOnlySpell) { retval = SpellManager.CanCast(spell, true); } else { retval = SpellManager.CanCast(spell, target, true); } if (FTWProps.IgnoreCanCast.Contains(spellname)) { if (debugspell) { FTWLogger.log(ds, "Ignoring cancast for {0}", spellname); } retval = true; } if (retval == false) { if (spell.IsSelfOnlySpell) { FTWLogger.debug(Color.Orange, "CanCast returned false for {0} {1} on me", spellname, spell.Id); } else { FTWLogger.debug(Color.Orange, "CanCast returned false for {0} {1} on {2} health {3:0.0} dist {4:0.0}", spellname, spell.Id, target.SafeName(), target.HealthPercent, target.DistanceCalc()); } MyTimer.Stop(fnname); if (debugspell) { FTWLogger.log(ds, "CanCast returned false"); } return(false); } } // Leave early if still casting if (!castnow && (Me.IsCasting || Me.IsChanneling)) { if (debugspell) { FTWLogger.log(ds, "Still casting"); } return(false); } // Definitely going to cast - Stop casting if (Me.IsCasting || Me.IsChanneling) { if (FTWProps.CastOver.Contains(FTWProps.lastspellcastname)) { FTWLogger.log("Currently casting {0} - casting over it with {1}", FTWProps.lastspellcastname, spellname); } else if (castnow) { SpellManager.StopCasting(); } } // Stop moving if (spell.CastTime > 0 && !FTWUtils.MovementDisabled()) { WoWMovement.MoveStop(); } // Clear target spell, if one is up if (StyxWoW.Me.CurrentPendingCursorSpell != null) { Lua.DoString("SpellStopTargeting()"); } if (spell.IsTankOnlySpell() == true) { // Spells that are only ever cast on tanks, such as Misdirection and Tricks of the Trade. if (FTWProps.people.Count > 1 && FTWProps.tank.Guid != Me.Guid && FTWProps.tank.IsDead == false) { FTWLogger.log("cast tank-only spell {0} on {1}", spell.Name, FTWProps.tank.SafeName()); if (SpellManager.CanCast(spell, FTWProps.tank)) { FTWLogger.log("nope - failed cancast check."); retval = SpellManager.Cast(spell, FTWProps.tank); } } else if (StyxWoW.Me.GotAlivePet && Me.Pet.IsDead == false) { FTWLogger.log("cast tank-only spell {0} on pet", spell.Name); if (SpellManager.CanCast(spell, Me.Pet)) { FTWLogger.log("nope - failed cancast check."); retval = SpellManager.Cast(spell, Me.Pet); } } else { retval = false; } } else if (spell.IsAreaSpell() == false) { // Most normal spells if (spell.IsSelfOnlySpell) { retval = SpellManager.Cast(spell); } else { retval = SpellManager.Cast(spell, target); } // Post-spell processing if (spellname == "Feign Death") { while (Me.HasAura("Feign Death")) { List <WoWUnit> mobs = (List <WoWUnit>)FTWCoreUnits.GetNearbyUnfriendlyUnits(); if (mobs.Count == 0 && StyxWoW.Me.HealthPercent > 95) { break; } Thread.Sleep(10); } } } else { // Area spell WoWPoint loc; if (spell.IsSelfOnlySpell && !spell.Name.EndsWith(" Trap")) { loc = StyxWoW.Me.Location; } else { loc = target.Location; } retval = SpellManager.Cast(spell); DateTime stoptime = DateTime.Now.AddMilliseconds(500); while (DateTime.Now < stoptime) { if (StyxWoW.Me.CurrentPendingCursorSpell != null) { break; } Thread.Sleep(10); } SpellManager.ClickRemoteLocation(loc); stoptime = DateTime.Now.AddMilliseconds(500); while (DateTime.Now < stoptime) { // Clear target spell, if one is up if (StyxWoW.Me.CurrentPendingCursorSpell == null) { break; } Thread.Sleep(10); } // Clear target spell, if one is up if (StyxWoW.Me.CurrentPendingCursorSpell != null) { Lua.DoString("SpellStopTargeting()"); } } // Add current spell to CastOnce dictionary if (castonce == true) { FTWProps.CastOnce[spellname + target.Guid.ToString()] = DateTime.Now.AddSeconds(FTWProps.CastOnceTime); } // Wait until spell starts casting if (spell.CastTime > 0) { DateTime dt = DateTime.Now.AddSeconds(1.5); while (DateTime.Now < dt && !StyxWoW.Me.IsCasting && !StyxWoW.Me.IsChanneling) { Thread.Sleep(10); } } // Save spell cooldown if (retval == true) { FTWLogger.log(" {0} {1} on {2} {3} health {4:0} dist {5:0.0} aggro {6} {7} (addscount {8})", firstpart, spellname, target.SafeName(), target.ShortGuid(), target.HealthPercent, target.DistanceCalc(), (int)target.ThreatInfo.ThreatStatus, target.ThreatInfo.ThreatStatus, FTWProps.adds.Count); FTWCoreStatus.SaveCooldown(spellname); FTWProps.MovementCooldown = DateTime.Now.Add(TimeSpan.FromMilliseconds(Math.Min(1500, spell.CastTime))); FTWProps.not_in_los_attempts = 0; } else { FTWLogger.log(Color.Gray, " - Couldn't cast spell {0} on {1} {2} at dist {3:0.0}", spellname, target.SafeName(), target.ShortGuid(), target.DistanceCalc()); } if (FTWProps.Stuns.Contains(spellname)) { // Don't attack again for 3 seconds - hb is slow in noticing cc's. Blacklist.Add(target.Guid, BlacklistFlags.Combat, TimeSpan.FromSeconds(3)); if (target.Guid == StyxWoW.Me.CurrentTargetGuid) { StyxWoW.Me.ClearTarget(); } } MyTimer.Stop(fnname); return(retval); }
public void Pulse() { string fnname = "FTWCore.Pulse"; MyTimer.Start(fnname); try { if (!FTWUtils.ValidState()) { // Do nothing FTWLogger.log(Color.Red, "Not valid state"); } else { // Get lists of interesting people // Party or raid members FTWProps.people = FTWCoreUnits.GetGroup(); // Tank FTWProps.tank = FTWCoreUnits.GetTank(); // Healer FTWProps.healer = FTWCoreUnits.GetHealer(); // Party members within range FTWProps.closePeople = FTWProps.people.Where(p => p.DistanceCalc() <= 40).ToList(); // Party members within range, in line of sight, that need healing FTWProps.healTarget = FTWCoreUnits.GetHealTarget(); // Diseased party member // Only tanks and me get dispels, the rest can stay sick. // A necessary optimization to keep the speed up. FTWProps.cleanseTarget = (from unit in FTWProps.closePeople where unit.Role().Contains("Tank") || unit.Guid == StyxWoW.Me.Guid where unit.EligibleForCleanse() == true orderby unit.HealWeight() select unit).FirstOrDefault(); // Dead party member outside combat FTWProps.reviveTarget = (from unit in FTWProps.closePeople where unit.IsDead || unit.IsGhost select unit).FirstOrDefault(); if (FTWProps.reviveTarget != null) { FTWLogger.log(Color.Red, "{0} is dead - revive him if you can.", FTWProps.reviveTarget.SafeName()); } // Get average health of all party members. try { FTWProps.avgHealth = (int)(from p in FTWProps.closePeople where p.IsDead == false select p.HealthPercent).Average(); } catch (Exception ex) { FTWProps.avgHealth = (int)StyxWoW.Me.HealthPercent; } // Get mobs fighting us FTWProps.adds = FTWCoreUnits.GetAdds(); // Get mobs fighting us, that we're not targeting WoWUnit pettarget = null; if (StyxWoW.Me.GotAlivePet) { pettarget = StyxWoW.Me.Pet.CurrentTarget; } FTWProps.nontargetadds = (from unit in FTWProps.adds where unit != StyxWoW.Me.CurrentTarget && unit != pettarget select unit).ToList(); // Get a mob fighting us, that we're not targeting. FTWProps.add = FTWProps.nontargetadds.FirstOrDefault(); } } catch (Exception ex) { FTWLogger.log(Color.Red, ex.ToString()); } MyTimer.Stop(fnname); }