婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > 詳解用 python-docx 創(chuàng)建浮動圖片

詳解用 python-docx 創(chuàng)建浮動圖片

熱門標(biāo)簽:萬利達(dá)百貨商場地圖標(biāo)注 惠州龍門400電話要怎么申請 上海企業(yè)外呼系統(tǒng) 外呼電信系統(tǒng) 熱門電銷機(jī)器人 河南虛擬外呼系統(tǒng)公司 電話機(jī)器人哪里有賣 okcc外呼系統(tǒng)怎么調(diào)速度 智能機(jī)器人電銷神器

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

問題提出

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

Word中的設(shè)置

我們先嘗試在Office Word中,手動解決上述問題。具備基礎(chǔ)的Word使用經(jīng)驗(yàn)即可知,通過設(shè)置圖片版式來控制圖片的浮動和具體位置。

上圖版式設(shè)置中的文本環(huán)繞樣式,大體可以分為三類:

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

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

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

綜上,我們需要實(shí)現(xiàn)精確定位襯于文本下方的圖片版式。

docx背后的xml

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

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

內(nèi)聯(lián)圖片片段:

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>

對比發(fā)現(xiàn)以下相同/相似點(diǎn):

  • 兩類圖片都放在w:drawing>節(jié)點(diǎn)下:內(nèi)聯(lián)圖片wp:inline>,浮動圖片wp:anchor>
  • 具備相同的內(nèi)容節(jié)點(diǎn):wp:extent>wp:docPr>a:graphic>

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

wp:anchor>節(jié)點(diǎn)的behindDoc屬性表明圖片版式為襯于文本下方

wp:positionH>wp:positionV>節(jié)點(diǎn)表明水平和豎直絕對定位方式,其中:

  • relativeFrom屬性指定用于定位的參考對象
  • 子節(jié)點(diǎn)wp:posOffset>指定具體坐標(biāo)值

從內(nèi)聯(lián)圖片開始

從xml的結(jié)構(gòu)對比來看,我們完全可以根據(jù)python-docx對內(nèi)聯(lián)圖片的實(shí)現(xiàn)來插入浮動圖片。于是,從插入內(nèi)聯(lián)圖片的代碼入手:

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

python-docx安裝文件夾site-packages/docx進(jìn)行內(nèi)容搜索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)

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

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)

于是進(jìn)入CT_Inline類(限于篇幅,刪除了前兩個(gè)類方法newnew_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類的三個(gè)方法,即可將它們聯(lián)系上:

  • _inline_xml()方法給出內(nèi)聯(lián)圖片wp:inline>的xml結(jié)構(gòu)。
  • new()方法調(diào)用_inline_xml(),并為其中的子節(jié)點(diǎn)例如wp:extent>wp:docPr>賦值。
  • new_pic_inline()調(diào)用new(),同時(shí)拼接CT_Picture類的結(jié)果(節(jié)點(diǎn)pic:pic>,即圖片的具體內(nèi)容)到a:graphicData>節(jié)點(diǎn)中去。

綜上,實(shí)現(xiàn)了內(nèi)聯(lián)圖片的完整xml結(jié)構(gòu)。

插入浮動圖片

從xml結(jié)構(gòu)的對比及上述python-docx對內(nèi)聯(lián)圖片的實(shí)現(xiàn),得到創(chuàng)建浮動圖片的思路:

  • 初始化wp:anchor>結(jié)構(gòu),例如behindDoc="1"指定圖片版式為襯于文本下方
  • 使用類似的代碼填充wp:anchor>元素,尤其是wp:extent>wp:docPr>pic:pic>
  • 填充wp:positionH>wp:positionV>精確定位圖片

具體實(shí)踐中發(fā)現(xiàn)還有關(guān)鍵的一步——注冊xml標(biāo)簽名稱到對應(yīng)的類,例如wp:inline>CT_Inline

# docx.oxml.__init__.py
register_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 wrapping
styles and position modes per `CT_Anchor._anchor_xml()`.
'''
 
from docx.oxml import parse_xml, register_element_cls
from docx.oxml.ns import nsdecls
from docx.oxml.shape import CT_Picture
from docx.oxml.xmlchemy import BaseOxmlElement, OneAndOnlyOne
 
# refer to docx.oxml.shape.CT_Inline
class 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 passed
        as parameters.
        """
        anchor = parse_xml(cls._anchor_xml(pos_x, pos_y))
        anchor.extent.cx = cx
        anchor.extent.cy = cy
        anchor.docPr.id = shape_id
        anchor.docPr.name = 'Picture %d' % shape_id
        anchor.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` element
        specified by the argument values.
        """
        pic_id = 0  # Word doesn't seem to use this, but does not omit it
        pic = 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_inline
def 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.filename    
    return CT_Anchor.new_pic_anchor(shape_id, rId, filename, cx, cy, pos_x, pos_y)
# refer to docx.text.run.add_picture
def 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__.py
register_element_cls('wp:anchor', CT_Anchor)

示例

最后,來一個(gè)例子看看結(jié)果吧:

from docx import Document
from docx.shared import Inches, Pt
from 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')

作者:crazyhat,Python及科學(xué)計(jì)算愛好者

到此這篇關(guān)于詳解用 python-docx 創(chuàng)建浮動圖片的文章就介紹到這了,更多相關(guān)python-docx 浮動圖片內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • 安裝python-docx后,無法在pycharm中導(dǎo)入的解決方案
  • Python安裝docx依賴包教程
  • Python-docx 實(shí)現(xiàn)整體修改或者部分修改文字的大小和字體類型
  • 使用Python docx修改word關(guān)鍵詞顏色的操作
  • Python中docx2txt庫的使用說明
  • python docx的超鏈接網(wǎng)址和鏈接文本操作
  • python 實(shí)現(xiàn)docx與doc文件的互相轉(zhuǎn)換

標(biāo)簽:淮安 周口 綿陽 綏化 秦皇島 合肥 周口 百色

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《詳解用 python-docx 創(chuàng)建浮動圖片》,本文關(guān)鍵詞  詳解,用,python-docx,創(chuàng)建,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《詳解用 python-docx 創(chuàng)建浮動圖片》相關(guān)的同類信息!
  • 本頁收集關(guān)于詳解用 python-docx 創(chuàng)建浮動圖片的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    五月天欧美精品| 99re热视频这里只精品| 国产伦理精品不卡| 国产拍揄自揄精品视频麻豆| 国产激情视频一区二区三区欧美| 国产精品美女久久久久aⅴ| 91在线观看视频| 日韩激情一二三区| 精品国产乱码久久久久久蜜臀| 日韩精品亚洲一区二区三区免费| 精品久久久久一区| 亚洲va国产天堂va久久en| 日韩欧美中文字幕精品| 天天综合网天天综合色| 欧美一区二区三区日韩| 91免费国产视频网站| 日韩精品欧美精品| 国产三级一区二区三区| 欧洲av在线精品| www.亚洲人| 日韩精品电影在线观看| 国产视频不卡一区| 亚洲成人综合在线| 亚洲欧美偷拍另类a∨色屁股| 欧美喷潮久久久xxxxx| 日韩av成人高清| 色综合网站在线| 国产激情一区二区三区桃花岛亚洲| 有坂深雪av一区二区精品| 国产欧美一区视频| 国产三级精品三级在线专区| 欧美精品丝袜久久久中文字幕| 国产成人免费av在线| 国产精品69毛片高清亚洲| 亚洲高清免费一级二级三级| 最新热久久免费视频| 亚洲欧洲www| 久久色.com| 欧美一卡二卡三卡| 7799精品视频| 91麻豆精品91久久久久久清纯 | av在线不卡观看免费观看| 蜜臀av性久久久久蜜臀aⅴ四虎| 国产亚洲欧美日韩在线一区| 精品国产髙清在线看国产毛片 | 色婷婷久久久久swag精品| 午夜av一区二区| 亚洲第一福利一区| 一区二区三区在线免费视频| 精品久久久久久久久久久院品网| 久久久蜜桃精品| 国产欧美日韩另类视频免费观看| 日韩一级黄色片| 中文字幕国产精品一区二区| 欧美韩日一区二区三区| 国产欧美一区二区三区沐欲| 亚洲免费av高清| 亚洲综合另类小说| 日韩激情在线观看| 成人av网站在线观看免费| a在线欧美一区| 欧美午夜精品理论片a级按摩| 久久免费视频一区| 国产精品无人区| 亚洲一区二区美女| 国产不卡免费视频| 97精品视频在线观看自产线路二| 91国内精品野花午夜精品| 欧美成人a视频| 久久久精品tv| 一区二区成人在线观看| 大桥未久av一区二区三区中文| 成人精品小蝌蚪| 欧美视频精品在线| 亚洲欧洲精品一区二区三区| 午夜激情一区二区三区| 捆绑变态av一区二区三区 | 岛国一区二区三区| 在线视频你懂得一区| 久久在线免费观看| 亚洲欧美激情小说另类| 午夜av一区二区三区| 99精品在线免费| 欧美精品少妇一区二区三区| 久久久久久久久岛国免费| 一区二区三区在线视频免费观看| 久久er99热精品一区二区| 国产福利电影一区二区三区| 日韩欧美综合一区| 中文字幕在线观看一区二区| 午夜精品一区二区三区免费视频| 国产成人在线影院| 欧美日韩精品一区视频| 国产色综合一区| 天天操天天干天天综合网| 国产综合色视频| 在线看国产一区二区| 精品久久久久久久久久久久久久久 | 日韩女优视频免费观看| 亚洲综合一区二区精品导航| 国产成人av电影在线| 欧美日韩aaa| 亚洲精品综合在线| 精品在线一区二区| 欧美日本在线观看| 爽爽淫人综合网网站| 91免费看片在线观看| 久久久久久久久免费| 国产一区二区日韩精品| 日韩欧美成人激情| 午夜在线成人av| 91精品久久久久久久99蜜桃 | 亚洲精品欧美二区三区中文字幕| 国产九色精品成人porny| 国产日韩欧美不卡在线| 韩国成人在线视频| 欧美一级理论片| 日本欧美一区二区| 欧美在线看片a免费观看| 国产人久久人人人人爽| 99精品在线观看视频| 久久久国产精品午夜一区ai换脸| 久久疯狂做爰流白浆xx| 日本一区二区免费在线观看视频| 免费在线看一区| 91精品在线观看入口| 国产成人综合在线播放| 精品精品国产高清a毛片牛牛| 视频一区二区欧美| 91精品国产麻豆| 精品在线免费观看| 久久夜色精品国产噜噜av| 久草精品在线观看| 日韩午夜在线观看视频| 麻豆精品在线观看| 欧美视频日韩视频| 国产91清纯白嫩初高中在线观看| 久久久精品欧美丰满| 视频一区中文字幕| 日本一区二区三区四区| 成人18视频日本| 亚洲一区二区综合| 久久久久88色偷偷免费| 波多野结衣中文字幕一区| 日韩vs国产vs欧美| 欧美精品一区二区三区视频| 欧美亚洲国产一区在线观看网站 | 午夜精品成人在线视频| 久久久精品黄色| 94色蜜桃网一区二区三区| 一区二区三区电影在线播| 久久久久久久综合日本| 欧美日韩一区三区四区| 久久99精品一区二区三区三区| 日韩午夜av电影| 欧美三级韩国三级日本三斤| 五月天中文字幕一区二区| 欧美在线三级电影| 成人h动漫精品一区二区| 亚洲欧美经典视频| 色欧美日韩亚洲| 国产不卡视频在线观看| 一本色道亚洲精品aⅴ| 麻豆91在线播放| 五月天中文字幕一区二区| 一区二区三区免费观看| 国产精品国产三级国产三级人妇| 日韩三级在线观看| 欧美日韩成人在线一区| 在线观看不卡一区| 在线一区二区观看| 99久久综合国产精品| 成人禁用看黄a在线| av电影在线观看完整版一区二区| 国产成人在线视频网站| 国产成人精品一区二区三区四区| 国产伦精品一区二区三区视频青涩| 日韩av一区二| 九色综合狠狠综合久久| 美腿丝袜亚洲色图| 青青草国产成人99久久| 日本欧美在线看| 久久99精品国产| 国产精品亚洲一区二区三区在线| 国产成人av一区二区三区在线观看| 国产一区二区伦理片| 国产v日产∨综合v精品视频| 成人av电影在线网| 91麻豆国产在线观看| 欧美午夜精品一区二区三区 | 久久综合一区二区| 中文乱码免费一区二区| 亚洲免费av高清| 日日摸夜夜添夜夜添国产精品| 久久精品久久综合| 国产成人精品亚洲777人妖| 99国产精品久久久久久久久久| 欧美综合欧美视频| 欧美大片日本大片免费观看| 亚洲国产成人私人影院tom|