2021-02-07 22:51:20 -05:00
|
|
|
package xls
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/binary"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"math"
|
|
|
|
"time"
|
|
|
|
"unicode/utf16"
|
|
|
|
)
|
|
|
|
|
|
|
|
func (b *WorkBook) Sheets() []string {
|
2021-02-08 11:02:37 -05:00
|
|
|
res := make([]string, 0, len(b.sheets))
|
|
|
|
for _, s := range b.sheets {
|
|
|
|
if (s.HiddenState & 0x03) == 0 {
|
|
|
|
res = append(res, s.Name)
|
|
|
|
}
|
2021-02-07 22:51:20 -05:00
|
|
|
}
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *WorkBook) Get(sheetName string) (*WorkSheet, error) {
|
|
|
|
for _, s := range b.sheets {
|
|
|
|
if s.Name == sheetName {
|
|
|
|
ss := b.pos2substream[int64(s.Position)]
|
|
|
|
ws := &WorkSheet{
|
|
|
|
b: b, s: s, ss: ss,
|
2021-02-09 01:28:22 -05:00
|
|
|
iterRow: -1,
|
2021-02-07 22:51:20 -05:00
|
|
|
}
|
|
|
|
return ws, ws.parse()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, errors.New("xls: sheet not found")
|
|
|
|
}
|
|
|
|
|
|
|
|
type WorkSheet struct {
|
|
|
|
b *WorkBook
|
|
|
|
s *boundSheet
|
|
|
|
ss int
|
|
|
|
err error
|
|
|
|
|
2021-02-10 01:32:41 -05:00
|
|
|
minRow int
|
|
|
|
maxRow int // maximum valid row index (0xFFFF)
|
|
|
|
minCol int
|
|
|
|
maxCol int // maximum valid column index (0xFF)
|
2021-02-07 22:51:20 -05:00
|
|
|
rows []*row
|
2021-02-08 11:02:37 -05:00
|
|
|
empty bool
|
2021-02-07 22:51:20 -05:00
|
|
|
|
|
|
|
iterRow int
|
2021-02-09 01:11:41 -05:00
|
|
|
iterMC int
|
2021-02-07 22:51:20 -05:00
|
|
|
}
|
|
|
|
|
2021-02-09 01:11:41 -05:00
|
|
|
type staticCellType rune
|
2021-02-07 22:51:20 -05:00
|
|
|
|
2021-02-09 01:11:41 -05:00
|
|
|
const (
|
|
|
|
staticBlank staticCellType = 0
|
2021-02-07 22:51:20 -05:00
|
|
|
|
2021-02-09 01:11:41 -05:00
|
|
|
// marks a continuation column within a merged cell.
|
|
|
|
continueColumnMerged staticCellType = '→'
|
|
|
|
// marks the last column of a merged cell.
|
|
|
|
endColumnMerged staticCellType = '⇥'
|
|
|
|
|
|
|
|
// marks a continuation row within a merged cell.
|
|
|
|
continueRowMerged staticCellType = '↓'
|
|
|
|
// marks the last row of a merged cell.
|
|
|
|
endRowMerged staticCellType = '⤓'
|
|
|
|
)
|
|
|
|
|
|
|
|
func (s staticCellType) String() string {
|
|
|
|
if s == 0 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return string([]rune{rune(s)})
|
2021-02-07 22:51:20 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
type row struct {
|
|
|
|
// each value must be one of: int, float64, string, or time.Time
|
|
|
|
cols []interface{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *WorkSheet) placeValue(rowIndex, colIndex int, val interface{}) {
|
2021-02-10 01:32:41 -05:00
|
|
|
if colIndex > s.maxCol || rowIndex > s.maxRow {
|
2021-02-09 13:59:58 -05:00
|
|
|
// invalid
|
|
|
|
return
|
2021-02-07 22:51:20 -05:00
|
|
|
}
|
|
|
|
|
2021-02-09 13:59:58 -05:00
|
|
|
// ensure we always have a complete matrix
|
2021-02-07 22:51:20 -05:00
|
|
|
for len(s.rows) <= rowIndex {
|
2021-02-10 01:32:41 -05:00
|
|
|
emptyRow := make([]interface{}, s.maxCol+1)
|
|
|
|
for i := 0; i <= s.maxCol; i++ {
|
2021-02-07 22:51:20 -05:00
|
|
|
emptyRow[i] = staticBlank
|
|
|
|
}
|
|
|
|
s.rows = append(s.rows, &row{emptyRow})
|
|
|
|
}
|
|
|
|
|
|
|
|
s.rows[rowIndex].cols[colIndex] = val
|
|
|
|
}
|
|
|
|
|
2021-02-08 11:02:37 -05:00
|
|
|
func (s *WorkSheet) IsEmpty() bool {
|
|
|
|
return s.empty
|
|
|
|
}
|
|
|
|
|
2021-02-07 22:51:20 -05:00
|
|
|
func (s *WorkSheet) parse() error {
|
2021-02-10 01:32:41 -05:00
|
|
|
inSubstream := 0
|
|
|
|
for idx, r := range s.b.substreams[s.ss] {
|
|
|
|
if inSubstream > 0 {
|
|
|
|
if r.RecType == RecTypeEOF {
|
|
|
|
inSubstream--
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
switch r.RecType {
|
|
|
|
case RecTypeBOF:
|
|
|
|
if idx > 0 {
|
|
|
|
inSubstream++
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
case RecTypeWsBool:
|
2021-02-08 11:02:37 -05:00
|
|
|
if (r.Data[1] & 0x10) != 0 {
|
|
|
|
// it's a dialog
|
|
|
|
return nil
|
|
|
|
}
|
2021-02-10 01:32:41 -05:00
|
|
|
|
|
|
|
case RecTypeDimensions:
|
|
|
|
bb := bytes.NewReader(r.Data)
|
|
|
|
var minRow, maxRow uint32
|
|
|
|
var minCol, maxCol uint16
|
|
|
|
|
|
|
|
// max = 0-based index of the row AFTER the last valid index
|
|
|
|
binary.Read(bb, binary.LittleEndian, &minRow)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &maxRow) // max = 0x010000
|
|
|
|
binary.Read(bb, binary.LittleEndian, &minCol)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &maxCol) // max = 0x000100
|
|
|
|
//log.Printf("dimensions: %d,%d + %dx%d", minRow&0x0000FFFF, minCol,
|
|
|
|
// (maxRow&0x0000FFFF)-(minRow&0x0000FFFF), maxCol-minCol)
|
|
|
|
if minRow > 0x0000FFFF || maxRow > 0x00010000 {
|
|
|
|
log.Println("invalid dimensions")
|
|
|
|
}
|
|
|
|
if minCol > 0x00FF || maxCol > 0x0100 {
|
|
|
|
log.Println("invalid dimensions")
|
|
|
|
}
|
|
|
|
s.minRow = int(uint64(minRow) & 0x0FFFF)
|
|
|
|
s.maxRow = int(uint64(maxRow)&0x1FFFF) - 1 // translate to last valid index
|
|
|
|
s.minCol = int(uint64(minCol) & 0x000FF)
|
|
|
|
s.maxCol = int(uint64(maxCol)&0x001FF) - 1 // translate to last valid index
|
|
|
|
//log.Println(s.minCol, s.maxCol)
|
|
|
|
if (maxRow-minRow) == 0 || (maxCol-minCol) == 0 {
|
|
|
|
s.empty = true
|
|
|
|
} else {
|
|
|
|
// pre-allocate cells
|
|
|
|
s.placeValue(s.maxRow, s.maxCol, staticBlank)
|
|
|
|
}
|
|
|
|
|
|
|
|
case RecTypeRow:
|
|
|
|
bb := bytes.NewReader(r.Data)
|
|
|
|
row := &shRow{}
|
|
|
|
binary.Read(bb, binary.LittleEndian, row)
|
|
|
|
if (row.Reserved & 0xFFFF) != 0 {
|
|
|
|
log.Println("invalid Row spec")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
//log.Printf("row spec: %+v", *row)
|
2021-02-08 11:02:37 -05:00
|
|
|
}
|
|
|
|
}
|
2021-02-10 01:32:41 -05:00
|
|
|
inSubstream = 0
|
2021-02-08 11:02:37 -05:00
|
|
|
|
|
|
|
var formulaRow, formulaCol uint16
|
2021-02-08 15:36:08 -05:00
|
|
|
for ridx, r := range s.b.substreams[s.ss] {
|
2021-02-10 01:32:41 -05:00
|
|
|
if inSubstream > 0 {
|
|
|
|
if r.RecType == RecTypeEOF {
|
|
|
|
inSubstream--
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2021-02-07 22:51:20 -05:00
|
|
|
bb := bytes.NewReader(r.Data)
|
2021-02-08 15:36:08 -05:00
|
|
|
//log.Println(ridx, r.RecType)
|
|
|
|
|
|
|
|
// sec 2.1.7.20.6 Common Productions ABNF:
|
|
|
|
/*
|
|
|
|
CELLTABLE = 1*(1*Row *CELL 1*DBCell) *EntExU2
|
|
|
|
CELL = FORMULA / Blank / MulBlank / RK / MulRk / BoolErr / Number / LabelSst
|
|
|
|
FORMULA = [Uncalced] Formula [Array / Table / ShrFmla / SUB] [String *Continue]
|
|
|
|
|
|
|
|
Not parsed form the list above:
|
|
|
|
DBCell, EntExU2, Uncalced, Array, Table,ShrFmla
|
|
|
|
NB: no idea what "SUB" is
|
|
|
|
*/
|
2021-02-07 22:51:20 -05:00
|
|
|
|
|
|
|
switch r.RecType {
|
2021-02-10 01:32:41 -05:00
|
|
|
case RecTypeBOF:
|
|
|
|
if ridx > 0 {
|
|
|
|
inSubstream++
|
2021-02-08 11:02:37 -05:00
|
|
|
continue
|
|
|
|
}
|
2021-02-07 22:51:20 -05:00
|
|
|
case RecTypeBlank:
|
|
|
|
var rowIndex, colIndex uint16
|
|
|
|
binary.Read(bb, binary.LittleEndian, &rowIndex)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &colIndex)
|
2021-02-08 11:02:37 -05:00
|
|
|
//log.Printf("blank spec: %d %d", rowIndex, colIndex)
|
|
|
|
|
|
|
|
case RecTypeBoolErr:
|
|
|
|
var rowIndex, colIndex, ixfe uint16
|
|
|
|
binary.Read(bb, binary.LittleEndian, &rowIndex)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &colIndex)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &ixfe)
|
|
|
|
if r.Data[7] == 0 {
|
|
|
|
bv := false
|
|
|
|
if r.Data[6] == 1 {
|
|
|
|
bv = true
|
|
|
|
}
|
|
|
|
s.placeValue(int(rowIndex), int(colIndex), bv)
|
|
|
|
//log.Printf("bool/error spec: %d %d %+v", rowIndex, colIndex, bv)
|
|
|
|
} else {
|
|
|
|
be, ok := berrLookup[r.Data[6]]
|
|
|
|
if !ok {
|
|
|
|
be = "<unknown error>"
|
|
|
|
}
|
|
|
|
s.placeValue(int(rowIndex), int(colIndex), be)
|
|
|
|
//log.Printf("bool/error spec: %d %d %s", rowIndex, colIndex, be)
|
|
|
|
}
|
|
|
|
|
2021-02-07 22:51:20 -05:00
|
|
|
case RecTypeMulBlank:
|
|
|
|
var rowIndex, firstCol uint16
|
|
|
|
binary.Read(bb, binary.LittleEndian, &rowIndex)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &firstCol)
|
2021-02-08 11:02:37 -05:00
|
|
|
// nrk := int((r.RecSize - 6) / 6)
|
|
|
|
// log.Printf("row blanks spec: %d %d %d", rowIndex, firstCol, nrk)
|
|
|
|
|
2021-02-07 22:51:20 -05:00
|
|
|
case RecTypeMulRk:
|
|
|
|
mr := &shMulRK{}
|
|
|
|
nrk := int((r.RecSize - 6) / 6)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &mr.RowIndex)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &mr.FirstCol)
|
|
|
|
mr.Values = make([]RkRec, nrk)
|
|
|
|
for i := 0; i < nrk; i++ {
|
|
|
|
rr := RkRec{}
|
|
|
|
binary.Read(bb, binary.LittleEndian, &rr)
|
|
|
|
mr.Values[i] = rr
|
|
|
|
|
|
|
|
var rval interface{}
|
|
|
|
if rr.Value.IsInteger() {
|
|
|
|
rval = rr.Value.Int()
|
|
|
|
} else {
|
|
|
|
rval = rr.Value.Float64()
|
2021-02-10 01:33:50 -05:00
|
|
|
|
|
|
|
fno := s.b.xfs[rr.IXFCell]
|
|
|
|
format := s.b.formats[fno]
|
|
|
|
if formatIsDateTime(fno, format) {
|
|
|
|
dt, ok := getDateTime(fno, format, rr.Value.Float64(), s.b.dateMode == 1)
|
|
|
|
if ok {
|
|
|
|
rval = dt
|
|
|
|
}
|
|
|
|
//log.Println("apply format: ", format, rval, int(mr.RowIndex), int(mr.FirstCol)+i)
|
|
|
|
}
|
2021-02-07 22:51:20 -05:00
|
|
|
}
|
|
|
|
s.placeValue(int(mr.RowIndex), int(mr.FirstCol)+i, rval)
|
|
|
|
}
|
|
|
|
binary.Read(bb, binary.LittleEndian, &mr.LastCol)
|
2021-02-08 11:02:37 -05:00
|
|
|
//log.Printf("mulrow spec: %+v", *mr)
|
2021-02-07 22:51:20 -05:00
|
|
|
|
|
|
|
case RecTypeNumber:
|
|
|
|
var rowIndex, colIndex, ixfe uint16
|
|
|
|
var xnum uint64
|
|
|
|
binary.Read(bb, binary.LittleEndian, &rowIndex)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &colIndex)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &ixfe)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &xnum)
|
|
|
|
value := math.Float64frombits(xnum)
|
2021-02-10 01:33:50 -05:00
|
|
|
var rval interface{} = value
|
|
|
|
|
|
|
|
fno := s.b.xfs[ixfe]
|
|
|
|
format := s.b.formats[fno]
|
|
|
|
if formatIsDateTime(fno, format) {
|
|
|
|
dt, ok := getDateTime(fno, format, value, s.b.dateMode == 1)
|
|
|
|
if ok {
|
|
|
|
rval = dt
|
|
|
|
}
|
|
|
|
//log.Println("apply format: ", format, rval)
|
|
|
|
}
|
|
|
|
s.placeValue(int(rowIndex), int(colIndex), rval)
|
2021-02-08 11:02:37 -05:00
|
|
|
//log.Printf("Number spec: %d %d = %f", rowIndex, colIndex, value)
|
2021-02-07 22:51:20 -05:00
|
|
|
|
|
|
|
case RecTypeRK:
|
|
|
|
var rowIndex, colIndex uint16
|
|
|
|
binary.Read(bb, binary.LittleEndian, &rowIndex)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &colIndex)
|
|
|
|
rr := RkRec{}
|
|
|
|
binary.Read(bb, binary.LittleEndian, &rr)
|
|
|
|
|
|
|
|
var rval interface{}
|
|
|
|
if rr.Value.IsInteger() {
|
|
|
|
rval = rr.Value.Int()
|
|
|
|
} else {
|
|
|
|
rval = rr.Value.Float64()
|
2021-02-10 01:33:50 -05:00
|
|
|
|
|
|
|
fno := s.b.xfs[rr.IXFCell]
|
|
|
|
format := s.b.formats[fno]
|
|
|
|
if formatIsDateTime(fno, format) {
|
|
|
|
dt, ok := getDateTime(fno, format, rr.Value.Float64(), s.b.dateMode == 1)
|
|
|
|
if ok {
|
|
|
|
rval = dt
|
|
|
|
}
|
|
|
|
//log.Println("apply format: ", format, rval)
|
|
|
|
}
|
2021-02-07 22:51:20 -05:00
|
|
|
}
|
|
|
|
s.placeValue(int(rowIndex), int(colIndex), rval)
|
2021-02-08 11:02:37 -05:00
|
|
|
//log.Printf("RK spec: %d %d = %s", rowIndex, colIndex, rr.Value.String())
|
2021-02-07 22:51:20 -05:00
|
|
|
|
|
|
|
case RecTypeFormula:
|
2021-02-08 11:02:37 -05:00
|
|
|
var ixfe uint16
|
|
|
|
binary.Read(bb, binary.LittleEndian, &formulaRow)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &formulaCol)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &ixfe)
|
|
|
|
fdata := r.Data[6:]
|
|
|
|
if fdata[6] == 0xFF && r.Data[7] == 0xFF {
|
|
|
|
switch fdata[0] {
|
|
|
|
case 0:
|
|
|
|
// string in next record
|
|
|
|
case 1:
|
|
|
|
// boolean
|
|
|
|
bv := false
|
|
|
|
if fdata[2] != 0 {
|
|
|
|
bv = true
|
|
|
|
}
|
|
|
|
s.placeValue(int(formulaRow), int(formulaCol), bv)
|
|
|
|
case 2:
|
|
|
|
// error value
|
|
|
|
be, ok := berrLookup[fdata[2]]
|
|
|
|
if !ok {
|
|
|
|
be = "<unknown error>"
|
|
|
|
}
|
|
|
|
s.placeValue(int(formulaRow), int(formulaCol), be)
|
|
|
|
case 3:
|
|
|
|
// blank string
|
|
|
|
default:
|
|
|
|
log.Println("unknown formula value type")
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
var xnum uint64
|
|
|
|
binary.Read(bb, binary.LittleEndian, &xnum)
|
|
|
|
value := math.Float64frombits(xnum)
|
2021-02-10 01:33:50 -05:00
|
|
|
var rval interface{} = value
|
|
|
|
|
|
|
|
fno := s.b.xfs[ixfe]
|
|
|
|
format := s.b.formats[fno]
|
|
|
|
if formatIsDateTime(fno, format) {
|
|
|
|
dt, ok := getDateTime(fno, format, value, s.b.dateMode == 1)
|
|
|
|
if ok {
|
|
|
|
rval = dt
|
|
|
|
}
|
|
|
|
//log.Println("apply format: ", format, rval)
|
|
|
|
}
|
|
|
|
s.placeValue(int(formulaRow), int(formulaCol), rval)
|
2021-02-08 11:02:37 -05:00
|
|
|
}
|
|
|
|
//log.Printf("formula spec: %d %d ~~ %+v", formulaRow, formulaCol, r.Data)
|
2021-02-07 22:51:20 -05:00
|
|
|
|
|
|
|
case RecTypeString:
|
2021-02-08 11:02:37 -05:00
|
|
|
var charCount uint16
|
|
|
|
var flags byte
|
2021-02-07 22:51:20 -05:00
|
|
|
binary.Read(bb, binary.LittleEndian, &charCount)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &flags)
|
2021-02-08 11:02:37 -05:00
|
|
|
fstr := ""
|
2021-02-07 22:51:20 -05:00
|
|
|
if (flags & 1) == 0 {
|
2021-02-08 11:02:37 -05:00
|
|
|
fstr = string(r.Data[3:])
|
2021-02-07 22:51:20 -05:00
|
|
|
} else {
|
|
|
|
us := make([]uint16, charCount)
|
|
|
|
binary.Read(bb, binary.LittleEndian, us)
|
2021-02-08 11:02:37 -05:00
|
|
|
fstr = string(utf16.Decode(us))
|
2021-02-07 22:51:20 -05:00
|
|
|
}
|
2021-02-08 15:36:08 -05:00
|
|
|
|
|
|
|
if (ridx + 1) < len(s.b.substreams[s.ss]) {
|
|
|
|
ridx2 := ridx + 1
|
|
|
|
nrecs := len(s.b.substreams[s.ss])
|
|
|
|
for ridx2 < nrecs {
|
|
|
|
r2 := s.b.substreams[s.ss][ridx2]
|
|
|
|
if r2.RecType != RecTypeContinue {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if (r2.Data[0] & 1) == 0 {
|
|
|
|
fstr += string(r2.Data[1:])
|
|
|
|
} else {
|
|
|
|
bb2 := bytes.NewReader(r2.Data[1:])
|
|
|
|
us := make([]uint16, len(r2.Data)-1)
|
|
|
|
binary.Read(bb2, binary.LittleEndian, us)
|
|
|
|
fstr += string(utf16.Decode(us))
|
|
|
|
}
|
|
|
|
ridx2++
|
|
|
|
}
|
|
|
|
}
|
2021-02-10 01:33:50 -05:00
|
|
|
// TODO: does formula record formatted dates as pre-computed strings?
|
2021-02-08 11:02:37 -05:00
|
|
|
s.placeValue(int(formulaRow), int(formulaCol), fstr)
|
2021-02-07 22:51:20 -05:00
|
|
|
|
|
|
|
case RecTypeLabelSst:
|
|
|
|
var rowIndex, colIndex, ixfe uint16
|
|
|
|
var sstIndex uint32
|
|
|
|
binary.Read(bb, binary.LittleEndian, &rowIndex)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &colIndex)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &ixfe)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &sstIndex)
|
2021-02-08 11:02:37 -05:00
|
|
|
if int(sstIndex) > len(s.b.strings) {
|
2021-02-08 15:36:08 -05:00
|
|
|
return errors.New("xls: invalid sst index")
|
2021-02-08 11:02:37 -05:00
|
|
|
}
|
2021-02-07 22:51:20 -05:00
|
|
|
s.placeValue(int(rowIndex), int(colIndex), s.b.strings[sstIndex])
|
2021-02-08 11:02:37 -05:00
|
|
|
//log.Printf("SST spec: %d %d = [%d] %s", rowIndex, colIndex, sstIndex, s.b.strings[sstIndex])
|
2021-02-07 22:51:20 -05:00
|
|
|
|
|
|
|
case RecTypeHLink:
|
|
|
|
loc := &shRef8{}
|
|
|
|
binary.Read(bb, binary.LittleEndian, loc)
|
2021-02-10 01:32:41 -05:00
|
|
|
if int(loc.FirstCol) > s.maxCol {
|
2021-02-08 15:36:08 -05:00
|
|
|
//log.Println("invalid hyperlink column")
|
|
|
|
continue
|
|
|
|
}
|
2021-02-10 01:32:41 -05:00
|
|
|
if int(loc.FirstRow) > s.maxRow {
|
2021-02-08 15:36:08 -05:00
|
|
|
//log.Println("invalid hyperlink row")
|
|
|
|
continue
|
|
|
|
}
|
2021-02-09 13:59:58 -05:00
|
|
|
if loc.LastRow == 0xFFFF {
|
|
|
|
loc.LastRow = uint16(s.maxRow)
|
|
|
|
}
|
|
|
|
if loc.LastCol == 0xFF {
|
2021-02-10 01:32:41 -05:00
|
|
|
loc.LastCol = uint16(s.maxCol)
|
2021-02-09 13:59:58 -05:00
|
|
|
}
|
2021-02-07 22:51:20 -05:00
|
|
|
|
2021-02-08 23:10:24 -05:00
|
|
|
displayText, linkText, err := decodeHyperlinks(bb)
|
|
|
|
if err != nil {
|
|
|
|
log.Println(err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2021-02-09 01:11:41 -05:00
|
|
|
// apply merge cell rules
|
2021-02-10 01:32:41 -05:00
|
|
|
for rn := int(loc.FirstRow); rn <= int(loc.LastRow); rn++ {
|
|
|
|
for cn := int(loc.FirstCol); cn <= int(loc.LastCol); cn++ {
|
|
|
|
if rn == int(loc.FirstRow) && cn == int(loc.FirstCol) {
|
|
|
|
s.placeValue(rn, cn, displayText+" <"+linkText+">")
|
|
|
|
} else if cn == int(loc.FirstCol) {
|
2021-02-09 01:11:41 -05:00
|
|
|
// first and last column MAY be the same
|
2021-02-10 01:32:41 -05:00
|
|
|
if rn == int(loc.LastRow) {
|
|
|
|
s.placeValue(rn, cn, endRowMerged)
|
2021-02-09 01:11:41 -05:00
|
|
|
} else {
|
2021-02-10 01:32:41 -05:00
|
|
|
s.placeValue(rn, cn, continueRowMerged)
|
2021-02-09 01:11:41 -05:00
|
|
|
}
|
2021-02-10 01:32:41 -05:00
|
|
|
} else if cn == int(loc.LastCol) {
|
2021-02-09 01:11:41 -05:00
|
|
|
// first and last column are NOT the same
|
2021-02-10 01:32:41 -05:00
|
|
|
s.placeValue(rn, cn, endColumnMerged)
|
2021-02-09 01:11:41 -05:00
|
|
|
} else {
|
2021-02-10 01:32:41 -05:00
|
|
|
s.placeValue(rn, cn, continueColumnMerged)
|
2021-02-09 01:11:41 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-02-07 22:51:20 -05:00
|
|
|
|
|
|
|
case RecTypeMergeCells:
|
|
|
|
var cmcs uint16
|
|
|
|
binary.Read(bb, binary.LittleEndian, &cmcs)
|
|
|
|
mcRefs := make([]shRef8, cmcs)
|
|
|
|
binary.Read(bb, binary.LittleEndian, &mcRefs)
|
2021-02-09 01:11:41 -05:00
|
|
|
for _, loc := range mcRefs {
|
2021-02-09 13:59:58 -05:00
|
|
|
if loc.LastRow == 0xFFFF {
|
|
|
|
loc.LastRow = uint16(s.maxRow)
|
|
|
|
}
|
|
|
|
if loc.LastCol == 0xFF {
|
2021-02-10 01:32:41 -05:00
|
|
|
loc.LastCol = uint16(s.maxCol)
|
2021-02-09 13:59:58 -05:00
|
|
|
}
|
2021-02-10 01:32:41 -05:00
|
|
|
for rn := int(loc.FirstRow); rn <= int(loc.LastRow); rn++ {
|
|
|
|
for cn := int(loc.FirstCol); cn <= int(loc.LastCol); cn++ {
|
|
|
|
if rn == int(loc.FirstRow) && cn == int(loc.FirstCol) {
|
2021-02-09 01:11:41 -05:00
|
|
|
// should be a value there already!
|
2021-02-10 01:32:41 -05:00
|
|
|
} else if cn == int(loc.FirstCol) {
|
2021-02-09 01:11:41 -05:00
|
|
|
// first and last column MAY be the same
|
2021-02-10 01:32:41 -05:00
|
|
|
if rn == int(loc.LastRow) {
|
|
|
|
s.placeValue(rn, cn, endRowMerged)
|
2021-02-09 01:11:41 -05:00
|
|
|
} else {
|
2021-02-10 01:32:41 -05:00
|
|
|
s.placeValue(rn, cn, continueRowMerged)
|
2021-02-09 01:11:41 -05:00
|
|
|
}
|
2021-02-10 01:32:41 -05:00
|
|
|
} else if cn == int(loc.LastCol) {
|
2021-02-09 01:11:41 -05:00
|
|
|
// first and last column are NOT the same
|
2021-02-10 01:32:41 -05:00
|
|
|
s.placeValue(rn, cn, endColumnMerged)
|
2021-02-09 01:11:41 -05:00
|
|
|
} else {
|
2021-02-10 01:32:41 -05:00
|
|
|
s.placeValue(rn, cn, continueColumnMerged)
|
2021-02-09 01:11:41 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-02-07 22:51:20 -05:00
|
|
|
|
2021-02-08 15:36:08 -05:00
|
|
|
case RecTypeContinue:
|
|
|
|
// the only situation so far is when used in RecTypeString above
|
|
|
|
|
2021-02-07 22:51:20 -05:00
|
|
|
default:
|
2021-02-08 11:02:37 -05:00
|
|
|
//log.Println("worksheet", r.RecType, r.RecSize)
|
2021-02-07 22:51:20 -05:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Err returns the last error that occured.
|
|
|
|
func (s *WorkSheet) Err() error {
|
|
|
|
return s.err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Next advances to the next row of content.
|
|
|
|
// It MUST be called prior to any Scan().
|
|
|
|
func (s *WorkSheet) Next() bool {
|
|
|
|
s.iterRow++
|
|
|
|
return s.iterRow < len(s.rows)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *WorkSheet) Strings() []string {
|
|
|
|
currow := s.rows[s.iterRow]
|
|
|
|
res := make([]string, len(currow.cols))
|
|
|
|
for i, col := range currow.cols {
|
|
|
|
res[i] = fmt.Sprint(col)
|
|
|
|
}
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scan extracts values from the row into the provided arguments
|
2021-02-08 11:02:37 -05:00
|
|
|
// Arguments must be pointers to one of 5 supported types:
|
|
|
|
// bool, int, float64, string, or time.Time
|
2021-02-07 22:51:20 -05:00
|
|
|
func (s *WorkSheet) Scan(args ...interface{}) error {
|
|
|
|
currow := s.rows[s.iterRow]
|
|
|
|
|
|
|
|
for i, a := range args {
|
|
|
|
switch v := a.(type) {
|
2021-02-08 11:02:37 -05:00
|
|
|
case *bool:
|
|
|
|
*v = currow.cols[i].(bool)
|
2021-02-07 22:51:20 -05:00
|
|
|
case *int:
|
|
|
|
*v = currow.cols[i].(int)
|
|
|
|
case *float64:
|
|
|
|
*v = currow.cols[i].(float64)
|
|
|
|
case *string:
|
|
|
|
*v = currow.cols[i].(string)
|
|
|
|
case *time.Time:
|
|
|
|
*v = currow.cols[i].(time.Time)
|
|
|
|
default:
|
|
|
|
return ErrInvalidType
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ErrInvalidType is returned by Scan for invalid arguments.
|
2021-02-08 11:02:37 -05:00
|
|
|
var ErrInvalidType = errors.New("xls: Scan only supports *bool, *int, *float64, *string, *time.Time arguments")
|
|
|
|
|
|
|
|
var berrLookup = map[byte]string{
|
|
|
|
0x00: "#NULL!",
|
|
|
|
0x07: "#DIV/0!",
|
|
|
|
0x0F: "#VALUE!",
|
|
|
|
0x17: "#REF!",
|
|
|
|
0x1D: "#NAME?",
|
|
|
|
0x24: "#NUM!",
|
|
|
|
0x2A: "#N/A",
|
|
|
|
0x2B: "#GETTING_DATA",
|
|
|
|
}
|