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>
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>
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();
}
}
}
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);
}
}
zomato-bill.html
): Defines the layout and format of the Zomato bill using placeholders for dynamic content, such as customer details and item lists.PdfService
): Handles rendering the Thymeleaf template into HTML and converting it to a PDF using Flying Saucer.BillController
): Provides a REST endpoint /api/bills/generate
which, when called, generates the PDF and sends it as a downloadable response.http://localhost:8080/api/bills/generate
.