当前位置: 首页 > news >正文

python reportlab模块----操作PDF文件

reportlab模块----操作PDF文件

  • 一. 安装模块
  • 二. reportlab相关介绍
  • 三. 扩展canvas类
  • 四. 水平写入完整代码
  • 五. 垂直写入完整代码

一. 安装模块

pip install reportlab

二. reportlab相关介绍

# 1. letter 生成A4纸张尺寸
from reportlab.lib.pagesizes import letter
print(letter)    # (612.0, 792.0)  X轴大小,Y轴大小

# 2. TTFont 设置字体
from reportlab.pdfbase import pdfmetrics        # 注册字体方法
from reportlab.pdfbase.ttfonts import TTFont	# 获取字体类
pdfmetrics.registerFont(TTFont('song', 'simsun.ttc'))

# 3. reportlab自带的颜色库
from reportlab.lib import colors

# 4. Canvas画布,相当于工作区
from reportlab.pdfgen import canvas
canvas.Canvas('新PDF文件名称.pdf', pagesize=letter)  # letter=A4纸大小

# 5. 设置字体
c = canvas.Canvas(filename, pagesize=letter)
c.setFont('song', 12)  # 参数1:已注册字体名称,参数2:字体大小

# 6. TableStyle 给表格设置样式
tab_style = TableStyle([
	('FONTNAME', (0, 0), (-1, 0), 'song-Bold', 12),		 # 表头字体
	('TOPPADDING', (0, 1), (-1, -1), 1),				 # 表头顶部填充
	('BOTTOMPADDING', (0, 0), (-1, 0), 1),				 # 表头底部填充
	('BACKGROUND', (0, 0), (-1, 0), colors.grey),        # 表头背景色
    ('TEXTCOLOR', (0, 0), (-1, 0), colors.whitesmoke),   # 表头文字颜色
    ('ALIGN', (0, 0), (-1, -1), 'CENTER'),               # 单元格文本居中
    ('VALIGN', (0, 0), (-1, -1), 'MIDDLE'),              # 单元格文本垂直居中
    ("FONT", (0, 0), (-1, -1), 'song', 8),               # 单元格字体
    ('BACKGROUND', (0, 1), (-1, -1), colors.beige),      # 行背景色
    ('BOX', (0, 0), (-1, -1), 0.5, colors.black),        # 表格边框颜色
    ('INNERGRID', (0, 0), (-1, -1), 0.5, colors.black),  # 表格单元格颜色
])
# ('FONTNAME', (0,     0),     (-1,    0),     'song-Bold', 12)
#   样式名称,   (开始列,开始行), (结束列, 结束行), 字体,         字体大小)   对应关系

# 7. Table 绘制表格
# Table(数据,style=样式,colWidths=[列宽],rowHeights=[行高])
table_data= [
	('姓名', '性别', '年龄', '民族'),
	('张三', '男', 20, '汉'),
	('李四', '男', 21, '汉'),
	('王小小', '女', 18, '汉'),
]
colWidths = [20, 15, 13, 15]
rowHeights = [29]*len(table_data)
x = 30
y = 600
table1 = Table(table_data, style=table_style, colWidths=colWidths, rowHeights=rowHeights)
table1.wrapOn(self, letter[0], letter[1]) 
table1.drawOn(self, x, y)  # 起始点(x,y)

# 8. 添加图片
filepath = 'd:\images\tttt.png'
image = ImageReader(filepath)
c.drawImage(image, x, y, width=120, height=60)  
# 起始点(x,y) width=图片宽度,height=高度
 
# 9. 添加文本
c.drawString(x, y, text) # 起始点(x,y)

# 10. 垂直写入内容的 语法糖
def translate(func):
    def wrapper(*args, **kwargs):
        ox = letter[0] / 2
        oy = letter[1] / 2
        args[0].saveState()
        args[0].translate(ox, oy)   # 将中心点设置为工作区原点
        args[0].rotate(90)			# 然后按照原点顺时针旋转90度
        func(args[0], *args[1:], **kwargs)
        args[0].restoreState()
    return wrapper

三. 扩展canvas类

from reportlab.lib.pagesizes import letter
from reportlab.pdfgen import canvas
from reportlab.pdfbase import pdfmetrics
from reportlab.pdfbase.ttfonts import TTFont
from reportlab.platypus import Table, TableStyle
from reportlab.lib import colors
from reportlab.lib.utils import ImageReader

# 水平写入扩展
class PDFCanvas(canvas.Canvas):
    FONTS_DIR = r"D:\fonts"
    font_path = os.path.join(FONTS_DIR, 'simsun.ttc')
    pdfmetrics.registerFont(TTFont('song', font_path))

    def __init__(self, filename, **kwargs):
        super().__init__(filename, **kwargs)

    def draw_image(self, filename, **kwargs):
        IMAGES_DIR = r'C:\下载图片'
        filepath = os.path.join(IMAGES_DIR, filename)
        x = kwargs.get('x', 20)                                 # x轴位置
        y = kwargs.get('y', 756)                                # y轴位置
        width = kwargs.get('width', 48)                         # 图片宽度
        height = kwargs.get('height', 18)
        image = ImageReader(filepath)
        self.drawImage(image, x, y, width=width, height=height)

    def draw_text(self, text, **kwargs):
        x = kwargs.get('x', None)
        y = kwargs.get('y', None)
        font = kwargs.get('font', 'song')
        font_size = kwargs.get('font_size', 12)
        self.setFont(font, font_size)
        text_width = self.stringWidth(text, font, font_size)
        x = x if x else (letter[0] - text_width) / 2
        self.drawString(x, y, text)

    def draw_table(self, table_data, **kwargs):
        title = kwargs.get('title', '')
        if title:
            table_data = [(title,), *table_data]
        x = kwargs.get('x', 30)  # 表格X轴位置
        y = kwargs.get('y', 650)  # 表格Y轴位置
        colWidths = kwargs.get('colWidths', 120)
        table_style = kwargs.get('table_style', tab_style)  # 表格样式
        table_h = Table(table_data, style=table_style, colWidths=colWidths, rowHeights=[29]*len(table_data))
        table_h.wrapOn(self, letter[0], letter[1])
        table_h.drawOn(self, x, y)


# 垂直写入扩展
def translate(func):
    def wrapper(*args, **kwargs):
        ox = letter[0] / 2
        oy = letter[1] / 2
        args[0].saveState()
        args[0].translate(ox, oy)   # 将中心点设置为工作区原点
        args[0].rotate(90)			# 然后按照原点顺时针旋转90度
        func(args[0], *args[1:], **kwargs)
        args[0].restoreState()
    return wrapper


class PDFCanvas(canvas.Canvas):

    font_path = os.path.join(FONTS_DIR, 'simsun.ttc')
    pdfmetrics.registerFont(TTFont('song', font_path))

    def __init__(self, filename, **kwargs):
        super().__init__(filename, **kwargs)

    @translate
    def draw_image(self, filename, **kwargs):
    	IMAGES_DIR = r'C:\下载图片'
        filepath = os.path.join(IMAGES_DIR, filename)
        x = kwargs.get('x', 20)                                 # x轴位置
        y = kwargs.get('y', 756)                                # y轴位置
        width = kwargs.get('width', 48)                         # 图片宽度
        height = kwargs.get('height', 18)                       # 图像路径
        x1 = -(letter[1] / 2 - y)
        y = letter[0] / 2 - x
        image = ImageReader(filepath)
        self.drawImage(image, x1, y, width=width, height=height)

    @translate
    def draw_text(self, text, **kwargs):
        x = kwargs.get('x', 20)
        y = kwargs.get('y', None)
        font = kwargs.get('font', 'song')
        font_size = kwargs.get('font_size', 12)
        self.setFont(font, font_size)
        text_width = self.stringWidth(text, font, font_size)
        x1 = -(letter[1] / 2 - y) if y else -int(text_width / 2)
        y = letter[0] / 2 - x
        self.drawString(x1, y, text)

    @translate
    def draw_table(self, table_data, **kwargs):
        title = kwargs.get('title', '')
        if title:
            table_data = [(title,), *table_data]
        x = kwargs.get('x', 30)  # 表格X轴位置
        y = kwargs.get('y', 650)  # 表格Y轴位置
        colWidths = kwargs.get('colWidths', 120)
        table_style = kwargs.get('table_style', tab_style)  # 表格样式
        x1 = -(letter[1] / 2 - y)
        y = letter[0] / 2 - x
        table1 = Table(table_data, style=table_style, colWidths=colWidths, rowHeights=[29]*len(table_data))
        table1.wrapOn(self, letter[0], letter[1])
        table1.drawOn(self, x1, y)

四. 水平写入完整代码

import os
from reportlab.lib.pagesizes import letter
from reportlab.pdfgen import canvas
from reportlab.pdfbase import pdfmetrics
from reportlab.pdfbase.ttfonts import TTFont
from reportlab.platypus import Table, TableStyle
from reportlab.lib import colors
from reportlab.lib.utils import ImageReader

# 水平写入内容
tab_style = TableStyle([
	('FONTNAME', (0, 0), (-1, 0), 'song-Bold', 12),		 # 表头字体
	('TOPPADDING', (0, 1), (-1, -1), 1),				 # 表头顶部填充
	('BOTTOMPADDING', (0, 0), (-1, 0), 1),				 # 表头底部填充
	('BACKGROUND', (0, 0), (-1, 0), colors.grey),        # 表头背景色
    ('TEXTCOLOR', (0, 0), (-1, 0), colors.whitesmoke),   # 表头文字颜色
    ('ALIGN', (0, 0), (-1, -1), 'CENTER'),               # 单元格文本居中
    ('VALIGN', (0, 0), (-1, -1), 'MIDDLE'),              # 单元格文本垂直居中
    ("FONT", (0, 0), (-1, -1), 'song', 8),               # 单元格字体
    ('BACKGROUND', (0, 1), (-1, -1), colors.beige),      # 行背景色
    ('BOX', (0, 0), (-1, -1), 0.5, colors.black),        # 表格边框颜色
    ('INNERGRID', (0, 0), (-1, -1), 0.5, colors.black),  # 表格单元格颜色
])

class PDFCanvas(canvas.Canvas):
    FONTS_DIR = r"D:\fonts"
    font_path = os.path.join(FONTS_DIR, 'simsun.ttc')
    pdfmetrics.registerFont(TTFont('song', font_path))

    def __init__(self, filename, **kwargs):
        super().__init__(filename, **kwargs)

    def draw_image(self, filename, **kwargs):
        IMAGES_DIR = r'C:\下载图片'
        filepath = os.path.join(IMAGES_DIR, filename)
        x = kwargs.get('x', 20)                                 # x轴位置
        y = kwargs.get('y', 756)                                # y轴位置
        width = kwargs.get('width', 48)                         # 图片宽度
        height = kwargs.get('height', 18)
        image = ImageReader(filepath)
        self.drawImage(image, x, y, width=width, height=height)

    def draw_text(self, text, **kwargs):
        x = kwargs.get('x', None)
        y = kwargs.get('y', None)
        font = kwargs.get('font', 'song')
        font_size = kwargs.get('font_size', 12)
        self.setFont(font, font_size)
        text_width = self.stringWidth(text, font, font_size)
        x = x if x else (letter[0] - text_width) / 2
        self.drawString(x, y, text)

    def draw_table(self, table_data, **kwargs):
        title = kwargs.get('title', '')
        if title:
            table_data = [(title,), *table_data]
        x = kwargs.get('x', 30)  # 表格X轴位置
        y = kwargs.get('y', 650)  # 表格Y轴位置
        colWidths = kwargs.get('colWidths', 120)
        table_style = kwargs.get('table_style', tab_style)  # 表格样式
        table_h = Table(table_data, style=table_style, colWidths=colWidths, rowHeights=[29]*len(table_data))
        table_h.wrapOn(self, letter[0], letter[1])
        table_h.drawOn(self, x, y)


if __name__ == '__main__':
    c = PDFCanvas('水平写入.pdf', pagesize=letter)
    image_name = '123.png'
    c.draw_image(image_name, x=70, y=400, width=320, height=240)
    text1 = '我现在需要回答用户关于“reportlab letter”的问题'
    c.draw_text(text1,  y=700, font_size=16)
    table1 = [
        ('姓名', '性别', '年龄', '民族'),
        ('张三', '男', 20, '汉'),
        ('李四', '男', 21, '汉'),
        ('王小小', '女', 18, '汉'),
    ]

    c.draw_table(table1, x=100, y=200, colWidths=[80, 80, 80, 80])
    c.save()

在这里插入图片描述

五. 垂直写入完整代码

import os
from reportlab.lib.pagesizes import letter
from reportlab.pdfgen import canvas
from reportlab.pdfbase import pdfmetrics
from reportlab.pdfbase.ttfonts import TTFont
from reportlab.platypus import Table, TableStyle
from reportlab.lib import colors
from reportlab.lib.utils import ImageReader


# 垂直写入内容
def translate(func):
    def wrapper(*args, **kwargs):
        ox = letter[0] / 2
        oy = letter[1] / 2
        args[0].saveState()
        args[0].translate(ox, oy)   # 将中心点设置为工作区原点
        args[0].rotate(90)			# 然后按照原点顺时针旋转90度
        func(args[0], *args[1:], **kwargs)
        args[0].restoreState()
    return wrapper
    
tab_style = TableStyle([
	('FONTNAME', (0, 0), (-1, 0), 'song-Bold', 12),		 # 表头字体
	('TOPPADDING', (0, 1), (-1, -1), 1),				 # 表头顶部填充
	('BOTTOMPADDING', (0, 0), (-1, 0), 1),				 # 表头底部填充
	('BACKGROUND', (0, 0), (-1, 0), colors.grey),        # 表头背景色
    ('TEXTCOLOR', (0, 0), (-1, 0), colors.whitesmoke),   # 表头文字颜色
    ('ALIGN', (0, 0), (-1, -1), 'CENTER'),               # 单元格文本居中
    ('VALIGN', (0, 0), (-1, -1), 'MIDDLE'),              # 单元格文本垂直居中
    ("FONT", (0, 0), (-1, -1), 'song', 8),               # 单元格字体
    ('BACKGROUND', (0, 1), (-1, -1), colors.beige),      # 行背景色
    ('BOX', (0, 0), (-1, -1), 0.5, colors.black),        # 表格边框颜色
    ('INNERGRID', (0, 0), (-1, -1), 0.5, colors.black),  # 表格单元格颜色
])


class PDFCanvas(canvas.Canvas):
    FONTS_DIR = r"D:\fonts"
    font_path = os.path.join(FONTS_DIR, 'simsun.ttc')
    pdfmetrics.registerFont(TTFont('song', font_path))

    def __init__(self, filename, **kwargs):
        super().__init__(filename, **kwargs)

    @translate
    def draw_image(self, filename, **kwargs):
        IMAGES_DIR = r'C:\下载图片'
        filepath = os.path.join(IMAGES_DIR, filename)
        x = kwargs.get('x', 20)                                 # x轴位置
        y = kwargs.get('y', 756)                                # y轴位置
        width = kwargs.get('width', 48)                         # 图片宽度
        height = kwargs.get('height', 18)                       # 图像路径
        x1 = -(letter[1] / 2 - y)
        y = letter[0] / 2 - x
        image = ImageReader(filepath)
        self.drawImage(image, x1, y, width=width, height=height)

    @translate
    def draw_text(self, text, **kwargs):
        x = kwargs.get('x', 20)
        y = kwargs.get('y', None)
        font = kwargs.get('font', 'song')
        font_size = kwargs.get('font_size', 12)
        self.setFont(font, font_size)
        text_width = self.stringWidth(text, font, font_size)
        x1 = -(letter[1] / 2 - y) if y else -int(text_width / 2)
        y = letter[0] / 2 - x
        self.drawString(x1, y, text)

    @translate
    def draw_table(self, table_data, **kwargs):
        title = kwargs.get('title', '')
        if title:
            table_data = [(title,), *table_data]
        x = kwargs.get('x', 30)  # 表格X轴位置
        y = kwargs.get('y', 650)  # 表格Y轴位置
        colWidths = kwargs.get('colWidths', 120)
        table_style = kwargs.get('table_style', tab_style)  # 表格样式
        x1 = -(letter[1] / 2 - y)
        y = letter[0] / 2 - x
        table1 = Table(table_data, style=table_style, colWidths=colWidths, rowHeights=[29]*len(table_data))
        table1.wrapOn(self, letter[0], letter[1])
        table1.drawOn(self, x1, y)


if __name__ == '__main__':
    c = PDFCanvas('垂直写入.pdf', pagesize=letter)
    image_name = '123.png'
    c.draw_image(image_name, x=280, y=100, width=320, height=200)
    text1 = '我现在需要回答用户关于“reportlab letter”的问题'
    c.draw_text(text1, x=50, font_size=16)
    table1 = [
        ('姓名', '性别', '年龄', '民族'),
        ('张三', '男', 20, '汉'),
        ('李四', '男', 21, '汉'),
        ('王小小', '女', 18, '汉'),
    ]

    c.draw_table(table1, x=400, y=100, colWidths=[80, 80, 80, 80])
    c.save()

在这里插入图片描述

http://www.dtcms.com/a/122123.html

相关文章:

  • 请回答集成测试和系统测试的区别,以及它们的应用场景主要是什么?
  • 解决华硕主板Z890m下载ubuntu20.04后没有以太网问题
  • 【免费】在线工具网址
  • CSS空间转换
  • 第十四届蓝桥杯Scratch03月stema选拔赛真题——绘制彩虹
  • pycharm编译部署智能合约(solcx与web3搭配)
  • crossorigin是什么?【你了解前端的crossorigin吗?!】
  • mapbox基础,加载F4Map二维地图
  • Scala相关知识学习总结6
  • GPT-4o-image模型:开启AI图片编辑新时代
  • Windows Terminal 美化增强攻略 2.0:打造个性化高效开发环境(快捷键介绍、编程语言环境、starship美化、高效命令行工具)
  • TCP的三次握手和四次挥手
  • Visual Studio未能加载相应的Package包弹窗报错
  • HarmonyOS:使用geoLocationManager (位置服务)获取位置信息
  • ES6规范新特性总结
  • WebView2最低支持.NET frame4.5,win7系统
  • Linux基础IO(三)之访问文件的本质
  • GaussDB使用指南
  • FFmpeg安装和使用
  • # 基于BERT的文本分类
  • 龙蜥社区荣获 OS2ATC 2025 “最具影响力开源创新贡献奖”
  • spark数据清洗案例:流量统计
  • 力扣刷题-热题100题-第34题(c++、python)
  • 机器学习代码基础——ML2 使用梯度下降的线性回归
  • 暑假实习面试复盘
  • Vue框架的Diff算法
  • 使用Ollama通过预训练模型获取句子向量(rest api方式)
  • GDB调试程序的基本命令和用法(Qt程序为例)
  • 三月份面试感触
  • OpenCV链接库失败,报错 无法解析的外部符号