pdf_image_tools.py 2.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263
  1. from io import BytesIO
  2. import cv2
  3. import fitz
  4. import numpy as np
  5. from PIL import Image
  6. from magic_pdf.data.data_reader_writer import DataWriter
  7. from magic_pdf.libs.commons import join_path
  8. from magic_pdf.libs.hash_utils import compute_sha256
  9. def cut_image(bbox: tuple, page_num: int, page: fitz.Page, return_path, imageWriter: DataWriter):
  10. """从第page_num页的page中,根据bbox进行裁剪出一张jpg图片,返回图片路径 save_path:需要同时支持s3和本地,
  11. 图片存放在save_path下,文件名是:
  12. {page_num}_{bbox[0]}_{bbox[1]}_{bbox[2]}_{bbox[3]}.jpg , bbox内数字取整。"""
  13. # 拼接文件名
  14. filename = f'{page_num}_{int(bbox[0])}_{int(bbox[1])}_{int(bbox[2])}_{int(bbox[3])}'
  15. # 老版本返回不带bucket的路径
  16. img_path = join_path(return_path, filename) if return_path is not None else None
  17. # 新版本生成平铺路径
  18. img_hash256_path = f'{compute_sha256(img_path)}.jpg'
  19. # 将坐标转换为fitz.Rect对象
  20. rect = fitz.Rect(*bbox)
  21. # 配置缩放倍数为3倍
  22. zoom = fitz.Matrix(3, 3)
  23. # 截取图片
  24. pix = page.get_pixmap(clip=rect, matrix=zoom)
  25. byte_data = pix.tobytes(output='jpeg', jpg_quality=95)
  26. imageWriter.write(img_hash256_path, byte_data)
  27. return img_hash256_path
  28. def cut_image_to_pil_image(bbox: tuple, page: fitz.Page, mode="pillow"):
  29. # 将坐标转换为fitz.Rect对象
  30. rect = fitz.Rect(*bbox)
  31. # 配置缩放倍数为3倍
  32. zoom = fitz.Matrix(3, 3)
  33. # 截取图片
  34. pix = page.get_pixmap(clip=rect, matrix=zoom)
  35. if mode == "cv2":
  36. # 直接转换为numpy数组供cv2使用
  37. img_array = np.frombuffer(pix.samples, dtype=np.uint8).reshape(pix.height, pix.width, pix.n)
  38. # PyMuPDF使用RGB顺序,而cv2使用BGR顺序
  39. if pix.n == 3 or pix.n == 4:
  40. image_result = cv2.cvtColor(img_array, cv2.COLOR_RGB2BGR)
  41. else:
  42. image_result = img_array
  43. elif mode == "pillow":
  44. # 将字节数据转换为文件对象
  45. image_file = BytesIO(pix.tobytes(output='png'))
  46. # 使用 Pillow 打开图像
  47. image_result = Image.open(image_file)
  48. else:
  49. raise ValueError(f"mode: {mode} is not supported.")
  50. return image_result