npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

@docmentis/udoc-viewer

v0.6.34

Published

Universal document viewer for the web — open-source, framework-agnostic viewer powered by a built-from-scratch WebAssembly engine for high-fidelity rendering across PDF, DOCX, PPTX, SVG, and images.

Downloads

19,082

Readme

@docmentis/udoc-viewer

Universal document viewer for the web.

Open-source, framework-agnostic viewer powered by a built-from-scratch WebAssembly engine for high-fidelity rendering across PDF, DOCX, PPTX, SVG, and images.

npm version license

Live Demo · Guide · Changelog · Report Issue


Why udoc-viewer?

Most web document viewers only handle PDF, rely on server-side rendering, or require expensive commercial licenses. udoc-viewer is different:

  • Truly universal — PDF, Word, PowerPoint, SVG, and images in a single viewer, with more formats coming
  • High fidelity — powered by a custom Rust/WebAssembly rendering engine, not PDF.js
  • Client-side only — everything runs in the browser, no server round-trips
  • Framework agnostic — works with React, Vue, Angular, Svelte, or plain HTML
  • Free for commercial use — MIT-licensed wrapper, free WASM engine

Supported Formats

| Format | Extensions | | ------ | ------------------------------------------------------------------------------------------------- | | PDF | .pdf | | DOCX | .docx | | PPTX | .pptx | | Vector | .svg, .wmf, .emf | | Images | .png, .jpg, .jpeg, .gif, .webp, .bmp, .tif, .tiff, .ico, .tga, .ppm, .pgm, .pbm, .hdr, .exr, .qoi |

Quick Start

Install

npm install @docmentis/udoc-viewer

Basic Usage

import { UDocClient } from "@docmentis/udoc-viewer";

// Create a client (loads the WASM engine)
const client = await UDocClient.create();

// Create a viewer attached to a container element
const viewer = await client.createViewer({
    container: "#viewer",
});

// Load a document
await viewer.load("https://example.com/document.pdf");

// Clean up when done
viewer.destroy();
client.destroy();

HTML

<div id="viewer" style="width: 100%; height: 600px;"></div>

<script type="module">
    import { UDocClient } from "@docmentis/udoc-viewer";

    const client = await UDocClient.create();
    const viewer = await client.createViewer({ container: "#viewer" });
    await viewer.load("/path/to/document.pdf");
</script>

React

import { useEffect, useRef } from "react";
import { UDocClient } from "@docmentis/udoc-viewer";

function DocumentViewer({ src }) {
    const containerRef = useRef(null);

    useEffect(() => {
        let client, viewer;

        (async () => {
            client = await UDocClient.create();
            viewer = await client.createViewer({
                container: containerRef.current,
            });
            await viewer.load(src);
        })();

        return () => {
            viewer?.destroy();
            client?.destroy();
        };
    }, [src]);

    return <div ref={containerRef} style={{ width: "100%", height: "600px" }} />;
}

Examples

Full working examples for every major framework are in the examples/ directory:

| Example | Stack | | ---------------------------------------------------- | ------------------- | | vanilla | TypeScript + Vite | | react-vite | React + Vite | | vue-vite | Vue + Vite | | svelte-vite | Svelte 5 + Vite | | angular | Angular 19 | | nextjs-webpack | Next.js + Webpack | | nextjs-turbopack | Next.js + Turbopack | | nuxt | Nuxt 3 |

Features

  • 📄 Multi-format rendering — PDF, DOCX, PPTX, and images in one unified viewer
  • 🎯 High-fidelity output — custom Rust rendering engine compiled to WebAssembly
  • 🔍 Zoom & navigation — toolbar with zoom controls, page thumbnails, and keyboard navigation
  • 🔎 Full-text search — search with match highlighting, navigation, and fuzzy matching for AI citations
  • 🌓 Dark mode — built-in light/dark theme with system preference support
  • 🎨 Customizable — override colors and styles via CSS variables
  • 📱 Responsive — works on desktop and mobile browsers
  • 🌊 Streaming — pages render progressively as the document loads
  • 🔒 Private — documents never leave the browser; no server upload required

API Reference

Client Options

const client = await UDocClient.create({
    // License key for commercial use (optional)
    // Enables licensed features such as hiding the attribution link
    license: "eyJ2Ijox...",

    // Custom base URL for worker and WASM files (optional)
    // Expected files: {baseUrl}/worker.js and {baseUrl}/udoc_bg.wasm
    baseUrl: "https://cdn.example.com/udoc/",

    // Disable anonymous telemetry reporting (default: false)
    // Requires a valid license with the "no_telemetry" feature
    disableTelemetry: false,

    // Disable checking npm registry for a newer version on startup (default: false)
    // When enabled (default), a background check logs a console reminder if a newer version is available
    // Never blocks initialization
    disableUpdateCheck: false,

    // Enable Google Fonts for automatic font fetching (default: true)
    // When enabled, missing fonts are fetched from Google Fonts on-demand during rendering
    googleFonts: true,

    // Register custom font URLs for on-demand fetching during layout (optional)
    // Supports OTF, TTF, WOFF, and WOFF2 formats
    // Registered fonts take priority over Google Fonts
    fonts: [
        { typeface: "Roboto", bold: false, italic: false, url: "https://cdn.example.com/Roboto-Regular.woff2" },
        { typeface: "Roboto", bold: true, italic: false, url: "https://cdn.example.com/Roboto-Bold.woff2" },
        { typeface: "Roboto", bold: false, italic: true, url: "https://cdn.example.com/Roboto-Italic.woff2" },
        { typeface: "Roboto", bold: true, italic: true, url: "https://cdn.example.com/Roboto-BoldItalic.woff2" },
    ],
});

Viewer Options

const viewer = await client.createViewer({
    // Container element or CSS selector (required for UI mode, omit for headless)
    container: "#viewer",

    // --- View modes ---

    // Scroll mode: 'continuous' or 'spread' (default: 'continuous')
    scrollMode: "continuous",

    // Layout mode: 'single-page', 'double-page', 'double-page-odd-right', 'double-page-odd-left'
    // (default: 'single-page')
    layoutMode: "single-page",

    // Initial page rotation: 0, 90, 180, or 270 (default: 0)
    pageRotation: 0,

    // Spacing mode: 'all', 'none', 'spread-only', 'page-only' (default: 'all')
    spacingMode: "all",

    // --- Zoom ---

    // Zoom mode: 'fit-spread-width', 'fit-spread-width-max', 'fit-spread-height', 'fit-spread', 'custom'
    // (default: 'fit-spread-width')
    zoomMode: "fit-spread-width",

    // Initial zoom level (when zoomMode is 'custom', default: 1)
    zoom: 1,

    // Custom zoom steps for zoom in/out
    // (default: [0.1, 0.25, 0.5, 0.75, 1, 1.25, 1.5, 2, 3, 4, 5])
    zoomSteps: [0.1, 0.25, 0.5, 0.75, 1, 1.25, 1.5, 2, 3, 4, 5],

    // Zoom range limits (default: 0.1 and 5)
    minZoom: 0.1,
    maxZoom: 5,

    // --- Scroll alignment ---

    // Default scroll alignment for navigation (outline, links, goToDestination)
    // 'top' | 'center' | 'bottom' | 'nearest' (default: 'top')
    navigationScrollAlignment: "top",

    // Default scroll alignment for search result navigation
    // 'top' | 'center' | 'bottom' | 'nearest' (default: 'center')
    searchScrollAlignment: "center",

    // --- Spacing & layout ---

    // Spacing between pages in pixels (default: 10)
    pageSpacing: 10,

    // Spacing between spreads in pixels (default: 20)
    spreadSpacing: 20,

    // Width of thumbnail images in pixels (default: 150)
    // Height is derived automatically from the page aspect ratio
    thumbnailWidth: 150,

    // Target display DPI (default: 96)
    dpi: 96,

    // --- Theme ---

    // Color theme: 'light', 'dark', or 'system' (default: 'light')
    theme: "light",

    // Hide the theme toggle button (default: false)
    disableThemeSwitching: false,

    // --- Features ---

    // Disable text selection and copying (default: false)
    disableTextSelection: false,

    // --- Panels ---

    // Initially active panel, or null for no panel (default: null)
    // Left panels: 'thumbnail', 'outline', 'bookmarks', 'layers', 'attachments'
    // Right panels: 'search', 'comments'
    activePanel: null,

    // Disable individual panel tabs (all default to false)
    disableLeftPanel: false, // Disable the entire left panel area
    disableRightPanel: false, // Disable the entire right panel area
    disableThumbnails: false, // Disable the thumbnails tab
    disableOutline: false, // Disable the outline tab
    disableBookmarks: false, // Disable the bookmarks tab
    disableLayers: false, // Disable the layers tab
    disableAttachments: false, // Disable the attachments tab
    disableSearch: false, // Disable the search panel
    disableComments: false, // Disable the comments panel

    // --- UI visibility ---

    // Hide the top toolbar (default: false)
    hideToolbar: false,

    // Hide the floating toolbar (page nav, zoom, view mode) (default: false)
    hideFloatingToolbar: false,

    // Remove the fullscreen button (default: false)
    disableFullscreen: false,

    // --- Advanced ---

    // Enable performance tracking (default: false)
    enablePerformanceCounter: false,

    // Callback for performance log entries (called when enablePerformanceCounter is true)
    onPerformanceLog: (entry) => console.log(entry),

    // Hide the "Powered by docMentis" attribution link (default: false)
    // Requires a valid license with the "no_attribution" feature
    hideAttribution: true,

    // Hide the loading overlay shown during document download and processing (default: false)
    // Requires a valid license with the "no_attribution" feature
    hideLoadingOverlay: true,
});

Loading Documents

The viewer accepts multiple document sources:

// From URL
await viewer.load("https://example.com/document.pdf");

// From File object (e.g., from file input)
await viewer.load(file);

// From raw bytes
await viewer.load(new Uint8Array(buffer));

// Close current document
viewer.close();

Navigation

// Get current page (1-based)
const page = viewer.currentPage;

// Go to a specific page (1-based)
viewer.goToPage(5);

// Navigate to next/previous page
viewer.nextPage();
viewer.previousPage();

// Navigate to a destination (from outline)
viewer.goToDestination(destination);

// Navigate with scroll alignment override
viewer.goToDestination(destination, { scrollAlignment: "center" });

// Change default scroll alignment at runtime
viewer.setNavigationScrollAlignment("center"); // 'top' | 'center' | 'bottom' | 'nearest'
viewer.setSearchScrollAlignment("top");

Document Information

// Check if document is loaded
if (viewer.isLoaded) {
    // Get page count
    const total = viewer.pageCount;

    // Get document metadata
    const meta = viewer.metadata;
    console.log(meta?.title, meta?.author);

    // Get page dimensions (0-based index)
    const info = await viewer.getPageInfo(0);
    console.log(`Page 1: ${info.width} x ${info.height} points`);

    // Get document outline (table of contents)
    const outline = await viewer.getOutline();

    // Get annotations on a page (0-based index)
    const annotations = await viewer.getPageAnnotations(0);

    // Get plain text of a page (matches what search indexes)
    const text = await viewer.getPageText(0);
}

Extracting Page Text

getPageText(page) returns the flat text of a page exactly as the search engine sees it. Glyph runs are concatenated in visual order; spaces and tabs become " ", line breaks and paragraph ends become "\n", and inline drawings become "\uFFFC" (object replacement character). Works uniformly across PDF, DOCX, PPTX, and XLSX.

// Extract text from a single page (0-based index)
const pageText = await viewer.getPageText(0);

// Extract the full document
const pages: string[] = [];
for (let i = 0; i < viewer.pageCount; i++) {
    pages.push(await viewer.getPageText(i));
}
const fullText = pages.join("\n\n");

Programmatic Viewer Control

Control zoom, view modes, and fullscreen programmatically — useful when toolbars are hidden:

// Zoom
viewer.zoomIn();
viewer.zoomOut();
viewer.setZoom(1.5); // 150%
viewer.setZoomMode("fit-spread-width");
viewer.setMinZoom(0.5); // clamp minimum to 50%
viewer.setMaxZoom(3); // clamp maximum to 300%
console.log(viewer.zoom); // current zoom level
console.log(viewer.zoomMode); // current zoom mode

// View modes
viewer.setScrollMode("continuous"); // 'continuous' | 'spread'
viewer.setLayoutMode("double-page"); // 'single-page' | 'double-page' | ...
viewer.setPageRotation(90); // 0 | 90 | 180 | 270
viewer.setSpacingMode("none"); // 'all' | 'none' | 'spread-only' | 'page-only'

// Theme
viewer.setTheme("dark"); // 'light' | 'dark' | 'system'
viewer.setThemeSwitchingEnabled(false); // hide theme toggle button
console.log(viewer.theme); // current theme

// Text selection
viewer.setTextSelectionEnabled(false); // disable text selection

// Fullscreen
viewer.setFullscreen(true);
console.log(viewer.isFullscreen);

UI Visibility Control

Show, hide, or disable UI components at runtime:

// Toolbar visibility
viewer.setToolbarVisible(false);
viewer.setFloatingToolbarVisible(false);

// Fullscreen button
viewer.setFullscreenEnabled(false);

// Disable entire panel areas
viewer.setLeftPanelEnabled(false);
viewer.setRightPanelEnabled(false);

// Disable individual panel tabs
// Panels: 'thumbnail', 'outline', 'bookmarks', 'layers', 'attachments', 'search', 'comments'
viewer.setPanelEnabled("thumbnail", false);
viewer.setPanelEnabled("search", false);

// Open/close panels programmatically
viewer.openPanel("outline");
viewer.closePanel();

Programmatic Search

Search document text programmatically. Works with or without the built-in search panel — ideal for building custom search UIs. Highlight overlays are rendered automatically on matching pages.

// Search and await results (returns Promise<SearchMatch[]>)
const matches = await viewer.search("hello world");
console.log(`Found ${matches.length} matches`);

// Search with options
const matches = await viewer.search("hello", { caseSensitive: true });

// Fuzzy matching — strips all whitespace, control characters, pipes (|),
// and zero-width characters from both the query and document text before
// comparison. Designed for AI-generated citations where spacing or
// separators may differ from the original document text.
const matches = await viewer.search("cell A | cell B", { fuzzy: true });

// Restrict search to a page range (inclusive, 0-based).
// Both text loading and match collection are limited to the range —
// useful for large documents when only a subset is of interest.
const matches = await viewer.search("invoice", { pageRange: [10, 20] });

// Each search() call is self-contained: the range is reset unless
// explicitly provided, so a prior scoped search can't leak into a
// later broad one.
await viewer.search("invoice", { pageRange: [10, 20] }); // pages 10-20
await viewer.search("payment"); // whole doc

// Note: the built-in search panel always searches the entire document.
// pageRange is only honored when calling the API directly.

// Each match contains location, highlight rects, and context snippet
for (const match of matches) {
    console.log(`Page ${match.pageIndex + 1}: "${match.context[1]}"`);
    console.log(`  Before: "${match.context[0]}", After: "${match.context[2]}"`);
    console.log(`  Rects:`, match.rects); // bounding boxes in PDF points
}

// Navigate between matches (scrolls viewport to match)
viewer.searchNext();
viewer.searchPrev();

// Navigate with a per-call scroll alignment override
viewer.searchNext({ scrollAlignment: "center" });
viewer.searchPrev({ scrollAlignment: "top" });

// Jump to a specific match by index
viewer.setSearchActiveIndex(5);
viewer.setSearchActiveIndex(5, { scrollAlignment: "center" });

// Read current state
console.log(viewer.searchMatches); // SearchMatch[]
console.log(viewer.searchActiveIndex); // number (-1 if none)

// Listen for incremental updates (fires as pages load and on navigation)
viewer.on("search:change", ({ matches, activeIndex }) => {
    updateMySearchUI(matches, activeIndex);
});

// Clear search
viewer.clearSearch();

Custom Search UI Example

Disable the built-in search panel and build your own:

const viewer = await client.createViewer({
    container: "#viewer",
    disableSearch: true, // hide built-in search panel
});

await viewer.load("document.pdf");

// Wire up your own search input
searchInput.addEventListener("input", async () => {
    const matches = await viewer.search(searchInput.value);
    renderResultsList(matches); // your custom UI
});

// Wire up next/prev buttons
nextBtn.addEventListener("click", () => viewer.searchNext());
prevBtn.addEventListener("click", () => viewer.searchPrev());

CSS Customization

The viewer uses CSS custom properties (variables) for all colors, shadows, and borders. Since no Shadow DOM is used, you can override any variable from your own stylesheet:

/* Override the primary color */
.udoc-viewer-root {
    --udoc-primary: #e91e63;
    --udoc-primary-hover: #c2185b;
}

/* Override dark theme colors */
.udoc-viewer-root.udoc-viewer-dark {
    --udoc-primary: #f48fb1;
    --udoc-primary-hover: #f06292;
}

Available CSS Variables

| Variable | Description | Light Default | Dark Default | | -------------------------------- | ---------------------- | ----------------------------- | ---------------------------- | | Backgrounds | | | | | --udoc-bg-viewport | Viewport background | #e0e0e0 | #1a1a1a | | --udoc-bg-surface | Page / card surface | #fff | #2d2d2d | | --udoc-bg-panel | Side panel background | #f5f5f5 | #252525 | | --udoc-bg-panel-tabs | Panel tab bar | #e8e8e8 | #1e1e1e | | --udoc-bg-input | Input fields | #fff | #3a3a3a | | --udoc-bg-overlay | Modal overlay | rgba(0,0,0,0.5) | rgba(0,0,0,0.7) | | Text | | | | | --udoc-text-primary | Primary text | rgba(0,0,0,0.8) | rgba(255,255,255,0.87) | | --udoc-text-secondary | Secondary text | rgba(0,0,0,0.7) | rgba(255,255,255,0.7) | | --udoc-text-muted | Muted text | rgba(0,0,0,0.5) | rgba(255,255,255,0.5) | | --udoc-text-disabled | Disabled text | rgba(0,0,0,0.25) | rgba(255,255,255,0.25) | | --udoc-text-placeholder | Placeholder text | #999 | #777 | | --udoc-text-on-primary | Text on primary color | #fff | #fff | | Primary color | | | | | --udoc-primary | Primary / accent color | #0066cc | #4da6ff | | --udoc-primary-hover | Primary hover state | #0052a3 | #80bfff | | --udoc-primary-focus-ring | Focus ring color | rgba(0,102,204,0.2) | rgba(77,166,255,0.25) | | Borders | | | | | --udoc-border | Default border | #ddd | #444 | | --udoc-border-input | Input border | #ccc | #555 | | --udoc-border-light | Light border | #eee | #3a3a3a | | Shadows | | | | | --udoc-shadow-page | Page shadow | 0 2px 8px rgba(0,0,0,0.15) | 0 2px 8px rgba(0,0,0,0.4) | | --udoc-shadow-toolbar | Toolbar shadow | 0 2px 12px rgba(0,0,0,0.15) | 0 2px 12px rgba(0,0,0,0.4) | | --udoc-shadow-dropdown | Dropdown shadow | 0 4px 16px rgba(0,0,0,0.2) | 0 4px 16px rgba(0,0,0,0.5) | | Search | | | | | --udoc-search-highlight | Search match highlight | rgba(255,200,0,0.35) | rgba(255,200,0,0.4) | | --udoc-search-highlight-active | Active match highlight | rgba(255,140,0,0.6) | rgba(255,140,0,0.65) | | Selection | | | | | --udoc-text-selection | Text selection color | rgba(0,120,215,0.3) | rgba(77,166,255,0.35) | | Scrollbar | | | | | --udoc-scrollbar-thumb | Scrollbar thumb | rgba(0,0,0,0.3) | rgba(255,255,255,0.3) | | --udoc-scrollbar-thumb-hover | Scrollbar thumb hover | rgba(0,0,0,0.5) | rgba(255,255,255,0.5) | | Errors | | | | | --udoc-error-bg | Error background | #fef2f2 | #3a1c1c | | --udoc-error-border | Error border | #fecaca | #6b2c2c | | --udoc-error-text | Error text | #dc2626 | #f87171 | | Progress | | | | | --udoc-progress-track | Progress bar track | #e5e7eb | #404040 | | --udoc-progress-fill | Progress bar fill | #0066cc | #4da6ff |

The full list of variables is defined in src/ui/viewer/styles.css. All viewer styles are scoped under .udoc-viewer-root, so your overrides won't leak into the rest of the page.

Events

// Document loaded
const unsubscribe = viewer.on("document:load", ({ pageCount }) => {
    console.log(`Loaded ${pageCount} pages`);
});

// Document closed
viewer.on("document:close", () => {
    console.log("Document closed");
});

// Page changed
viewer.on("page:change", ({ page, previousPage }) => {
    console.log(`Page ${previousPage} -> ${page}`);
});

// Panel opened/closed
viewer.on("panel:change", ({ panel, previousPanel }) => {
    console.log(`Panel: ${previousPanel} -> ${panel}`);
});

// UI component visibility changed
viewer.on("ui:visibilityChange", ({ component, visible }) => {
    console.log(`${component} is now ${visible ? "visible" : "hidden"}`);
});

// Download progress
viewer.on("download:progress", ({ loaded, total, percent }) => {
    console.log(`Downloaded ${loaded}/${total} bytes (${percent}%)`);
});

// Search results changed (matches found or active match navigated)
viewer.on("search:change", ({ matches, activeIndex }) => {
    console.log(`${matches.length} matches, active: ${activeIndex}`);
});

// Error occurred
viewer.on("error", ({ error, phase }) => {
    console.error(`Error during ${phase}:`, error);
});

// Unsubscribe
unsubscribe();

Document Export

// Export document as raw bytes
const bytes = await viewer.toBytes();

// Download document as a file
await viewer.download("document.pdf");

Document Composition

Compose new documents by cherry-picking and rotating pages:

// Create a new document from pages of existing documents
const [newDoc] = await client.compose([
    [
        { doc: viewerA, pages: "1-3" },
        { doc: viewerB, pages: "5", rotation: 90 },
    ],
]);

// Export the composed document
const bytes = await newDoc.toBytes();
await newDoc.download("composed.pdf");

Document Utilities

// Split a document by its outline (table of contents)
const { viewers, sections } = await client.splitByOutline(source, {
    maxLevel: 2,
    splitMidPage: false,
});

// Extract images from a document
const images = await client.extractImages(source, {
    convertRawToPng: true,
});

// Extract fonts from a document
const fonts = await client.extractFonts(source);

// Compress a document
const compressed = await client.compress(source);

// Decompress a document
const decompressed = await client.decompress(source);

Font Management

udoc-viewer automatically handles fonts for document rendering. By default, missing fonts are fetched from Google Fonts on-demand. You can also register custom fonts for full control.

Font resolution order: Custom registered fonts are resolved first, then Google Fonts (if enabled). Supported font formats: OTF, TTF, WOFF, and WOFF2.

// Option 1: Register fonts declaratively at client creation
const client = await UDocClient.create({
    googleFonts: true, // default, enable Google Fonts fallback
    fonts: [
        { typeface: "CustomFont", bold: false, italic: false, url: "https://cdn.example.com/CustomFont-Regular.woff2" },
        { typeface: "CustomFont", bold: true, italic: false, url: "https://cdn.example.com/CustomFont-Bold.woff2" },
    ],
});

// Option 2: Register fonts programmatically (before loading documents)
await client.registerFonts([
    { typeface: "CustomFont", bold: false, italic: false, url: "https://cdn.example.com/CustomFont-Regular.woff2" },
]);

// Option 3: Disable Google Fonts entirely (only use registered fonts)
const client = await UDocClient.create({ googleFonts: false });

Parsing Font Info

When users upload custom font files, you typically don't know the typeface name or whether a file is the bold/italic variant. parseFontInfo extracts that metadata from the raw font binary, so you can store it in your own database alongside the font file. Later, when initializing the viewer, you retrieve that metadata and pass it to registerFonts.

Step 1: At upload time — parse the font and store the metadata + file in your database/storage.

// User uploads a font file
const fontBytes = new Uint8Array(await fontFile.arrayBuffer());
const info = await client.parseFontInfo(fontBytes);
// info = { typeface: "Roboto", bold: true, italic: false }

// Store font file to your storage (e.g., S3, GCS) and save metadata to your database
await storage.upload(`fonts/${fontFile.name}`, fontBytes);
await db.fonts.insert({
    typeface: info.typeface,
    bold: info.bold,
    italic: info.italic,
    url: `https://cdn.example.com/fonts/${fontFile.name}`,
});

Step 2: At viewing time — retrieve stored metadata and register fonts before loading a document.

// Fetch font entries from your database
const fonts = await db.fonts.list();

// Register all fonts with the viewer
await client.registerFonts(fonts.map((f) => ({ typeface: f.typeface, bold: f.bold, italic: f.italic, url: f.url })));

// Now load the document — registered fonts will be used during rendering
await viewer.load(documentSource);

Supported font formats: OTF, TTF, WOFF, and WOFF2.

Font Usage

After rendering, you can inspect how each font request in the document was resolved — which font was matched, its source, and any glyph-fallback fonts used during text shaping.

// Query font usage after at least one page has been rendered
const fontUsage = await viewer.getFontUsage();

for (const entry of fontUsage) {
    // What the document requested
    const spec =
        "typeface" in entry.spec
            ? `${entry.spec.typeface} (bold=${entry.spec.bold}, italic=${entry.spec.italic})`
            : `fontId=${entry.spec.fontId}`;

    // How it was resolved
    const resolved = entry.resolved;
    console.log(`${spec} → ${resolved.familyName} [${resolved.source}]`);

    // Any additional fonts used via glyph fallback
    for (const fb of entry.fallbacks) {
        console.log(`  fallback: ${fb.familyName} [${fb.source}]`);
    }
}

Font usage is populated incrementally as pages are rendered. To react to changes, listen to the font:usageChange event:

viewer.on("font:usageChange", ({ entries }) => {
    console.log(`Font usage updated: ${entries.length} font specs resolved`);
});

The source field on each resolved font indicates where it came from: "embedded" (bundled in the document), "standard" (built-in standard font), "googleFonts", "url", "local", or { custom: string } for registered custom fonts.

Headless Rendering

Render pages to images without UI:

// Create headless viewer (no container)
const viewer = await client.createViewer();
await viewer.load(pdfBytes);

// Render page to ImageData (0-based page index)
const imageData = await viewer.renderPage(0, { scale: 2 });

// Render to Blob
const blob = await viewer.renderPage(0, {
    format: "blob",
    imageType: "image/png",
});

// Render to data URL
const dataUrl = await viewer.renderPage(0, {
    format: "data-url",
    imageType: "image/jpeg",
    quality: 0.9,
});

// Render thumbnail
const thumb = await viewer.renderThumbnail(0, { scale: 1 });

Password-Protected Documents

When a password-protected document is loaded in UI mode, the viewer automatically prompts the user to enter the password. For headless mode, you can handle it programmatically:

await viewer.load(source);

if (await viewer.needsPassword()) {
    const success = await viewer.authenticate("my-password");
    if (!success) {
        console.error("Incorrect password");
    }
}

How It Works

udoc-viewer uses a custom document processing engine written in Rust, compiled to WebAssembly. Documents are parsed and rendered entirely in the browser with near-native performance — no PDF.js, no iframe hacks, no server-side conversion.

The JavaScript wrapper (@docmentis/udoc-viewer) is MIT-licensed and open source. The WASM rendering engine is free to use, including in commercial applications. See LICENSE for details.

Browser Support

| Browser | Supported | | ------------- | --------- | | Chrome / Edge | ✅ 80+ | | Firefox | ✅ 80+ | | Safari | ✅ 15+ |

Requires WebAssembly support.

Telemetry

udoc-viewer collects anonymous, non-personally-identifiable usage data to help us understand SDK adoption and prioritize format support. Telemetry fires once per document open.

What we collect:

| Field | Description | Example | | ---------------- | ------------------------------------------------------ | -------------- | | domain | Hostname of the embedding website | example.com | | format | Document format | pdf | | size_bucket | File size in units of 100 KB (floor(bytes / 100000)) | 3 | | viewer_version | SDK version string | 0.5.19 | | license_hash | SHA-256 hash of the license key (empty if none) | a1b2c3... | | license_info | Non-sensitive license metadata (identifier, validity) | — | | distinct_id | Anonymous random UUID stored in localStorage | f47ac10b-... |

Opting out

Telemetry can be disabled with a license that includes the no_telemetry feature:

const client = await UDocClient.create({
    license: "eyJ2Ijox...",
    disableTelemetry: true,
});

Without a qualifying license the flag is silently ignored and a warning is logged to the console. To obtain a license, contact [email protected].

What we do NOT collect:

  • Document content, filenames, or URLs
  • User identity, cookies, or session data (the UUID above is random and not linked to any account)
  • IP addresses (disabled at the collection endpoint)
  • Any other personally identifiable information

Branding & Attribution

udoc-viewer is free to use, including in commercial applications. A "Powered by docMentis" attribution link is shown by default.

To remove the attribution, contact [email protected] to obtain a license key, then pass it when creating the client:

const client = await UDocClient.create({ license: "eyJ2Ijox..." });
const viewer = await client.createViewer({
    container: "#viewer",
    hideAttribution: true,
});

The hideAttribution option is only honored when the license includes the no_attribution feature. Without a valid license, the attribution link will remain visible.

Contributing

We welcome bug reports, feature requests, and pull requests.

License

The JavaScript/TypeScript source code is licensed under the MIT License.

The WebAssembly binary (src/wasm/udoc_bg.wasm) is distributed under the docMentis WASM Runtime License -- free to use with the docMentis Viewer in commercial and non-commercial applications.

Links