pub mod barcode;
pub mod chart;
pub mod error;
pub mod font;
pub mod image_loader;
pub mod layout;
pub mod model;
pub mod pdf;
pub mod qrcode;
pub mod style;
pub mod svg;
pub mod template;
pub mod text;
#[cfg(feature = "wasm")]
pub mod wasm;
#[cfg(feature = "wasm-raw")]
pub mod wasm_raw;
pub use error::FormeError;
pub use layout::LayoutInfo;
pub use model::{
CertificationConfig, ColumnDef, ColumnWidth, FontEntry, PatternType, RedactionPattern,
RedactionRegion, TextRun,
};
pub use model::{ChartDataPoint, ChartSeries, DotPlotGroup};
pub use model::{Document, Metadata, Node, NodeKind, PageConfig, PageSize};
pub use style::Style;
use font::FontContext;
use layout::LayoutEngine;
use pdf::PdfWriter;
pub fn certify_pdf(
pdf_bytes: &[u8],
config: &model::CertificationConfig,
) -> Result<Vec<u8>, FormeError> {
pdf::certify::certify_pdf(pdf_bytes, config)
}
pub fn redact_pdf(
pdf_bytes: &[u8],
regions: &[model::RedactionRegion],
) -> Result<Vec<u8>, FormeError> {
pdf::redaction::redact_pdf(pdf_bytes, regions)
}
pub fn find_text_regions(
pdf_bytes: &[u8],
patterns: &[model::RedactionPattern],
) -> Result<Vec<RedactionRegion>, FormeError> {
pdf::redaction::find_text_regions(pdf_bytes, patterns)
}
pub fn redact_text(
pdf_bytes: &[u8],
patterns: &[model::RedactionPattern],
) -> Result<Vec<u8>, FormeError> {
pdf::redaction::redact_text(pdf_bytes, patterns)
}
pub fn merge_pdfs(pdfs: &[&[u8]]) -> Result<Vec<u8>, FormeError> {
pdf::merge::merge_pdfs(pdfs)
}
pub fn render(document: &Document) -> Result<Vec<u8>, FormeError> {
let mut font_context = FontContext::new();
register_document_fonts(&mut font_context, &document.fonts);
let engine = LayoutEngine::new();
let mut pages = engine.layout(document, &font_context);
for _ in 0..2 {
let needed = digits_for_count(pages.len());
if needed == font_context.sentinel_digit_count() {
break;
}
font_context.set_sentinel_digit_count(needed);
pages = engine.layout(document, &font_context);
}
let writer = PdfWriter::new();
let tagged = document.tagged
|| document.pdf_ua
|| matches!(document.pdfa, Some(model::PdfAConformance::A2a));
let pdf = writer.write(
&pages,
&document.metadata,
&font_context,
tagged,
document.pdfa.as_ref(),
document.pdf_ua,
document.embedded_data.as_deref(),
document.flatten_forms,
)?;
let pdf = if let Some(ref sig_config) = document.certification {
pdf::certify::certify_pdf(&pdf, sig_config)?
} else {
pdf
};
Ok(pdf)
}
pub fn render_with_layout(document: &Document) -> Result<(Vec<u8>, LayoutInfo), FormeError> {
let mut font_context = FontContext::new();
register_document_fonts(&mut font_context, &document.fonts);
let engine = LayoutEngine::new();
let mut pages = engine.layout(document, &font_context);
for _ in 0..2 {
let needed = digits_for_count(pages.len());
if needed == font_context.sentinel_digit_count() {
break;
}
font_context.set_sentinel_digit_count(needed);
pages = engine.layout(document, &font_context);
}
let layout_info = LayoutInfo::from_pages(&pages);
let writer = PdfWriter::new();
let tagged = document.tagged
|| document.pdf_ua
|| matches!(document.pdfa, Some(model::PdfAConformance::A2a));
let pdf = writer.write(
&pages,
&document.metadata,
&font_context,
tagged,
document.pdfa.as_ref(),
document.pdf_ua,
document.embedded_data.as_deref(),
document.flatten_forms,
)?;
let pdf = if let Some(ref sig_config) = document.certification {
pdf::certify::certify_pdf(&pdf, sig_config)?
} else {
pdf
};
Ok((pdf, layout_info))
}
fn digits_for_count(n: usize) -> u32 {
if n < 10 {
1
} else if n < 100 {
2
} else if n < 1000 {
3
} else {
4
}
}
fn register_document_fonts(font_context: &mut FontContext, fonts: &[FontEntry]) {
use base64::Engine as _;
let b64 = base64::engine::general_purpose::STANDARD;
for entry in fonts {
let bytes = if let Some(comma_pos) = entry.src.find(',') {
b64.decode(&entry.src[comma_pos + 1..]).ok()
} else {
b64.decode(&entry.src).ok()
};
if let Some(data) = bytes {
font_context
.registry_mut()
.register(&entry.family, entry.weight, entry.italic, data);
}
}
}
pub fn render_json(json: &str) -> Result<Vec<u8>, FormeError> {
let document: Document = serde_json::from_str(json)?;
render(&document)
}
pub fn render_json_with_layout(json: &str) -> Result<(Vec<u8>, LayoutInfo), FormeError> {
let document: Document = serde_json::from_str(json)?;
render_with_layout(&document)
}
pub fn render_template(template_json: &str, data_json: &str) -> Result<Vec<u8>, FormeError> {
let template: serde_json::Value = serde_json::from_str(template_json)?;
let data: serde_json::Value = serde_json::from_str(data_json)?;
let resolved = template::evaluate_template(&template, &data)?;
let document: Document = serde_json::from_value(resolved)?;
render(&document)
}
pub fn render_template_with_layout(
template_json: &str,
data_json: &str,
) -> Result<(Vec<u8>, LayoutInfo), FormeError> {
let template: serde_json::Value = serde_json::from_str(template_json)?;
let data: serde_json::Value = serde_json::from_str(data_json)?;
let resolved = template::evaluate_template(&template, &data)?;
let document: Document = serde_json::from_value(resolved)?;
render_with_layout(&document)
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_digits_for_count() {
assert_eq!(digits_for_count(0), 1);
assert_eq!(digits_for_count(1), 1);
assert_eq!(digits_for_count(9), 1);
assert_eq!(digits_for_count(10), 2);
assert_eq!(digits_for_count(99), 2);
assert_eq!(digits_for_count(100), 3);
assert_eq!(digits_for_count(999), 3);
assert_eq!(digits_for_count(1000), 4);
}
}