use std::io::{Read, Write};
use crate::tbc::{
Class, Gender, Race,
};
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord, Default)]
pub struct CMSG_CHAR_CREATE {
pub name: String,
pub race: Race,
pub class: Class,
pub gender: Gender,
pub skin_color: u8,
pub face: u8,
pub hair_style: u8,
pub hair_color: u8,
pub facial_hair: u8,
}
impl CMSG_CHAR_CREATE {
pub const OUTFIT_ID_VALUE: u8 = 0x00;
}
impl crate::private::Sealed for CMSG_CHAR_CREATE {}
impl CMSG_CHAR_CREATE {
fn read_inner(mut r: &mut &[u8], body_size: u32) -> Result<Self, crate::errors::ParseErrorKind> {
if !(10..=265).contains(&body_size) {
return Err(crate::errors::ParseErrorKind::InvalidSize);
}
let name = {
let name = crate::util::read_c_string_to_vec(&mut r)?;
String::from_utf8(name)?
};
let race = crate::util::read_u8_le(&mut r)?.try_into()?;
let class = crate::util::read_u8_le(&mut r)?.try_into()?;
let gender = crate::util::read_u8_le(&mut r)?.try_into()?;
let skin_color = crate::util::read_u8_le(&mut r)?;
let face = crate::util::read_u8_le(&mut r)?;
let hair_style = crate::util::read_u8_le(&mut r)?;
let hair_color = crate::util::read_u8_le(&mut r)?;
let facial_hair = crate::util::read_u8_le(&mut r)?;
let _outfit_id = crate::util::read_u8_le(&mut r)?;
Ok(Self {
name,
race,
class,
gender,
skin_color,
face,
hair_style,
hair_color,
facial_hair,
})
}
}
impl crate::Message for CMSG_CHAR_CREATE {
const OPCODE: u32 = 0x0036;
#[cfg(feature = "print-testcase")]
fn message_name(&self) -> &'static str {
"CMSG_CHAR_CREATE"
}
#[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 CMSG_CHAR_CREATE {{").unwrap();
writeln!(s, " name = \"{}\";", self.name).unwrap();
writeln!(s, " race = {};", self.race.as_test_case_value()).unwrap();
writeln!(s, " class = {};", self.class.as_test_case_value()).unwrap();
writeln!(s, " gender = {};", self.gender.as_test_case_value()).unwrap();
writeln!(s, " skin_color = {};", self.skin_color).unwrap();
writeln!(s, " face = {};", self.face).unwrap();
writeln!(s, " hair_style = {};", self.hair_style).unwrap();
writeln!(s, " hair_color = {};", self.hair_color).unwrap();
writeln!(s, " facial_hair = {};", self.facial_hair).unwrap();
writeln!(s, "}} [").unwrap();
let [a, b] = (u16::try_from(self.size() + 4).unwrap()).to_be_bytes();
writeln!(s, " {a:#04X}, {b:#04X}, /* size */").unwrap();
let [a, b, c, d] = 54_u32.to_le_bytes();
writeln!(s, " {a:#04X}, {b:#04X}, {c:#04X}, {d:#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, self.name.len() + 1, "name", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "race", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "class", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "gender", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "skin_color", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "face", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "hair_style", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "hair_color", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "facial_hair", " ");
crate::util::write_bytes(&mut s, &mut bytes, 1, "outfit_id", " ");
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> {
assert_ne!(self.name.as_bytes().iter().next_back(), Some(&0_u8), "String `name` must not be null-terminated.");
w.write_all(self.name.as_bytes())?;
w.write_all(&[0])?;
w.write_all(&(self.race.as_int().to_le_bytes()))?;
w.write_all(&(self.class.as_int().to_le_bytes()))?;
w.write_all(&(self.gender.as_int().to_le_bytes()))?;
w.write_all(&self.skin_color.to_le_bytes())?;
w.write_all(&self.face.to_le_bytes())?;
w.write_all(&self.hair_style.to_le_bytes())?;
w.write_all(&self.hair_color.to_le_bytes())?;
w.write_all(&self.facial_hair.to_le_bytes())?;
w.write_all(&Self::OUTFIT_ID_VALUE.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(54, "CMSG_CHAR_CREATE", body_size, a))
}
}
#[cfg(feature = "tbc")]
impl crate::tbc::ClientMessage for CMSG_CHAR_CREATE {}
impl CMSG_CHAR_CREATE {
pub(crate) fn size(&self) -> usize {
self.name.len() + 1 + 1 + 1 + 1 + 1 + 1 + 1 + 1 + 1 + 1 }
}