use pdf_syntax::object::dict::keys::{BASE_STATE, D, OCGS, OCMD, OCPROPERTIES, OFF, ON, P, TYPE};
use pdf_syntax::object::{Array, Dict, Name, ObjectIdentifier};
use std::collections::HashSet;
pub(crate) struct OcgState {
inactive_ocgs: HashSet<ObjectIdentifier>,
visibility_stack: Vec<bool>,
}
impl OcgState {
fn dummy() -> Self {
Self {
inactive_ocgs: HashSet::default(),
visibility_stack: vec![],
}
}
pub(crate) fn from_catalog(catalog: &Dict<'_>) -> Self {
let Some(oc_properties) = catalog.get::<Dict<'_>>(OCPROPERTIES) else {
return Self::dummy();
};
let Some(config) = oc_properties.get::<Dict<'_>>(D) else {
return Self::dummy();
};
let mut inactive = HashSet::new();
let base_state = config
.get::<Name>(BASE_STATE)
.and_then(|b| BaseState::from_name(b.as_ref()));
if base_state.unwrap_or(BaseState::On) == BaseState::Off
&& let Some(ocgs) = oc_properties.get::<Array<'_>>(OCGS)
{
for item in ocgs.raw_iter() {
if let Some(ref_) = item.as_obj_ref() {
let id: ObjectIdentifier = ref_.into();
inactive.insert(id);
}
}
}
let mut read_ocg_array = |key, insert_active: bool| {
if let Some(arr) = config.get::<Array<'_>>(key) {
for item in arr.raw_iter() {
if let Some(ref_) = item.as_obj_ref() {
let id: ObjectIdentifier = ref_.into();
if insert_active {
inactive.remove(&id);
} else {
inactive.insert(id);
}
}
}
}
};
read_ocg_array(ON, true);
read_ocg_array(OFF, false);
Self {
inactive_ocgs: inactive,
visibility_stack: Vec::new(),
}
}
pub(crate) fn begin_single_oc(&mut self, ocg_id: ObjectIdentifier) {
let is_active = !self.inactive_ocgs.contains(&ocg_id);
let visible = self.is_visible() && is_active;
self.visibility_stack.push(visible);
}
pub(crate) fn begin_ocmd(&mut self, ocmd: &Dict<'_>) {
let policy = ocmd
.get::<Name>(P)
.and_then(|n| OcmdPolicy::from_name(n.as_ref()))
.unwrap_or(OcmdPolicy::AnyOn);
let mut ocg_ids: Vec<ObjectIdentifier> = Vec::new();
if let Some(arr) = ocmd.get::<Array<'_>>(OCGS) {
for item in arr.raw_iter() {
if let Some(ref_) = item.as_obj_ref() {
ocg_ids.push(ref_.into());
}
}
} else if let Some(ref_) = ocmd.get_ref(OCGS) {
ocg_ids.push(ref_.into());
}
let is_active = if ocg_ids.is_empty() {
true
} else {
match policy {
OcmdPolicy::AllOn => ocg_ids.iter().all(|id| !self.inactive_ocgs.contains(id)),
OcmdPolicy::AnyOn => ocg_ids.iter().any(|id| !self.inactive_ocgs.contains(id)),
OcmdPolicy::AnyOff => ocg_ids.iter().any(|id| self.inactive_ocgs.contains(id)),
OcmdPolicy::AllOff => ocg_ids.iter().all(|id| self.inactive_ocgs.contains(id)),
}
};
let visible = self.is_visible() && is_active;
self.visibility_stack.push(visible);
}
pub(crate) fn begin_ocg(&mut self, props: &Dict<'_>, ref_id: ObjectIdentifier) {
match props.get::<Name>(TYPE).as_deref() {
Some(OCMD) => self.begin_ocmd(props),
_ => self.begin_single_oc(ref_id),
}
}
pub(crate) fn begin_marked_content(&mut self) {
let visible = self.is_visible();
self.visibility_stack.push(visible);
}
pub(crate) fn end_marked_content(&mut self) {
self.visibility_stack.pop();
}
pub(crate) fn is_visible(&self) -> bool {
self.visibility_stack.last().copied().unwrap_or(true)
}
}
impl Default for OcgState {
fn default() -> Self {
Self::dummy()
}
}
#[derive(Debug, PartialEq, Eq, Copy, Clone)]
enum BaseState {
On,
Off,
Unchanged,
}
impl BaseState {
fn from_name(name: &[u8]) -> Option<Self> {
match name {
b"ON" => Some(Self::On),
b"OFF" => Some(Self::Off),
b"Unchanged" => Some(Self::Unchanged),
_ => None,
}
}
}
#[derive(Debug, PartialEq, Eq, Copy, Clone)]
enum OcmdPolicy {
AllOn,
AnyOn,
AnyOff,
AllOff,
}
impl OcmdPolicy {
fn from_name(name: &[u8]) -> Option<Self> {
match name {
b"AllOn" => Some(Self::AllOn),
b"AnyOn" => Some(Self::AnyOn),
b"AnyOff" => Some(Self::AnyOff),
b"AllOff" => Some(Self::AllOff),
_ => None,
}
}
}