mirror of
https://github.com/unidoc/unioffice.git
synced 2025-04-25 13:48:53 +08:00

* MATCH, IFS, MAXA, MINA * OFFSET fixed * ISBLANK, ISERR, ISERROR, ISEVEN ,ISFORMULA, ISNONTEXT, ISNUMBER, ISODD, ISTEXT * ISLEAPYEAR, ISLOGICAL, ISNA, ISREF * FIND, FINDB * SEARCH, SEARCHB * CONCAT, CONCATENATE * YEAR, YEARFRAC * CONCAT is fixed, now TRUE and FALSE are concatenated instead of 1 and 0 in case of boolean results * NOW, TODAY, TIME, TIMEVALUE * DATE * DATEDIF
164 lines
5.0 KiB
Go
164 lines
5.0 KiB
Go
// Copyright 2017 FoxyUtils ehf. All rights reserved.
|
|
//
|
|
// Use of this source code is governed by the terms of the Affero GNU General
|
|
// Public License version 3.0 as published by the Free Software Foundation and
|
|
// appearing in the file LICENSE included in the packaging of this file. A
|
|
// commercial license can be purchased on https://unidoc.io.
|
|
|
|
package zippkg
|
|
|
|
import (
|
|
"archive/zip"
|
|
"bytes"
|
|
"encoding/xml"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"sort"
|
|
"strings"
|
|
|
|
"github.com/unidoc/unioffice"
|
|
|
|
"github.com/unidoc/unioffice/algo"
|
|
"github.com/unidoc/unioffice/schema/soo/pkg/relationships"
|
|
)
|
|
|
|
// RelationsPathFor returns the relations path for a given filename.
|
|
func RelationsPathFor(path string) string {
|
|
sp := strings.Split(path, "/")
|
|
pathPortion := strings.Join(sp[0:len(sp)-1], "/")
|
|
filePortion := sp[len(sp)-1]
|
|
pathPortion += "/_rels/"
|
|
filePortion += ".rels"
|
|
return pathPortion + filePortion
|
|
}
|
|
|
|
// Decode unmarshals the content of a *zip.File as XML to a given destination.
|
|
func Decode(f *zip.File, dest interface{}) error {
|
|
rc, err := f.Open()
|
|
if err != nil {
|
|
return fmt.Errorf("error reading %s: %s", f.Name, err)
|
|
}
|
|
defer rc.Close()
|
|
dec := xml.NewDecoder(rc)
|
|
if err := dec.Decode(dest); err != nil {
|
|
return fmt.Errorf("error decoding %s: %s", f.Name, err)
|
|
}
|
|
|
|
// this ensures that relationship ID is increasing, which we apparently rely
|
|
// on....
|
|
if ds, ok := dest.(*relationships.Relationships); ok {
|
|
for i, r := range ds.Relationship {
|
|
switch r.TypeAttr {
|
|
// Common
|
|
case unioffice.OfficeDocumentTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.OfficeDocumentType
|
|
case unioffice.StylesTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.StylesType
|
|
case unioffice.ThemeTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.ThemeType
|
|
case unioffice.SettingsTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.SettingsType
|
|
case unioffice.ImageTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.ImageType
|
|
case unioffice.CommentsTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.CommentsType
|
|
case unioffice.ThumbnailTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.ThumbnailType
|
|
case unioffice.DrawingTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.DrawingType
|
|
case unioffice.ChartTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.ChartType
|
|
case unioffice.ExtendedPropertiesTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.ExtendedPropertiesType
|
|
case unioffice.CustomXMLTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.CustomXMLType
|
|
|
|
// SML
|
|
case unioffice.WorksheetTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.WorksheetType
|
|
case unioffice.SharedStingsTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.SharedStingsType
|
|
case unioffice.TableTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.TableType
|
|
|
|
// WML
|
|
case unioffice.HeaderTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.HeaderType
|
|
case unioffice.FooterTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.FooterType
|
|
case unioffice.NumberingTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.NumberingType
|
|
case unioffice.FontTableTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.FontTableType
|
|
case unioffice.WebSettingsTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.WebSettingsType
|
|
case unioffice.FootNotesTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.FootNotesType
|
|
case unioffice.EndNotesTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.EndNotesType
|
|
|
|
// PML
|
|
case unioffice.SlideTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.SlideType
|
|
|
|
// VML
|
|
case unioffice.VMLDrawingTypeStrict:
|
|
ds.Relationship[i].TypeAttr = unioffice.VMLDrawingType
|
|
}
|
|
}
|
|
|
|
sort.Slice(ds.Relationship, func(i, j int) bool {
|
|
lhs := ds.Relationship[i]
|
|
rhs := ds.Relationship[j]
|
|
return algo.NaturalLess(lhs.IdAttr, rhs.IdAttr)
|
|
})
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// AddFileFromDisk reads a file from disk and adds it at a given path to a zip file.
|
|
func AddFileFromDisk(z *zip.Writer, zipPath, diskPath string) error {
|
|
w, err := z.Create(zipPath)
|
|
if err != nil {
|
|
return fmt.Errorf("error creating %s: %s", zipPath, err)
|
|
}
|
|
f, err := os.Open(diskPath)
|
|
if err != nil {
|
|
return fmt.Errorf("error opening %s: %s", diskPath, err)
|
|
}
|
|
_, err = io.Copy(w, f)
|
|
return err
|
|
}
|
|
|
|
// AddFileFromBytes takes a byte array and adds it at a given path to a zip file.
|
|
func AddFileFromBytes(z *zip.Writer, zipPath string, data []byte) error {
|
|
w, err := z.Create(zipPath)
|
|
if err != nil {
|
|
return fmt.Errorf("error creating %s: %s", zipPath, err)
|
|
}
|
|
_, err = io.Copy(w, bytes.NewReader(data))
|
|
return err
|
|
}
|
|
|
|
// ExtractToDiskTmp extracts a zip file to a temporary file in a given path,
|
|
// returning the name of the file.
|
|
func ExtractToDiskTmp(f *zip.File, path string) (string, error) {
|
|
tmpFile, err := ioutil.TempFile(path, "zz")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
defer tmpFile.Close()
|
|
rc, err := f.Open()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
defer rc.Close()
|
|
_, err = io.Copy(tmpFile, rc)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return tmpFile.Name(), nil
|
|
}
|