2016-08-30 11:51:31 +08:00
|
|
|
package excelize
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/xml"
|
|
|
|
"fmt"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2016-09-02 11:54:52 +08:00
|
|
|
// NewSheet provice function to greate a new sheet by given index, when
|
|
|
|
// creating a new XLSX file, the default sheet will be create, when you
|
|
|
|
// create a new file, you need to ensure that the index is continuous.
|
2016-08-30 11:51:31 +08:00
|
|
|
func NewSheet(file []FileList, index int, name string) []FileList {
|
|
|
|
// Update docProps/app.xml
|
2016-09-02 11:54:52 +08:00
|
|
|
file = setAppXML(file)
|
2016-08-30 11:51:31 +08:00
|
|
|
// Update [Content_Types].xml
|
|
|
|
file = setContentTypes(file, index)
|
|
|
|
// Create new sheet /xl/worksheets/sheet%d.xml
|
|
|
|
file = setSheet(file, index)
|
|
|
|
// Update xl/_rels/workbook.xml.rels
|
|
|
|
file = addXlsxWorkbookRels(file, index)
|
|
|
|
// Update xl/workbook.xml
|
|
|
|
file = setWorkbook(file, index, name)
|
|
|
|
return file
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read and update property of contents type of XLSX
|
|
|
|
func setContentTypes(file []FileList, index int) []FileList {
|
|
|
|
var content xlsxTypes
|
2016-09-02 11:54:52 +08:00
|
|
|
xml.Unmarshal([]byte(readXML(file, `[Content_Types].xml`)), &content)
|
2016-08-30 11:51:31 +08:00
|
|
|
content.Overrides = append(content.Overrides, xlsxOverride{
|
2016-08-30 21:54:28 +08:00
|
|
|
PartName: `/xl/worksheets/sheet` + strconv.Itoa(index) + `.xml`,
|
2016-08-30 11:51:31 +08:00
|
|
|
ContentType: "application/vnd.openxmlformats-officedocument.spreadsheetml.worksheet+xml",
|
|
|
|
})
|
|
|
|
output, err := xml.MarshalIndent(content, "", "")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
return saveFileList(file, `[Content_Types].xml`, string(output))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update sheet property by given index
|
|
|
|
func setSheet(file []FileList, index int) []FileList {
|
|
|
|
var xlsx xlsxWorksheet
|
|
|
|
xlsx.Dimension.Ref = "A1"
|
|
|
|
xlsx.SheetViews.SheetView = append(xlsx.SheetViews.SheetView, xlsxSheetView{
|
2016-09-02 11:54:52 +08:00
|
|
|
WorkbookViewID: 0,
|
2016-08-30 11:51:31 +08:00
|
|
|
})
|
|
|
|
output, err := xml.MarshalIndent(xlsx, "", "")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
2016-08-30 21:54:28 +08:00
|
|
|
path := `xl/worksheets/sheet` + strconv.Itoa(index) + `.xml`
|
2016-08-30 11:51:31 +08:00
|
|
|
return saveFileList(file, path, replaceRelationshipsID(replaceWorkSheetsRelationshipsNameSpace(string(output))))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update workbook property of XLSX
|
|
|
|
func setWorkbook(file []FileList, index int, name string) []FileList {
|
|
|
|
var content xlsxWorkbook
|
2016-09-02 11:54:52 +08:00
|
|
|
xml.Unmarshal([]byte(readXML(file, `xl/workbook.xml`)), &content)
|
2016-08-30 11:51:31 +08:00
|
|
|
|
|
|
|
rels := readXlsxWorkbookRels(file)
|
2016-09-02 11:54:52 +08:00
|
|
|
rID := len(rels.Relationships)
|
2016-08-30 11:51:31 +08:00
|
|
|
content.Sheets.Sheet = append(content.Sheets.Sheet, xlsxSheet{
|
|
|
|
Name: name,
|
2016-09-02 11:54:52 +08:00
|
|
|
SheetID: strconv.Itoa(index),
|
|
|
|
ID: "rId" + strconv.Itoa(rID),
|
2016-08-30 11:51:31 +08:00
|
|
|
})
|
|
|
|
output, err := xml.MarshalIndent(content, "", "")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
return saveFileList(file, `xl/workbook.xml`, replaceRelationshipsNameSpace(string(output)))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read and unmarshal workbook relationships of XLSX
|
|
|
|
func readXlsxWorkbookRels(file []FileList) xlsxWorkbookRels {
|
|
|
|
var content xlsxWorkbookRels
|
2016-09-02 11:54:52 +08:00
|
|
|
xml.Unmarshal([]byte(readXML(file, `xl/_rels/workbook.xml.rels`)), &content)
|
2016-08-30 11:51:31 +08:00
|
|
|
return content
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update workbook relationships property of XLSX
|
|
|
|
func addXlsxWorkbookRels(file []FileList, sheet int) []FileList {
|
|
|
|
content := readXlsxWorkbookRels(file)
|
2016-09-02 11:54:52 +08:00
|
|
|
rID := len(content.Relationships) + 1
|
2016-08-30 11:51:31 +08:00
|
|
|
content.Relationships = append(content.Relationships, xlsxWorkbookRelation{
|
2016-09-02 11:54:52 +08:00
|
|
|
ID: "rId" + strconv.Itoa(rID),
|
2016-08-30 21:54:28 +08:00
|
|
|
Target: `worksheets/sheet` + strconv.Itoa(sheet) + `.xml`,
|
2016-08-30 11:51:31 +08:00
|
|
|
Type: "http://schemas.openxmlformats.org/officeDocument/2006/relationships/worksheet",
|
|
|
|
})
|
|
|
|
output, err := xml.MarshalIndent(content, "", "")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
return saveFileList(file, `xl/_rels/workbook.xml.rels`, string(output))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update docProps/app.xml file of XML
|
2016-09-02 11:54:52 +08:00
|
|
|
func setAppXML(file []FileList) []FileList {
|
|
|
|
return saveFileList(file, `docProps/app.xml`, templateDocpropsApp)
|
2016-08-30 11:51:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Some tools that read XLSX files have very strict requirements about
|
|
|
|
// the structure of the input XML. In particular both Numbers on the Mac
|
|
|
|
// and SAS dislike inline XML namespace declarations, or namespace
|
|
|
|
// prefixes that don't match the ones that Excel itself uses. This is a
|
|
|
|
// problem because the Go XML library doesn't multiple namespace
|
|
|
|
// declarations in a single element of a document. This function is a
|
|
|
|
// horrible hack to fix that after the XML marshalling is completed.
|
|
|
|
func replaceRelationshipsNameSpace(workbookMarshal string) string {
|
|
|
|
// newWorkbook := strings.Replace(workbookMarshal, `xmlns:relationships="http://schemas.openxmlformats.org/officeDocument/2006/relationships" relationships:id`, `r:id`, -1)
|
|
|
|
// Dirty hack to fix issues #63 and #91; encoding/xml currently
|
|
|
|
// "doesn't allow for additional namespaces to be defined in the
|
|
|
|
// root element of the document," as described by @tealeg in the
|
|
|
|
// comments for #63.
|
|
|
|
oldXmlns := `<workbook xmlns="http://schemas.openxmlformats.org/spreadsheetml/2006/main">`
|
|
|
|
newXmlns := `<workbook xmlns="http://schemas.openxmlformats.org/spreadsheetml/2006/main" xmlns:r="http://schemas.openxmlformats.org/officeDocument/2006/relationships">`
|
|
|
|
return strings.Replace(workbookMarshal, oldXmlns, newXmlns, -1)
|
|
|
|
}
|
|
|
|
|
|
|
|
// replace relationships ID in worksheets/sheet%d.xml
|
|
|
|
func replaceRelationshipsID(workbookMarshal string) string {
|
|
|
|
rids := strings.Replace(workbookMarshal, `<drawing rid="" />`, ``, -1)
|
|
|
|
return strings.Replace(rids, `<drawing rid="`, `<drawing r:id="`, -1)
|
|
|
|
}
|
|
|
|
|
2016-09-02 11:54:52 +08:00
|
|
|
// SetActiveSheet provide function to set default active sheet of XLSX by given index
|
2016-08-30 11:51:31 +08:00
|
|
|
func SetActiveSheet(file []FileList, index int) []FileList {
|
|
|
|
var content xlsxWorkbook
|
|
|
|
if index < 1 {
|
|
|
|
index = 1
|
|
|
|
}
|
2016-09-02 11:54:52 +08:00
|
|
|
index--
|
|
|
|
xml.Unmarshal([]byte(readXML(file, `xl/workbook.xml`)), &content)
|
2016-08-30 11:51:31 +08:00
|
|
|
if len(content.BookViews.WorkBookView) > 0 {
|
|
|
|
content.BookViews.WorkBookView[0].ActiveTab = index
|
|
|
|
} else {
|
|
|
|
content.BookViews.WorkBookView = append(content.BookViews.WorkBookView, xlsxWorkBookView{
|
|
|
|
ActiveTab: index,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
sheets := len(content.Sheets.Sheet)
|
|
|
|
output, err := xml.MarshalIndent(content, "", "")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
file = saveFileList(file, `xl/workbook.xml`, workBookCompatibility(replaceRelationshipsNameSpace(string(output))))
|
2016-09-02 11:54:52 +08:00
|
|
|
index++
|
2016-08-30 11:51:31 +08:00
|
|
|
for i := 0; i < sheets; i++ {
|
|
|
|
xlsx := xlsxWorksheet{}
|
|
|
|
sheetIndex := i + 1
|
2016-08-30 21:54:28 +08:00
|
|
|
path := `xl/worksheets/sheet` + strconv.Itoa(sheetIndex) + `.xml`
|
2016-09-02 11:54:52 +08:00
|
|
|
xml.Unmarshal([]byte(readXML(file, path)), &xlsx)
|
2016-08-30 11:51:31 +08:00
|
|
|
if index == sheetIndex {
|
|
|
|
if len(xlsx.SheetViews.SheetView) > 0 {
|
|
|
|
xlsx.SheetViews.SheetView[0].TabSelected = true
|
|
|
|
} else {
|
|
|
|
xlsx.SheetViews.SheetView = append(xlsx.SheetViews.SheetView, xlsxSheetView{
|
|
|
|
TabSelected: true,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if len(xlsx.SheetViews.SheetView) > 0 {
|
|
|
|
xlsx.SheetViews.SheetView[0].TabSelected = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sheet, err := xml.MarshalIndent(xlsx, "", "")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
file = saveFileList(file, path, replaceRelationshipsID(replaceWorkSheetsRelationshipsNameSpace(string(sheet))))
|
|
|
|
}
|
|
|
|
return file
|
|
|
|
}
|
|
|
|
|
|
|
|
// Replace xl/workbook.xml XML tags to self-closing for compatible Office Excel 2007
|
|
|
|
func workBookCompatibility(workbookMarshal string) string {
|
|
|
|
workbookMarshal = strings.Replace(workbookMarshal, `xmlns:relationships="http://schemas.openxmlformats.org/officeDocument/2006/relationships" relationships:id="`, `r:id="`, -1)
|
|
|
|
workbookMarshal = strings.Replace(workbookMarshal, `></sheet>`, ` />`, -1)
|
|
|
|
workbookMarshal = strings.Replace(workbookMarshal, `></workbookView>`, ` />`, -1)
|
|
|
|
workbookMarshal = strings.Replace(workbookMarshal, `></fileVersion>`, ` />`, -1)
|
|
|
|
workbookMarshal = strings.Replace(workbookMarshal, `></workbookPr>`, ` />`, -1)
|
|
|
|
workbookMarshal = strings.Replace(workbookMarshal, `></definedNames>`, ` />`, -1)
|
|
|
|
workbookMarshal = strings.Replace(workbookMarshal, `></calcPr>`, ` />`, -1)
|
|
|
|
workbookMarshal = strings.Replace(workbookMarshal, `></workbookProtection>`, ` />`, -1)
|
|
|
|
workbookMarshal = strings.Replace(workbookMarshal, `></fileRecoveryPr>`, ` />`, -1)
|
|
|
|
return workbookMarshal
|
|
|
|
}
|