2024-11-22 16:33:58 +08:00
|
|
|
package office
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"regexp"
|
|
|
|
"strconv"
|
|
|
|
|
|
|
|
"github.com/ssgo/u"
|
|
|
|
"github.com/xuri/excelize/v2"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Excel struct {
|
|
|
|
filename string
|
|
|
|
password string
|
|
|
|
excel *excelize.File
|
|
|
|
}
|
|
|
|
|
|
|
|
func OpenExcel(filename string, password *string) (*Excel, error) {
|
|
|
|
xls := &Excel{
|
|
|
|
filename: filename,
|
|
|
|
password: u.String(password),
|
|
|
|
}
|
|
|
|
var err error
|
|
|
|
if u.FileExists(filename) {
|
|
|
|
if xls.excel, err = excelize.OpenFile(filename, excelize.Options{Password: xls.password}); err != nil {
|
|
|
|
xls.excel = excelize.NewFile()
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
xls.excel = excelize.NewFile()
|
|
|
|
}
|
|
|
|
return xls, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (xls *Excel) Save(filename, password *string) error {
|
|
|
|
if filename != nil {
|
|
|
|
xls.filename = *filename
|
|
|
|
}
|
|
|
|
if password != nil {
|
|
|
|
xls.password = *password
|
|
|
|
}
|
|
|
|
return xls.excel.SaveAs(xls.filename, excelize.Options{Password: xls.password})
|
|
|
|
}
|
|
|
|
|
2024-12-13 19:27:15 +08:00
|
|
|
func (xls *Excel) WriteToBuffer() ([]byte, error) {
|
|
|
|
buf, err := xls.excel.WriteToBuffer()
|
|
|
|
return buf.Bytes(), err
|
|
|
|
}
|
|
|
|
|
2024-11-22 16:33:58 +08:00
|
|
|
var isIntMatcher = regexp.MustCompile(`^\d{1,8}$`)
|
|
|
|
var isFloatMatcher = regexp.MustCompile(`^[\d.]{1,8}$`)
|
|
|
|
|
|
|
|
func (xls *Excel) Set(sheetName string, table [][]any, start, end *string) error {
|
2024-12-13 19:27:15 +08:00
|
|
|
sheetName = xls.GetSheetName(sheetName)
|
2024-11-22 16:33:58 +08:00
|
|
|
startX, startY := ParseCellID(u.String(start))
|
|
|
|
endX, endY := ParseCellID(u.String(end))
|
|
|
|
for y, row := range table {
|
|
|
|
if endY > 0 && startY+y > endY {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
for x, v := range row {
|
|
|
|
if endX > 0 && startX+x > endX {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
// fmt.Println(startX+x, startY+y, MakeCellID(startX+x, startY+y), reflect.TypeOf(v).String(), v)
|
|
|
|
if err := xls.excel.SetCellValue(sheetName, MakeCellID(startX+x, startY+y), v); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// fmt.Println(xls.excel.GetCellType(sheetName, MakeCellID(startX+x, startY+y)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-12-13 19:27:15 +08:00
|
|
|
func (xls *Excel) RemoveSheet(sheetName string) error {
|
|
|
|
sheetName = xls.GetSheetName(sheetName)
|
|
|
|
return xls.excel.DeleteSheet(sheetName)
|
|
|
|
}
|
|
|
|
|
2024-11-22 16:33:58 +08:00
|
|
|
func (xls *Excel) Sheets() []string {
|
|
|
|
return xls.excel.GetSheetList()
|
|
|
|
}
|
|
|
|
|
2024-12-13 19:27:15 +08:00
|
|
|
func (xls *Excel) SetColWidth(sheetName string, startCol string, endCol string, width float64) error {
|
|
|
|
sheetName = xls.GetSheetName(sheetName)
|
|
|
|
return xls.excel.SetColWidth(sheetName, startCol, endCol, width)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (xls *Excel) SetColWidths(sheetName string, widths []float64) error {
|
|
|
|
sheetName = xls.GetSheetName(sheetName)
|
|
|
|
for i, w := range widths {
|
|
|
|
col := MakeColID(i)
|
|
|
|
if err := xls.excel.SetColWidth(sheetName, col, col, w); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (xls *Excel) MakeStyle(style excelize.Style) (int, error) {
|
|
|
|
fmt.Println(u.JsonP(style), 111)
|
|
|
|
return xls.excel.NewStyle(&style)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (xls *Excel) SetCellStyle(sheetName string, start string, end string, styleID int) error {
|
|
|
|
sheetName = xls.GetSheetName(sheetName)
|
|
|
|
return xls.excel.SetCellStyle(sheetName, start, end, styleID)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (xls *Excel) SetPanes(sheetName string, panes excelize.Panes) error {
|
|
|
|
sheetName = xls.GetSheetName(sheetName)
|
|
|
|
return xls.excel.SetPanes(sheetName, &panes)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (xls *Excel) SetAutoFilter(sheetName string, start, end string, option *[]excelize.AutoFilterOptions) error {
|
|
|
|
sheetName = xls.GetSheetName(sheetName)
|
|
|
|
var opt []excelize.AutoFilterOptions
|
|
|
|
if option != nil {
|
|
|
|
opt = *option
|
|
|
|
}
|
|
|
|
return xls.excel.AutoFilter(sheetName, start+":"+end, opt)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (xls *Excel) GetSheetName(sheetName string) string {
|
|
|
|
if sheetName == "" {
|
|
|
|
sheetName = "Sheet1"
|
|
|
|
}
|
|
|
|
if isIntMatcher.MatchString(sheetName) {
|
2024-11-22 16:33:58 +08:00
|
|
|
sheetName = xls.excel.GetSheetName(u.Int(sheetName))
|
2024-12-13 19:27:15 +08:00
|
|
|
} else {
|
|
|
|
if idx, err := xls.excel.GetSheetIndex(sheetName); err != nil || idx == -1 {
|
|
|
|
_, _ = xls.excel.NewSheet(sheetName)
|
|
|
|
}
|
2024-11-22 16:33:58 +08:00
|
|
|
}
|
2024-12-13 19:27:15 +08:00
|
|
|
return sheetName
|
|
|
|
}
|
2024-11-22 16:33:58 +08:00
|
|
|
|
2024-12-13 19:27:15 +08:00
|
|
|
func (xls *Excel) Get(sheetName string, start, end *string) ([][]any, error) {
|
|
|
|
sheetName = xls.GetSheetName(sheetName)
|
2024-11-22 16:33:58 +08:00
|
|
|
rows, err := xls.excel.GetRows(sheetName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
startX, startY := ParseCellID(u.String(start))
|
|
|
|
endX, endY := ParseCellID(u.String(end))
|
|
|
|
result := make([][]any, 0)
|
|
|
|
for y, row := range rows {
|
|
|
|
if startY > 0 && y < startY {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if endY > 0 && y > endY {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
// result[y-startY] = make([]any, len(row)-startX)
|
|
|
|
rowData := make([]any, 0)
|
|
|
|
for x, v := range row {
|
|
|
|
if startX > 0 && x < startX {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if endX > 0 && x > endX {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
cellType, err := xls.excel.GetCellType(sheetName, MakeCellID(x, y))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
switch cellType {
|
|
|
|
case excelize.CellTypeInlineString, excelize.CellTypeSharedString:
|
|
|
|
rowData = append(rowData, v)
|
|
|
|
case excelize.CellTypeFormula:
|
|
|
|
// TODO 获得公式结果
|
|
|
|
// formula, err := xls.excel.GetCellFormula(sheetName, MakeCellID(j, i))
|
|
|
|
rowData = append(rowData, v)
|
|
|
|
case excelize.CellTypeNumber:
|
|
|
|
if isFloatMatcher.MatchString(v) {
|
|
|
|
rowData = append(rowData, u.Float64(v))
|
|
|
|
} else {
|
|
|
|
rowData = append(rowData, u.Int64(v))
|
|
|
|
}
|
|
|
|
case excelize.CellTypeBool:
|
|
|
|
rowData = append(rowData, u.Bool(v))
|
|
|
|
case excelize.CellTypeDate:
|
|
|
|
// TODO 获得日期结果
|
|
|
|
rowData = append(rowData, v)
|
|
|
|
case excelize.CellTypeUnset:
|
|
|
|
if isFloatMatcher.MatchString(v) {
|
|
|
|
rowData = append(rowData, u.Float64(v))
|
|
|
|
} else if isIntMatcher.MatchString(v) {
|
|
|
|
rowData = append(rowData, u.Int64(v))
|
|
|
|
} else {
|
|
|
|
rowData = append(rowData, v)
|
|
|
|
}
|
|
|
|
case excelize.CellTypeError:
|
|
|
|
rowData = append(rowData, v)
|
|
|
|
default:
|
|
|
|
rowData = append(rowData, v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
result = append(result, rowData)
|
|
|
|
}
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (xls *Excel) SetData(sheetName string, data []map[string]any, start, end *string) error {
|
|
|
|
table, err := xls.Get(sheetName, start, end)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fieldIndex := map[string]int{}
|
|
|
|
fieldNum := 0
|
|
|
|
if len(table) > 0 {
|
|
|
|
fieldNum = len(table[0])
|
|
|
|
for i, v := range table[0] {
|
|
|
|
fieldIndex[u.String(v)] = i
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, item := range data {
|
|
|
|
if len(table) <= i+1 {
|
|
|
|
table = append(table, []any{})
|
|
|
|
}
|
|
|
|
for k, v := range item {
|
|
|
|
idx, idxOk := fieldIndex[k]
|
|
|
|
if !idxOk {
|
|
|
|
idx = fieldNum
|
|
|
|
fieldIndex[k] = idx
|
|
|
|
fieldNum++
|
|
|
|
|
|
|
|
for len(table[0]) < idx+1 {
|
|
|
|
table[0] = append(table[0], "")
|
|
|
|
}
|
|
|
|
table[0][idx] = k
|
|
|
|
}
|
|
|
|
for len(table[i+1]) < idx+1 {
|
|
|
|
table[i+1] = append(table[i+1], "")
|
|
|
|
}
|
|
|
|
table[i+1][idx] = v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return xls.Set(sheetName, table, start, end)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (xls *Excel) GetData(sheetName string, start, end *string) ([]map[string]any, error) {
|
|
|
|
rows, err := xls.Get(sheetName, start, end)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
fields := []string{}
|
|
|
|
if len(rows) > 0 {
|
|
|
|
for _, v := range rows[0] {
|
|
|
|
v1 := u.String(v)
|
|
|
|
if v1 == "" {
|
|
|
|
v1 = MakeCellID(len(fields), 0)
|
|
|
|
}
|
|
|
|
fields = append(fields, v1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
data := make([]map[string]any, 0)
|
|
|
|
for i := 1; i < len(rows); i++ {
|
|
|
|
row := map[string]any{}
|
|
|
|
for j := 0; j < len(fields); j++ {
|
|
|
|
if j < len(rows[i]) {
|
|
|
|
row[fields[j]] = rows[i][j]
|
|
|
|
} else {
|
|
|
|
row[fields[j]] = ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
data = append(data, row)
|
|
|
|
}
|
|
|
|
return data, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func MakeCellID(col, row int) string {
|
|
|
|
colName := ""
|
|
|
|
for col >= 0 {
|
|
|
|
i := col % 26
|
|
|
|
colName = string(rune(i+65)) + colName
|
|
|
|
col = col/26 - 1
|
|
|
|
}
|
|
|
|
return fmt.Sprint(colName, row+1)
|
|
|
|
}
|
|
|
|
|
2024-12-13 19:27:15 +08:00
|
|
|
func MakeColID(col int) string {
|
|
|
|
colName := ""
|
|
|
|
for col >= 0 {
|
|
|
|
i := col % 26
|
|
|
|
colName = string(rune(i+65)) + colName
|
|
|
|
col = col/26 - 1
|
|
|
|
}
|
|
|
|
return fmt.Sprint(colName)
|
|
|
|
}
|
|
|
|
|
2024-11-22 16:33:58 +08:00
|
|
|
func ParseCellID(cell string) (col, row int) {
|
|
|
|
row = 0
|
|
|
|
col = 0
|
|
|
|
for i, c := range cell {
|
|
|
|
if c >= 'A' && c <= 'Z' {
|
|
|
|
col = col*26 + int(c-'A'+1)
|
|
|
|
} else if c >= 'a' && c <= 'z' {
|
|
|
|
col = col*26 + int(c-'a'+1)
|
|
|
|
} else {
|
|
|
|
row, _ = strconv.Atoi(cell[i:])
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
col--
|
|
|
|
row--
|
|
|
|
if col < 0 {
|
|
|
|
col = 0
|
|
|
|
}
|
|
|
|
if row < 0 {
|
|
|
|
row = 0
|
|
|
|
}
|
|
|
|
return col, row
|
|
|
|
}
|