123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639 |
- package excelize
- import (
- "bytes"
- "encoding/json"
- "encoding/xml"
- "fmt"
- "image"
- "io"
- "io/ioutil"
- "os"
- "path"
- "path/filepath"
- "strconv"
- "strings"
- )
- func parseFormatPictureSet(formatSet string) (*formatPicture, error) {
- format := formatPicture{
- FPrintsWithSheet: true,
- FLocksWithSheet: false,
- NoChangeAspect: false,
- Autofit: false,
- OffsetX: 0,
- OffsetY: 0,
- XScale: 1.0,
- YScale: 1.0,
- }
- err := json.Unmarshal(parseFormatSet(formatSet), &format)
- return &format, err
- }
- func (f *File) AddPicture(sheet, cell, picture, format string) error {
- var err error
-
- if _, err = os.Stat(picture); os.IsNotExist(err) {
- return err
- }
- ext, ok := supportImageTypes[path.Ext(picture)]
- if !ok {
- return ErrImgExt
- }
- file, _ := ioutil.ReadFile(picture)
- _, name := filepath.Split(picture)
- return f.AddPictureFromBytes(sheet, cell, format, name, ext, file)
- }
- func (f *File) AddPictureFromBytes(sheet, cell, format, name, extension string, file []byte) error {
- var drawingHyperlinkRID int
- var hyperlinkType string
- ext, ok := supportImageTypes[extension]
- if !ok {
- return ErrImgExt
- }
- formatSet, err := parseFormatPictureSet(format)
- if err != nil {
- return err
- }
- img, _, err := image.DecodeConfig(bytes.NewReader(file))
- if err != nil {
- return err
- }
-
- ws, err := f.workSheetReader(sheet)
- if err != nil {
- return err
- }
-
- drawingID := f.countDrawings() + 1
- drawingXML := "xl/drawings/drawing" + strconv.Itoa(drawingID) + ".xml"
- drawingID, drawingXML = f.prepareDrawing(ws, drawingID, sheet, drawingXML)
- drawingRels := "xl/drawings/_rels/drawing" + strconv.Itoa(drawingID) + ".xml.rels"
- mediaStr := ".." + strings.TrimPrefix(f.addMedia(file, ext), "xl")
- drawingRID := f.addRels(drawingRels, SourceRelationshipImage, mediaStr, hyperlinkType)
-
- if formatSet.Hyperlink != "" && formatSet.HyperlinkType != "" {
- if formatSet.HyperlinkType == "External" {
- hyperlinkType = formatSet.HyperlinkType
- }
- drawingHyperlinkRID = f.addRels(drawingRels, SourceRelationshipHyperLink, formatSet.Hyperlink, hyperlinkType)
- }
- err = f.addDrawingPicture(sheet, drawingXML, cell, name, img.Width, img.Height, drawingRID, drawingHyperlinkRID, formatSet)
- if err != nil {
- return err
- }
- f.addContentTypePart(drawingID, "drawings")
- f.addSheetNameSpace(sheet, SourceRelationship)
- return err
- }
- func (f *File) deleteSheetRelationships(sheet, rID string) {
- name, ok := f.sheetMap[trimSheetName(sheet)]
- if !ok {
- name = strings.ToLower(sheet) + ".xml"
- }
- var rels = "xl/worksheets/_rels/" + strings.TrimPrefix(name, "xl/worksheets/") + ".rels"
- sheetRels := f.relsReader(rels)
- if sheetRels == nil {
- sheetRels = &xlsxRelationships{}
- }
- for k, v := range sheetRels.Relationships {
- if v.ID == rID {
- sheetRels.Relationships = append(sheetRels.Relationships[:k], sheetRels.Relationships[k+1:]...)
- }
- }
- f.Relationships[rels] = sheetRels
- }
- func (f *File) addSheetLegacyDrawing(sheet string, rID int) {
- xlsx, _ := f.workSheetReader(sheet)
- xlsx.LegacyDrawing = &xlsxLegacyDrawing{
- RID: "rId" + strconv.Itoa(rID),
- }
- }
- func (f *File) addSheetDrawing(sheet string, rID int) {
- xlsx, _ := f.workSheetReader(sheet)
- xlsx.Drawing = &xlsxDrawing{
- RID: "rId" + strconv.Itoa(rID),
- }
- }
- func (f *File) addSheetPicture(sheet string, rID int) {
- xlsx, _ := f.workSheetReader(sheet)
- xlsx.Picture = &xlsxPicture{
- RID: "rId" + strconv.Itoa(rID),
- }
- }
- func (f *File) countDrawings() int {
- c1, c2 := 0, 0
- for k := range f.XLSX {
- if strings.Contains(k, "xl/drawings/drawing") {
- c1++
- }
- }
- for rel := range f.Drawings {
- if strings.Contains(rel, "xl/drawings/drawing") {
- c2++
- }
- }
- if c1 < c2 {
- return c2
- }
- return c1
- }
- func (f *File) addDrawingPicture(sheet, drawingXML, cell, file string, width, height, rID, hyperlinkRID int, formatSet *formatPicture) error {
- col, row, err := CellNameToCoordinates(cell)
- if err != nil {
- return err
- }
- if formatSet.Autofit {
- width, height, col, row, err = f.drawingResize(sheet, cell, float64(width), float64(height), formatSet)
- if err != nil {
- return err
- }
- } else {
- width = int(float64(width) * formatSet.XScale)
- height = int(float64(height) * formatSet.YScale)
- }
- col--
- row--
- colStart, rowStart, colEnd, rowEnd, x2, y2 :=
- f.positionObjectPixels(sheet, col, row, formatSet.OffsetX, formatSet.OffsetY, width, height)
- content, cNvPrID := f.drawingParser(drawingXML)
- twoCellAnchor := xdrCellAnchor{}
- twoCellAnchor.EditAs = formatSet.Positioning
- from := xlsxFrom{}
- from.Col = colStart
- from.ColOff = formatSet.OffsetX * EMU
- from.Row = rowStart
- from.RowOff = formatSet.OffsetY * EMU
- to := xlsxTo{}
- to.Col = colEnd
- to.ColOff = x2 * EMU
- to.Row = rowEnd
- to.RowOff = y2 * EMU
- twoCellAnchor.From = &from
- twoCellAnchor.To = &to
- pic := xlsxPic{}
- pic.NvPicPr.CNvPicPr.PicLocks.NoChangeAspect = formatSet.NoChangeAspect
- pic.NvPicPr.CNvPr.ID = cNvPrID
- pic.NvPicPr.CNvPr.Descr = file
- pic.NvPicPr.CNvPr.Name = "Picture " + strconv.Itoa(cNvPrID)
- if hyperlinkRID != 0 {
- pic.NvPicPr.CNvPr.HlinkClick = &xlsxHlinkClick{
- R: SourceRelationship.Value,
- RID: "rId" + strconv.Itoa(hyperlinkRID),
- }
- }
- pic.BlipFill.Blip.R = SourceRelationship.Value
- pic.BlipFill.Blip.Embed = "rId" + strconv.Itoa(rID)
- pic.SpPr.PrstGeom.Prst = "rect"
- twoCellAnchor.Pic = &pic
- twoCellAnchor.ClientData = &xdrClientData{
- FLocksWithSheet: formatSet.FLocksWithSheet,
- FPrintsWithSheet: formatSet.FPrintsWithSheet,
- }
- content.TwoCellAnchor = append(content.TwoCellAnchor, &twoCellAnchor)
- f.Drawings[drawingXML] = content
- return err
- }
- func (f *File) countMedia() int {
- count := 0
- for k := range f.XLSX {
- if strings.Contains(k, "xl/media/image") {
- count++
- }
- }
- return count
- }
- func (f *File) addMedia(file []byte, ext string) string {
- count := f.countMedia()
- for name, existing := range f.XLSX {
- if !strings.HasPrefix(name, "xl/media/image") {
- continue
- }
- if bytes.Equal(file, existing) {
- return name
- }
- }
- media := "xl/media/image" + strconv.Itoa(count+1) + ext
- f.XLSX[media] = file
- return media
- }
- func (f *File) setContentTypePartImageExtensions() {
- var imageTypes = map[string]bool{"jpeg": false, "png": false, "gif": false, "tiff": false}
- content := f.contentTypesReader()
- for _, v := range content.Defaults {
- _, ok := imageTypes[v.Extension]
- if ok {
- imageTypes[v.Extension] = true
- }
- }
- for k, v := range imageTypes {
- if !v {
- content.Defaults = append(content.Defaults, xlsxDefault{
- Extension: k,
- ContentType: "image/" + k,
- })
- }
- }
- }
- func (f *File) setContentTypePartVMLExtensions() {
- vml := false
- content := f.contentTypesReader()
- for _, v := range content.Defaults {
- if v.Extension == "vml" {
- vml = true
- }
- }
- if !vml {
- content.Defaults = append(content.Defaults, xlsxDefault{
- Extension: "vml",
- ContentType: ContentTypeVML,
- })
- }
- }
- func (f *File) addContentTypePart(index int, contentType string) {
- setContentType := map[string]func(){
- "comments": f.setContentTypePartVMLExtensions,
- "drawings": f.setContentTypePartImageExtensions,
- }
- partNames := map[string]string{
- "chart": "/xl/charts/chart" + strconv.Itoa(index) + ".xml",
- "chartsheet": "/xl/chartsheets/sheet" + strconv.Itoa(index) + ".xml",
- "comments": "/xl/comments" + strconv.Itoa(index) + ".xml",
- "drawings": "/xl/drawings/drawing" + strconv.Itoa(index) + ".xml",
- "table": "/xl/tables/table" + strconv.Itoa(index) + ".xml",
- "pivotTable": "/xl/pivotTables/pivotTable" + strconv.Itoa(index) + ".xml",
- "pivotCache": "/xl/pivotCache/pivotCacheDefinition" + strconv.Itoa(index) + ".xml",
- "sharedStrings": "/xl/sharedStrings.xml",
- }
- contentTypes := map[string]string{
- "chart": ContentTypeDrawingML,
- "chartsheet": ContentTypeSpreadSheetMLChartsheet,
- "comments": ContentTypeSpreadSheetMLComments,
- "drawings": ContentTypeDrawing,
- "table": ContentTypeSpreadSheetMLTable,
- "pivotTable": ContentTypeSpreadSheetMLPivotTable,
- "pivotCache": ContentTypeSpreadSheetMLPivotCacheDefinition,
- "sharedStrings": ContentTypeSpreadSheetMLSharedStrings,
- }
- s, ok := setContentType[contentType]
- if ok {
- s()
- }
- content := f.contentTypesReader()
- for _, v := range content.Overrides {
- if v.PartName == partNames[contentType] {
- return
- }
- }
- content.Overrides = append(content.Overrides, xlsxOverride{
- PartName: partNames[contentType],
- ContentType: contentTypes[contentType],
- })
- }
- func (f *File) getSheetRelationshipsTargetByID(sheet, rID string) string {
- name, ok := f.sheetMap[trimSheetName(sheet)]
- if !ok {
- name = strings.ToLower(sheet) + ".xml"
- }
- var rels = "xl/worksheets/_rels/" + strings.TrimPrefix(name, "xl/worksheets/") + ".rels"
- sheetRels := f.relsReader(rels)
- if sheetRels == nil {
- sheetRels = &xlsxRelationships{}
- }
- for _, v := range sheetRels.Relationships {
- if v.ID == rID {
- return v.Target
- }
- }
- return ""
- }
- func (f *File) GetPicture(sheet, cell string) (string, []byte, error) {
- col, row, err := CellNameToCoordinates(cell)
- if err != nil {
- return "", nil, err
- }
- col--
- row--
- ws, err := f.workSheetReader(sheet)
- if err != nil {
- return "", nil, err
- }
- if ws.Drawing == nil {
- return "", nil, err
- }
- target := f.getSheetRelationshipsTargetByID(sheet, ws.Drawing.RID)
- drawingXML := strings.Replace(target, "..", "xl", -1)
- _, ok := f.XLSX[drawingXML]
- if !ok {
- return "", nil, err
- }
- drawingRelationships := strings.Replace(
- strings.Replace(target, "../drawings", "xl/drawings/_rels", -1), ".xml", ".xml.rels", -1)
- return f.getPicture(row, col, drawingXML, drawingRelationships)
- }
- func (f *File) DeletePicture(sheet, cell string) (err error) {
- col, row, err := CellNameToCoordinates(cell)
- if err != nil {
- return
- }
- col--
- row--
- ws, err := f.workSheetReader(sheet)
- if err != nil {
- return
- }
- if ws.Drawing == nil {
- return
- }
- drawingXML := strings.Replace(f.getSheetRelationshipsTargetByID(sheet, ws.Drawing.RID), "..", "xl", -1)
- return f.deleteDrawing(col, row, drawingXML, "Pic")
- }
- func (f *File) getPicture(row, col int, drawingXML, drawingRelationships string) (ret string, buf []byte, err error) {
- var (
- wsDr *xlsxWsDr
- ok bool
- deWsDr *decodeWsDr
- drawRel *xlsxRelationship
- deTwoCellAnchor *decodeTwoCellAnchor
- )
- wsDr, _ = f.drawingParser(drawingXML)
- if ret, buf = f.getPictureFromWsDr(row, col, drawingRelationships, wsDr); len(buf) > 0 {
- return
- }
- deWsDr = new(decodeWsDr)
- if err = f.xmlNewDecoder(bytes.NewReader(namespaceStrictToTransitional(f.readXML(drawingXML)))).
- Decode(deWsDr); err != nil && err != io.EOF {
- err = fmt.Errorf("xml decode error: %s", err)
- return
- }
- err = nil
- for _, anchor := range deWsDr.TwoCellAnchor {
- deTwoCellAnchor = new(decodeTwoCellAnchor)
- if err = f.xmlNewDecoder(strings.NewReader("<decodeTwoCellAnchor>" + anchor.Content + "</decodeTwoCellAnchor>")).
- Decode(deTwoCellAnchor); err != nil && err != io.EOF {
- err = fmt.Errorf("xml decode error: %s", err)
- return
- }
- if err = nil; deTwoCellAnchor.From != nil && deTwoCellAnchor.Pic != nil {
- if deTwoCellAnchor.From.Col == col && deTwoCellAnchor.From.Row == row {
- drawRel = f.getDrawingRelationships(drawingRelationships, deTwoCellAnchor.Pic.BlipFill.Blip.Embed)
- if _, ok = supportImageTypes[filepath.Ext(drawRel.Target)]; ok {
- ret, buf = filepath.Base(drawRel.Target), f.XLSX[strings.Replace(drawRel.Target, "..", "xl", -1)]
- return
- }
- }
- }
- }
- return
- }
- func (f *File) getPictureFromWsDr(row, col int, drawingRelationships string, wsDr *xlsxWsDr) (ret string, buf []byte) {
- var (
- ok bool
- anchor *xdrCellAnchor
- drawRel *xlsxRelationship
- )
- for _, anchor = range wsDr.TwoCellAnchor {
- if anchor.From != nil && anchor.Pic != nil {
- if anchor.From.Col == col && anchor.From.Row == row {
- if drawRel = f.getDrawingRelationships(drawingRelationships,
- anchor.Pic.BlipFill.Blip.Embed); drawRel != nil {
- if _, ok = supportImageTypes[filepath.Ext(drawRel.Target)]; ok {
- ret, buf = filepath.Base(drawRel.Target), f.XLSX[strings.Replace(drawRel.Target, "..", "xl", -1)]
- return
- }
- }
- }
- }
- }
- return
- }
- func (f *File) getDrawingRelationships(rels, rID string) *xlsxRelationship {
- if drawingRels := f.relsReader(rels); drawingRels != nil {
- for _, v := range drawingRels.Relationships {
- if v.ID == rID {
- return &v
- }
- }
- }
- return nil
- }
- func (f *File) drawingsWriter() {
- for path, d := range f.Drawings {
- if d != nil {
- v, _ := xml.Marshal(d)
- f.saveFileList(path, v)
- }
- }
- }
- func (f *File) drawingResize(sheet string, cell string, width, height float64, formatSet *formatPicture) (w, h, c, r int, err error) {
- var mergeCells []MergeCell
- mergeCells, err = f.GetMergeCells(sheet)
- if err != nil {
- return
- }
- var rng []int
- var inMergeCell bool
- if c, r, err = CellNameToCoordinates(cell); err != nil {
- return
- }
- cellWidth, cellHeight := f.getColWidth(sheet, c), f.getRowHeight(sheet, r)
- for _, mergeCell := range mergeCells {
- if inMergeCell {
- continue
- }
- if inMergeCell, err = f.checkCellInArea(cell, mergeCell[0]); err != nil {
- return
- }
- if inMergeCell {
- rng, _ = areaRangeToCoordinates(mergeCell.GetStartAxis(), mergeCell.GetEndAxis())
- _ = sortCoordinates(rng)
- }
- }
- if inMergeCell {
- cellWidth, cellHeight = 0, 0
- c, r = rng[0], rng[1]
- for col := rng[0] - 1; col < rng[2]; col++ {
- cellWidth += f.getColWidth(sheet, col)
- }
- for row := rng[1] - 1; row < rng[3]; row++ {
- cellHeight += f.getRowHeight(sheet, row)
- }
- }
- if float64(cellWidth) < width {
- asp := float64(cellWidth) / width
- width, height = float64(cellWidth), height*asp
- }
- if float64(cellHeight) < height {
- asp := float64(cellHeight) / height
- height, width = float64(cellHeight), width*asp
- }
- width, height = width-float64(formatSet.OffsetX), height-float64(formatSet.OffsetY)
- w, h = int(width*formatSet.XScale), int(height*formatSet.YScale)
- return
- }
|