mirror of
https://github.com/lifegpc/msg-tool.git
synced 2026-06-07 13:28:47 +08:00
Add basic emote psb file export support
This commit is contained in:
@@ -1,3 +1,4 @@
|
||||
//! Emote images
|
||||
pub mod dref;
|
||||
pub mod pimg;
|
||||
pub mod psb;
|
||||
|
||||
242
src/scripts/emote/psb.rs
Normal file
242
src/scripts/emote/psb.rs
Normal file
@@ -0,0 +1,242 @@
|
||||
//! Basic Handle for all emote PSB files.
|
||||
use crate::ext::io::*;
|
||||
use crate::ext::psb::*;
|
||||
use crate::scripts::base::*;
|
||||
use crate::types::*;
|
||||
use crate::utils::encoding::*;
|
||||
use crate::utils::img::*;
|
||||
use anyhow::Result;
|
||||
use base64::Engine;
|
||||
use emote_psb::*;
|
||||
use libtlg_rs::*;
|
||||
use serde::{Deserialize, Serialize};
|
||||
use std::collections::HashMap;
|
||||
use std::io::{Read, Seek, Write};
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct PsbBuilder {}
|
||||
|
||||
impl PsbBuilder {
|
||||
pub fn new() -> Self {
|
||||
Self {}
|
||||
}
|
||||
}
|
||||
|
||||
impl ScriptBuilder for PsbBuilder {
|
||||
fn default_encoding(&self) -> Encoding {
|
||||
Encoding::Utf8
|
||||
}
|
||||
|
||||
fn build_script(
|
||||
&self,
|
||||
buf: Vec<u8>,
|
||||
_filename: &str,
|
||||
encoding: Encoding,
|
||||
_archive_encoding: Encoding,
|
||||
config: &ExtraConfig,
|
||||
_archive: Option<&Box<dyn Script>>,
|
||||
) -> Result<Box<dyn Script>> {
|
||||
Ok(Box::new(Psb::new(MemReader::new(buf), encoding, config)?))
|
||||
}
|
||||
|
||||
fn build_script_from_reader(
|
||||
&self,
|
||||
reader: Box<dyn ReadSeek>,
|
||||
_filename: &str,
|
||||
encoding: Encoding,
|
||||
_archive_encoding: Encoding,
|
||||
config: &ExtraConfig,
|
||||
_archive: Option<&Box<dyn Script>>,
|
||||
) -> Result<Box<dyn Script>> {
|
||||
Ok(Box::new(Psb::new(reader, encoding, config)?))
|
||||
}
|
||||
|
||||
fn build_script_from_file(
|
||||
&self,
|
||||
filename: &str,
|
||||
encoding: Encoding,
|
||||
_archive_encoding: Encoding,
|
||||
config: &ExtraConfig,
|
||||
_archive: Option<&Box<dyn Script>>,
|
||||
) -> Result<Box<dyn Script>> {
|
||||
let file = std::fs::File::open(filename)?;
|
||||
let f = std::io::BufReader::new(file);
|
||||
Ok(Box::new(Psb::new(f, encoding, config)?))
|
||||
}
|
||||
|
||||
fn extensions(&self) -> &'static [&'static str] {
|
||||
&[]
|
||||
}
|
||||
|
||||
fn script_type(&self) -> &'static ScriptType {
|
||||
&ScriptType::EmotePsb
|
||||
}
|
||||
|
||||
fn is_this_format(&self, _filename: &str, buf: &[u8], buf_len: usize) -> Option<u8> {
|
||||
if buf_len >= 4 && buf.starts_with(b"PSB\0") {
|
||||
return Some(10);
|
||||
}
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct Psb {
|
||||
psb: VirtualPsbFixed,
|
||||
encoding: Encoding,
|
||||
config: ExtraConfig,
|
||||
}
|
||||
|
||||
impl Psb {
|
||||
pub fn new<R: Read + Seek>(
|
||||
reader: R,
|
||||
encoding: Encoding,
|
||||
config: &ExtraConfig,
|
||||
) -> Result<Self> {
|
||||
let mut psb = PsbReader::open_psb(reader)
|
||||
.map_err(|e| anyhow::anyhow!("Failed to open psb file: {:?}", e))?;
|
||||
let psb = psb
|
||||
.load()
|
||||
.map_err(|e| anyhow::anyhow!("Failed to load psb: {:?}", e))?
|
||||
.to_psb_fixed();
|
||||
Ok(Self {
|
||||
psb,
|
||||
encoding,
|
||||
config: config.clone(),
|
||||
})
|
||||
}
|
||||
|
||||
fn output_resource(
|
||||
&self,
|
||||
folder_path: &std::path::PathBuf,
|
||||
path: String,
|
||||
data: &[u8],
|
||||
) -> Result<Resource> {
|
||||
let mut res = Resource { path, tlg: None };
|
||||
if self.config.psb_process_tlg && is_valid_tlg(&data) {
|
||||
let tlg = load_tlg(MemReaderRef::new(&data))?;
|
||||
res.tlg = Some(TlgInfo::from_tlg(&tlg, self.encoding));
|
||||
let outtype = self.config.image_type.unwrap_or(ImageOutputType::Png);
|
||||
res.path = {
|
||||
let mut pb = std::path::PathBuf::from(&res.path);
|
||||
pb.set_extension(outtype.as_ref());
|
||||
pb.to_string_lossy().to_string()
|
||||
};
|
||||
let path = folder_path.join(&res.path);
|
||||
let img = ImageData {
|
||||
width: tlg.width as u32,
|
||||
height: tlg.height as u32,
|
||||
color_type: match tlg.color {
|
||||
TlgColorType::Bgr24 => ImageColorType::Bgr,
|
||||
TlgColorType::Bgra32 => ImageColorType::Bgra,
|
||||
TlgColorType::Grayscale8 => ImageColorType::Grayscale,
|
||||
},
|
||||
depth: 8,
|
||||
data: tlg.data,
|
||||
};
|
||||
encode_img(img, outtype, &path.to_string_lossy(), &self.config)?;
|
||||
} else {
|
||||
let path = folder_path.join(&res.path);
|
||||
std::fs::write(&path, data)?;
|
||||
}
|
||||
Ok(res)
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Deserialize, Serialize)]
|
||||
struct TlgInfo {
|
||||
metadata: HashMap<String, String>,
|
||||
}
|
||||
|
||||
impl TlgInfo {
|
||||
fn from_tlg(tlg: &Tlg, encoding: Encoding) -> Self {
|
||||
let mut metadata = HashMap::new();
|
||||
for (k, v) in &tlg.tags {
|
||||
let k = if let Ok(s) = decode_to_string(encoding, &k, true) {
|
||||
s
|
||||
} else {
|
||||
format!(
|
||||
"base64:{}",
|
||||
base64::engine::general_purpose::STANDARD.encode(k)
|
||||
)
|
||||
};
|
||||
let v = if let Ok(s) = decode_to_string(encoding, &v, true) {
|
||||
s
|
||||
} else {
|
||||
format!(
|
||||
"base64:{}",
|
||||
base64::engine::general_purpose::STANDARD.encode(v)
|
||||
)
|
||||
};
|
||||
metadata.insert(k, v);
|
||||
}
|
||||
Self { metadata }
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Deserialize, Serialize)]
|
||||
struct Resource {
|
||||
path: String,
|
||||
tlg: Option<TlgInfo>,
|
||||
}
|
||||
|
||||
impl Script for Psb {
|
||||
fn default_output_script_type(&self) -> OutputScriptType {
|
||||
OutputScriptType::Custom
|
||||
}
|
||||
|
||||
fn is_output_supported(&self, output: OutputScriptType) -> bool {
|
||||
matches!(output, OutputScriptType::Custom)
|
||||
}
|
||||
|
||||
fn default_format_type(&self) -> FormatOptions {
|
||||
FormatOptions::None
|
||||
}
|
||||
|
||||
fn custom_output_extension<'a>(&'a self) -> &'a str {
|
||||
"json"
|
||||
}
|
||||
|
||||
fn custom_export(&self, filename: &std::path::Path, encoding: Encoding) -> Result<()> {
|
||||
let mut data = self.psb.to_json();
|
||||
let mut resources = Vec::new();
|
||||
let mut extra_resources = Vec::new();
|
||||
let folder_path = {
|
||||
let mut pb = filename.to_path_buf();
|
||||
pb.set_extension("");
|
||||
pb
|
||||
};
|
||||
if self.psb.resources().len() > 0 || self.psb.extra().len() > 0 {
|
||||
std::fs::create_dir_all(&folder_path)?;
|
||||
}
|
||||
for (i, data) in self.psb.resources().iter().enumerate() {
|
||||
let i = i as u64;
|
||||
let res_name = self
|
||||
.psb
|
||||
.root()
|
||||
.find_resource_key(i)
|
||||
.map(|s| s.to_string())
|
||||
.unwrap_or(format!("res_{}", i));
|
||||
let res = self.output_resource(&folder_path, res_name, data)?;
|
||||
resources.push(res);
|
||||
}
|
||||
for (i, data) in self.psb.extra().iter().enumerate() {
|
||||
let i = i as u64;
|
||||
let res_name = self
|
||||
.psb
|
||||
.root()
|
||||
.find_resource_key(i)
|
||||
.map(|s| format!("extra_{}", s))
|
||||
.unwrap_or(format!("extra_res_{}", i));
|
||||
let res = self.output_resource(&folder_path, res_name, data)?;
|
||||
extra_resources.push(res);
|
||||
}
|
||||
data["resources"] = json::parse(&serde_json::to_string(&resources)?)?;
|
||||
data["extra_resources"] = json::parse(&serde_json::to_string(&extra_resources)?)?;
|
||||
let s = json::stringify_pretty(data, 2);
|
||||
let s = encode_string(encoding, &s, false)?;
|
||||
let mut file = std::fs::File::create(filename)?;
|
||||
file.write_all(&s)?;
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
@@ -128,6 +128,8 @@ lazy_static::lazy_static! {
|
||||
Box::new(favorite::hcb::HcbScriptBuilder::new()),
|
||||
#[cfg(feature = "silky")]
|
||||
Box::new(silky::map::MapBuilder::new()),
|
||||
#[cfg(feature = "emote-img")]
|
||||
Box::new(emote::psb::PsbBuilder::new()),
|
||||
];
|
||||
/// A list of all script extensions.
|
||||
pub static ref ALL_EXTS: Vec<String> =
|
||||
|
||||
Reference in New Issue
Block a user