Skip to content

Commit 0072bb7

Browse files
committed
resolve the issue corrupted xlsx after deleting formula of cell, reference qax-os#346
1 parent e780e41 commit 0072bb7

10 files changed

+121
-4
lines changed

calcchain.go

+55
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
// Copyright 2016 - 2019 The excelize Authors. All rights reserved. Use of
2+
// this source code is governed by a BSD-style license that can be found in
3+
// the LICENSE file.
4+
//
5+
// Package excelize providing a set of functions that allow you to write to
6+
// and read from XLSX files. Support reads and writes XLSX file generated by
7+
// Microsoft Excel™ 2007 and later. Support save file without losing original
8+
// charts of XLSX. This library needs Go version 1.8 or later.
9+
10+
package excelize
11+
12+
import "encoding/xml"
13+
14+
// calcChainReader provides a function to get the pointer to the structure
15+
// after deserialization of xl/calcChain.xml.
16+
func (f *File) calcChainReader() *xlsxCalcChain {
17+
if f.CalcChain == nil {
18+
var c xlsxCalcChain
19+
_ = xml.Unmarshal(namespaceStrictToTransitional(f.readXML("xl/calcChain.xml")), &c)
20+
f.CalcChain = &c
21+
}
22+
return f.CalcChain
23+
}
24+
25+
// calcChainWriter provides a function to save xl/calcChain.xml after
26+
// serialize structure.
27+
func (f *File) calcChainWriter() {
28+
if f.CalcChain != nil {
29+
output, _ := xml.Marshal(f.CalcChain)
30+
f.saveFileList("xl/calcChain.xml", output)
31+
}
32+
}
33+
34+
// deleteCalcChain provides a function to remove cell reference on the
35+
// calculation chain.
36+
func (f *File) deleteCalcChain(axis string) {
37+
calc := f.calcChainReader()
38+
if calc != nil {
39+
for i, c := range calc.C {
40+
if c.R == axis {
41+
calc.C = append(calc.C[:i], calc.C[i+1:]...)
42+
}
43+
}
44+
}
45+
if len(calc.C) == 0 {
46+
f.CalcChain = nil
47+
delete(f.XLSX, "xl/calcChain.xml")
48+
content := f.contentTypesReader()
49+
for k, v := range content.Overrides {
50+
if v.PartName == "/xl/calcChain.xml" {
51+
content.Overrides = append(content.Overrides[:k], content.Overrides[k+1:]...)
52+
}
53+
}
54+
}
55+
}

cell.go

+5
Original file line numberDiff line numberDiff line change
@@ -305,6 +305,11 @@ func (f *File) SetCellFormula(sheet, axis, formula string) {
305305
completeRow(xlsx, rows, cell)
306306
completeCol(xlsx, rows, cell)
307307

308+
if formula == "" {
309+
xlsx.SheetData.Row[xAxis].C[yAxis].F = nil
310+
f.deleteCalcChain(axis)
311+
return
312+
}
308313
if xlsx.SheetData.Row[xAxis].C[yAxis].F != nil {
309314
xlsx.SheetData.Row[xAxis].C[yAxis].F.Content = formula
310315
} else {

col.go

+4
Original file line numberDiff line numberDiff line change
@@ -322,6 +322,10 @@ func (f *File) InsertCol(sheet, column string) {
322322
//
323323
// xlsx.RemoveCol("Sheet1", "C")
324324
//
325+
// Use this method with caution, which will affect changes in references such
326+
// as formulas, charts, and so on. If there is any referenced value of the
327+
// worksheet, it will cause a file error when you open it. The excelize only
328+
// partially updates these references currently.
325329
func (f *File) RemoveCol(sheet, column string) {
326330
xlsx := f.workSheetReader(sheet)
327331
for r := range xlsx.SheetData.Row {

excelize.go

+3-1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ import (
2525
type File struct {
2626
checked map[string]bool
2727
sheetMap map[string]string
28+
CalcChain *xlsxCalcChain
2829
ContentTypes *xlsxTypes
2930
Path string
3031
SharedStrings *xlsxSST
@@ -201,7 +202,8 @@ func (f *File) UpdateLinkedValue() {
201202
// row: Index number of the row we're inserting/deleting before
202203
// offset: Number of rows/column to insert/delete negative values indicate deletion
203204
//
204-
// TODO: adjustPageBreaks, adjustComments, adjustDataValidations, adjustProtectedCells
205+
// TODO: adjustCalcChain, adjustPageBreaks, adjustComments,
206+
// adjustDataValidations, adjustProtectedCells
205207
//
206208
func (f *File) adjustHelper(sheet string, column, row, offset int) {
207209
xlsx := f.workSheetReader(sheet)

excelize_test.go

+11-1
Original file line numberDiff line numberDiff line change
@@ -342,8 +342,18 @@ func TestSetCellFormula(t *testing.T) {
342342
xlsx.SetCellFormula("Sheet1", "C19", "SUM(Sheet2!D2,Sheet2!D9)")
343343
// Test set cell formula with illegal rows number.
344344
xlsx.SetCellFormula("Sheet1", "C", "SUM(Sheet2!D2,Sheet2!D9)")
345+
assert.NoError(t, xlsx.SaveAs(filepath.Join("test", "TestSetCellFormula1.xlsx")))
345346

346-
assert.NoError(t, xlsx.SaveAs(filepath.Join("test", "TestSetCellFormula.xlsx")))
347+
xlsx, err = OpenFile(filepath.Join("test", "CalcChain.xlsx"))
348+
if !assert.NoError(t, err) {
349+
t.FailNow()
350+
}
351+
// Test remove cell formula.
352+
xlsx.SetCellFormula("Sheet1", "A1", "")
353+
assert.NoError(t, xlsx.SaveAs(filepath.Join("test", "TestSetCellFormula2.xlsx")))
354+
// Test remove all cell formula.
355+
xlsx.SetCellFormula("Sheet1", "B1", "")
356+
assert.NoError(t, xlsx.SaveAs(filepath.Join("test", "TestSetCellFormula3.xlsx")))
347357
}
348358

349359
func TestSetSheetBackground(t *testing.T) {

file.go

+1
Original file line numberDiff line numberDiff line change
@@ -92,6 +92,7 @@ func (f *File) WriteToBuffer() (*bytes.Buffer, error) {
9292
f.workbookRelsWriter()
9393
f.worksheetWriter()
9494
f.styleSheetWriter()
95+
f.calcChainWriter()
9596
for path, content := range f.XLSX {
9697
fi, err := zw.Create(path)
9798
if err != nil {

rows.go

+13-1
Original file line numberDiff line numberDiff line change
@@ -343,6 +343,10 @@ func (f *File) GetRowOutlineLevel(sheet string, rowIndex int) uint8 {
343343
//
344344
// xlsx.RemoveRow("Sheet1", 2)
345345
//
346+
// Use this method with caution, which will affect changes in references such
347+
// as formulas, charts, and so on. If there is any referenced value of the
348+
// worksheet, it will cause a file error when you open it. The excelize only
349+
// partially updates these references currently.
346350
func (f *File) RemoveRow(sheet string, row int) {
347351
if row < 0 {
348352
return
@@ -375,15 +379,23 @@ func (f *File) InsertRow(sheet string, row int) {
375379
//
376380
// xlsx.DuplicateRow("Sheet1", 2)
377381
//
382+
// Use this method with caution, which will affect changes in references such
383+
// as formulas, charts, and so on. If there is any referenced value of the
384+
// worksheet, it will cause a file error when you open it. The excelize only
385+
// partially updates these references currently.
378386
func (f *File) DuplicateRow(sheet string, row int) {
379387
f.DuplicateRowTo(sheet, row, row+1)
380388
}
381389

382390
// DuplicateRowTo inserts a copy of specified row at specified row position
383-
// movig down exists rows aftet target position
391+
// moving down exists rows after target position
384392
//
385393
// xlsx.DuplicateRowTo("Sheet1", 2, 7)
386394
//
395+
// Use this method with caution, which will affect changes in references such
396+
// as formulas, charts, and so on. If there is any referenced value of the
397+
// worksheet, it will cause a file error when you open it. The excelize only
398+
// partially updates these references currently.
387399
func (f *File) DuplicateRowTo(sheet string, row, row2 int) {
388400
if row <= 0 || row2 <= 0 || row == row2 {
389401
return

sheet.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -900,7 +900,7 @@ func (p *PageLayoutPaperSize) getPageLayout(ps *xlsxPageSetUp) {
900900
// 40 | German standard fanfold (8.5 in. by 12 in.)
901901
// 41 | German legal fanfold (8.5 in. by 13 in.)
902902
// 42 | ISO B4 (250 mm by 353 mm)
903-
// 43 | Japanese double postcard (200 mm by 148 mm)
903+
// 43 | Japanese postcard (100 mm by 148 mm)
904904
// 44 | Standard paper (9 in. by 11 in.)
905905
// 45 | Standard paper (10 in. by 11 in.)
906906
// 46 | Standard paper (15 in. by 11 in.)

test/CalcChain.xlsx

5.82 KB
Binary file not shown.

xmlCalcChain.go

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
// Copyright 2016 - 2019 The excelize Authors. All rights reserved. Use of
2+
// this source code is governed by a BSD-style license that can be found in
3+
// the LICENSE file.
4+
//
5+
// Package excelize providing a set of functions that allow you to write to
6+
// and read from XLSX files. Support reads and writes XLSX file generated by
7+
// Microsoft Excel™ 2007 and later. Support save file without losing original
8+
// charts of XLSX. This library needs Go version 1.8 or later.
9+
10+
package excelize
11+
12+
import "encoding/xml"
13+
14+
// xlsxCalcChain directly maps the calcChain element. This element represents the root of the calculation chain.
15+
type xlsxCalcChain struct {
16+
XMLName xml.Name `xml:"http://schemas.openxmlformats.org/spreadsheetml/2006/main calcChain"`
17+
C []xlsxCalcChainC `xml:"c"`
18+
}
19+
20+
// xlsxCalcChainC directly maps the c element.
21+
type xlsxCalcChainC struct {
22+
R string `xml:"r,attr"`
23+
I int `xml:"i,attr"`
24+
L bool `xml:"l,attr,omitempty"`
25+
S bool `xml:"s,attr,omitempty"`
26+
T bool `xml:"t,attr,omitempty"`
27+
A bool `xml:"a,attr,omitempty"`
28+
}

0 commit comments

Comments
 (0)