d2/lib/ppt/presentation.go

125 lines
2.6 KiB
Go
Raw Normal View History

2023-04-04 14:13:47 +00:00
package ppt
import (
"archive/zip"
"bytes"
_ "embed"
"fmt"
"image/png"
"os"
)
2023-04-04 21:00:45 +00:00
// TODO: comments / references / assumptions
// TODO: update core files with metadata
type Presentation struct {
2023-04-04 14:13:47 +00:00
Slides []*Slide
}
type Slide struct {
2023-04-04 21:47:22 +00:00
Title string
Image []byte
ImageWidth int
ImageHeight int
ImageTop int
ImageLeft int
2023-04-04 14:13:47 +00:00
}
2023-04-04 21:00:45 +00:00
func NewPresentation() *Presentation {
return &Presentation{}
2023-04-04 14:13:47 +00:00
}
2023-04-04 21:47:22 +00:00
func (p *Presentation) AddSlide(title string, pngContent []byte) error {
2023-04-04 14:13:47 +00:00
src, err := png.Decode(bytes.NewReader(pngContent))
if err != nil {
return fmt.Errorf("error decoding PNG image: %v", err)
}
2023-04-04 21:00:45 +00:00
var width, height, top, left int
2023-04-04 14:13:47 +00:00
srcSize := src.Bounds().Size()
2023-04-04 21:00:45 +00:00
// compute the size and position to fit the slide
if srcSize.X > srcSize.Y {
2023-04-04 21:47:22 +00:00
width = IMAGE_WIDTH
2023-04-04 21:00:45 +00:00
height = int(float64(width) * (float64(srcSize.X) / float64(srcSize.Y)))
left = 0
2023-04-04 21:47:22 +00:00
top = (IMAGE_HEIGHT - height) / 2
2023-04-04 21:00:45 +00:00
} else {
2023-04-04 21:47:22 +00:00
height = IMAGE_HEIGHT
2023-04-04 21:00:45 +00:00
width = int(float64(height) * (float64(srcSize.X) / float64(srcSize.Y)))
top = 0
2023-04-04 21:47:22 +00:00
left = (IMAGE_WIDTH - width) / 2
2023-04-04 21:00:45 +00:00
}
2023-04-04 14:13:47 +00:00
p.Slides = append(p.Slides, &Slide{
2023-04-04 21:47:22 +00:00
Title: title,
Image: pngContent,
ImageWidth: width,
ImageHeight: height,
ImageTop: top,
ImageLeft: left,
2023-04-04 14:13:47 +00:00
})
return nil
}
2023-04-04 21:00:45 +00:00
func (p *Presentation) SaveTo(filePath string) error {
2023-04-04 14:13:47 +00:00
f, err := os.Create(filePath)
if err != nil {
return err
}
defer f.Close()
zipFile := zip.NewWriter(f)
defer zipFile.Close()
copyPptxTemplateTo(zipFile)
var slideFileNames []string
for i, slide := range p.Slides {
imageId := fmt.Sprintf("slide%dImage", i+1)
slideFileName := fmt.Sprintf("slide%d", i+1)
slideFileNames = append(slideFileNames, slideFileName)
imageWriter, err := zipFile.Create(fmt.Sprintf("ppt/media/%s.png", imageId))
if err != nil {
return err
}
_, err = imageWriter.Write(slide.Image)
if err != nil {
return err
}
err = addFile(zipFile, fmt.Sprintf("ppt/slides/_rels/%s.xml.rels", slideFileName), getRelsSlideXml(imageId))
if err != nil {
return err
}
// TODO: center the image?
2023-04-04 21:47:22 +00:00
err = addFile(
zipFile,
fmt.Sprintf("ppt/slides/%s.xml", slideFileName),
getSlideXml(slide.Title, imageId, slide.ImageTop, slide.ImageLeft, slide.ImageWidth, slide.ImageHeight),
)
2023-04-04 14:13:47 +00:00
if err != nil {
return err
}
}
err = addFile(zipFile, "[Content_Types].xml", getContentTypesXml(slideFileNames))
if err != nil {
return err
}
err = addFile(zipFile, "ppt/_rels/presentation.xml.rels", getPresentationXmlRels(slideFileNames))
if err != nil {
return err
}
err = addFile(zipFile, "ppt/presentation.xml", getPresentationXml(slideFileNames))
if err != nil {
return err
}
return nil
}