|
|
|
package excelize
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"reflect"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
_ "image/jpeg"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestConcurrency(t *testing.T) {
|
|
|
|
f, err := OpenFile(filepath.Join("test", "Book1.xlsx"))
|
|
|
|
assert.NoError(t, err)
|
|
|
|
wg := new(sync.WaitGroup)
|
|
|
|
for i := 1; i <= 5; i++ {
|
|
|
|
wg.Add(1)
|
|
|
|
go func(val int, t *testing.T) {
|
|
|
|
// Concurrency set cell value
|
|
|
|
assert.NoError(t, f.SetCellValue("Sheet1", fmt.Sprintf("A%d", val), val))
|
|
|
|
assert.NoError(t, f.SetCellValue("Sheet1", fmt.Sprintf("B%d", val), strconv.Itoa(val)))
|
|
|
|
// Concurrency get cell value
|
|
|
|
_, err := f.GetCellValue("Sheet1", fmt.Sprintf("A%d", val))
|
|
|
|
assert.NoError(t, err)
|
|
|
|
// Concurrency set rows
|
|
|
|
assert.NoError(t, f.SetSheetRow("Sheet1", "B6", &[]interface{}{
|
|
|
|
" Hello",
|
|
|
|
[]byte("World"), 42, int8(1<<8/2 - 1), int16(1<<16/2 - 1), int32(1<<32/2 - 1),
|
|
|
|
int64(1<<32/2 - 1), float32(42.65418), -42.65418, float32(42), float64(42),
|
|
|
|
uint(1<<32 - 1), uint8(1<<8 - 1), uint16(1<<16 - 1), uint32(1<<32 - 1),
|
|
|
|
uint64(1<<32 - 1), true, complex64(5 + 10i),
|
|
|
|
}))
|
|
|
|
// Concurrency create style
|
|
|
|
style, err := f.NewStyle(`{"font":{"color":"#1265BE","underline":"single"}}`)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
// Concurrency set cell style
|
|
|
|
assert.NoError(t, f.SetCellStyle("Sheet1", "A3", "A3", style))
|
|
|
|
// Concurrency add picture
|
|
|
|
assert.NoError(t, f.AddPicture("Sheet1", "F21", filepath.Join("test", "images", "excel.jpg"),
|
|
|
|
`{"x_offset": 10, "y_offset": 10, "hyperlink": "https://github.com/xuri/excelize", "hyperlink_type": "External", "positioning": "oneCell"}`))
|
|
|
|
// Concurrency get cell picture
|
|
|
|
name, raw, err := f.GetPicture("Sheet1", "A1")
|
|
|
|
assert.Equal(t, "", name)
|
|
|
|
assert.Nil(t, raw)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
// Concurrency iterate rows
|
|
|
|
rows, err := f.Rows("Sheet1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
for rows.Next() {
|
|
|
|
_, err := rows.Columns()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
// Concurrency iterate columns
|
|
|
|
cols, err := f.Cols("Sheet1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
for rows.Next() {
|
|
|
|
_, err := cols.Rows()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
wg.Done()
|
|
|
|
}(i, t)
|
|
|
|
}
|
|
|
|
wg.Wait()
|
|
|
|
val, err := f.GetCellValue("Sheet1", "A1")
|
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
assert.Equal(t, "1", val)
|
|
|
|
assert.NoError(t, f.SaveAs(filepath.Join("test", "TestConcurrency.xlsx")))
|
|
|
|
assert.NoError(t, f.Close())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCheckCellInArea(t *testing.T) {
|
|
|
|
f := NewFile()
|
|
|
|
expectedTrueCellInAreaList := [][2]string{
|
|
|
|
{"c2", "A1:AAZ32"},
|
|
|
|
{"B9", "A1:B9"},
|
|
|
|
{"C2", "C2:C2"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, expectedTrueCellInArea := range expectedTrueCellInAreaList {
|
|
|
|
cell := expectedTrueCellInArea[0]
|
|
|
|
area := expectedTrueCellInArea[1]
|
|
|
|
ok, err := f.checkCellInArea(cell, area)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Truef(t, ok,
|
|
|
|
"Expected cell %v to be in area %v, got false\n", cell, area)
|
|
|
|
}
|
|
|
|
|
|
|
|
expectedFalseCellInAreaList := [][2]string{
|
|
|
|
{"c2", "A4:AAZ32"},
|
|
|
|
{"C4", "D6:A1"}, // weird case, but you never know
|
|
|
|
{"AEF42", "BZ40:AEF41"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, expectedFalseCellInArea := range expectedFalseCellInAreaList {
|
|
|
|
cell := expectedFalseCellInArea[0]
|
|
|
|
area := expectedFalseCellInArea[1]
|
|
|
|
ok, err := f.checkCellInArea(cell, area)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Falsef(t, ok,
|
|
|
|
"Expected cell %v not to be inside of area %v, but got true\n", cell, area)
|
|
|
|
}
|
Huge refactorig for consistent col/row numbering (#356)
* Huge refactorig for consistent col/row numbering
Started from simply changing ToALphaString()/TitleToNumber() logic and related fixes.
But have to go deeper, do fixes, after do related fixes and again and again.
Major improvements:
1. Tests made stronger again (But still be weak).
2. "Empty" returns for incorrect input replaces with panic.
3. Check for correct col/row/cell naming & addressing by default.
4. Removed huge amount of duplicated code.
5. Removed ToALphaString(), TitleToNumber() and it helpers functions at all,
and replaced with SplitCellName(), JoinCellName(), ColumnNameToNumber(), ColumnNumberToName(), CellNameToCoordinates(), CoordinatesToCellName().
6. Minor fixes for internal variable naming for code readability (ex. col, row for input params, colIdx, rowIdx for slice indexes etc).
* Formatting fixes
6 years ago
|
|
|
|
|
|
|
ok, err := f.checkCellInArea("A1", "A:B")
|
|
|
|
assert.EqualError(t, err, newCellNameToCoordinatesError("A", newInvalidCellNameError("A")).Error())
|
|
|
|
assert.False(t, ok)
|
|
|
|
|
|
|
|
ok, err = f.checkCellInArea("AA0", "Z0:AB1")
|
|
|
|
assert.EqualError(t, err, newCellNameToCoordinatesError("AA0", newInvalidCellNameError("AA0")).Error())
|
|
|
|
assert.False(t, ok)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSetCellFloat(t *testing.T) {
|
|
|
|
sheet := "Sheet1"
|
|
|
|
t.Run("with no decimal", func(t *testing.T) {
|
|
|
|
f := NewFile()
|
|
|
|
assert.NoError(t, f.SetCellFloat(sheet, "A1", 123.0, -1, 64))
|
|
|
|
assert.NoError(t, f.SetCellFloat(sheet, "A2", 123.0, 1, 64))
|
|
|
|
val, err := f.GetCellValue(sheet, "A1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "123", val, "A1 should be 123")
|
|
|
|
val, err = f.GetCellValue(sheet, "A2")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "123", val, "A2 should be 123")
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("with a decimal and precision limit", func(t *testing.T) {
|
|
|
|
f := NewFile()
|
|
|
|
assert.NoError(t, f.SetCellFloat(sheet, "A1", 123.42, 1, 64))
|
|
|
|
val, err := f.GetCellValue(sheet, "A1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "123.4", val, "A1 should be 123.4")
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("with a decimal and no limit", func(t *testing.T) {
|
|
|
|
f := NewFile()
|
|
|
|
assert.NoError(t, f.SetCellFloat(sheet, "A1", 123.42, -1, 64))
|
|
|
|
val, err := f.GetCellValue(sheet, "A1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "123.42", val, "A1 should be 123.42")
|
|
|
|
})
|
|
|
|
f := NewFile()
|
|
|
|
assert.EqualError(t, f.SetCellFloat(sheet, "A", 123.42, -1, 64), newCellNameToCoordinatesError("A", newInvalidCellNameError("A")).Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSetCellValue(t *testing.T) {
|
|
|
|
f := NewFile()
|
|
|
|
assert.EqualError(t, f.SetCellValue("Sheet1", "A", time.Now().UTC()), newCellNameToCoordinatesError("A", newInvalidCellNameError("A")).Error())
|
|
|
|
assert.EqualError(t, f.SetCellValue("Sheet1", "A", time.Duration(1e13)), newCellNameToCoordinatesError("A", newInvalidCellNameError("A")).Error())
|
|
|
|
// Test set cell value with column and row style inherit
|
|
|
|
style1, err := f.NewStyle(&Style{NumFmt: 2})
|
|
|
|
assert.NoError(t, err)
|
|
|
|
style2, err := f.NewStyle(&Style{NumFmt: 9})
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.NoError(t, f.SetColStyle("Sheet1", "B", style1))
|
|
|
|
assert.NoError(t, f.SetRowStyle("Sheet1", 1, 1, style2))
|
|
|
|
assert.NoError(t, f.SetCellValue("Sheet1", "B1", 0.5))
|
|
|
|
assert.NoError(t, f.SetCellValue("Sheet1", "B2", 0.5))
|
|
|
|
B1, err := f.GetCellValue("Sheet1", "B1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "50%", B1)
|
|
|
|
B2, err := f.GetCellValue("Sheet1", "B2")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "0.50", B2)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSetCellValues(t *testing.T) {
|
|
|
|
f := NewFile()
|
|
|
|
err := f.SetCellValue("Sheet1", "A1", time.Date(2010, time.December, 31, 0, 0, 0, 0, time.UTC))
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
v, err := f.GetCellValue("Sheet1", "A1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, v, "12/31/10 00:00")
|
|
|
|
|
|
|
|
// test date value lower than min date supported by Excel
|
|
|
|
err = f.SetCellValue("Sheet1", "A1", time.Date(1600, time.December, 31, 0, 0, 0, 0, time.UTC))
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
v, err = f.GetCellValue("Sheet1", "A1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, v, "1600-12-31T00:00:00Z")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSetCellBool(t *testing.T) {
|
|
|
|
f := NewFile()
|
|
|
|
assert.EqualError(t, f.SetCellBool("Sheet1", "A", true), newCellNameToCoordinatesError("A", newInvalidCellNameError("A")).Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSetCellTime(t *testing.T) {
|
|
|
|
date, err := time.Parse(time.RFC3339Nano, "2009-11-10T23:00:00Z")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
for location, expected := range map[string]string{
|
|
|
|
"America/New_York": "40127.75",
|
|
|
|
"Asia/Shanghai": "40128.291666666664",
|
|
|
|
"Europe/London": "40127.958333333336",
|
|
|
|
"UTC": "40127.958333333336",
|
|
|
|
} {
|
|
|
|
timezone, err := time.LoadLocation(location)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
_, b, isNum, err := setCellTime(date.In(timezone), false)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, true, isNum)
|
|
|
|
assert.Equal(t, expected, b)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetCellValue(t *testing.T) {
|
|
|
|
// Test get cell value without r attribute of the row.
|
|
|
|
f := NewFile()
|
|
|
|
sheetData := `<worksheet xmlns="http://schemas.openxmlformats.org/spreadsheetml/2006/main"><sheetData>%s</sheetData></worksheet>`
|
|
|
|
|
|
|
|
f.Sheet.Delete("xl/worksheets/sheet1.xml")
|
|
|
|
f.Pkg.Store("xl/worksheets/sheet1.xml", []byte(fmt.Sprintf(sheetData, `<row r="3"><c t="str"><v>A3</v></c></row><row><c t="str"><v>A4</v></c><c t="str"><v>B4</v></c></row><row r="7"><c t="str"><v>A7</v></c><c t="str"><v>B7</v></c></row><row><c t="str"><v>A8</v></c><c t="str"><v>B8</v></c></row>`)))
|
|
|
|
f.checked = nil
|
|
|
|
cells := []string{"A3", "A4", "B4", "A7", "B7"}
|
|
|
|
rows, err := f.GetRows("Sheet1")
|
|
|
|
assert.Equal(t, [][]string{nil, nil, {"A3"}, {"A4", "B4"}, nil, nil, {"A7", "B7"}, {"A8", "B8"}}, rows)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
for _, cell := range cells {
|
|
|
|
value, err := f.GetCellValue("Sheet1", cell)
|
|
|
|
assert.Equal(t, cell, value)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
cols, err := f.GetCols("Sheet1")
|
|
|
|
assert.Equal(t, [][]string{{"", "", "A3", "A4", "", "", "A7", "A8"}, {"", "", "", "B4", "", "", "B7", "B8"}}, cols)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
f.Sheet.Delete("xl/worksheets/sheet1.xml")
|
|
|
|
f.Pkg.Store("xl/worksheets/sheet1.xml", []byte(fmt.Sprintf(sheetData, `<row r="2"><c r="A2" t="str"><v>A2</v></c></row><row r="2"><c r="B2" t="str"><v>B2</v></c></row>`)))
|
|
|
|
f.checked = nil
|
|
|
|
cell, err := f.GetCellValue("Sheet1", "A2")
|
|
|
|
assert.Equal(t, "A2", cell)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
f.Sheet.Delete("xl/worksheets/sheet1.xml")
|
|
|
|
f.Pkg.Store("xl/worksheets/sheet1.xml", []byte(fmt.Sprintf(sheetData, `<row r="2"><c r="A2" t="str"><v>A2</v></c></row><row r="2"><c r="B2" t="str"><v>B2</v></c></row>`)))
|
|
|
|
f.checked = nil
|
|
|
|
rows, err = f.GetRows("Sheet1")
|
|
|
|
assert.Equal(t, [][]string{nil, {"A2", "B2"}}, rows)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
f.Sheet.Delete("xl/worksheets/sheet1.xml")
|
|
|
|
f.Pkg.Store("xl/worksheets/sheet1.xml", []byte(fmt.Sprintf(sheetData, `<row r="1"><c r="A1" t="str"><v>A1</v></c></row><row r="1"><c r="B1" t="str"><v>B1</v></c></row>`)))
|
|
|
|
f.checked = nil
|
|
|
|
rows, err = f.GetRows("Sheet1")
|
|
|
|
assert.Equal(t, [][]string{{"A1", "B1"}}, rows)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
f.Sheet.Delete("xl/worksheets/sheet1.xml")
|
|
|
|
f.Pkg.Store("xl/worksheets/sheet1.xml", []byte(fmt.Sprintf(sheetData, `<row><c t="str"><v>A3</v></c></row><row><c t="str"><v>A4</v></c><c t="str"><v>B4</v></c></row><row r="7"><c t="str"><v>A7</v></c><c t="str"><v>B7</v></c></row><row><c t="str"><v>A8</v></c><c t="str"><v>B8</v></c></row>`)))
|
|
|
|
f.checked = nil
|
|
|
|
rows, err = f.GetRows("Sheet1")
|
|
|
|
assert.Equal(t, [][]string{{"A3"}, {"A4", "B4"}, nil, nil, nil, nil, {"A7", "B7"}, {"A8", "B8"}}, rows)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
f.Sheet.Delete("xl/worksheets/sheet1.xml")
|
|
|
|
f.Pkg.Store("xl/worksheets/sheet1.xml", []byte(fmt.Sprintf(sheetData, `<row r="0"><c r="H6" t="str"><v>H6</v></c><c r="A1" t="str"><v>r0A6</v></c><c r="F4" t="str"><v>F4</v></c></row><row><c r="A1" t="str"><v>A6</v></c><c r="B1" t="str"><v>B6</v></c><c r="C1" t="str"><v>C6</v></c></row><row r="3"><c r="A3"><v>100</v></c><c r="B3" t="str"><v>B3</v></c></row>`)))
|
|
|
|
f.checked = nil
|
|
|
|
cell, err = f.GetCellValue("Sheet1", "H6")
|
|
|
|
assert.Equal(t, "H6", cell)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
rows, err = f.GetRows("Sheet1")
|
|
|
|
assert.Equal(t, [][]string{
|
|
|
|
{"A6", "B6", "C6"},
|
|
|
|
nil,
|
|
|
|
{"100", "B3"},
|
|
|
|
{"", "", "", "", "", "F4"},
|
|
|
|
nil,
|
|
|
|
{"", "", "", "", "", "", "", "H6"},
|
|
|
|
}, rows)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
f.Sheet.Delete("xl/worksheets/sheet1.xml")
|
|
|
|
f.Pkg.Store("xl/worksheets/sheet1.xml", []byte(fmt.Sprintf(sheetData, `<row r="1">
|
|
|
|
<c r="A1"><v>2422.3000000000002</v></c>
|
|
|
|
<c r="B1"><v>2422.3000000000002</v></c>
|
|
|
|
<c r="C1"><v>12.4</v></c>
|
|
|
|
<c r="D1"><v>964</v></c>
|
|
|
|
<c r="E1"><v>1101.5999999999999</v></c>
|
|
|
|
<c r="F1"><v>275.39999999999998</v></c>
|
|
|
|
<c r="G1"><v>68.900000000000006</v></c>
|
|
|
|
<c r="H1"><v>44385.208333333336</v></c>
|
|
|
|
<c r="I1"><v>5.0999999999999996</v></c>
|
|
|
|
<c r="J1"><v>5.1100000000000003</v></c>
|
|
|
|
<c r="K1"><v>5.0999999999999996</v></c>
|
|
|
|
<c r="L1"><v>5.1109999999999998</v></c>
|
|
|
|
<c r="M1"><v>5.1111000000000004</v></c>
|
|
|
|
<c r="N1"><v>2422.012345678</v></c>
|
|
|
|
<c r="O1"><v>2422.0123456789</v></c>
|
|
|
|
<c r="P1"><v>12.012345678901</v></c>
|
|
|
|
<c r="Q1"><v>964</v></c>
|
|
|
|
<c r="R1"><v>1101.5999999999999</v></c>
|
|
|
|
<c r="S1"><v>275.39999999999998</v></c>
|
|
|
|
<c r="T1"><v>68.900000000000006</v></c>
|
|
|
|
<c r="U1"><v>8.8880000000000001E-2</v></c>
|
|
|
|
<c r="V1"><v>4.0000000000000003e-5</v></c>
|
|
|
|
<c r="W1"><v>2422.3000000000002</v></c>
|
|
|
|
<c r="X1"><v>1101.5999999999999</v></c>
|
|
|
|
<c r="Y1"><v>275.39999999999998</v></c>
|
|
|
|
<c r="Z1"><v>68.900000000000006</v></c>
|
|
|
|
<c r="AA1"><v>1.1000000000000001</v></c>
|
|
|
|
<c r="AA2"><v>1234567890123_4</v></c>
|
|
|
|
<c r="AA3"><v>123456789_0123_4</v></c>
|
|
|
|
</row>`)))
|
|
|
|
f.checked = nil
|
|
|
|
rows, err = f.GetRows("Sheet1")
|
|
|
|
assert.Equal(t, [][]string{{
|
|
|
|
"2422.3",
|
|
|
|
"2422.3",
|
|
|
|
"12.4",
|
|
|
|
"964",
|
|
|
|
"1101.6",
|
|
|
|
"275.4",
|
|
|
|
"68.9",
|
|
|
|
"44385.2083333333",
|
|
|
|
"5.1",
|
|
|
|
"5.11",
|
|
|
|
"5.1",
|
|
|
|
"5.111",
|
|
|
|
"5.1111",
|
|
|
|
"2422.012345678",
|
|
|
|
"2422.0123456789",
|
|
|
|
"12.012345678901",
|
|
|
|
"964",
|
|
|
|
"1101.6",
|
|
|
|
"275.4",
|
|
|
|
"68.9",
|
|
|
|
"0.08888",
|
|
|
|
"0.00004",
|
|
|
|
"2422.3",
|
|
|
|
"1101.6",
|
|
|
|
"275.4",
|
|
|
|
"68.9",
|
|
|
|
"1.1",
|
|
|
|
"1234567890123_4",
|
|
|
|
"123456789_0123_4",
|
|
|
|
}}, rows)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetCellType(t *testing.T) {
|
|
|
|
f := NewFile()
|
|
|
|
cellType, err := f.GetCellType("Sheet1", "A1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, CellTypeUnset, cellType)
|
|
|
|
assert.NoError(t, f.SetCellValue("Sheet1", "A1", "A1"))
|
|
|
|
cellType, err = f.GetCellType("Sheet1", "A1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, CellTypeString, cellType)
|
|
|
|
_, err = f.GetCellType("Sheet1", "A")
|
|
|
|
assert.EqualError(t, err, newCellNameToCoordinatesError("A", newInvalidCellNameError("A")).Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetValueFrom(t *testing.T) {
|
|
|
|
f := NewFile()
|
|
|
|
c := xlsxC{T: "s"}
|
|
|
|
value, err := c.getValueFrom(f, f.sharedStringsReader(), false)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "", value)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetCellFormula(t *testing.T) {
|
|
|
|
// Test get cell formula on not exist worksheet.
|
|
|
|
f := NewFile()
|
|
|
|
_, err := f.GetCellFormula("SheetN", "A1")
|
|
|
|
assert.EqualError(t, err, "sheet SheetN is not exist")
|
|
|
|
|
|
|
|
// Test get cell formula on no formula cell.
|
|
|
|
assert.NoError(t, f.SetCellValue("Sheet1", "A1", true))
|
|
|
|
_, err = f.GetCellFormula("Sheet1", "A1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
// Test get cell shared formula
|
|
|
|
f = NewFile()
|
|
|
|
sheetData := `<worksheet xmlns="http://schemas.openxmlformats.org/spreadsheetml/2006/main"><sheetData><row r="1"><c r="A1"><v>1</v></c><c r="B1"><f>2*A1</f></c></row><row r="2"><c r="A2"><v>2</v></c><c r="B2"><f t="shared" ref="B2:B7" si="0">%s</f></c></row><row r="3"><c r="A3"><v>3</v></c><c r="B3"><f t="shared" si="0"/></c></row><row r="4"><c r="A4"><v>4</v></c><c r="B4"><f t="shared" si="0"/></c></row><row r="5"><c r="A5"><v>5</v></c><c r="B5"><f t="shared" si="0"/></c></row><row r="6"><c r="A6"><v>6</v></c><c r="B6"><f t="shared" si="0"/></c></row><row r="7"><c r="A7"><v>7</v></c><c r="B7"><f t="shared" si="0"/></c></row></sheetData></worksheet>`
|
|
|
|
|
|
|
|
for sharedFormula, expected := range map[string]string{
|
|
|
|
`2*A2`: `2*A3`,
|
|
|
|
`2*A1A`: `2*A2A`,
|
|
|
|
`2*$A$2+LEN("")`: `2*$A$2+LEN("")`,
|
|
|
|
} {
|
|
|
|
f.Sheet.Delete("xl/worksheets/sheet1.xml")
|
|
|
|
f.Pkg.Store("xl/worksheets/sheet1.xml", []byte(fmt.Sprintf(sheetData, sharedFormula)))
|
|
|
|
formula, err := f.GetCellFormula("Sheet1", "B3")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, expected, formula)
|
|
|
|
}
|
|
|
|
|
|
|
|
f.Sheet.Delete("xl/worksheets/sheet1.xml")
|
|
|
|
f.Pkg.Store("xl/worksheets/sheet1.xml", []byte(`<worksheet xmlns="http://schemas.openxmlformats.org/spreadsheetml/2006/main"><sheetData><row r="2"><c r="B2"><f t="shared" si="0"></f></c></row></sheetData></worksheet>`))
|
|
|
|
formula, err := f.GetCellFormula("Sheet1", "B2")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "", formula)
|
|
|
|
}
|
|
|
|
|
|
|
|
func ExampleFile_SetCellFloat() {
|
|
|
|
f := NewFile()
|
|
|
|
x := 3.14159265
|
|
|
|
if err := f.SetCellFloat("Sheet1", "A1", x, 2, 64); err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
val, _ := f.GetCellValue("Sheet1", "A1")
|
|
|
|
fmt.Println(val)
|
|
|
|
// Output: 3.14
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkSetCellValue(b *testing.B) {
|
|
|
|
values := []string{"First", "Second", "Third", "Fourth", "Fifth", "Sixth"}
|
|
|
|
cols := []string{"A", "B", "C", "D", "E", "F"}
|
|
|
|
f := NewFile()
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 1; i <= b.N; i++ {
|
|
|
|
for j := 0; j < len(values); j++ {
|
|
|
|
if err := f.SetCellValue("Sheet1", cols[j]+strconv.Itoa(i), values[j]); err != nil {
|
|
|
|
b.Error(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOverflowNumericCell(t *testing.T) {
|
|
|
|
f, err := OpenFile(filepath.Join("test", "OverflowNumericCell.xlsx"))
|
|
|
|
if !assert.NoError(t, err) {
|
|
|
|
t.FailNow()
|
|
|
|
}
|
|
|
|
val, err := f.GetCellValue("Sheet1", "A1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
// GOARCH=amd64 - all ok; GOARCH=386 - actual: "-2147483648"
|
|
|
|
assert.Equal(t, "8595602512225", val, "A1 should be 8595602512225")
|
|
|
|
assert.NoError(t, f.Close())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSetCellFormula(t *testing.T) {
|
|
|
|
f, err := OpenFile(filepath.Join("test", "Book1.xlsx"))
|
|
|
|
if !assert.NoError(t, err) {
|
|
|
|
t.FailNow()
|
|
|
|
}
|
|
|
|
|
|
|
|
assert.NoError(t, f.SetCellFormula("Sheet1", "B19", "SUM(Sheet2!D2,Sheet2!D11)"))
|
|
|
|
assert.NoError(t, f.SetCellFormula("Sheet1", "C19", "SUM(Sheet2!D2,Sheet2!D9)"))
|
|
|
|
|
|
|
|
// Test set cell formula with illegal rows number.
|
|
|
|
assert.EqualError(t, f.SetCellFormula("Sheet1", "C", "SUM(Sheet2!D2,Sheet2!D9)"), newCellNameToCoordinatesError("C", newInvalidCellNameError("C")).Error())
|
|
|
|
|
|
|
|
assert.NoError(t, f.SaveAs(filepath.Join("test", "TestSetCellFormula1.xlsx")))
|
|
|
|
assert.NoError(t, f.Close())
|
|
|
|
|
|
|
|
f, err = OpenFile(filepath.Join("test", "CalcChain.xlsx"))
|
|
|
|
if !assert.NoError(t, err) {
|
|
|
|
t.FailNow()
|
|
|
|
}
|
|
|
|
// Test remove cell formula.
|
|
|
|
assert.NoError(t, f.SetCellFormula("Sheet1", "A1", ""))
|
|
|
|
assert.NoError(t, f.SaveAs(filepath.Join("test", "TestSetCellFormula2.xlsx")))
|
|
|
|
// Test remove all cell formula.
|
|
|
|
assert.NoError(t, f.SetCellFormula("Sheet1", "B1", ""))
|
|
|
|
assert.NoError(t, f.SaveAs(filepath.Join("test", "TestSetCellFormula3.xlsx")))
|
|
|
|
assert.NoError(t, f.Close())
|
|
|
|
|
|
|
|
// Test set shared formula for the cells.
|
|
|
|
f = NewFile()
|
|
|
|
for r := 1; r <= 5; r++ {
|
|
|
|
assert.NoError(t, f.SetSheetRow("Sheet1", fmt.Sprintf("A%d", r), &[]interface{}{r, r + 1}))
|
|
|
|
}
|
|
|
|
formulaType, ref := STCellFormulaTypeShared, "C1:C5"
|
|
|
|
assert.NoError(t, f.SetCellFormula("Sheet1", "C1", "=A1+B1", FormulaOpts{Ref: &ref, Type: &formulaType}))
|
|
|
|
sharedFormulaSpreadsheet := filepath.Join("test", "TestSetCellFormula4.xlsx")
|
|
|
|
assert.NoError(t, f.SaveAs(sharedFormulaSpreadsheet))
|
|
|
|
|
|
|
|
f, err = OpenFile(sharedFormulaSpreadsheet)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
ref = "D1:D5"
|
|
|
|
assert.NoError(t, f.SetCellFormula("Sheet1", "D1", "=A1+C1", FormulaOpts{Ref: &ref, Type: &formulaType}))
|
|
|
|
ref = ""
|
|
|
|
assert.EqualError(t, f.SetCellFormula("Sheet1", "D1", "=A1+C1", FormulaOpts{Ref: &ref, Type: &formulaType}), ErrParameterInvalid.Error())
|
|
|
|
assert.NoError(t, f.SaveAs(filepath.Join("test", "TestSetCellFormula5.xlsx")))
|
|
|
|
|
|
|
|
// Test set table formula for the cells.
|
|
|
|
f = NewFile()
|
|
|
|
for idx, row := range [][]interface{}{{"A", "B", "C"}, {1, 2}} {
|
|
|
|
assert.NoError(t, f.SetSheetRow("Sheet1", fmt.Sprintf("A%d", idx+1), &row))
|
|
|
|
}
|
|
|
|
assert.NoError(t, f.AddTable("Sheet1", "A1", "C2", `{"table_name":"Table1","table_style":"TableStyleMedium2"}`))
|
|
|
|
formulaType = STCellFormulaTypeDataTable
|
|
|
|
assert.NoError(t, f.SetCellFormula("Sheet1", "C2", "=SUM(Table1[[A]:[B]])", FormulaOpts{Type: &formulaType}))
|
|
|
|
assert.NoError(t, f.SaveAs(filepath.Join("test", "TestSetCellFormula6.xlsx")))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetCellRichText(t *testing.T) {
|
|
|
|
f := NewFile()
|
|
|
|
|
|
|
|
runsSource := []RichTextRun{
|
|
|
|
{
|
|
|
|
Text: "a\n",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Text: "b",
|
|
|
|
Font: &Font{
|
|
|
|
Underline: "single",
|
|
|
|
Color: "ff0000",
|
|
|
|
Bold: true,
|
|
|
|
Italic: true,
|
|
|
|
Family: "Times New Roman",
|
|
|
|
Size: 100,
|
|
|
|
Strike: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
assert.NoError(t, f.SetCellRichText("Sheet1", "A1", runsSource))
|
|
|
|
assert.NoError(t, f.SetCellValue("Sheet1", "A2", false))
|
|
|
|
|
|
|
|
runs, err := f.GetCellRichText("Sheet1", "A2")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, []RichTextRun(nil), runs)
|
|
|
|
|
|
|
|
runs, err = f.GetCellRichText("Sheet1", "A1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Equal(t, runsSource[0].Text, runs[0].Text)
|
|
|
|
assert.Nil(t, runs[0].Font)
|
|
|
|
assert.NotNil(t, runs[1].Font)
|
|
|
|
|
|
|
|
runsSource[1].Font.Color = strings.ToUpper(runsSource[1].Font.Color)
|
|
|
|
assert.True(t, reflect.DeepEqual(runsSource[1].Font, runs[1].Font), "should get the same font")
|
|
|
|
|
|
|
|
// Test get cell rich text when string item index overflow
|
|
|
|
ws, ok := f.Sheet.Load("xl/worksheets/sheet1.xml")
|
|
|
|
assert.True(t, ok)
|
|
|
|
ws.(*xlsxWorksheet).SheetData.Row[0].C[0].V = "2"
|
|
|
|
runs, err = f.GetCellRichText("Sheet1", "A1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, 0, len(runs))
|
|
|
|
// Test get cell rich text when string item index is negative
|
|
|
|
ws, ok = f.Sheet.Load("xl/worksheets/sheet1.xml")
|
|
|
|
assert.True(t, ok)
|
|
|
|
ws.(*xlsxWorksheet).SheetData.Row[0].C[0].V = "-1"
|
|
|
|
runs, err = f.GetCellRichText("Sheet1", "A1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, 0, len(runs))
|
|
|
|
// Test get cell rich text on invalid string item index
|
|
|
|
ws, ok = f.Sheet.Load("xl/worksheets/sheet1.xml")
|
|
|
|
assert.True(t, ok)
|
|
|
|
ws.(*xlsxWorksheet).SheetData.Row[0].C[0].V = "x"
|
|
|
|
_, err = f.GetCellRichText("Sheet1", "A1")
|
|
|
|
assert.EqualError(t, err, "strconv.Atoi: parsing \"x\": invalid syntax")
|
|
|
|
// Test set cell rich text on not exists worksheet
|
|
|
|
_, err = f.GetCellRichText("SheetN", "A1")
|
|
|
|
assert.EqualError(t, err, "sheet SheetN is not exist")
|
|
|
|
// Test set cell rich text with illegal cell coordinates
|
|
|
|
_, err = f.GetCellRichText("Sheet1", "A")
|
|
|
|
assert.EqualError(t, err, newCellNameToCoordinatesError("A", newInvalidCellNameError("A")).Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSetCellRichText(t *testing.T) {
|
|
|
|
f := NewFile()
|
|
|
|
assert.NoError(t, f.SetRowHeight("Sheet1", 1, 35))
|
|
|
|
assert.NoError(t, f.SetColWidth("Sheet1", "A", "A", 44))
|
|
|
|
richTextRun := []RichTextRun{
|
|
|
|
{
|
|
|
|
Text: "bold",
|
|
|
|
Font: &Font{
|
|
|
|
Bold: true,
|
|
|
|
Color: "2354e8",
|
|
|
|
Family: "Times New Roman",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Text: " and ",
|
|
|
|
Font: &Font{
|
|
|
|
Family: "Times New Roman",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Text: "italic ",
|
|
|
|
Font: &Font{
|
|
|
|
Bold: true,
|
|
|
|
Color: "e83723",
|
|
|
|
Italic: true,
|
|
|
|
Family: "Times New Roman",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Text: "text with color and font-family, ",
|
|
|
|
Font: &Font{
|
|
|
|
Bold: true,
|
|
|
|
Color: "2354e8",
|
|
|
|
Family: "Times New Roman",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Text: "\r\nlarge text with ",
|
|
|
|
Font: &Font{
|
|
|
|
Size: 14,
|
|
|
|
Color: "ad23e8",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Text: "strike",
|
|
|
|
Font: &Font{
|
|
|
|
Color: "e89923",
|
|
|
|
Strike: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Text: " superscript",
|
|
|
|
Font: &Font{
|
|
|
|
Color: "dbc21f",
|
|
|
|
VertAlign: "superscript",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Text: " and ",
|
|
|
|
Font: &Font{
|
|
|
|
Size: 14,
|
|
|
|
Color: "ad23e8",
|
|
|
|
VertAlign: "BASELINE",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Text: "underline",
|
|
|
|
Font: &Font{
|
|
|
|
Color: "23e833",
|
|
|
|
Underline: "single",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Text: " subscript.",
|
|
|
|
Font: &Font{
|
|
|
|
Color: "017505",
|
|
|
|
VertAlign: "subscript",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
assert.NoError(t, f.SetCellRichText("Sheet1", "A1", richTextRun))
|
|
|
|
assert.NoError(t, f.SetCellRichText("Sheet1", "A2", richTextRun))
|
|
|
|
style, err := f.NewStyle(&Style{
|
|
|
|
Alignment: &Alignment{
|
|
|
|
WrapText: true,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.NoError(t, f.SetCellStyle("Sheet1", "A1", "A1", style))
|
|
|
|
assert.NoError(t, f.SaveAs(filepath.Join("test", "TestSetCellRichText.xlsx")))
|
|
|
|
// Test set cell rich text on not exists worksheet
|
|
|
|
assert.EqualError(t, f.SetCellRichText("SheetN", "A1", richTextRun), "sheet SheetN is not exist")
|
|
|
|
// Test set cell rich text with illegal cell coordinates
|
|
|
|
assert.EqualError(t, f.SetCellRichText("Sheet1", "A", richTextRun), newCellNameToCoordinatesError("A", newInvalidCellNameError("A")).Error())
|
|
|
|
richTextRun = []RichTextRun{{Text: strings.Repeat("s", TotalCellChars+1)}}
|
|
|
|
// Test set cell rich text with characters over the maximum limit
|
|
|
|
assert.EqualError(t, f.SetCellRichText("Sheet1", "A1", richTextRun), ErrCellCharsLength.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestFormattedValue2(t *testing.T) {
|
|
|
|
f := NewFile()
|
|
|
|
v := f.formattedValue(0, "43528", false)
|
|
|
|
assert.Equal(t, "43528", v)
|
|
|
|
|
|
|
|
v = f.formattedValue(15, "43528", false)
|
|
|
|
assert.Equal(t, "43528", v)
|
|
|
|
|
|
|
|
v = f.formattedValue(1, "43528", false)
|
|
|
|
assert.Equal(t, "43528", v)
|
|
|
|
customNumFmt := "[$-409]MM/DD/YYYY"
|
|
|
|
_, err := f.NewStyle(&Style{
|
|
|
|
CustomNumFmt: &customNumFmt,
|
|
|
|
})
|
|
|
|
assert.NoError(t, err)
|
|
|
|
v = f.formattedValue(1, "43528", false)
|
|
|
|
assert.Equal(t, "03/04/2019", v)
|
|
|
|
|
|
|
|
// formatted value with no built-in number format ID
|
|
|
|
numFmtID := 5
|
|
|
|
f.Styles.CellXfs.Xf = append(f.Styles.CellXfs.Xf, xlsxXf{
|
|
|
|
NumFmtID: &numFmtID,
|
|
|
|
})
|
|
|
|
v = f.formattedValue(2, "43528", false)
|
|
|
|
assert.Equal(t, "43528", v)
|
|
|
|
|
|
|
|
// formatted value with invalid number format ID
|
|
|
|
f.Styles.CellXfs.Xf = append(f.Styles.CellXfs.Xf, xlsxXf{
|
|
|
|
NumFmtID: nil,
|
|
|
|
})
|
|
|
|
_ = f.formattedValue(3, "43528", false)
|
|
|
|
|
|
|
|
// formatted value with empty number format
|
|
|
|
f.Styles.NumFmts = nil
|
|
|
|
f.Styles.CellXfs.Xf = append(f.Styles.CellXfs.Xf, xlsxXf{
|
|
|
|
NumFmtID: &numFmtID,
|
|
|
|
})
|
|
|
|
v = f.formattedValue(1, "43528", false)
|
|
|
|
assert.Equal(t, "43528", v)
|
|
|
|
|
|
|
|
// formatted decimal value with build-in number format ID
|
|
|
|
styleID, err := f.NewStyle(&Style{
|
|
|
|
NumFmt: 1,
|
|
|
|
})
|
|
|
|
assert.NoError(t, err)
|
|
|
|
v = f.formattedValue(styleID, "310.56", false)
|
|
|
|
assert.Equal(t, "311", v)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSharedStringsError(t *testing.T) {
|
|
|
|
f, err := OpenFile(filepath.Join("test", "Book1.xlsx"), Options{UnzipXMLSizeLimit: 128})
|
|
|
|
assert.NoError(t, err)
|
|
|
|
tempFile, ok := f.tempFiles.Load(defaultXMLPathSharedStrings)
|
|
|
|
assert.True(t, ok)
|
|
|
|
f.tempFiles.Store(defaultXMLPathSharedStrings, "")
|
|
|
|
assert.Equal(t, "1", f.getFromStringItem(1))
|
|
|
|
// Cleanup undelete temporary files
|
|
|
|
assert.NoError(t, os.Remove(tempFile.(string)))
|
|
|
|
// Test reload the file error on set cell cell and rich text. The error message was different between macOS and Windows.
|
|
|
|
err = f.SetCellValue("Sheet1", "A19", "A19")
|
|
|
|
assert.Error(t, err)
|
|
|
|
|
|
|
|
f.tempFiles.Store(defaultXMLPathSharedStrings, "")
|
|
|
|
err = f.SetCellRichText("Sheet1", "A19", []RichTextRun{})
|
|
|
|
assert.Error(t, err)
|
|
|
|
assert.NoError(t, f.Close())
|
|
|
|
|
|
|
|
f, err = OpenFile(filepath.Join("test", "Book1.xlsx"), Options{UnzipXMLSizeLimit: 128})
|
|
|
|
assert.NoError(t, err)
|
|
|
|
rows, err := f.Rows("Sheet1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
const maxUint16 = 1<<16 - 1
|
|
|
|
currentRow := 0
|
|
|
|
for rows.Next() {
|
|
|
|
currentRow++
|
|
|
|
if currentRow == 19 {
|
|
|
|
_, err := rows.Columns()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
// Test get cell value from string item with invalid offset
|
|
|
|
f.sharedStringItem[1] = []uint{maxUint16 - 1, maxUint16}
|
|
|
|
assert.Equal(t, "1", f.getFromStringItem(1))
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert.NoError(t, rows.Close())
|
|
|
|
// Test shared string item temporary files has been closed before close the workbook
|
|
|
|
assert.NoError(t, f.sharedStringTemp.Close())
|
|
|
|
assert.Error(t, f.Close())
|
|
|
|
// Cleanup undelete temporary files
|
|
|
|
f.tempFiles.Range(func(k, v interface{}) bool {
|
|
|
|
return assert.NoError(t, os.Remove(v.(string)))
|
|
|
|
})
|
|
|
|
|
|
|
|
f, err = OpenFile(filepath.Join("test", "Book1.xlsx"), Options{UnzipXMLSizeLimit: 128})
|
|
|
|
assert.NoError(t, err)
|
|
|
|
rows, err = f.Rows("Sheet1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
currentRow = 0
|
|
|
|
for rows.Next() {
|
|
|
|
currentRow++
|
|
|
|
if currentRow == 19 {
|
|
|
|
_, err := rows.Columns()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert.NoError(t, rows.Close())
|
|
|
|
assert.NoError(t, f.sharedStringTemp.Close())
|
|
|
|
// Test shared string item temporary files has been closed before set the cell value
|
|
|
|
assert.Error(t, f.SetCellValue("Sheet1", "A1", "A1"))
|
|
|
|
assert.Error(t, f.Close())
|
|
|
|
// Cleanup undelete temporary files
|
|
|
|
f.tempFiles.Range(func(k, v interface{}) bool {
|
|
|
|
return assert.NoError(t, os.Remove(v.(string)))
|
|
|
|
})
|
|
|
|
}
|