Step 1: Add Dependencies

In your pom.xml, add dependencies for Thymeleaf, Flying Saucer, and iText (which Flying Saucer uses for PDF generation). Here’s what the dependencies would look like:

<dependencies>
    <!-- Spring Boot Starter for Thymeleaf -->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-thymeleaf</artifactId>
    </dependency>

    <!-- Flying Saucer PDF Renderer -->
    <dependency>
        <groupId>org.xhtmlrenderer</groupId>
        <artifactId>flying-saucer-pdf-openpdf</artifactId>
        <version>9.1.22</version>
    </dependency>

    <!-- iText (for PDF generation) -->
    <dependency>
        <groupId>com.lowagie</groupId>
        <artifactId>itext</artifactId>
        <version>2.1.7</version>
    </dependency>
</dependencies>

Step 2: Create the Thymeleaf Template

Create an HTML file in your resources folder under src/main/resources/templates/ named zomato-bill.html. This file represents the structure and layout of the Zomato bill.

<!DOCTYPE html>
<html xmlns:th="<http://www.thymeleaf.org>">
<head>
    <meta charset="UTF-8">
    <style>
        body { font-family: Arial, sans-serif; }
        .header { text-align: center; margin-bottom: 20px; }
        .bill-details { width: 100%; margin: 20px 0; }
        .bill-details th, .bill-details td { padding: 10px; text-align: left; border-bottom: 1px solid #ddd; }
        .total { font-weight: bold; }
    </style>
</head>
<body>
    <div class="header">
        <h2>Zomato Order Receipt</h2>
        <p>Order ID: <span th:text="${orderId}"></span></p>
    </div>

    <h3>Customer Details</h3>
    <p><strong>Name:</strong> <span th:text="${customerName}"></span></p>
    <p><strong>Address:</strong> <span th:text="${customerAddress}"></span></p>

    <h3>Order Details</h3>
    <table class="bill-details">
        <thead>
            <tr>
                <th>Item</th>
                <th>Quantity</th>
                <th>Price</th>
            </tr>
        </thead>
        <tbody>
            <tr th:each="item : ${items}">
                <td th:text="${item.name}"></td>
                <td th:text="${item.quantity}"></td>
                <td th:text="${item.price}"></td>
            </tr>
        </tbody>
    </table>

    <h3>Total Amount</h3>
    <p class="total">₹<span th:text="${totalAmount}"></span></p>
</body>
</html>

Step 3: Create a Service to Render the Template as a PDF

Now we need to create a service that will use Thymeleaf to render the HTML and Flying Saucer to convert the rendered HTML into a PDF.

import org.springframework.stereotype.Service;
import org.springframework.ui.Model;
import org.springframework.web.servlet.view.freemarker.FreeMarkerConfig;
import org.thymeleaf.TemplateEngine;
import org.thymeleaf.context.Context;
import org.xhtmlrenderer.pdf.ITextRenderer;

import java.io.ByteArrayOutputStream;
import java.util.List;
import java.util.Map;

@Service
public class PdfService {

    private final TemplateEngine templateEngine;

    public PdfService(TemplateEngine templateEngine) {
        this.templateEngine = templateEngine;
    }

    public byte[] generatePdf(String templateName, Map<String, Object> data) throws Exception {
        // Render HTML with Thymeleaf
        Context context = new Context();
        context.setVariables(data);
        String htmlContent = templateEngine.process(templateName, context);

        // Generate PDF with Flying Saucer
        try (ByteArrayOutputStream baos = new ByteArrayOutputStream()) {
            ITextRenderer renderer = new ITextRenderer();
            renderer.setDocumentFromString(htmlContent);
            renderer.layout();
            renderer.createPDF(baos);
            return baos.toByteArray();
        }
    }
}

Step 4: Create a Controller to Expose the REST API Endpoint

Next, create a REST API endpoint in a controller class to receive the request and generate the PDF.

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;

import javax.servlet.http.HttpServletResponse;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

@RestController
@RequestMapping("/api/bills")
public class BillController {

    @Autowired
    private PdfService pdfService;

    @GetMapping("/generate")
    public void generateZomatoBillPdf(HttpServletResponse response) throws Exception {
        // Sample data for the Zomato bill
        Map<String, Object> data = new HashMap<>();
        data.put("orderId", "ZOM12345");
        data.put("customerName", "John Doe");
        data.put("customerAddress", "123 Main St, Springfield");
        data.put("items", List.of(
                Map.of("name", "Pizza", "quantity", 2, "price", 500),
                Map.of("name", "Burger", "quantity", 1, "price", 200),
                Map.of("name", "Coke", "quantity", 3, "price", 150)
        ));
        data.put("totalAmount", 1000);

        // Generate PDF
        byte[] pdfBytes = pdfService.generatePdf("zomato-bill", data);

        // Set response headers
        response.setContentType("application/pdf");
        response.setHeader("Content-Disposition", "attachment; filename=zomato-bill.pdf");
        response.getOutputStream().write(pdfBytes);
    }
}

Step 5: Explanation of Components

  1. Thymeleaf Template (zomato-bill.html): Defines the layout and format of the Zomato bill using placeholders for dynamic content, such as customer details and item lists.
  2. PDF Service (PdfService): Handles rendering the Thymeleaf template into HTML and converting it to a PDF using Flying Saucer.
  3. REST Controller (BillController): Provides a REST endpoint /api/bills/generate which, when called, generates the PDF and sends it as a downloadable response.

Step 6: Testing the API