mod any;
mod mipmap_or_drawable;
mod res_or_string;
mod types;
pub use any::*;
pub use mipmap_or_drawable::*;
pub use res_or_string::*;
use serde::{
de::{self, Visitor},
Deserialize, Deserializer, Serialize, Serializer,
};
use std::{
fmt,
io::{Read, Write},
marker::PhantomData,
str::FromStr,
};
pub use types::*;
use yaserde::{YaDeserialize, YaSerialize};
pub trait ResourceType: FromStr {
fn new(name: &str, package: Option<String>) -> Resource<Self> {
Resource {
name: name.to_string(),
package,
phantom: PhantomData,
}
}
fn resource_type() -> &'static str;
}
#[derive(Debug, PartialEq, Eq, Clone)]
pub struct Resource<T: ResourceType> {
name: String,
package: Option<String>,
phantom: PhantomData<T>,
}
impl<T: ResourceType> Resource<T> {
pub fn new(name: &str) -> Self {
Self {
name: name.to_string(),
package: None,
phantom: PhantomData,
}
}
pub fn new_with_package(name: &str, package: Option<String>) -> Self {
Self {
name: name.to_string(),
package,
phantom: PhantomData,
}
}
pub fn name(&self) -> &str {
&self.name
}
pub fn resource_type(&self) -> &'static str {
T::resource_type()
}
}
impl<T: ResourceType> ToString for Resource<T> {
fn to_string(&self) -> String {
if let Some(package) = &self.package {
format!("@{}:{}/{}", package, T::resource_type(), self.name)
} else {
format!("@{}/{}", T::resource_type(), self.name)
}
}
}
impl<T: ResourceType> Serialize for Resource<T> {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
serializer.serialize_str(&self.to_string())
}
}
impl<T: ResourceType> YaSerialize for Resource<T> {
fn serialize<W: Write>(&self, writer: &mut yaserde::ser::Serializer<W>) -> Result<(), String> {
if let Some(package) = &self.package {
let _ret = writer.write(xml::writer::XmlEvent::characters(&format!(
"@{}:{}/{}",
package,
T::resource_type(),
self.name
)));
} else {
let _ret = writer.write(xml::writer::XmlEvent::characters(&format!(
"@{}/{}",
T::resource_type(),
self.name
)));
}
Ok(())
}
fn serialize_attributes(
&self,
attributes: Vec<xml::attribute::OwnedAttribute>,
namespace: xml::namespace::Namespace,
) -> Result<
(
Vec<xml::attribute::OwnedAttribute>,
xml::namespace::Namespace,
),
String,
> {
Ok((attributes, namespace))
}
}
struct ResourceVisitor<T: ResourceType> {
phantom: PhantomData<T>,
}
impl<T: ResourceType> ResourceVisitor<T> {
pub fn new() -> Self {
ResourceVisitor {
phantom: PhantomData,
}
}
}
impl<'de, T: ResourceType> Visitor<'de> for ResourceVisitor<T> {
type Value = Resource<T>;
fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str(&format!(
"an {} resource in format @[package:]{}/resource_name",
T::resource_type(),
T::resource_type()
))
}
fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
where
E: de::Error,
{
parse_resource_with_type(v).map_err(E::custom)
}
}
impl<'de, T: ResourceType> Deserialize<'de> for Resource<T> {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: Deserializer<'de>,
{
deserializer.deserialize_string(ResourceVisitor::new())
}
}
impl<T: ResourceType> YaDeserialize for Resource<T> {
fn deserialize<R: Read>(reader: &mut yaserde::de::Deserializer<R>) -> Result<Self, String> {
loop {
match reader.next_event()? {
xml::reader::XmlEvent::StartElement { .. } => {}
xml::reader::XmlEvent::Characters(ref text_content) => {
return parse_resource_with_type(text_content);
}
_ => {
break;
}
}
}
Err("Unable to parse attribute".to_string())
}
}
fn parse_resource(resource: &str) -> Result<(Option<String>, String, String), String> {
if resource.is_empty() {
return Err("value of attribute is empty".to_string());
};
let split_str: Vec<_> = resource.split('/').collect();
if split_str.len() != 2 {
return Err(
"a wrong resource format, expected format @[package:]resource_type/resource_name"
.to_string(),
);
};
let first_part = split_str.first().unwrap(); let resource_type = &first_part[1..];
let split_type: Vec<_> = resource_type.split(':').collect();
let (resource_type, package) = if split_type.len() == 2 {
(split_type[1], Some(split_type[0].to_string()))
} else {
(split_type[0], None)
};
let resource_name = split_str.get(1).unwrap(); Ok((
package,
resource_type.to_string(),
resource_name.to_string(),
))
}
fn parse_resource_with_type<T: ResourceType>(resource: &str) -> Result<Resource<T>, String> {
let (package, resource_type, resource_name) = parse_resource(resource)?;
if resource_type != T::resource_type() {
return Err(format!(
"a wrong resource type, expected @[package:]{}/{}, found {}",
T::resource_type(),
resource_name,
resource
));
};
Ok(Resource {
name: resource_name,
package,
phantom: PhantomData,
})
}