An unofficial collection of APIs used in FreeJam games and mods
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

194 lines
7.1KB

  1. use chrono::{naive::NaiveDate, Datelike};
  2. use std::io::{Read, Write};
  3. use crate::techblox::{EntityHeader, BlockGroupEntity, parse_i64, parse_u32, Parsable, SerializedEntityDescriptor,
  4. SerializedFlyCamEntity, SerializedPhysicsCameraEntity};
  5. use crate::techblox::blocks::{lookup_hashname, SerializedWireEntity, SerializedGlobalWireSettingsEntity};
  6. /// A collection of cubes and other data from a GameSave.techblox file
  7. //#[derive(Clone)]
  8. pub struct GameSave {
  9. /// Game version that this save was created by.
  10. /// This may affect how the rest of the save file was parsed.
  11. pub version: NaiveDate,
  12. /// Time when file was saved, corresponding to ticks since 0 AD
  13. /// https://docs.microsoft.com/en-us/dotnet/api/system.datetime.ticks?view=netframework-4.7.2
  14. /// Not used for deserialization so not required to be sensible.
  15. pub ticks: i64,
  16. /// Amount of cubes present in the save data, as claimed by the file header.
  17. pub cube_len: u32,
  18. /// Maximum block entity identifier in the game save.
  19. /// Not used for deserialization so not required to be correct.
  20. pub max_entity_id: u32,
  21. /// Amount of block groups, as claimed by the file header.
  22. pub group_len: u32,
  23. /// Entity group descriptors for block group entities.
  24. pub group_headers: Vec<EntityHeader>,
  25. /// Block group entities.
  26. pub cube_groups: Vec<BlockGroupEntity>,
  27. /// Entity group descriptors for block entities.
  28. pub cube_headers: Vec<EntityHeader>,
  29. /// Blocks
  30. pub cube_entities: Vec<Box<dyn SerializedEntityDescriptor>>,
  31. /// Amount of wires in the save data, as claimed by the file.
  32. pub wire_len: u32,
  33. /// Entity group descriptor for wire entities.
  34. pub wire_headers: Vec<EntityHeader>,
  35. /// Wires
  36. pub wire_entities: Vec<SerializedWireEntity>,
  37. /// Entity group descriptor for wire settings
  38. pub wire_settings_header: EntityHeader,
  39. /// Wire settings
  40. pub wire_settings_entity: SerializedGlobalWireSettingsEntity,
  41. /// Entity group descriptor for player fly camera
  42. pub flycam_header: EntityHeader,
  43. /// Player edit mode fly camera
  44. pub flycam_entity: SerializedFlyCamEntity,
  45. /// Entity group descriptor for player simulation mode camera
  46. pub phycam_header: EntityHeader,
  47. /// Player simulation mode camera
  48. pub phycam_entity: SerializedPhysicsCameraEntity,
  49. }
  50. impl Parsable for GameSave {
  51. /// Process a Techblox save file from raw bytes.
  52. fn parse(data: &mut dyn Read) -> std::io::Result<Self> {
  53. // parse version
  54. let year = parse_u32(data)?; // parsed as i32 in-game for some reason
  55. let month = parse_u32(data)?;
  56. let day = parse_u32(data)?;
  57. let date = NaiveDate::from_ymd(year as i32, month, day);
  58. let ticks = parse_i64(data)?; // unused
  59. let cube_count = parse_u32(data)?; // parsed as i32 in-game for some reason
  60. let max_e_id = parse_u32(data)?; // unused
  61. let group_count = parse_u32(data)?; // parsed as i32 in-game for some reason
  62. // parse block groups
  63. let mut groups_h = Vec::<EntityHeader>::with_capacity(group_count as usize);
  64. let mut groups_e = Vec::<BlockGroupEntity>::with_capacity(group_count as usize);
  65. for _i in 0..group_count {
  66. groups_h.push(EntityHeader::parse(data)?);
  67. groups_e.push(BlockGroupEntity::parse(data)?);
  68. }
  69. // parse cube data
  70. let mut cubes_h = Vec::<EntityHeader>::with_capacity(cube_count as usize);
  71. let mut cubes_e = Vec::<Box<dyn SerializedEntityDescriptor>>::with_capacity(cube_count as usize);
  72. for _i in 0..cube_count {
  73. let header = EntityHeader::parse(data)?;
  74. let hash = header.hash;
  75. cubes_h.push(header);
  76. cubes_e.push(lookup_hashname(hash, data)?);
  77. }
  78. // parse wire data
  79. let wire_count = parse_u32(data)?;
  80. let mut wires_h = Vec::<EntityHeader>::with_capacity(wire_count as usize);
  81. let mut wires_e = Vec::<SerializedWireEntity>::with_capacity(wire_count as usize);
  82. for _i in 0..wire_count {
  83. wires_h.push(EntityHeader::parse(data)?);
  84. wires_e.push(SerializedWireEntity::parse(data)?);
  85. }
  86. // parse global wire settings
  87. let wire_settings_h = EntityHeader::parse(data)?;
  88. let wire_settings_e = SerializedGlobalWireSettingsEntity::parse(data)?;
  89. // parse player cameras
  90. let flycam_h = EntityHeader::parse(data)?;
  91. let flycam_e = SerializedFlyCamEntity::parse(data)?;
  92. let phycam_h = EntityHeader::parse(data)?;
  93. let phycam_e = SerializedPhysicsCameraEntity::parse(data)?;
  94. // build struct
  95. Ok(Self {
  96. version: date,
  97. ticks: ticks,
  98. cube_len: cube_count,
  99. max_entity_id: max_e_id,
  100. group_len: group_count,
  101. group_headers: groups_h,
  102. cube_groups: groups_e,
  103. cube_headers: cubes_h,
  104. cube_entities: cubes_e,
  105. wire_len: wire_count,
  106. wire_headers: wires_h,
  107. wire_entities: wires_e,
  108. wire_settings_header: wire_settings_h,
  109. wire_settings_entity: wire_settings_e,
  110. flycam_header: flycam_h,
  111. flycam_entity: flycam_e,
  112. phycam_header: phycam_h,
  113. phycam_entity: phycam_e,
  114. })
  115. }
  116. fn dump(&self, writer: &mut dyn Write) -> std::io::Result<usize> {
  117. let mut write_count: usize = 0;
  118. // version
  119. write_count += self.version.year().dump(writer)?;
  120. write_count += self.version.month().dump(writer)?;
  121. write_count += self.version.day().dump(writer)?;
  122. // unused separator \/
  123. write_count += self.ticks.dump(writer)?;
  124. write_count += self.cube_len.dump(writer)?;
  125. // unused separator \/
  126. write_count += self.max_entity_id.dump(writer)?;
  127. write_count += self.group_len.dump(writer)?;
  128. // dump block groups
  129. for i in 0..self.group_len as usize {
  130. write_count += self.group_headers[i].dump(writer)?;
  131. write_count += self.cube_groups[i].dump(writer)?;
  132. }
  133. // dump cube data
  134. for i in 0..self.cube_len as usize {
  135. write_count += self.cube_headers[i].dump(writer)?;
  136. write_count += self.cube_entities[i].dump(writer)?;
  137. }
  138. // dump wire data
  139. write_count += self.wire_len.dump(writer)?;
  140. for i in 0..self.wire_len as usize {
  141. write_count += self.wire_headers[i].dump(writer)?;
  142. write_count += self.wire_entities[i].dump(writer)?;
  143. }
  144. // dump global wire settings
  145. write_count += self.wire_settings_header.dump(writer)?;
  146. write_count += self.wire_settings_entity.dump(writer)?;
  147. // dump player cameras
  148. write_count += self.flycam_header.dump(writer)?;
  149. write_count += self.flycam_entity.dump(writer)?;
  150. write_count += self.phycam_header.dump(writer)?;
  151. write_count += self.phycam_entity.dump(writer)?;
  152. Ok(write_count)
  153. }
  154. }
  155. impl std::string::ToString for GameSave {
  156. fn to_string(&self) -> String {
  157. format!("{}g {}c {}w (v{})", self.group_len, self.cube_len, self.wire_len, self.version)
  158. }
  159. }