use std::io::{Read, Write};
use crate::Guid;
use crate::tbc::{
HitInfo, SpellSchool,
};
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord, Default)]
pub struct SMSG_SPELLNONMELEEDAMAGELOG {
pub target: Guid,
pub attacker: Guid,
pub spell: u32,
pub damage: u32,
pub school: SpellSchool,
pub absorbed_damage: u32,
pub resisted: u32,
pub periodic_log: bool,
pub unused: u8,
pub blocked: u32,
pub hit_info: HitInfo,
pub extend_flag: u8,
}
impl crate::private::Sealed for SMSG_SPELLNONMELEEDAMAGELOG {}
impl SMSG_SPELLNONMELEEDAMAGELOG {
fn read_inner(mut r: &mut &[u8], body_size: u32) -> Result<Self, crate::errors::ParseErrorKind> {
if !(30..=46).contains(&body_size) {
return Err(crate::errors::ParseErrorKind::InvalidSize);
}
let target = crate::util::read_packed_guid(&mut r)?;
let attacker = crate::util::read_packed_guid(&mut r)?;
let spell = crate::util::read_u32_le(&mut r)?;
let damage = crate::util::read_u32_le(&mut r)?;
let school = crate::util::read_u8_le(&mut r)?.try_into()?;
let absorbed_damage = crate::util::read_u32_le(&mut r)?;
let resisted = crate::util::read_u32_le(&mut r)?;
let periodic_log = crate::util::read_bool_u8(&mut r)?;
let unused = crate::util::read_u8_le(&mut r)?;
let blocked = crate::util::read_u32_le(&mut r)?;
let hit_info = crate::util::read_u32_le(&mut r)?.try_into()?;
let extend_flag = crate::util::read_u8_le(&mut r)?;
Ok(Self {
target,
attacker,
spell,
damage,
school,
absorbed_damage,
resisted,
periodic_log,
unused,
blocked,
hit_info,
extend_flag,
})
}
}
impl crate::Message for SMSG_SPELLNONMELEEDAMAGELOG {
const OPCODE: u32 = 0x0250;
#[cfg(feature = "print-testcase")]
fn message_name(&self) -> &'static str {
"SMSG_SPELLNONMELEEDAMAGELOG"
}
#[cfg(feature = "print-testcase")]
fn to_test_case_string(&self) -> Option<String> {
use std::fmt::Write;
use crate::traits::Message;
let mut s = String::new();
writeln!(s, "test SMSG_SPELLNONMELEEDAMAGELOG {{").unwrap();
writeln!(s, " target = {};", self.target.guid()).unwrap();
writeln!(s, " attacker = {};", self.attacker.guid()).unwrap();
writeln!(s, " spell = {};", self.spell).unwrap();
writeln!(s, " damage = {};", self.damage).unwrap();
writeln!(s, " school = {};", self.school.as_test_case_value()).unwrap();
writeln!(s, " absorbed_damage = {};", self.absorbed_damage).unwrap();
writeln!(s, " resisted = {};", self.resisted).unwrap();
writeln!(s, " periodic_log = {};", if self.periodic_log { "TRUE" } else { "FALSE" }).unwrap();
writeln!(s, " unused = {};", self.unused).unwrap();
writeln!(s, " blocked = {};", self.blocked).unwrap();
writeln!(s, " hit_info = {};", self.hit_info.as_test_case_value()).unwrap();
writeln!(s, " extend_flag = {};", self.extend_flag).unwrap();
writeln!(s, "}} [").unwrap();
let [a, b] = (u16::try_from(self.size() + 2).unwrap()).to_be_bytes();
writeln!(s, " {a:#04X}, {b:#04X}, /* size */").unwrap();
let [a, b] = 592_u16.to_le_bytes();
writeln!(s, " {a:#04X}, {b:#04X}, /* opcode */").unwrap();
let mut bytes: Vec<u8> = Vec::new();
self.write_into_vec(&mut bytes).unwrap();
let mut bytes = bytes.into_iter();
crate::util::write_bytes(&mut s, &mut bytes, crate::util::packed_guid_size(&self.target), "target", " ");
crate::util::write_bytes(&mut s, &mut bytes, crate::util::packed_guid_size(&self.attacker), "attacker", " ");
crate::util::write_bytes(&mut s, &mut bytes, 4, "spell", " ");
crate::util::write_bytes(&mut s, &mut bytes, 4, "damage", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "school", " ");
crate::util::write_bytes(&mut s, &mut bytes, 4, "absorbed_damage", " ");
crate::util::write_bytes(&mut s, &mut bytes, 4, "resisted", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "periodic_log", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "unused", " ");
crate::util::write_bytes(&mut s, &mut bytes, 4, "blocked", " ");
crate::util::write_bytes(&mut s, &mut bytes, 4, "hit_info", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "extend_flag", " ");
writeln!(s, "] {{").unwrap();
writeln!(s, " versions = \"{}\";", std::env::var("WOWM_TEST_CASE_WORLD_VERSION").unwrap_or("2.4.3".to_string())).unwrap();
writeln!(s, "}}\n").unwrap();
Some(s)
}
fn size_without_header(&self) -> u32 {
self.size() as u32
}
fn write_into_vec(&self, mut w: impl Write) -> Result<(), std::io::Error> {
crate::util::write_packed_guid(&self.target, &mut w)?;
crate::util::write_packed_guid(&self.attacker, &mut w)?;
w.write_all(&self.spell.to_le_bytes())?;
w.write_all(&self.damage.to_le_bytes())?;
w.write_all(&(self.school.as_int().to_le_bytes()))?;
w.write_all(&self.absorbed_damage.to_le_bytes())?;
w.write_all(&self.resisted.to_le_bytes())?;
w.write_all(u8::from(self.periodic_log).to_le_bytes().as_slice())?;
w.write_all(&self.unused.to_le_bytes())?;
w.write_all(&self.blocked.to_le_bytes())?;
w.write_all(&(self.hit_info.as_int().to_le_bytes()))?;
w.write_all(&self.extend_flag.to_le_bytes())?;
Ok(())
}
fn read_body<S: crate::private::Sealed>(r: &mut &[u8], body_size: u32) -> Result<Self, crate::errors::ParseError> {
Self::read_inner(r, body_size).map_err(|a| crate::errors::ParseError::new(592, "SMSG_SPELLNONMELEEDAMAGELOG", body_size, a))
}
}
#[cfg(feature = "tbc")]
impl crate::tbc::ServerMessage for SMSG_SPELLNONMELEEDAMAGELOG {}
impl SMSG_SPELLNONMELEEDAMAGELOG {
pub(crate) const fn size(&self) -> usize {
crate::util::packed_guid_size(&self.target) + crate::util::packed_guid_size(&self.attacker) + 4 + 4 + 1 + 4 + 4 + 1 + 1 + 4 + 4 + 1 }
}