日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

詳解用 python-docx 創建浮動圖片

瀏覽:23日期:2022-06-29 11:35:32

相信大家對python-docx這個常用的操作docx文檔的庫都不陌生,它支持以內聯形狀(Inline Shape)的形式插入圖片,即圖片和文本之間沒有重疊,遵循流動版式(flow layout)。但是,截至最新的0.8.10版本,python-docx尚不支持插入浮動圖片(floating picture)。這顯然不能滿足豐富多彩的文檔樣式的需要,因此本文探究基于python-docx插入浮動圖片——剖析xml、追蹤源碼,最后得到完整代碼。

問題提出

作者在嘗試實現PDF文檔轉docx(pdf2docx:https://github.com/dothinking/pdf2docx,開發中)的過程中遇到一個需求:根據背景圖片在PDF頁面的具體位置(例如左上角坐標和圖片區域的長寬),將其重現到docx頁面的相應位置。考慮到背景圖片與文本的重疊,這就需要實現精確定位的浮動圖片,參考下圖示例。

詳解用 python-docx 創建浮動圖片

Word中的設置

我們先嘗試在Office Word中,手動解決上述問題。具備基礎的Word使用經驗即可知,通過設置圖片版式來控制圖片的浮動和具體位置。

詳解用 python-docx 創建浮動圖片

上圖版式設置中的文本環繞樣式,大體可以分為三類:

分類 文本重疊 自由定位 樣式名稱 嵌入型 否 否 In line with text 環繞型 否 是 Square, Tight, Through, Top and bottom 完全浮動 是 是 behind text, In front of text

例如最常見的嵌入型圖片,它占據了整行區域,我們既不能將其與文字重疊,也不能自由放置它的位置,而是由頁面排版自動確定。對于環繞型圖片,文本可以進入圖片所在行,但是無法與之重疊;并且,我們可以用鼠標自由拖動其位置。完全浮動型圖片則可以浮于文本上方或者襯于文本下方,同時支持隨意放置其位置。

如果需要精確定位,則可在圖片版式的位置(Position)選項卡進行設置。它提供了多種定位方式,例如絕對定位——根據圖片左上角點距離水平和豎直參考的坐標值來定位。至于參考對象,可以是頁面(Page)本身,這樣(0, 0)就是頁面左上角;也可以是邊距(Margin),此時(0, 0)即為正文區域的左上角。

綜上,我們需要實現精確定位襯于文本下方的圖片版式。

docx背后的xml

我們還知道,docx文檔的背后是xml格式的數據,python-docx正是通過處理xml的方式來讀寫word文檔。所以,接下來先手工創建word文檔,然后查看圖片部分的xml內容。

作為對比,首先分別創建一個普通嵌入型圖片文件和一個襯于文本下方的浮動型圖片文件。然后執行查看步驟:右鍵docx文件 | 7-zip打開壓縮包 | word | document.xml,復制文件內容并格式化xml,得到如下的關于圖片部分的片段。為了便于對比分析,刪除了一些節點屬性。

內聯圖片片段:

<w:drawing> <wp:inline><wp:extent cx='3297600' cy='2782800'/><wp:effectExtent l='0' t='0' r='0' b='0'/><wp:docPr id='1' name='Picture 1'/><wp:cNvGraphicFramePr> <a:graphicFrameLocks/></wp:cNvGraphicFramePr><a:graphic> <a:graphicData><pic:pic> <!-- more pic content --></pic:pic> </a:graphicData></a:graphic> </wp:inline></w:drawing>

浮動圖片片段:

<w:drawing> <wp:anchor behindDoc='1' locked='0' layoutInCell='1' allowOverlap='1'><wp:simplePos x='0' y='0'/><wp:positionH relativeFrom='page'> <wp:posOffset>285750</wp:posOffset></wp:positionH><wp:positionV relativeFrom='page'> <wp:posOffset>457200</wp:posOffset></wp:positionV><wp:extent cx='3297600' cy='2782800'/><wp:effectExtent l='0' t='0' r='0' b='0'/><wp:wrapNone/><wp:docPr id='1' name='Picture 1'/><wp:cNvGraphicFramePr> <a:graphicFrameLocks/></wp:cNvGraphicFramePr><a:graphic> <a:graphicData><pic:pic> <!-- more pic content --></pic:pic> </a:graphicData></a:graphic> </wp:anchor></w:drawing>

對比發現以下相同/相似點:

兩類圖片都放在<w:drawing>節點下:內聯圖片<wp:inline>,浮動圖片<wp:anchor> 具備相同的內容節點:<wp:extent>、<wp:docPr>、<a:graphic>等

除此之外,浮動圖片還有一些獨有特征,并且我們可以從命名上猜測和解讀:

<wp:anchor>節點的behindDoc屬性表明圖片版式為襯于文本下方

<wp:positionH>和<wp:positionV>節點表明水平和豎直絕對定位方式,其中:

relativeFrom屬性指定用于定位的參考對象 子節點<wp:posOffset>指定具體坐標值從內聯圖片開始

從xml的結構對比來看,我們完全可以根據python-docx對內聯圖片的實現來插入浮動圖片。于是,從插入內聯圖片的代碼入手:

from docx import Documentfrom docx.shared import Pt document = Document()document.add_picture(’image.jpg’, width=Pt(200))document.save(’output.docx’)

從python-docx安裝文件夾site-packages/docx進行內容搜索add_picture,得到docx.text.run.add_picture原始定義處:

def add_picture(self, image_path_or_stream, width=None, height=None): inline = self.part.new_pic_inline(image_path_or_stream, width, height) self._r.add_drawing(inline) return InlineShape(inline)

繼續搜索new_pic_inline得到docx.parts.story.BaseStoryPart.new_pic_inline。從注釋可知這是利用CT_Inline類創建<wp:inline>元素,因此后續創建浮動圖片的<wp:anchor>可以在此基礎上修改。

def new_pic_inline(self, image_descriptor, width, height): '''Return a newly-created `w:inline` element. The element contains the image specified by *image_descriptor* and is scaled based on the values of *width* and *height*. ''' rId, image = self.get_or_add_image(image_descriptor) cx, cy = image.scaled_dimensions(width, height) shape_id, filename = self.next_id, image.filename return CT_Inline.new_pic_inline(shape_id, rId, filename, cx, cy)

于是進入CT_Inline類(限于篇幅,刪除了前兩個類方法new和new_pic_inline的具體代碼)——終于見到了一開始探索的xml代碼:

class CT_Inline(BaseOxmlElement): ''' ``<w:inline>`` element, container for an inline shape. ''' @classmethod def new(cls, cx, cy, shape_id, pic):pass @classmethod def new_pic_inline(cls, shape_id, rId, filename, cx, cy):pass @classmethod def _inline_xml(cls):return ( ’<wp:inline %s>n’ ’ <wp:extent cx='914400' cy='914400'/>n’ ’ <wp:docPr id='666' name='unnamed'/>n’ ’ <wp:cNvGraphicFramePr>n’ ’ <a:graphicFrameLocks noChangeAspect='1'/>n’ ’ </wp:cNvGraphicFramePr>n’ ’ <a:graphic>n’ ’ <a:graphicData uri='URI not set'/>n’ ’ </a:graphic>n’ ’</wp:inline>’ % nsdecls(’wp’, ’a’, ’pic’, ’r’))

簡單掃一下CT_Inline類的三個方法,即可將它們聯系上:

_inline_xml()方法給出內聯圖片<wp:inline>的xml結構。 new()方法調用_inline_xml(),并為其中的子節點例如<wp:extent>和<wp:docPr>賦值。 new_pic_inline()調用new(),同時拼接CT_Picture類的結果(節點<pic:pic>,即圖片的具體內容)到<a:graphicData>節點中去。

綜上,實現了內聯圖片的完整xml結構。

插入浮動圖片

從xml結構的對比及上述python-docx對內聯圖片的實現,得到創建浮動圖片的思路:

初始化<wp:anchor>結構,例如behindDoc='1'指定圖片版式為襯于文本下方 使用類似的代碼填充<wp:anchor>元素,尤其是<wp:extent>、<wp:docPr>和<pic:pic> 填充<wp:positionH>和<wp:positionV>精確定位圖片

具體實踐中發現還有關鍵的一步——注冊xml標簽名稱到對應的類,例如<wp:inline>和CT_Inline:

# docx.oxml.__init__.pyregister_element_cls(’wp:inline’, CT_Inline)

綜上,利用python-docx插入浮動圖片(襯于文本下方、頁面定位)的完整代碼如下:

# -*- coding: utf-8 -*- # filename: add_float_picture.py ’’’Implement floating image based on python-docx.- Text wrapping style: BEHIND TEXT <wp:anchor behindDoc='1'>- Picture position: top-left corner of PAGE `<wp:positionH relativeFrom='page'>`.Create a docx sample (Layout | Positions | More Layout Options) and explore the source xml (Open as a zip | word | document.xml) to implement other text wrappingstyles and position modes per `CT_Anchor._anchor_xml()`.’’’ from docx.oxml import parse_xml, register_element_clsfrom docx.oxml.ns import nsdeclsfrom docx.oxml.shape import CT_Picturefrom docx.oxml.xmlchemy import BaseOxmlElement, OneAndOnlyOne # refer to docx.oxml.shape.CT_Inlineclass CT_Anchor(BaseOxmlElement): ''' ``<w:anchor>`` element, container for a floating image. ''' extent = OneAndOnlyOne(’wp:extent’) docPr = OneAndOnlyOne(’wp:docPr’) graphic = OneAndOnlyOne(’a:graphic’) @classmethod def new(cls, cx, cy, shape_id, pic, pos_x, pos_y):'''Return a new ``<wp:anchor>`` element populated with the values passedas parameters.'''anchor = parse_xml(cls._anchor_xml(pos_x, pos_y))anchor.extent.cx = cxanchor.extent.cy = cyanchor.docPr.id = shape_idanchor.docPr.name = ’Picture %d’ % shape_idanchor.graphic.graphicData.uri = ( ’http://schemas.openxmlformats.org/drawingml/2006/picture’)anchor.graphic.graphicData._insert_pic(pic)return anchor @classmethod def new_pic_anchor(cls, shape_id, rId, filename, cx, cy, pos_x, pos_y):'''Return a new `wp:anchor` element containing the `pic:pic` elementspecified by the argument values.'''pic_id = 0 # Word doesn’t seem to use this, but does not omit itpic = CT_Picture.new(pic_id, filename, rId, cx, cy)anchor = cls.new(cx, cy, shape_id, pic, pos_x, pos_y)anchor.graphic.graphicData._insert_pic(pic)return anchor @classmethod def _anchor_xml(cls, pos_x, pos_y):return ( ’<wp:anchor distT='0' distB='0' distL='0' distR='0' simplePos='0' relativeHeight='0' n’ ’ behindDoc='1' locked='0' layoutInCell='1' allowOverlap='1' n’ ’ %s>n’ ’ <wp:simplePos x='0' y='0'/>n’ ’ <wp:positionH relativeFrom='page'>n’ ’ <wp:posOffset>%d</wp:posOffset>n’ ’ </wp:positionH>n’ ’ <wp:positionV relativeFrom='page'>n’ ’ <wp:posOffset>%d</wp:posOffset>n’ ’ </wp:positionV>n’’ <wp:extent cx='914400' cy='914400'/>n’ ’ <wp:wrapNone/>n’ ’ <wp:docPr id='666' name='unnamed'/>n’ ’ <wp:cNvGraphicFramePr>n’ ’ <a:graphicFrameLocks noChangeAspect='1'/>n’ ’ </wp:cNvGraphicFramePr>n’ ’ <a:graphic>n’ ’ <a:graphicData uri='URI not set'/>n’ ’ </a:graphic>n’ ’</wp:anchor>’ % ( nsdecls(’wp’, ’a’, ’pic’, ’r’), int(pos_x), int(pos_y) ))# refer to docx.parts.story.BaseStoryPart.new_pic_inlinedef new_pic_anchor(part, image_descriptor, width, height, pos_x, pos_y): '''Return a newly-created `w:anchor` element. The element contains the image specified by *image_descriptor* and is scaled based on the values of *width* and *height*. ''' rId, image = part.get_or_add_image(image_descriptor) cx, cy = image.scaled_dimensions(width, height) shape_id, filename = part.next_id, image.filenamereturn CT_Anchor.new_pic_anchor(shape_id, rId, filename, cx, cy, pos_x, pos_y)# refer to docx.text.run.add_picturedef add_float_picture(p, image_path_or_stream, width=None, height=None, pos_x=0, pos_y=0): '''Add float picture at fixed position `pos_x` and `pos_y` to the top-left point of page. ''' run = p.add_run() anchor = new_pic_anchor(run.part, image_path_or_stream, width, height, pos_x, pos_y) run._r.add_drawing(anchor)# refer to docx.oxml.__init__.pyregister_element_cls(’wp:anchor’, CT_Anchor)示例

最后,來一個例子看看結果吧:

from docx import Documentfrom docx.shared import Inches, Ptfrom add_float_picture import add_float_picture if __name__ == ’__main__’: document = Document() # add a floating picture p = document.add_paragraph() add_float_picture(p, ’test.png’, width=Inches(5.0), pos_x=Pt(20), pos_y=Pt(30)) # add text p.add_run(’Hello World ’*50) document.save(’output.docx’)

詳解用 python-docx 創建浮動圖片

作者:crazyhat,Python及科學計算愛好者

到此這篇關于詳解用 python-docx 創建浮動圖片的文章就介紹到這了,更多相關python-docx 浮動圖片內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Python 編程
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
欧美成人一二区| 久久一区精品| 1024精品一区二区三区| 久久精品理论片| 成人在线免费观看91| 日韩在线短视频| 91精品99| 蜜桃视频免费观看一区| 日本在线不卡视频| 国产精品久久久久av蜜臀| 精品日韩一区| 中文字幕系列一区| 欧美.日韩.国产.一区.二区| 久久国产高清| 国产精品香蕉| 精品丝袜在线| 午夜一区在线| 国产精品传媒麻豆hd| 日韩欧美综合| 亚洲资源网站| 成人在线免费观看网站| 欧美91视频| 日韩精品成人| 欧洲一级精品| 亚洲综合婷婷| 国产一区二区三区四区五区| 亚洲欧洲另类| 久久黄色影视| 欧美不卡高清一区二区三区| 一区二区三区四区日韩| 精品国产欧美| 午夜在线播放视频欧美| 精品丝袜久久| 蜜臀a∨国产成人精品| 精品中文字幕一区二区三区| 狠狠久久婷婷| 精品高清久久| 亚洲精品一级| 国产精品亚洲一区二区三区在线观看| 亚洲一区二区成人| 久久精品五月| 亚洲欧美专区| 欧美aa在线观看| 日本aⅴ免费视频一区二区三区| 日韩久久精品网| 日韩在线网址| 久久精品国产99久久| 国产美女撒尿一区二区| 欧美+日本+国产+在线a∨观看| 国产欧美大片| 久久亚洲图片| 91看片一区| 国产经典一区| 久久xxxx| 999国产精品视频| 国产精品色婷婷在线观看| 国产精品视区| 亚洲精品福利电影| 亚洲三级网站| 蜜桃成人av| 国产91在线播放精品| 日本va欧美va欧美va精品| 91精品啪在线观看国产18| 久久久亚洲欧洲日产| 日本va欧美va瓶| 在线一区电影| 啪啪国产精品| 国产va免费精品观看精品视频| 奇米狠狠一区二区三区| 亚洲综合丁香| 激情五月综合| 日韩精品一卡| 精品久久免费| 国产欧美日韩免费观看| 亚洲精品国产日韩| 美女国产精品| 在线一区免费| 婷婷中文字幕一区| 99精品电影| 美女福利一区二区三区| 福利一区二区三区视频在线观看| 国产日产精品_国产精品毛片| 亚洲色图网站| 亚洲aa在线| 亚洲精品女人| 日韩在线观看中文字幕| 亚洲精品一区二区在线播放∴| 黄色日韩精品| 黄页网站一区| 亚洲一区二区动漫| 国产亚洲一区在线| 亚洲一区区二区| 蜜桃视频一区二区| 日韩av影院| 国产精品国码视频| 美女精品一区二区| 国产在线观看91一区二区三区| 精品一区不卡| 亚洲国产成人二区| 欧美日韩一二| 亚洲一区国产| 日本伊人久久| 欧美天堂一区| 国产精品三p一区二区| 久久精品三级| 久久激情网站| 亚洲黄色在线| 视频一区视频二区在线观看| 综合一区av| 国产精品一区二区美女视频免费看| 国产精品欧美在线观看| 高清一区二区| 欧美日韩中文字幕一区二区三区| 成人av二区| 三级欧美在线一区| 日韩av一二三| 色一区二区三区四区| 日本免费久久| 亚洲福利国产| 亚州精品视频| 国际精品欧美精品| 免费视频一区三区| 亚洲精品日韩久久| 精品国产网站| 欧美精品自拍| 欧美精品福利| 日本不卡免费高清视频在线| 婷婷亚洲五月| 欧美日一区二区三区在线观看国产免 | 午夜国产一区二区| 在线观看亚洲精品福利片| 国产乱人伦丫前精品视频| 国产66精品| 爽好多水快深点欧美视频| 国产精品久久久久久妇女 | 日韩欧美高清一区二区三区| 欧美交a欧美精品喷水| 韩国久久久久久| 日韩中文字幕av电影| 欧美黑人做爰爽爽爽| 婷婷综合亚洲| 国产九一精品| 99tv成人| 国产精品亚洲产品| 亚洲欧洲另类| 国语对白精品一区二区| 免费成人在线观看| 天堂√中文最新版在线| 中文字幕av亚洲精品一部二部| 成人午夜在线| 亚洲免费成人av在线| 欧洲精品一区二区三区| 亚洲最新av| av一区在线| 国产精品白浆| 亚洲欧美日韩视频二区| 超级白嫩亚洲国产第一| 日韩三区四区| 一区三区视频| 黄色欧美在线| 欧美亚洲三区| 视频一区二区三区入口| 国产美女高潮在线| 久久精品99国产国产精| av不卡免费看| 91日韩欧美| 国产日本久久| 黄色精品网站| 水蜜桃精品av一区二区| 清纯唯美亚洲综合一区| 亚洲一区二区三区高清| 欧美成人基地| 久久精品五月| 久久国产尿小便嘘嘘| 99热精品在线观看| 999国产精品| 国产成人免费| 国产精品色在线网站| 亚洲免费福利一区| 夜夜嗨av一区二区三区网站四季av| 免费视频一区二区三区在线观看| 亚洲影院天堂中文av色| 99精品在线观看| 一区二区三区四区日本视频| 国产精品亚洲产品| 欧美一区网站| 蜜臀va亚洲va欧美va天堂| 免费av一区| 99久久久久国产精品| 中文在线а√天堂| 福利片在线一区二区 | 国产精品久久777777毛茸茸| 中文字幕乱码亚洲无线精品一区| 91九色精品| 极品日韩av| se01亚洲视频| 久久久久国产一区二区| 亚洲1234区| 久久婷婷av| 亚洲成人av观看|