mod attach;
mod convert;
mod image;
mod link;
mod metadata;
mod outline;
mod page;
mod paint;
mod shape;
mod tags;
mod text;
mod util;
pub use self::metadata::{Timestamp, Timezone};
use std::fmt::{self, Debug, Formatter};
use ecow::eco_format;
use krilla::configure::Validator;
use serde::{Deserialize, Serialize};
use typst_library::diag::{SourceResult, StrResult, bail};
use typst_library::foundations::Smart;
use typst_library::layout::{PageRanges, PagedDocument};
#[typst_macros::time(name = "pdf")]
pub fn pdf(document: &PagedDocument, options: &PdfOptions) -> SourceResult<Vec<u8>> {
convert::convert(document, options)
}
#[doc(hidden)]
pub fn pdf_tags(document: &PagedDocument, options: &PdfOptions) -> SourceResult<String> {
convert::tag_tree(document, options)
}
#[derive(Debug)]
pub struct PdfOptions<'a> {
pub ident: Smart<&'a str>,
pub timestamp: Option<Timestamp>,
pub page_ranges: Option<PageRanges>,
pub standards: PdfStandards,
pub tagged: bool,
}
impl PdfOptions<'_> {
pub(crate) fn is_pdf_ua(&self) -> bool {
self.standards.config.validator() == Validator::UA1
}
}
impl Default for PdfOptions<'_> {
fn default() -> Self {
Self {
ident: Smart::Auto,
timestamp: None,
page_ranges: None,
standards: PdfStandards::default(),
tagged: true,
}
}
}
#[derive(Clone)]
pub struct PdfStandards {
pub(crate) config: krilla::configure::Configuration,
}
impl PdfStandards {
pub fn new(list: &[PdfStandard]) -> StrResult<Self> {
use krilla::configure::{Configuration, PdfVersion, Validator};
let mut version: Option<PdfVersion> = None;
let mut set_version = |v: PdfVersion| -> StrResult<()> {
if let Some(prev) = version {
bail!(
"PDF cannot conform to {} and {} at the same time",
prev.as_str(),
v.as_str()
);
}
version = Some(v);
Ok(())
};
let mut validator = None;
let mut set_validator = |v: Validator| -> StrResult<()> {
if validator.is_some() {
bail!("Typst currently only supports one PDF substandard at a time");
}
validator = Some(v);
Ok(())
};
for standard in list {
match standard {
PdfStandard::V_1_4 => set_version(PdfVersion::Pdf14)?,
PdfStandard::V_1_5 => set_version(PdfVersion::Pdf15)?,
PdfStandard::V_1_6 => set_version(PdfVersion::Pdf16)?,
PdfStandard::V_1_7 => set_version(PdfVersion::Pdf17)?,
PdfStandard::V_2_0 => set_version(PdfVersion::Pdf20)?,
PdfStandard::A_1b => set_validator(Validator::A1_B)?,
PdfStandard::A_1a => set_validator(Validator::A1_A)?,
PdfStandard::A_2b => set_validator(Validator::A2_B)?,
PdfStandard::A_2u => set_validator(Validator::A2_U)?,
PdfStandard::A_2a => set_validator(Validator::A2_A)?,
PdfStandard::A_3b => set_validator(Validator::A3_B)?,
PdfStandard::A_3u => set_validator(Validator::A3_U)?,
PdfStandard::A_3a => set_validator(Validator::A3_A)?,
PdfStandard::A_4 => set_validator(Validator::A4)?,
PdfStandard::A_4f => set_validator(Validator::A4F)?,
PdfStandard::A_4e => set_validator(Validator::A4E)?,
PdfStandard::Ua_1 => set_validator(Validator::UA1)?,
}
}
let config = match (version, validator) {
(Some(version), Some(validator)) => {
Configuration::new_with(validator, version).ok_or_else(|| {
eco_format!(
"{} is not compatible with {}",
version.as_str(),
validator.as_str()
)
})?
}
(Some(version), None) => Configuration::new_with_version(version),
(None, Some(validator)) => Configuration::new_with_validator(validator),
(None, None) => Configuration::new_with_version(PdfVersion::Pdf17),
};
Ok(Self { config })
}
}
impl Debug for PdfStandards {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
f.pad("PdfStandards(..)")
}
}
impl Default for PdfStandards {
fn default() -> Self {
use krilla::configure::{Configuration, PdfVersion};
Self {
config: Configuration::new_with_version(PdfVersion::Pdf17),
}
}
}
#[derive(Debug, Copy, Clone, Eq, PartialEq, Serialize, Deserialize)]
#[allow(non_camel_case_types)]
#[non_exhaustive]
pub enum PdfStandard {
#[serde(rename = "1.4")]
V_1_4,
#[serde(rename = "1.5")]
V_1_5,
#[serde(rename = "1.6")]
V_1_6,
#[serde(rename = "1.7")]
V_1_7,
#[serde(rename = "2.0")]
V_2_0,
#[serde(rename = "a-1b")]
A_1b,
#[serde(rename = "a-1a")]
A_1a,
#[serde(rename = "a-2b")]
A_2b,
#[serde(rename = "a-2u")]
A_2u,
#[serde(rename = "a-2a")]
A_2a,
#[serde(rename = "a-3b")]
A_3b,
#[serde(rename = "a-3u")]
A_3u,
#[serde(rename = "a-3a")]
A_3a,
#[serde(rename = "a-4")]
A_4,
#[serde(rename = "a-4f")]
A_4f,
#[serde(rename = "a-4e")]
A_4e,
#[serde(rename = "ua-1")]
Ua_1,
}