detect_footnote.py 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. from collections import Counter
  2. from magic_pdf.libs.commons import fitz # pyMuPDF库
  3. from magic_pdf.libs.coordinate_transform import get_scale_ratio
  4. def parse_footnotes_by_model(page_ID: int, page: fitz.Page, json_from_DocXchain_obj: dict, md_bookname_save_path=None, debug_mode=False):
  5. """
  6. :param page_ID: int类型,当前page在当前pdf文档中是第page_D页。
  7. :param page :fitz读取的当前页的内容
  8. :param res_dir_path: str类型,是每一个pdf文档,在当前.py文件的目录下生成一个与pdf文档同名的文件夹,res_dir_path就是文件夹的dir
  9. :param json_from_DocXchain_obj: dict类型,把pdf文档送入DocXChain模型中后,提取bbox,结果保存到pdf文档同名文件夹下的 page_ID.json文件中了。json_from_DocXchain_obj就是打开后的dict
  10. """
  11. #--------- 通过json_from_DocXchain来获取 footnote ---------#
  12. footnote_bbox_from_DocXChain = []
  13. xf_json = json_from_DocXchain_obj
  14. horizontal_scale_ratio, vertical_scale_ratio = get_scale_ratio(xf_json, page)
  15. # {0: 'title', # 标题
  16. # 1: 'figure', # 图片
  17. # 2: 'plain text', # 文本
  18. # 3: 'header', # 页眉
  19. # 4: 'page number', # 页码
  20. # 5: 'footnote', # 脚注
  21. # 6: 'footer', # 页脚
  22. # 7: 'table', # 表格
  23. # 8: 'table caption', # 表格描述
  24. # 9: 'figure caption', # 图片描述
  25. # 10: 'equation', # 公式
  26. # 11: 'full column', # 单栏
  27. # 12: 'sub column', # 多栏
  28. # 13: 'embedding', # 嵌入公式
  29. # 14: 'isolated'} # 单行公式
  30. for xf in xf_json['layout_dets']:
  31. L = xf['poly'][0] / horizontal_scale_ratio
  32. U = xf['poly'][1] / vertical_scale_ratio
  33. R = xf['poly'][2] / horizontal_scale_ratio
  34. D = xf['poly'][5] / vertical_scale_ratio
  35. # L += pageL # 有的页面,artBox偏移了。不在(0,0)
  36. # R += pageL
  37. # U += pageU
  38. # D += pageU
  39. L, R = min(L, R), max(L, R)
  40. U, D = min(U, D), max(U, D)
  41. # if xf['category_id'] == 5 and xf['score'] >= 0.3:
  42. if xf['category_id'] == 5 and xf['score'] >= 0.43: # 新的footnote阈值
  43. footnote_bbox_from_DocXChain.append((L, U, R, D))
  44. footnote_final_names = []
  45. footnote_final_bboxs = []
  46. footnote_ID = 0
  47. for L, U, R, D in footnote_bbox_from_DocXChain:
  48. if debug_mode:
  49. # cur_footnote = page.get_pixmap(clip=(L,U,R,D))
  50. new_footnote_name = "footnote_{}_{}.png".format(page_ID, footnote_ID) # 脚注name
  51. # cur_footnote.save(md_bookname_save_path + '/' + new_footnote_name) # 把脚注存储在新建的文件夹,并命名
  52. footnote_final_names.append(new_footnote_name) # 把脚注的名字存在list中
  53. footnote_final_bboxs.append((L, U, R, D))
  54. footnote_ID += 1
  55. footnote_final_bboxs.sort(key = lambda LURD: (LURD[1], LURD[0]))
  56. curPage_all_footnote_bboxs = footnote_final_bboxs
  57. return curPage_all_footnote_bboxs
  58. def need_remove(block):
  59. if 'lines' in block and len(block['lines']) > 0:
  60. # block中只有一行,且该行文本全是大写字母,或字体为粗体bold关键词,SB关键词,把这个block捞回来
  61. if len(block['lines']) == 1:
  62. if 'spans' in block['lines'][0] and len(block['lines'][0]['spans']) == 1:
  63. font_keywords = ['SB', 'bold', 'Bold']
  64. if block['lines'][0]['spans'][0]['text'].isupper() or any(keyword in block['lines'][0]['spans'][0]['font'] for keyword in font_keywords):
  65. return True
  66. for line in block['lines']:
  67. if 'spans' in line and len(line['spans']) > 0:
  68. for span in line['spans']:
  69. # 检测"keyword"是否在span中,忽略大小写
  70. if "keyword" in span['text'].lower():
  71. return True
  72. return False
  73. def parse_footnotes_by_rule(remain_text_blocks, page_height, page_id, main_text_font):
  74. """
  75. 根据给定的文本块、页高和页码,解析出符合规则的脚注文本块,并返回其边界框。
  76. Args:
  77. remain_text_blocks (list): 包含所有待处理的文本块的列表。
  78. page_height (float): 页面的高度。
  79. page_id (int): 页面的ID。
  80. Returns:
  81. list: 符合规则的脚注文本块的边界框列表。
  82. """
  83. # if page_id > 20:
  84. if page_id > 2: # 为保证精确度,先只筛选前3页
  85. return []
  86. else:
  87. # 存储每一行的文本块大小的列表
  88. line_sizes = []
  89. # 存储每个文本块的平均行大小
  90. block_sizes = []
  91. # 存储每一行的字体信息
  92. # font_names = []
  93. font_names = Counter()
  94. if len(remain_text_blocks) > 0:
  95. for block in remain_text_blocks:
  96. block_line_sizes = []
  97. # block_fonts = []
  98. block_fonts = Counter()
  99. for line in block['lines']:
  100. # 提取每个span的size属性,并计算行大小
  101. span_sizes = [span['size'] for span in line['spans'] if 'size' in span]
  102. if span_sizes:
  103. line_size = sum(span_sizes) / len(span_sizes)
  104. line_sizes.append(line_size)
  105. block_line_sizes.append(line_size)
  106. span_font = [(span['font'], len(span['text'])) for span in line['spans'] if 'font' in span and len(span['text']) > 0]
  107. if span_font:
  108. # main_text_font应该用基于字数最多的字体而不是span级别的统计
  109. # font_names.append(font_name for font_name in span_font)
  110. # block_fonts.append(font_name for font_name in span_font)
  111. for font, count in span_font:
  112. # font_names.extend([font] * count)
  113. # block_fonts.extend([font] * count)
  114. font_names[font] += count
  115. block_fonts[font] += count
  116. if block_line_sizes:
  117. # 计算文本块的平均行大小
  118. block_size = sum(block_line_sizes) / len(block_line_sizes)
  119. # block_font = collections.Counter(block_fonts).most_common(1)[0][0]
  120. block_font = block_fonts.most_common(1)[0][0]
  121. block_sizes.append((block, block_size, block_font))
  122. # 计算main_text_size
  123. main_text_size = Counter(line_sizes).most_common(1)[0][0]
  124. # 计算main_text_font
  125. # main_text_font = collections.Counter(font_names).most_common(1)[0][0]
  126. # main_text_font = font_names.most_common(1)[0][0]
  127. # 删除一些可能被误识别为脚注的文本块
  128. block_sizes = [(block, block_size, block_font) for block, block_size, block_font in block_sizes if not need_remove(block)]
  129. # 检测footnote_block 并返回 footnote_bboxes
  130. # footnote_bboxes = [block['bbox'] for block, block_size, block_font in block_sizes if
  131. # block['bbox'][1] > page_height * 0.6 and block_size < main_text_size
  132. # and (len(block['lines']) < 5 or block_font != main_text_font)]
  133. # and len(block['lines']) < 5]
  134. footnote_bboxes = [block['bbox'] for block, block_size, block_font in block_sizes if
  135. block['bbox'][1] > page_height * 0.6 and
  136. # 较为严格的规则
  137. block_size < main_text_size and
  138. (len(block['lines']) < 5 or
  139. block_font != main_text_font)]
  140. # 较为宽松的规则
  141. # sum([block_size < main_text_size,
  142. # len(block['lines']) < 5,
  143. # block_font != main_text_font])
  144. # >= 2]
  145. return footnote_bboxes
  146. else:
  147. return []