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')
Пожалуйста, дайте мне знать, если вам нужна другая помощь