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

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

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

瀏覽:20日期: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久久,三上悠亚国产精品一区二区三区
亚洲a级精品| 激情六月综合| 日韩美女精品| 亚洲一区二区免费看| 日韩高清不卡在线| 深夜福利视频一区二区| 亚洲自啪免费| 欧美日韩18| 国产精品99一区二区三| 91精品国产成人观看| 国产精品美女久久久浪潮软件| 日韩有吗在线观看| 国产精品不卡| 在线综合视频| 免费在线成人| 99成人超碰| 欧美日韩国产高清| 国产乱码精品一区二区三区四区 | 亚洲麻豆一区| 日本三级亚洲精品| 国产韩日影视精品| 久久精品国产99久久| 99国内精品| 久久中文欧美| 亚洲日韩视频| 国产乱码精品| 欧美三级精品| 99热精品久久| 日韩激情av在线| 福利在线免费视频| 国产综合精品| 三级久久三级久久久| 伊人久久婷婷| 亚洲欧美久久久| 国产亚洲毛片| 亚洲国产一区二区在线观看| 美女久久一区| 国产精东传媒成人av电影| 成人在线网站| 五月天激情综合网| 人人精品久久| 国产精品蜜芽在线观看| 日本强好片久久久久久aaa| 国内揄拍国内精品久久| 国产精品草草| 欧美一区网站| 免费观看在线综合| 国产精一区二区| 国产欧美午夜| 久久影视三级福利片| 国产精品毛片久久久| 国产日韩高清一区二区三区在线| 亚洲在线免费| 精品日韩毛片| 国产亚洲欧洲| 国产乱码精品| 国产亚洲一级| 欧美激情另类| 国产aⅴ精品一区二区三区久久| 欧美一级鲁丝片| 福利一区和二区| 欧美日一区二区| 四虎精品一区二区免费| 国产精品扒开腿做爽爽爽软件| 久久精选视频| 久久国产精品免费一区二区三区| 亚洲调教视频在线观看| 国产精品jk白丝蜜臀av小说| 欧美特黄视频| 国产成人精品一区二区三区免费| 99精品网站| 精品国产黄a∨片高清在线| 国产精品hd| 日韩亚洲一区在线| 亚洲精品第一| 久久精品一区| 99在线|亚洲一区二区| 欧美日韩亚洲一区二区三区在线 | 欧美91福利在线观看| 日韩国产在线观看一区| 久久九九精品| 久久久精品国产**网站| 亚洲欧洲日韩精品在线| 韩国久久久久久| 久久精品xxxxx| 国产精品日本| 久久九九电影| 中文一区一区三区高中清不卡免费| 欧美一区精品| 亚洲精品免费观看| 在线午夜精品| 久久久精品久久久久久96| 久久99免费视频| 日本一区二区三区中文字幕| av成人国产| 激情久久久久久久| 国产欧美一区二区三区精品酒店| 久久国产精品色av免费看| 日本在线高清| 成人美女视频| 国产精选一区| 宅男噜噜噜66国产日韩在线观看| 国产精品久久乐| 视频一区二区三区中文字幕| 欧美午夜精彩| 久久精品青草| 亚洲午夜天堂| 欧美激情在线精品一区二区三区| 97国产成人高清在线观看| 国产日产精品_国产精品毛片 | 黄色在线观看www| 国产亚洲午夜| 久久中文字幕一区二区三区| 91精品丝袜国产高跟在线| 婷婷亚洲精品| 欧美日韩精品一区二区三区视频| 亚洲精品高潮| 日韩三区四区| 国产视频一区免费看| 在线免费观看亚洲| 日本在线一区二区三区| 国产日韩欧美一区在线| 久久久免费人体| 国产乱人伦丫前精品视频| 激情综合婷婷| 欧美成人高清| 亚洲激情偷拍| av在线最新| 99久久亚洲精品| 免费中文字幕日韩欧美| 亚洲精品少妇| 麻豆视频观看网址久久| 99久久精品费精品国产| 日韩精品永久网址| 国产91在线播放精品| 精品视频免费| 国产白浆在线免费观看| 91视频久久| 99久久久久| а√天堂中文在线资源8| 美国欧美日韩国产在线播放| 91青青国产在线观看精品| 日韩精品视频一区二区三区| 一本综合精品| 亚州精品视频| 国产精品毛片久久久| www.com.cn成人| 日韩在线短视频| 美女久久网站| 日本国产欧美| 国产精品久久久久久久久免费高清| 国产精品sm| 精品72久久久久中文字幕| 欧美香蕉视频| 亚洲欧美一区在线| 久久亚洲欧洲| 天堂va在线高清一区| 亚洲精品福利| 国产精品xvideos88| 成人精品视频| 久久美女性网| 狠狠色狠狠色综合日日tαg| 午夜欧美精品| 欧美久久精品| 久久精品国产网站| 黑丝美女一区二区| 日韩三级精品| 精品女同一区二区三区在线观看| sm久久捆绑调教精品一区| 日韩午夜av| 日韩av不卡一区二区| 国产盗摄——sm在线视频| 日韩欧美一区二区三区免费观看| 国产精品嫩草99av在线| 欧美一级二区| 日韩一区精品字幕| 男人的天堂久久精品| 国产精品1区| 亚洲精品电影| 日韩成人av影视| 深夜福利一区| 国产精品1luya在线播放| 亚洲伊人影院| 日韩免费一区| 日韩av一二三| 在线一区欧美| 欧美 日韩 国产精品免费观看| 日韩一区二区三区在线看| 国产精品伦理久久久久久| 91大神在线观看线路一区| 亚洲激情婷婷| 亚洲91视频| 综合激情视频| 日韩专区欧美专区| 亚洲人成高清| 国产精品扒开腿做爽爽爽软件| 久久精品三级| 亚洲涩涩在线| 日韩高清在线观看一区二区| 你懂的国产精品永久在线|