use std::{borrow::Cow, collections::HashMap, fmt::Debug};
use anyhow::Result;
use bytemuck::{Pod, Zeroable};
use strum::{Display, EnumCount, FromRepr};
mod path;
pub mod schema;
mod value;
pub use path::{path, Path};
pub use schema::{ChildrenKey, FieldKey};
pub use value::{Value, ValueConversionError};
#[repr(u32)]
#[derive(Debug, Default, Clone, Copy, PartialEq, Eq, FromRepr, EnumCount, Display)]
pub enum SpecType {
#[default]
Unknown = 0,
Attribute = 1,
Connection = 2,
Expression = 3,
Mapper = 4,
MapperArg = 5,
Prim = 6,
PseudoRoot = 7,
Relationship = 8,
RelationshipTarget = 9,
Variant = 10,
VariantSet = 11,
}
#[repr(i32)]
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, FromRepr)]
#[cfg_attr(feature = "serde", derive(serde::Serialize))]
#[cfg_attr(feature = "serde", serde(rename_all = "camelCase"))]
pub enum Specifier {
Def,
Over,
Class,
}
#[repr(i32)]
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, FromRepr)]
#[cfg_attr(feature = "serde", derive(serde::Serialize))]
#[cfg_attr(feature = "serde", serde(rename_all = "camelCase"))]
pub enum Permission {
Public,
Private,
}
#[repr(i32)]
#[derive(Debug, Default, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, FromRepr)]
#[cfg_attr(feature = "serde", derive(serde::Serialize))]
#[cfg_attr(feature = "serde", serde(rename_all = "camelCase"))]
pub enum Variability {
#[default]
Varying,
Uniform,
}
#[repr(C)]
#[derive(Debug, Clone, Copy, PartialEq, Pod, Zeroable)]
#[cfg_attr(feature = "serde", derive(serde::Serialize))]
pub struct LayerOffset {
pub offset: f64,
pub scale: f64,
}
impl Default for LayerOffset {
fn default() -> Self {
Self {
offset: 0.0,
scale: 1.0,
}
}
}
impl LayerOffset {
#[inline]
pub fn new(offset: f64, scale: f64) -> Self {
Self { offset, scale }
}
#[inline]
pub fn is_valid(&self) -> bool {
self.offset.is_finite() && self.scale.is_finite()
}
}
#[derive(Debug, Default, Clone, PartialEq)]
#[cfg_attr(feature = "serde", derive(serde::Serialize))]
pub struct Payload {
#[cfg_attr(feature = "serde", serde(rename = "asset", skip_serializing_if = "String::is_empty"))]
pub asset_path: String,
#[cfg_attr(feature = "serde", serde(rename = "path", skip_serializing_if = "Path::is_empty"))]
pub prim_path: Path,
#[cfg_attr(
feature = "serde",
serde(rename = "layerOffset", skip_serializing_if = "Option::is_none")
)]
pub layer_offset: Option<LayerOffset>,
}
#[derive(Debug, Default, Clone, PartialEq)]
#[cfg_attr(feature = "serde", derive(serde::Serialize))]
pub struct Reference {
#[cfg_attr(feature = "serde", serde(rename = "asset", skip_serializing_if = "String::is_empty"))]
pub asset_path: String,
#[cfg_attr(feature = "serde", serde(rename = "path", skip_serializing_if = "Path::is_empty"))]
pub prim_path: Path,
#[cfg_attr(feature = "serde", serde(rename = "layerOffset"))]
pub layer_offset: LayerOffset,
#[cfg_attr(
feature = "serde",
serde(rename = "customData", skip_serializing_if = "HashMap::is_empty")
)]
pub custom_data: HashMap<String, Value>,
}
mod list_op;
pub use list_op::ListOp;
pub type IntListOp = ListOp<i32>;
pub type UintListOp = ListOp<u32>;
pub type Int64ListOp = ListOp<i64>;
pub type Uint64ListOp = ListOp<u64>;
pub type StringListOp = ListOp<String>;
pub type TokenListOp = ListOp<String>;
pub type PathListOp = ListOp<Path>;
pub type ReferenceListOp = ListOp<Reference>;
pub type PayloadListOp = ListOp<Payload>;
pub type TimeSampleMap = Vec<(f64, Value)>;
pub trait AbstractData {
fn has_spec(&self, path: &Path) -> bool;
fn has_field(&self, path: &Path, field: &str) -> bool;
fn spec_type(&self, path: &Path) -> Option<SpecType>;
fn get(&self, path: &Path, field: &str) -> Result<Cow<'_, Value>>;
fn list(&self, path: &Path) -> Option<Vec<String>>;
}
pub type LayerData = Box<dyn AbstractData>;
#[derive(Debug, Clone)]
pub struct Spec {
pub ty: SpecType,
pub fields: HashMap<String, Value>,
}
impl Spec {
pub fn new(ty: SpecType) -> Self {
Self {
ty,
fields: Default::default(),
}
}
#[inline]
pub fn add(&mut self, key: impl AsRef<str>, value: impl Into<Value>) {
self.fields.insert(key.as_ref().to_owned(), value.into());
}
}