Python-pptx: копировать слайд

Как я могу скопировать слайд?

Я создал шаблон слайда, и мне нужно скопировать его и редактировать формы каждой копии отдельно.

Или как я могу добавить свой слайд шаблона в presentation.slide_layouts?

7 ответов

Это то, что я нашел на GitHub, и это работает для меня. Я изменил пару вещей для своего проекта. Вам нужно будет импортировать шесть и скопировать. Я использую pptx-6.10

def duplicate_slide(pres,index):
        template = pres.slides[index]
        try:
            blank_slide_layout = pres.slide_layouts[12]
        except:
            blank_slide_layout = pres.slide_layouts[len(pres.slide_layouts)]

        copied_slide = pres.slides.add_slide(blank_slide_layout)

        for shp in template.shapes:
            el = shp.element
            newel = copy.deepcopy(el)
            copied_slide.shapes._spTree.insert_element_before(newel, 'p:extLst')

        for _, value in six.iteritems(template.part.rels):
            # Make sure we don't copy a notesSlide relation as that won't exist
            if "notesSlide" not in value.reltype:
                copied_slide.part.rels.add_relationship(value.reltype,
                                                value._target,
                                                value.rId)

        return copied_slide

Затем вы можете создать копию с передачей вашей презентации и указателем слайдов вашего шаблона:

copied_slide = duplicate_slide(pres, 4)

Я все еще работаю над редактированием фигур из скопированного слайда, и как только я продвинусь дальше в своем проекте, я могу обновить

Я хотел представить свой обходной путь для копирования слайдов. Я использую шаблон ppt и заполняю его. Я знаю, прежде чем заполнять слайды, какие слайды шаблона необходимо копировать и как часто. Затем я копирую слайды и сохраняю новый ppt с скопированными слайдами. После сохранения я могу открыть ppt с скопированными слайдами и использовать pptx для заполнения слайдов.

import win32com.client
ppt_instance = win32com.client.Dispatch('PowerPoint.Application')
#open the powerpoint presentation headless in background
read_only = True
has_title = False
window    = False
prs = ppt_instance.Presentations.open('path/ppt.pptx',read_only,has_title,window)

nr_slide = 1
insert_index = 1
prs.Slides(nr_slide).Copy()
prs.Slides.Paste(Index=insert_index)

prs.SaveAs('path/new_ppt.pptx')
prs.Close()

#kills ppt_instance
ppt_instance.Quit()
del ppt_instance

В этом случае первый слайд будет скопирован из презентации и вставлен после первого слайда той же презентации.

Надеюсь, это поможет некоторым из вас!

Так как я также нашел другой сценарий использования кода, предоставленного @d_bergeron, я просто хотел поделиться им здесь. В моем случае я хотел скопировать слайд из другой презентации в ту, которую я создал с помощью python-pptx:

В качестве аргумента я передаю объект Presentation(), который я создал с помощью python-pptx (prs = Presenation()).

from pptx import Presentation
import copy

def copy_slide_from_external_prs(prs):

    # copy from external presentation all objects into the existing presentation
    external_pres = Presentation("PATH/TO/PRES/TO/IMPORT/from.pptx")

    # specify the slide you want to copy the contents from
    ext_slide = external_pres.slides[0]

    # Define the layout you want to use from your generated pptx
    SLD_LAYOUT = 5
    slide_layout = prs.slide_layouts[SLD_LAYOUT]

    # create now slide, to copy contents to 
    curr_slide = prs.slides.add_slide(slide_layout)

    # now copy contents from external slide, but do not copy slide properties
    # e.g. slide layouts, etc., because these would produce errors, as diplicate
    # entries might be generated

    for shp in ext_slide.shapes:
        el = shp.element
        newel = copy.deepcopy(el)
        curr_slide.shapes._spTree.insert_element_before(newel, 'p:extLst')

    return prs

В основном я публикую его здесь, так как я искал способ скопировать внешний слайд в свою презентацию и оказался в этой теме.

Я отредактировал решение @n00by0815 и получил очень элегантный код, который также может копировать изображения без ошибок:

      # ATTENTNION: PPTX PACKAGE RUNS ONLY ON CERTAINS VERSION OF PYTHON (https://python-pptx.readthedocs.io/en/latest/user/install.html)

from pptx import Presentation
from pptx.util import Pt
from pptx.enum.text import PP_ALIGN
import copy
import os

DIR_PATH = os.path.dirname(os.path.realpath(__file__))

#modeled on https://stackoverflow.com/a/56074651/20159015 and https://stackoverflow.com/a/62921848/20159015
#this for some reason doesnt copy text properties (font size, alignment etc.)
def SlideCopyFromPasteInto(copyFromPres, slideIndex,  pasteIntoPres):

    # specify the slide you want to copy the contents from
    slide_to_copy = copyFromPres.slides[slideIndex]

    # Define the layout you want to use from your generated pptx

    slide_layout = pasteIntoPres.slide_layouts.get_by_name("Blank") # names of layouts can be found here under step 3: https://www.geeksforgeeks.org/how-to-change-slide-layout-in-ms-powerpoint/
    # it is important for slide_layout to be blank since you dont want these "Write your title here" or something like that textboxes
    # alternative: slide_layout = pasteIntoPres.slide_layouts[copyFromPres.slide_layouts.index(slide_to_copy.slide_layout)]
    
    # create now slide, to copy contents to 
    new_slide = pasteIntoPres.slides.add_slide(slide_layout)

    # create images dict
    imgDict = {}

    # now copy contents from external slide, but do not copy slide properties
    # e.g. slide layouts, etc., because these would produce errors, as diplicate
    # entries might be generated
    for shp in slide_to_copy.shapes:
        if 'Picture' in shp.name:
            # save image
            with open(shp.name+'.jpg', 'wb') as f:
                f.write(shp.image.blob)

            # add image to dict
            imgDict[shp.name+'.jpg'] = [shp.left, shp.top, shp.width, shp.height]
        else:
            # create copy of elem
            el = shp.element
            newel = copy.deepcopy(el)

            # add elem to shape tree
            new_slide.shapes._spTree.insert_element_before(newel, 'p:extLst')
    
    # things added first will be covered by things added last => since I want pictures to be in foreground, I will add them after others elements
    # you can change this if you want
    # add pictures
    for k, v in imgDict.items():
        new_slide.shapes.add_picture(k, v[0], v[1], v[2], v[3])
        os.remove(k)

    return new_slide # this returns slide so you can instantly work with it when it is pasted in presentation



templatePres = Presentation(f"{DIR_PATH}/template.pptx")

outputPres = Presentation() 
outputPres.slide_height, outputPres.slide_width = templatePres.slide_height, templatePres.slide_width
# this can sometimes cause problems. Alternative:
# outputPres = Presentation(f"{DIR_PATH}/template.pptx") and now delete all slides to have empty presentation

# if you just want to copy and paste slide:
SlideCopyFromPasteInto(templatePres,0,outputPres)

# if you want to edit slide that was just pasted in presentation:
pastedSlide = SlideCopyFromPasteInto(templatePres,0,outputPres)
pastedSlide.shapes.title.text = "My very cool title"

for shape in pastedSlide.shapes:

    if not(shape.has_text_frame): continue

    # easiest ways to edit text fields is to put some identifying text in them
    if shape.text_frame.text == "personName": # there is a text field with "personName" written into it
        shape.text_frame.text = "Brian"

    if shape.text_frame.text == "personSalary":
        shape.text_frame.text = str(brianSalary)

    # stylizing text need to be done after you change it
    shape.text_frame.paragraphs[0].font.size = Pt(80) 
    shape.text_frame.paragraphs[0].alignment = PP_ALIGN.CENTER

outputPres.save(f'{DIR_PATH}/output.pptx')

Я использовал ответ n00by0815, и он отлично работал, пока мне не пришлось копировать изображения. Вот моя адаптированная версия, которая обрабатывает изображения. Этот код создает локальную копию изображения, а затем добавляет ее на слайд. Я уверен, что есть способ чище, но он работает.

Вот еще один способ скопировать каждый слайд на один слайд PPTX для всей презентации, а затем вы можете использовать LibreOffice для преобразования каждой отдельной точки PowerPoint в изображение:

def get_slide_count(prs):
""" Get the number of slides in PPTX presentation """
    slidecount = 0
    for slide in prs.slides:
        slidecount += 1
    return slidecount


def delete_slide(prs, slide):
    """ Delete a slide out of a powerpoint presentation"""
    id_dict = { slide.id: [i, slide.rId] for i,slide in enumerate(prs.slides._sldIdLst) }
    slide_id = slide.slide_id
    prs.part.drop_rel(id_dict[slide_id][1])
    del prs.slides._sldIdLst[id_dict[slide_id][0]]


def get_single_slide_pres(prs, slidetokeep):
    for idx, slide in enumerate(prs.slides):
        if idx < slidetokeep:
            delete_slide(prs, slide)
        elif (idx > slidetokeep):
            delete_slide(prs, slide)
    prs.save(str(slidetokeep + 1) + ".pptx")


pptxfilepath = "test.pptx"
prs = Presentation(pptxfilepath)
slidecount = get_slide_count(prs)
for i in range(slidecount):
    prs_backup = Presentation(pptxfilepath)
    get_single_slide_pres(prs_backup, i)
    prs_backup = None

Извините за задержку, я был перемещен в другой проект. Я смог завершить свой проект ppt, используя несколько слайдов шаблонов и копируя их. В конце построения презентации я удаляю шаблоны. Чтобы захватить фигуры, вам нужно будет пройтись по слайду и найти название фигуры, которую вы ищете. Как только вы это вернули, вы можете отредактировать форму по мере необходимости. Я добавил версию функции add_text, которую я использую для заполнения shape.text_frame.

def find_shape_by_name(shapes,name):
        for shape in shapes:
            if shape.name == name:
                return shape
        return None

def add_text(shape,text,alignment=None):

    if alignment:
        shape.vertical_anchor = alignment

    tf = shape.text_frame
    tf.clear()
    run = tf.paragraphs[0].add_run()
    run.text = text if text else ''

Чтобы найти форму "слайд_титл"

slide_title = find_shape_by_name(slide.shapes,'slide_title')

Чтобы добавить текст в форму

add_text(slide_title,'TEST SLIDE')

Пожалуйста, дайте мне знать, если вам нужна другая помощь

Другие вопросы по тегам