diff options
author | Daniel Schadt <kingdread@gmx.de> | 2022-04-01 13:39:41 +0200 |
---|---|---|
committer | Daniel Schadt <kingdread@gmx.de> | 2022-04-01 13:39:41 +0200 |
commit | 65cfa7b4f73fa16d356969aaef1428b5e5ebd116 (patch) | |
tree | 4a57607560f7e32370ec471d0ea6e29413589318 /src/analyzers | |
parent | 92867533ffd5fdf44a7f6fe6f0b4b919bf533228 (diff) | |
parent | 15c29b2435c48117b0e70a80ebbfbe37937af7cb (diff) | |
download | evtclib-65cfa7b4f73fa16d356969aaef1428b5e5ebd116.tar.gz evtclib-65cfa7b4f73fa16d356969aaef1428b5e5ebd116.tar.bz2 evtclib-65cfa7b4f73fa16d356969aaef1428b5e5ebd116.zip |
Merge branch 'eod-strikes'
Diffstat (limited to 'src/analyzers')
-rw-r--r-- | src/analyzers/mod.rs | 11 | ||||
-rw-r--r-- | src/analyzers/strikes.rs | 284 |
2 files changed, 289 insertions, 6 deletions
diff --git a/src/analyzers/mod.rs b/src/analyzers/mod.rs index ed0f157..c27fefa 100644 --- a/src/analyzers/mod.rs +++ b/src/analyzers/mod.rs @@ -131,10 +131,11 @@ pub fn for_log<'l>(log: &'l Log) -> Option<Box<dyn Analyzer + 'l>> { | Encounter::SuperKodanBrothers | Encounter::FraenirOfJormag | Encounter::Boneskinner - | Encounter::WhisperOfJormag - | Encounter::CaptainMaiTrin - | Encounter::Ankka - | Encounter::MinisterLi - | Encounter::Dragonvoid => Some(Box::new(strikes::GenericStrike::new(log))), + | Encounter::WhisperOfJormag => Some(Box::new(strikes::GenericStrike::new(log))), + + Encounter::CaptainMaiTrin => Some(Box::new(strikes::CaptainMaiTrin::new(log))), + Encounter::Ankka => Some(Box::new(strikes::Ankka::new(log))), + Encounter::MinisterLi => Some(Box::new(strikes::MinisterLi::new(log))), + Encounter::Dragonvoid => Some(Box::new(strikes::Dragonvoid::new(log))), } } diff --git a/src/analyzers/strikes.rs b/src/analyzers/strikes.rs index 8c22c49..9244124 100644 --- a/src/analyzers/strikes.rs +++ b/src/analyzers/strikes.rs @@ -1,7 +1,8 @@ //! Analyzers for Strike Mission logs. use crate::{ analyzers::{helpers, Analyzer, Outcome}, - Log, + gamedata::Boss, + EventKind, Log, }; /// Analyzer for strikes. @@ -36,3 +37,284 @@ impl<'log> Analyzer for GenericStrike<'log> { Outcome::from_bool(helpers::boss_is_dead(self.log)) } } + +/// Analyzer for the Captain Mai Trin/Aetherblade Hideout strike. +#[derive(Debug, Clone, Copy)] +pub struct CaptainMaiTrin<'log> { + log: &'log Log, +} + +impl<'log> CaptainMaiTrin<'log> { + pub const ECHO_OF_SCARLET_BRIAR: u16 = 24_768; + /// Determined buff that is used in Mai Trin's Strike. + /// + /// Thanks to ArenaNet's consistency, there are multiple versions of the Determined buff in + /// use. + /// + /// The chat link for this buff is `[&Bn8DAAA=]`. + pub const DETERMINED_ID: u32 = 895; + + /// Create a new [`CaptainMaiTrin`] analyzer for the given log. + /// + /// **Do not** use this method unless you know what you are doing. Instead, rely on + /// [`Log::analyzer`]! + pub fn new(log: &'log Log) -> Self { + CaptainMaiTrin { log } + } +} + +impl<'log> Analyzer for CaptainMaiTrin<'log> { + fn log(&self) -> &Log { + self.log + } + + fn is_cm(&self) -> bool { + // EoD strike CMs are not implemented yet as of 2022-03-31 + false + } + + fn outcome(&self) -> Option<Outcome> { + check_reward!(self.log); + + let scarlet = self + .log + .characters() + .find(|npc| npc.id() == Self::ECHO_OF_SCARLET_BRIAR)?; + let mai = self + .log + .characters() + .find(|npc| npc.id() == Boss::CaptainMaiTrin as u16)?; + + for event in self.log.events() { + if let EventKind::BuffApplication { + destination_agent_addr, + buff_id, + .. + } = event.kind() + { + if *buff_id == Self::DETERMINED_ID + && *destination_agent_addr == mai.addr() + && event.time() > scarlet.first_aware() + { + return Some(Outcome::Success); + } + } + } + + Some(Outcome::Failure) + } +} + +/// Analyzer for the Ankka/Xunlai Jade Junkyard strike. +#[derive(Debug, Clone, Copy)] +pub struct Ankka<'log> { + log: &'log Log, +} + +impl<'log> Ankka<'log> { + /// Determined buff that is used in Ankka's Strike. + /// + /// Thanks to ArenaNet's consistency, there are multiple versions of the Determined buff in + /// use. + /// + /// The chat link for this buff is `[&Bn8DAAA=]`. + pub const DETERMINED_ID: u32 = CaptainMaiTrin::DETERMINED_ID; + /// The minimum duration of [`DETERMINED_ID`] buff applications. + pub const DURATION_CUTOFF: i32 = i32::MAX; + /// The expected number of times that Ankka needs to phase before we consider it a success. + pub const EXPECTED_PHASE_COUNT: usize = 3; + + /// Create a new [`Ankka`] analyzer for the given log. + /// + /// **Do not** use this method unless you know what you are doing. Instead, rely on + /// [`Log::analyzer`]! + pub fn new(log: &'log Log) -> Self { + Ankka { log } + } +} + +impl<'log> Analyzer for Ankka<'log> { + fn log(&self) -> &Log { + self.log + } + + fn is_cm(&self) -> bool { + // EoD strike CMs are not implemented yet as of 2022-03-31 + false + } + + fn outcome(&self) -> Option<Outcome> { + check_reward!(self.log); + + let ankka = self + .log + .characters() + .find(|npc| npc.id() == Boss::Ankka as u16)?; + + let phase_change_count = self + .log + .events() + .iter() + .filter(|event| { + if let EventKind::BuffApplication { + destination_agent_addr, + buff_id, + duration, + .. + } = event.kind() + { + *buff_id == Self::DETERMINED_ID + && *destination_agent_addr == ankka.addr() + && *duration == Self::DURATION_CUTOFF + } else { + false + } + }) + .count(); + + Outcome::from_bool(phase_change_count == Self::EXPECTED_PHASE_COUNT) + } +} + +/// Analyzer for the Minister Li/Kaineng Overlook strike. +#[derive(Debug, Clone, Copy)] +pub struct MinisterLi<'log> { + log: &'log Log, +} + +impl<'log> MinisterLi<'log> { + /// Determined buff that is used in Minister Li's Strike. + /// + /// Thanks to ArenaNet's consistency, there are multiple versions of the Determined buff in + /// use. + /// + /// The chat link for this buff is `[&BvoCAAA=]`. + pub const DETERMINED_ID: u32 = 762; + /// The minimum number of times that Minister Li needs to phase before we consider it a success. + pub const MINIMUM_PHASE_COUNT: usize = 3; + + /// Create a new [`MinisterLi`] analyzer for the given log. + /// + /// **Do not** use this method unless you know what you are doing. Instead, rely on + /// [`Log::analyzer`]! + pub fn new(log: &'log Log) -> Self { + MinisterLi { log } + } +} + +impl<'log> Analyzer for MinisterLi<'log> { + fn log(&self) -> &Log { + self.log + } + + fn is_cm(&self) -> bool { + // EoD strike CMs are not implemented yet as of 2022-03-31 + false + } + + fn outcome(&self) -> Option<Outcome> { + check_reward!(self.log); + + let li = self + .log + .characters() + .find(|npc| npc.id() == Boss::MinisterLi as u16)?; + + let phase_change_count = self + .log + .events() + .iter() + .filter(|event| { + if let EventKind::BuffApplication { + destination_agent_addr, + buff_id, + .. + } = event.kind() + { + *buff_id == Self::DETERMINED_ID && *destination_agent_addr == li.addr() + } else { + false + } + }) + .count(); + + Outcome::from_bool(phase_change_count >= Self::MINIMUM_PHASE_COUNT) + } +} + +/// Analyzer for the Dragonvoid/Harvest Temple strike. +#[derive(Debug, Clone, Copy)] +pub struct Dragonvoid<'log> { + log: &'log Log, +} + +impl<'log> Dragonvoid<'log> { + pub const EXPECTED_TARGET_OFF_COUNT: usize = 2; + + /// Create a new [`Dragonvoid`] analyzer for the given log. + /// + /// **Do not** use this method unless you know what you are doing. Instead, rely on + /// [`Log::analyzer`]! + pub fn new(log: &'log Log) -> Self { + Dragonvoid { log } + } +} + +impl<'log> Analyzer for Dragonvoid<'log> { + fn log(&self) -> &Log { + self.log + } + + fn is_cm(&self) -> bool { + // EoD strike CMs are not implemented yet as of 2022-03-31 + false + } + + fn outcome(&self) -> Option<Outcome> { + // check_reward is pointless because the reward is delayed. + + // First, we find the right agent_addr + let mut first_voids = None; + for event in self.log.events() { + if let EventKind::AttackTarget { + agent_addr, + parent_agent_addr, + .. + } = event.kind() + { + if first_voids.is_none() { + first_voids = Some(parent_agent_addr); + } else if first_voids != Some(parent_agent_addr) { + // We find the amount of target off switches that occurred after a target on + // switch. + let mut is_on = false; + let mut target_off_count = 0; + + // The nested loop over events is not ideal, but it is currently the easiest + // way to implement this logic without trying to cram it into a single loop. + for e in self.log.events() { + if let EventKind::Targetable { + agent_addr: taa, + targetable, + } = e.kind() + { + if *taa != *agent_addr { + continue; + } + if *targetable { + is_on = true; + } else if !targetable && is_on { + target_off_count += 1; + } + } + } + + if target_off_count == Self::EXPECTED_TARGET_OFF_COUNT { + return Some(Outcome::Success); + } + } + } + } + Some(Outcome::Failure) + } +} |