Guía R-007 – Hacer informes con PDFMAKE

En esta búsqueda de completar las funcionalidades de los desarrollos de React me faltaba disponer de una solución para generar informes.

Siempre pensé que de todas las soluciones que dispongo para realizar informes en PHP (publicado para PHPRunner), serían suficientes y así en parte lo es, porque todas ellas se pueden utilizar, pero deseaba disponer de alguna en React, sencilla, fácil de utilizar y muy rápida, y creo que la he conseguido. Este ejemplo es un caso sencillo de hacer la simulación de una factura con el producto PDFMake.

Objetivo

Disponer de una solución para crear informes, documentos, etc., principalmente en PDF desde el entorno de React.

DEMO:  https://fhumanes.com/factura-pdfmake/

Solución Técnica

Como os he indicado, he utilizado el producto PDFMake. En React me costó mucho incorporar las fuentes para utilizar en el PDF, pero con la IA DeepSeek y muchas pruebas de las soluciones que me aportaba, conseguí incorporar las fuentes y a partir de ahí, todo han sido «alegrías», ya que ví lo sencillo y potente de la solución, aunque no sirve para todo tipo de informe.

El ejemplo es básico, un botón y la elaboración de la factura cuyos datos son fijos, pero muy fácil de hacerlos dinámicos.

Una de las aportaciones que hace esta solución es que puedes «diseñar» o escribir la definición del informe en interactivo en este enlace (http://pdfmake.org/playground.html) y una vez que lo tengas «ajustado» puedes llevar esa definición al código de la aplicación. Esta alternativa ayuda mucho a las personas que van a realizar pocos informes. La documentación es buena y tiene muchos ejemplos .

El código que produce la factura es este:

import { React, useEffect } from 'react';
import  * as pdfMake from 'pdfmake/build/pdfmake';
import { loadFonts } from './utils/loadFonts';

// Formateado de los valores númericos en el informe
let decimal = new Intl.NumberFormat('de-DE', {
    style: 'currency',
    currency: 'eur',
    minimumFractionDigits: 2,
    maximumFractionDigits: 2
});

const Factura = () => {

  // Datos de la Factura
  const facturaData = {
    numero: 'F-2023-001',
    fecha: new Date().toLocaleDateString(),
    cliente: 'Juan Pérez',
    nit: '123456789',
    items: [
      { descripcion: 'Laptop HP', cantidad: 1, precio: 1200 },
      { descripcion: 'Mouse inalámbrico', cantidad: 2, precio: 25 },
      { descripcion: 'Teclado mecánico', cantidad: 1, precio: 80 },
      { descripcion: 'Laptop HP', cantidad: 1, precio: 1200 },
      { descripcion: 'Mouse inalámbrico', cantidad: 2, precio: 25 },
      { descripcion: 'Teclado mecánico', cantidad: 1, precio: 80 },
      { descripcion: 'Laptop HP', cantidad: 1, precio: 1200 },
      { descripcion: 'Mouse inalámbrico', cantidad: 2, precio: 25 },
      { descripcion: 'Teclado mecánico', cantidad: 1, precio: 80 },
      { descripcion: 'Laptop HP', cantidad: 1, precio: 1200 },
      { descripcion: 'Mouse inalámbrico', cantidad: 2, precio: 25 },
      { descripcion: 'Teclado mecánico', cantidad: 1, precio: 80 },
      { descripcion: 'Laptop HP', cantidad: 10, precio: 1200 },
      { descripcion: 'Mouse inalámbrico', cantidad: 20, precio: 25 },
      { descripcion: 'Teclado mecánico', cantidad: 30, precio: 80 },
      { descripcion: 'Laptop HP', cantidad: 1, precio: 1200 },
      { descripcion: 'Mouse inalámbrico', cantidad: 2, precio: 25 },
      { descripcion: 'Teclado mecánico', cantidad: 1, precio: 80 },
      { descripcion: 'Laptop HP', cantidad: 1, precio: 1200 },
      { descripcion: 'Mouse inalámbrico', cantidad: 2, precio: 25 },
      { descripcion: 'Teclado mecánico', cantidad: 1, precio: 80 },
    ],
  };

    useEffect(() => {
      loadFonts(); // Precarga las fuentes al montar el componente
    }, []);

  const generarFactura = async () => {
    await loadFonts(); // Asegura que las fuentes estén cargadas

    const subtotal = facturaData.items.reduce((sum, item) => sum + (item.precio * item.cantidad), 0);
    const iva = subtotal * 0.16;
    const total = subtotal + iva;

    // Definición del Informe
    const docDefinition = {
        // a string or { width: number, height: number }
        pageSize: 'A5',
        // by default we use portrait, you can change it to landscape if you wish
        pageOrientation: 'landscape',
        // [left, top, right, bottom] or [horizontal, vertical] or just a number for equal margins
        pageMargins: [ 50, 30, 20, 40 ],
        
        footer: function(currentPage, pageCount) { 
            return [
                {text: 'Página: '+currentPage.toString() + ' de ' + pageCount ,margin: [30,10,10,0],alignment: 'right' }
                ]
        },
        header: function(currentPage, pageCount, pageSize) {
            // you can apply any logic and return any valid pdfmake element
            return [
            { text: 'simple text',margin: [30,10,30,10], alignment: (currentPage % 2) ? 'left' : 'right' },
            { canvas: [ { type: 'rect', x: 170, y: 32, w: pageSize.width - 170, h: 40 } ] }
            ]
        },
            
      content: [
        {
            layout: 'noBorders', // optional
            table: {
              widths: ['auto', '*', 'auto'],
              body: [
                  [ 
                    { image: '' ,
                         width: 75 },
                  { text: '\nFACTURA/INVOICE', style: 'header' },
                  { text: `\nNúmero: ${facturaData.numero}`, margin: [0, 10], color: 'blue'},
                  ]
                  ]
            },
          },
          { text: '\n\n' },
        { text: `Fecha: ${facturaData.fecha}` },
        { text: `Cliente: ${facturaData.cliente}` },
        { text: `NIT: ${facturaData.nit}`, margin: [0, 0, 0, 20] },
        
        {
          layout: 'lightHorizontalLines', // optional
          table: {
            headerRows: 1,
            widths: ['*', 'auto', 'auto', 'auto'],
            body: [
                [
                    { text: 'Descripción', style: 'tableHeader' },
                    { text: 'Cantidad', style: 'tableHeader', alignment: 'center' },
                    { text: 'Precio', style: 'tableHeader', alignment: 'right' },
                    { text: 'Total', style: 'tableHeader', alignment: 'right' }
                ],
              ...facturaData.items.map(item => 
                [
                    item.descripcion,
                    { text: item.cantidad ,alignment: 'center'},
                    { text: decimal.format(item.precio),alignment: 'right'} ,
                    { text: decimal.format(item.cantidad * item.precio),alignment: 'right'} ,
                ]),
                  // Asegúrate de que cada fila tenga exactamente 4 elementos:
                [{ text: 'Subtotal:', alignment: 'right', colSpan: 3 },'', '',  { text: decimal.format(subtotal),alignment: 'right'}],
                [{ text: 'IVA (16%):', alignment: 'right', colSpan: 3 },'', '',  { text: decimal.format(iva),alignment: 'right'}],
                [{ text: 'Total:', alignment: 'right', colSpan: 3, bold: true },'', '',  { text: decimal.format(total),alignment: 'right', bold: true}],

            ],
          },
        },
    
      ],
      styles: {
        header: { 
            fontSize: 18, 
            font: 'Roboto', 
            bold: true,
            alignment: 'center' },
        tableHeader: {
            font: 'Roboto', 
            bold: true,
            fontSize: 13,
            fillColor: '#dddddd'
        } 
      },
    };

    pdfMake.createPdf(docDefinition).download('factura.pdf'); // Descarga directa
  };

  return (
    <div>
      <h2>Generador de Factura</h2>
      <button onClick={generarFactura}>Descargar Factura</button>
    </div>
  );
};

export default Factura;

En el ejemplo podréis ver imágenes (en este caso embebida en el documento), definición de cabeceras y pies de página con su contador de páginas, también distintas ediciones de números y textos.

Hay otras muchas características que no he incluido como gráficos QR, protección y cifrado del documento PDF, definición de características del documento PDF, etc.

Creo que lo he transmitido en el texto, me asombra la sencillez de la definición del documento y lo rápido y eficiente, que produce el documento, más cuando todo está desarrollado en JavaScript, que al principio, parecía que era para «mover texto en el navegador».

Como siempre , os dejo el ejemplo para que lo podáis descargar en vuestros PC’s y podáis hacer lo ajustes y pruebas que consideréis adecuado.

Adjuntos

Archivo Tamaño de archivo Descargas
zip factura-pdfmake - REACT 528 KB 23

Blog personal para facilitar soporte gratuito a usuarios de React y PHPRunner