Skip to content

kyasu1/pdforge

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

PDForge

A powerful and flexible PDF generation library written in Rust, inspired by pdfme. PDForge allows you to create sophisticated PDFs from JSON templates with support for multiple content types including text, tables, images, QR codes, and SVG graphics.

Features

  • Template-based PDF generation - Define PDF layouts using JSON templates
  • Multiple content types - Support for text, dynamic text, tables, images, QR codes, SVG graphics, and rectangles
  • Static schema support - Add headers, footers, and page elements that appear on all pages
  • Custom variable support - Static schemas support custom input variables for dynamic content
  • Special template variables - Built-in support for page numbering, dates, and timestamps
  • Font management - Easy font loading and management with support for Japanese fonts (Noto Sans JP, Noto Serif JP)
  • Dynamic content - Template rendering with variable substitution using Tera templating engine
  • Multi-page support - Generate PDFs with multiple pages from single templates
  • Advanced table rendering - Tables can span multiple pages automatically with proper pagination
  • Text styling with borders - Text elements support borders, backgrounds, and comprehensive styling options
  • Flexible styling - Comprehensive styling options for colors, borders, alignment, and spacing
  • Schema versioning - Templates now use schemaVersion field for better version management
  • High-quality output - Built on the robust printpdf library

Supported Schema Types

  • Text: Static text with font styling, borders, backgrounds, and precise positioning
  • Dynamic Text: Template-driven text with variable substitution
  • Table: Structured tables with headers, borders, cell styling, and multi-page spanning
  • QR Code: QR code generation with customizable size and positioning
  • Image: Image embedding with PNG, JPEG, and BMP support, plus CSS-like object-fit controls
  • SVG: Scalable vector graphics rendering
  • Rectangle: Geometric shapes with customizable styling
  • Group: Container for grouping and transforming multiple schemas together

Installation

Add PDForge to your Cargo.toml:

[dependencies]
pdforge = "0.9.1"

Quick Start

Basic Usage

use pdforge::PDForgeBuilder;

fn main() -> Result<(), Box<dyn std::error::Error>> {
    // Create a new PDF generator using the builder pattern
    let mut pdforge = PDForgeBuilder::new("My Document".to_string())
        .add_font("NotoSerifJP", "./assets/fonts/NotoSerifJP-Regular.ttf")?
        .add_font("NotoSansJP", "./assets/fonts/NotoSansJP-Regular.ttf")?
        .load_template("table-test", "./templates/table-test.json")?
        .build();

    // Generate PDF (empty inputs required for static templates)
    let bytes: Vec<u8> = pdforge.render("table-test", vec![vec![]], None, None)?;

    // Save to file
    std::fs::write("./output.pdf", bytes)?;
    Ok(())
}

With Command Line Arguments

use std::env;
use std::path::Path;

fn main() -> Result<(), Box<dyn std::error::Error>> {
    let args: Vec<String> = env::args().collect();
    
    if args.len() != 2 {
        eprintln!("Usage: {} <template_file>", args[0]);
        std::process::exit(1);
    }
    
    let template_file = &args[1];
    let template_path = Path::new(template_file);
    
    // Extract filename without extension for output
    let file_stem = template_path
        .file_stem()
        .and_then(|s| s.to_str())
        .unwrap_or("output");
    
    let mut pdforge = pdforge::PDForgeBuilder::new("CLI Example".to_string())
        .add_font("NotoSerifJP", "./assets/fonts/NotoSerifJP-Regular.ttf")?
        .add_font("NotoSansJP", "./assets/fonts/NotoSansJP-Regular.ttf")?
        .load_template("template", template_file)?
        .build();

    let bytes: Vec<u8> = pdforge.render("template", vec![vec![]], None, None)?;

    // Generate output filename based on input template
    let output_file = format!("{}.pdf", file_stem);
    std::fs::write(&output_file, bytes)?;
    
    println!("PDF generated: {}", output_file);
    Ok(())
}

With Dynamic Data

use std::collections::HashMap;
use pdforge::PDForgeBuilder;

fn main() -> Result<(), Box<dyn std::error::Error>> {
    let mut pdforge = PDForgeBuilder::new("Dynamic Document".to_string())
        .add_font("NotoSerifJP", "./assets/fonts/NotoSerifJP-Regular.ttf")?
        .add_font("NotoSansJP", "./assets/fonts/NotoSansJP-Regular.ttf")?
        .load_template("pawn-ticket", "./templates/pawn-ticket.json")?
        .build();

    // Prepare input data
    let mut input: HashMap<&'static str, String> = HashMap::new();
    input.insert("name", "田中太郎".to_string());
    input.insert("date", "2025-05-24".to_string());
    input.insert("amount", "100,000".to_string());

    let inputs = vec![vec![input]];
    let bytes: Vec<u8> = pdforge.render("pawn-ticket", inputs, None, None)?;

    std::fs::write("./dynamic_output.pdf", bytes)?;
    Ok(())
}

Template Structure

PDForge uses JSON templates to define PDF layouts. Here's the basic structure:

{
  "schemas": [
    [
      {
        "type": "text",
        "name": "title",
        "content": "Hello World",
        "position": { "x": 10, "y": 10 },
        "width": 100,
        "height": 20,
        "fontSize": 16,
        "fontName": "NotoSansJP"
      }
    ]
  ],
  "basePdf": {
    "width": 210,
    "height": 297,
    "padding": [20, 10, 20, 10],
    "staticSchema": [
      {
        "type": "text",
        "name": "header",
        "content": "Page {{currentPage}} of {{totalPages}}",
        "position": { "x": 10, "y": 5 },
        "width": 100,
        "height": 10,
        "fontSize": 10,
        "fontName": "NotoSansJP"
      }
    ]
  },
  "schemaVersion": "1.0.0"
}

Schema Types

Text Schema

{
  "type": "text",
  "name": "field1",
  "content": "Static text content",
  "position": { "x": 10, "y": 10 },
  "width": 100,
  "height": 20,
  "fontSize": 12,
  "fontName": "NotoSansJP",
  "fontColor": "#000000",
  "backgroundColor": "#f0f0f0",
  "borderColor": "#ff0000",
  "borderWidth": 1,
  "alignment": "left"
}

Text Border Support: Text elements now support borders with customizable color and width:

  • borderColor - Border color in CSS color format (hex, rgb, named colors, etc.)
  • borderWidth - Border thickness in points (pt)
  • backgroundColor - Background color for text elements

Both border and background features are optional and work together seamlessly.

Table Schema

{
  "type": "table",
  "name": "data_table",
  "position": { "x": 10, "y": 50 },
  "width": 190,
  "height": 100,
  "content": "[[\"Name\",\"City\",\"Description\"],[\"Alice\",\"New York\",\"Web designer\"]]",
  "showHead": true,
  "headWidthPercentages": [
    { "content": "Name", "percent": 30, "fontSize": 12, "alignment": "center" },
    { "content": "City", "percent": 20, "fontSize": 12, "alignment": "center" },
    {
      "content": "Description",
      "percent": 50,
      "fontSize": 12,
      "alignment": "left"
    }
  ],
  "tableStyles": {
    "borderWidth": 0.3,
    "borderColor": "#000000"
  },
  "headStyles": {
    "fontSize": 12,
    "fontName": "NotoSansJP",
    "fontColor": "#ffffff",
    "backgroundColor": "#2980ba"
  }
}

Multi-Page Tables: Tables automatically span multiple pages when content exceeds the available space. The library handles pagination, headers, and proper content flow across pages automatically.

Image Schema

{
  "type": "image",
  "name": "photo",
  "content": "...",
  "position": { "x": 10, "y": 50 },
  "width": 80,
  "height": 60,
  "objectFit": "contain"
}

Object-Fit Control: The objectFit property controls how images are scaled and positioned within their containers, following CSS object-fit specifications:

  • fill (default): Stretches the image to fill the container exactly, may distort aspect ratio
  • contain: Scales image to fit within container while preserving aspect ratio, may leave empty space
  • cover: Scales image to cover entire container while preserving aspect ratio, may crop parts of the image
  • none: Displays image at natural size (1:1 pixel mapping), may overflow or leave empty space
  • scale-down: Behaves like none if image fits naturally, otherwise behaves like contain

Examples:

{
  "type": "image",
  "name": "logo",
  "content": "data:image/png;base64,...",
  "position": { "x": 10, "y": 10 },
  "width": 100,
  "height": 80,
  "objectFit": "contain"
}
{
  "type": "image", 
  "name": "background",
  "content": "data:image/jpeg;base64,...",
  "position": { "x": 0, "y": 0 },
  "width": 210,
  "height": 150,
  "objectFit": "cover"
}

The object-fit feature provides precise control over image display, ensuring your images look exactly as intended regardless of their original dimensions or aspect ratios.

QR Code Schema

{
  "type": "qrCode",
  "name": "qr1",
  "content": "https://example.com",
  "position": { "x": 150, "y": 50 },
  "width": 30,
  "height": 30
}

SVG Schema

{
  "type": "svg",
  "name": "logo",
  "content": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"100\" height=\"100\"><circle cx=\"50\" cy=\"50\" r=\"40\" fill=\"blue\"/></svg>",
  "position": { "x": 10, "y": 100 },
  "width": 50,
  "height": 50
}

Group Schema

{
  "type": "group",
  "name": "header_group",
  "position": { "x": 10, "y": 10 },
  "width": 190,
  "height": 50,
  "rotate": 15,
  "schemas": [
    {
      "type": "text",
      "name": "title",
      "content": "Grouped Title",
      "position": { "x": 0, "y": 0 },
      "width": 100,
      "height": 20,
      "fontSize": 16,
      "fontName": "NotoSansJP"
    },
    {
      "type": "qrCode",
      "name": "group_qr",
      "content": "https://example.com",
      "position": { "x": 120, "y": 0 },
      "width": 30,
      "height": 30
    }
  ]
}

The Group schema allows you to:

  • Group multiple elements: Combine different schema types into a single container
  • Apply transformations: Rotate the entire group around its center point
  • Coordinate positioning: Position child elements relative to the group's origin
  • Maintain hierarchy: Organize complex layouts with nested structures

Static Schema Support

PDForge supports static schemas that appear on every page of your PDF. This is perfect for headers, footers, page numbers, and other recurring elements.

Custom Variables in Static Schema

Static schemas now support custom input variables, allowing you to pass dynamic data that appears on every page. This is useful for document titles, company names, or other metadata that should be consistent across all pages.

Basic Static Schema

{
  "basePdf": {
    "width": 210,
    "height": 297,
    "padding": [20, 10, 20, 10],
    "staticSchema": [
      {
        "type": "text",
        "name": "header",
        "content": "Document Title",
        "position": { "x": 20, "y": 10 },
        "width": 170,
        "height": 15,
        "fontSize": 14,
        "fontName": "NotoSansJP",
        "alignment": "center"
      },
      {
        "type": "text",
        "name": "footer",
        "content": "Generated on {{date}}",
        "position": { "x": 20, "y": 280 },
        "width": 170,
        "height": 10,
        "fontSize": 9,
        "fontName": "NotoSansJP",
        "alignment": "right"
      }
    ]
  }
}

Special Template Variables

Static schemas support special template variables that are automatically populated:

  • {{currentPage}} - Current page number (1-based)
  • {{totalPages}} - Total number of pages in the document
  • {{date}} - Current date in YYYY-MM-DD format
  • {{dateTime}} - Current date and time in YYYY-MM-DD HH:MM:SS format

Custom Variable Support

In addition to special variables, static schemas can also use custom variables from your input data. These variables are passed through the render_with_inputs method and can be referenced in static schema content:

use std::collections::HashMap;

let mut input: HashMap<&'static str, String> = HashMap::new();
input.insert("company_name", "Acme Corporation".to_string());
input.insert("document_type", "Invoice".to_string());

// Custom variables are available in static schemas
let bytes = pdforge.render("template", vec![vec![input]], None, None)?;

Then reference these in your static schema:

{
  "staticSchema": [
    {
      "type": "text", 
      "name": "company_header",
      "content": "{{company_name}} - {{document_type}}",
      "position": { "x": 20, "y": 10 },
      "width": 170,
      "height": 15,
      "fontSize": 16,
      "fontName": "NotoSansJP"
    }
  ]
}

Complete Static Schema Example

{
  "schemas": [
    [
      {
        "type": "table",
        "name": "data_table",
        "position": { "x": 20, "y": 30 },
        "width": 170,
        "height": 200,
        "showHead": true,
        "headWidthPercentages": [
          { "content": "Name", "percent": 40 },
          { "content": "Value", "percent": 60 }
        ],
        "fields": []
      }
    ]
  ],
  "basePdf": {
    "width": 210,
    "height": 297,
    "padding": [20, 20, 20, 20],
    "staticSchema": [
      {
        "type": "text",
        "name": "title",
        "content": "Monthly Report",
        "position": { "x": 20, "y": 10 },
        "width": 170,
        "height": 15,
        "fontSize": 16,
        "fontName": "NotoSansJP",
        "alignment": "center"
      },
      {
        "type": "text",
        "name": "page_number",
        "content": "Page {{currentPage}} of {{totalPages}}",
        "position": { "x": 150, "y": 280 },
        "width": 40,
        "height": 10,
        "fontSize": 9,
        "fontName": "NotoSansJP",
        "alignment": "right"
      },
      {
        "type": "text",
        "name": "generated_time",
        "content": "Generated: {{dateTime}}",
        "position": { "x": 20, "y": 280 },
        "width": 120,
        "height": 10,
        "fontSize": 8,
        "fontName": "NotoSansJP",
        "alignment": "left"
      }
    ]
  },
  "schemaVersion": "1.0.0"
}

This creates a document where:

  • Every page has a "Monthly Report" title at the top
  • Page numbers appear in the bottom right ("Page 1 of 3", "Page 2 of 3", etc.)
  • Generation timestamp appears in the bottom left
  • The main content (table) can span multiple pages, with static elements on each page

Examples

The project includes several examples and templates:

Running Examples

Use the simple example to generate PDFs from any template:

# Generate PDF from template
cargo run --example simple templates/table.json

# Multi-page table examples
cargo run --example simple templates/large-tables-spanning.json
cargo run --example simple templates/multi-table-fixed.json

# Static schema examples
cargo run --example simple templates/static-schema-test.json
cargo run --example static_schema_custom_test  # Custom variables in static schema
cargo run --example table_with_static_test     # Table with static elements
cargo run --example print-renews  # Multi-page table with static headers/footers

# Image examples
cargo run --example image assets/images/test-image-1.jpg assets/images/test-image-2.jpg
cargo run --example object_fit_test

# Other examples  
cargo run --example simple templates/multipage.json
cargo run --example simple templates/tiger-svg.json

Using the CLI Binary

You can also use the main binary to generate PDFs from templates:

# Generate PDF using main binary
cargo run --bin pdforge templates/static-schema-test.json

# This will create: examples/pdf/static-schema-test.pdf

Available Templates

  • table.json - Basic table example
  • large-tables-spanning.json - Comprehensive 4-page document with two large tables spanning multiple pages
    • Financial report table (76 rows) across pages 1-2
    • Inventory management table (57 rows) across pages 3-4
  • multi-table-fixed.json - 3-page document with multiple separate tables
  • static-schema-test.json - Demonstrates static schema with page numbering and timestamps
  • static-schema-custom-test.json - Demonstrates static schema with custom variables
  • table-with-static-test.json - Shows table content with static headers and custom variables
  • table-with-static-simple.json - Simplified example of tables with static elements
  • print-renews.json - Multi-page table with comprehensive static headers and footers
  • multipage.json - Multi-page document example
  • tiger-svg.json - SVG graphics example
  • svg.json - Basic SVG example
  • image-test.json - Image embedding with object-fit controls
  • object-fit-test.json - Demonstrates all five object-fit modes side by side
  • text-border-test.json - Showcases text border and background styling features

Generating PDFs

All PDF files are generated locally and are not tracked in the repository. To create a PDF:

cargo run --example simple <template-file>

This will create a PDF file in the examples/pdf/ directory with the same name as your template.

Font Support

PDForge includes comprehensive font support with a focus on Japanese typography:

Included Fonts

  • Noto Sans JP: Modern sans-serif Japanese font family
  • Noto Serif JP: Traditional serif Japanese font family
  • Hannari Mincho: Elegant Japanese mincho font
  • IPA Gothic: Standard Japanese gothic font

Adding Custom Fonts

let pdforge = PDForgeBuilder::new("Document".to_string())
    .add_font("CustomFont", "./path/to/custom-font.ttf")?
    .build();

Template Engine

PDForge uses the Tera templating engine for dynamic content rendering. This allows you to:

  • Insert variables: {{ variable_name }}
  • Use conditionals: {% if condition %}
  • Apply filters: {{ name | upper }}
  • Loop through data: {% for item in items %}

Built-in Template Variables

For static schemas, PDForge provides these special variables:

  • {{currentPage}} - Current page number (starts from 1)
  • {{totalPages}} - Total number of pages in the document
  • {{date}} - Current date (YYYY-MM-DD format)
  • {{dateTime}} - Current date and time (YYYY-MM-DD HH:MM:SS format)

These variables are automatically populated during PDF generation and are especially useful for headers, footers, and page numbering.

Dependencies

PDForge is built on top of several excellent Rust crates:

  • printpdf: Core PDF generation functionality
  • tera: Template engine for dynamic content
  • serde: JSON serialization/deserialization
  • time: Date and time handling for template variables
  • qrcode: QR code generation
  • image: Image processing and embedding
  • uuid: Unique identifier generation

Development

Building

cargo build

Running Tests

cargo test

Running Examples

# Generate PDFs from templates
cargo run --example simple templates/table.json
cargo run --example simple templates/large-tables-spanning.json
cargo run --example simple templates/multi-table-fixed.json

# Legacy examples (may require specific setup)
cargo run --example table
cargo run --example svg
cargo run --example pdfium

Project Structure

pdforge/
├── src/
│   ├── lib.rs              # Main library interface
│   ├── main.rs             # CLI binary for template processing
│   ├── font.rs             # Font management
│   ├── common.rs           # Common utilities
│   ├── utils.rs            # Helper functions
│   └── schemas/            # Schema implementations
│       ├── mod.rs          # Schema definitions with static schema support
│       ├── text.rs         # Text schema
│       ├── table.rs        # Table schema with multi-page support
│       ├── qrcode.rs       # QR code schema
│       ├── image.rs        # Image schema
│       ├── svg.rs          # SVG schema
│       └── rect.rs         # Rectangle schema
├── examples/               # Usage examples and CLI tools
│   ├── simple.rs          # Main CLI for template processing
│   ├── print-renews.rs    # Multi-page table with static schema example
│   └── ...                # Other specialized examples
├── templates/              # JSON template files
│   ├── large-tables-spanning.json  # Comprehensive multi-page tables
│   ├── multi-table-fixed.json      # Multiple tables example
│   ├── static-schema-test.json     # Static schema demonstration
│   ├── print-renews.json           # Multi-page with headers/footers
│   └── ...                # Other template examples
├── examples/pdf/          # Generated PDF output (gitignored)
├── assets/fonts/          # Font files (gitignored)
└── pdfium/                # PDFium integration

Contributing

Contributions are welcome! Please feel free to submit issues, feature requests, or pull requests.

License

This project is licensed under the MIT License - see the LICENSE file for details.

Recent Changes

Version 0.9.1

  • Breaking Change: HashMap keys changed to &'static str for improved performance and memory efficiency
  • API Update: All HashMap parameters now use &'static str keys instead of String keys:
    • inputs: Vec<Vec<HashMap<&'static str, String>>>
    • table_data: HashMap<&'static str, Vec<Vec<String>>>
    • static_inputs: HashMap<&'static str, String>
  • Performance Improvement: Reduced memory allocation and improved lookup performance
  • Examples Updated: All example files updated to use &'static str keys
  • Type Safety: Compile-time string literal validation for HashMap keys

Version 0.9.0

  • Breaking Change: Simplified render API - consolidated 7 different render methods into a single unified render() method
  • New API: render(template_name, inputs, table_data, static_inputs) where inputs is required, others are optional
  • Input Validation: Empty inputs vector now returns an error for better error handling
  • Code Cleanup: Removed redundant render methods from both lib.rs and schemas/mod.rs
  • Examples Updated: All example files updated to use the new simplified API

Version 0.8.2

  • Code Cleanup: Removed unused set_pdf_metadata method from lib.rs for cleaner API
  • Maintenance: Code simplification and cleanup

Version 0.8.1

  • Custom Variable Support: Static schemas now support custom input variables for dynamic content
  • Code Organization: Moved DrawRoundedRectangle utility from table.rs to rect.rs for better modularity
  • Schema Cleanup: Removed unused table content field to simplify schema structure
  • New Examples: Added examples demonstrating custom variable functionality with static schemas

Version 0.8.0

  • Breaking Change: Template schema version field renamed from pdfmeVersion to schemaVersion
  • All templates now use schemaVersion: "1.0.0" for consistency and proper version management
  • Updated to version 0.8.0 to reflect this schema format change

Acknowledgments

  • Inspired by pdfme - A powerful PDF generation library
  • Built with printpdf - A pure Rust PDF generation library
  • Font rendering powered by industry-standard font libraries

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Packages

No packages published

Contributors 2

  •  
  •  

Languages