2021-03-30 23:02:22 +08:00
|
|
|
// Copyright 2016 - 2021 The excelize Authors. All rights reserved. Use of
|
2018-09-14 00:44:23 +08:00
|
|
|
// this source code is governed by a BSD-style license that can be found in
|
|
|
|
// the LICENSE file.
|
|
|
|
//
|
|
|
|
// Package excelize providing a set of functions that allow you to write to
|
2020-06-22 00:14:56 +08:00
|
|
|
// and read from XLSX / XLSM / XLTM files. Supports reading and writing
|
2021-03-30 23:02:22 +08:00
|
|
|
// spreadsheet documents generated by Microsoft Excel™ 2007 and later. Supports
|
2020-06-22 00:14:56 +08:00
|
|
|
// complex components by high compatibility, and provided streaming API for
|
|
|
|
// generating or reading data from a worksheet with huge amounts of data. This
|
2021-04-04 15:29:43 +08:00
|
|
|
// library needs Go version 1.15 or later.
|
2018-09-14 00:58:48 +08:00
|
|
|
|
2018-07-30 22:09:41 +08:00
|
|
|
package excelize
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2021-07-31 00:31:51 +08:00
|
|
|
"math"
|
2018-07-30 22:09:41 +08:00
|
|
|
"strings"
|
2021-07-29 00:03:57 +08:00
|
|
|
"unicode/utf16"
|
2018-07-30 22:09:41 +08:00
|
|
|
)
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// DataValidationType defined the type of data validation.
|
2018-07-30 22:09:41 +08:00
|
|
|
type DataValidationType int
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// Data validation types.
|
2018-07-30 22:09:41 +08:00
|
|
|
const (
|
|
|
|
_DataValidationType = iota
|
2018-09-01 23:32:44 +08:00
|
|
|
typeNone // inline use
|
2018-07-30 22:09:41 +08:00
|
|
|
DataValidationTypeCustom
|
|
|
|
DataValidationTypeDate
|
|
|
|
DataValidationTypeDecimal
|
2018-09-01 23:32:44 +08:00
|
|
|
typeList // inline use
|
2018-07-30 22:09:41 +08:00
|
|
|
DataValidationTypeTextLeng
|
|
|
|
DataValidationTypeTime
|
|
|
|
// DataValidationTypeWhole Integer
|
|
|
|
DataValidationTypeWhole
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2021-07-31 00:31:51 +08:00
|
|
|
// dataValidationFormulaStrLen 255 characters
|
|
|
|
dataValidationFormulaStrLen = 255
|
2018-07-30 22:09:41 +08:00
|
|
|
)
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// DataValidationErrorStyle defined the style of data validation error alert.
|
2018-07-30 22:09:41 +08:00
|
|
|
type DataValidationErrorStyle int
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// Data validation error styles.
|
2018-07-30 22:09:41 +08:00
|
|
|
const (
|
|
|
|
_ DataValidationErrorStyle = iota
|
2018-09-01 19:38:30 +08:00
|
|
|
DataValidationErrorStyleStop
|
|
|
|
DataValidationErrorStyleWarning
|
|
|
|
DataValidationErrorStyleInformation
|
2018-07-30 22:09:41 +08:00
|
|
|
)
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// Data validation error styles.
|
2018-07-30 22:09:41 +08:00
|
|
|
const (
|
|
|
|
styleStop = "stop"
|
|
|
|
styleWarning = "warning"
|
|
|
|
styleInformation = "information"
|
|
|
|
)
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// DataValidationOperator operator enum.
|
2018-07-30 22:09:41 +08:00
|
|
|
type DataValidationOperator int
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// Data validation operators.
|
2018-07-30 22:09:41 +08:00
|
|
|
const (
|
|
|
|
_DataValidationOperator = iota
|
|
|
|
DataValidationOperatorBetween
|
|
|
|
DataValidationOperatorEqual
|
|
|
|
DataValidationOperatorGreaterThan
|
|
|
|
DataValidationOperatorGreaterThanOrEqual
|
|
|
|
DataValidationOperatorLessThan
|
|
|
|
DataValidationOperatorLessThanOrEqual
|
|
|
|
DataValidationOperatorNotBetween
|
|
|
|
DataValidationOperatorNotEqual
|
|
|
|
)
|
|
|
|
|
2021-07-31 00:31:51 +08:00
|
|
|
// formulaEscaper mimics the Excel escaping rules for data validation,
|
|
|
|
// which converts `"` to `""` instead of `"`.
|
|
|
|
var formulaEscaper = strings.NewReplacer(
|
|
|
|
`&`, `&`,
|
|
|
|
`<`, `<`,
|
|
|
|
`>`, `>`,
|
|
|
|
`"`, `""`,
|
|
|
|
)
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// NewDataValidation return data validation struct.
|
2018-07-30 22:09:41 +08:00
|
|
|
func NewDataValidation(allowBlank bool) *DataValidation {
|
|
|
|
return &DataValidation{
|
2018-09-01 23:32:44 +08:00
|
|
|
AllowBlank: allowBlank,
|
|
|
|
ShowErrorMessage: false,
|
|
|
|
ShowInputMessage: false,
|
2018-07-30 22:09:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// SetError set error notice.
|
2018-09-01 19:38:30 +08:00
|
|
|
func (dd *DataValidation) SetError(style DataValidationErrorStyle, title, msg string) {
|
|
|
|
dd.Error = &msg
|
|
|
|
dd.ErrorTitle = &title
|
2018-07-30 22:09:41 +08:00
|
|
|
strStyle := styleStop
|
|
|
|
switch style {
|
2018-09-01 19:38:30 +08:00
|
|
|
case DataValidationErrorStyleStop:
|
2018-07-30 22:09:41 +08:00
|
|
|
strStyle = styleStop
|
2018-09-01 19:38:30 +08:00
|
|
|
case DataValidationErrorStyleWarning:
|
2018-07-30 22:09:41 +08:00
|
|
|
strStyle = styleWarning
|
2018-09-01 19:38:30 +08:00
|
|
|
case DataValidationErrorStyleInformation:
|
2018-07-30 22:09:41 +08:00
|
|
|
strStyle = styleInformation
|
|
|
|
|
|
|
|
}
|
2018-09-01 23:32:44 +08:00
|
|
|
dd.ShowErrorMessage = true
|
2018-07-30 22:09:41 +08:00
|
|
|
dd.ErrorStyle = &strStyle
|
|
|
|
}
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// SetInput set prompt notice.
|
2018-09-01 19:38:30 +08:00
|
|
|
func (dd *DataValidation) SetInput(title, msg string) {
|
2018-09-01 23:32:44 +08:00
|
|
|
dd.ShowInputMessage = true
|
2018-09-01 19:38:30 +08:00
|
|
|
dd.PromptTitle = &title
|
|
|
|
dd.Prompt = &msg
|
2018-07-30 22:09:41 +08:00
|
|
|
}
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// SetDropList data validation list.
|
2018-07-30 22:09:41 +08:00
|
|
|
func (dd *DataValidation) SetDropList(keys []string) error {
|
2021-07-31 00:31:51 +08:00
|
|
|
formula := strings.Join(keys, ",")
|
2021-07-29 00:03:57 +08:00
|
|
|
if dataValidationFormulaStrLen < len(utf16.Encode([]rune(formula))) {
|
2021-07-31 00:31:51 +08:00
|
|
|
return ErrDataValidationFormulaLenth
|
2019-01-23 22:07:11 +08:00
|
|
|
}
|
2021-07-31 00:31:51 +08:00
|
|
|
dd.Formula1 = fmt.Sprintf(`<formula1>"%s"</formula1>`, formulaEscaper.Replace(formula))
|
2018-07-30 22:09:41 +08:00
|
|
|
dd.Type = convDataValidationType(typeList)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// SetRange provides function to set data validation range in drop list.
|
2020-12-10 13:37:34 +08:00
|
|
|
func (dd *DataValidation) SetRange(f1, f2 float64, t DataValidationType, o DataValidationOperator) error {
|
2021-07-31 00:31:51 +08:00
|
|
|
if math.Abs(f1) > math.MaxFloat32 || math.Abs(f2) > math.MaxFloat32 {
|
|
|
|
return ErrDataValidationRange
|
2018-07-30 22:09:41 +08:00
|
|
|
}
|
2021-07-31 00:31:51 +08:00
|
|
|
dd.Formula1 = fmt.Sprintf("<formula1>%.17g</formula1>", f1)
|
|
|
|
dd.Formula2 = fmt.Sprintf("<formula2>%.17g</formula2>", f2)
|
2018-07-30 22:09:41 +08:00
|
|
|
dd.Type = convDataValidationType(t)
|
|
|
|
dd.Operator = convDataValidationOperatior(o)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-09-13 10:38:01 +08:00
|
|
|
// SetSqrefDropList provides set data validation on a range with source
|
|
|
|
// reference range of the worksheet by given data validation object and
|
|
|
|
// worksheet name. The data validation object can be created by
|
|
|
|
// NewDataValidation function. For example, set data validation on
|
|
|
|
// Sheet1!A7:B8 with validation criteria source Sheet1!E1:E3 settings, create
|
|
|
|
// in-cell dropdown by allowing list source:
|
|
|
|
//
|
|
|
|
// dvRange := excelize.NewDataValidation(true)
|
|
|
|
// dvRange.Sqref = "A7:B8"
|
2019-07-22 22:36:44 +08:00
|
|
|
// dvRange.SetSqrefDropList("$E$1:$E$3", true)
|
2019-04-20 14:57:50 +08:00
|
|
|
// f.AddDataValidation("Sheet1", dvRange)
|
2018-09-13 10:38:01 +08:00
|
|
|
//
|
2018-09-04 13:40:53 +08:00
|
|
|
func (dd *DataValidation) SetSqrefDropList(sqref string, isCurrentSheet bool) error {
|
|
|
|
if isCurrentSheet {
|
2021-08-12 00:02:27 +08:00
|
|
|
dd.Formula1 = fmt.Sprintf("<formula1>%s</formula1>", sqref)
|
2018-09-04 13:40:53 +08:00
|
|
|
dd.Type = convDataValidationType(typeList)
|
|
|
|
return nil
|
|
|
|
}
|
2018-09-12 15:47:56 +08:00
|
|
|
return fmt.Errorf("cross-sheet sqref cell are not supported")
|
2018-09-04 13:40:53 +08:00
|
|
|
}
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// SetSqref provides function to set data validation range in drop list.
|
2018-07-30 22:09:41 +08:00
|
|
|
func (dd *DataValidation) SetSqref(sqref string) {
|
|
|
|
if dd.Sqref == "" {
|
|
|
|
dd.Sqref = sqref
|
|
|
|
} else {
|
|
|
|
dd.Sqref = fmt.Sprintf("%s %s", dd.Sqref, sqref)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// convDataValidationType get excel data validation type.
|
2018-07-30 22:09:41 +08:00
|
|
|
func convDataValidationType(t DataValidationType) string {
|
|
|
|
typeMap := map[DataValidationType]string{
|
|
|
|
typeNone: "none",
|
|
|
|
DataValidationTypeCustom: "custom",
|
|
|
|
DataValidationTypeDate: "date",
|
|
|
|
DataValidationTypeDecimal: "decimal",
|
|
|
|
typeList: "list",
|
|
|
|
DataValidationTypeTextLeng: "textLength",
|
|
|
|
DataValidationTypeTime: "time",
|
|
|
|
DataValidationTypeWhole: "whole",
|
|
|
|
}
|
|
|
|
|
|
|
|
return typeMap[t]
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// convDataValidationOperatior get excel data validation operator.
|
2018-07-30 22:09:41 +08:00
|
|
|
func convDataValidationOperatior(o DataValidationOperator) string {
|
|
|
|
typeMap := map[DataValidationOperator]string{
|
|
|
|
DataValidationOperatorBetween: "between",
|
|
|
|
DataValidationOperatorEqual: "equal",
|
|
|
|
DataValidationOperatorGreaterThan: "greaterThan",
|
|
|
|
DataValidationOperatorGreaterThanOrEqual: "greaterThanOrEqual",
|
|
|
|
DataValidationOperatorLessThan: "lessThan",
|
|
|
|
DataValidationOperatorLessThanOrEqual: "lessThanOrEqual",
|
|
|
|
DataValidationOperatorNotBetween: "notBetween",
|
|
|
|
DataValidationOperatorNotEqual: "notEqual",
|
|
|
|
}
|
|
|
|
|
|
|
|
return typeMap[o]
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-09-01 23:32:44 +08:00
|
|
|
// AddDataValidation provides set data validation on a range of the worksheet
|
|
|
|
// by given data validation object and worksheet name. The data validation
|
|
|
|
// object can be created by NewDataValidation function.
|
|
|
|
//
|
|
|
|
// Example 1, set data validation on Sheet1!A1:B2 with validation criteria
|
2018-09-02 01:44:32 +08:00
|
|
|
// settings, show error alert after invalid data is entered with "Stop" style
|
2018-09-01 23:32:44 +08:00
|
|
|
// and custom title "error body":
|
|
|
|
//
|
|
|
|
// dvRange := excelize.NewDataValidation(true)
|
2018-09-02 01:44:32 +08:00
|
|
|
// dvRange.Sqref = "A1:B2"
|
2018-09-01 23:32:44 +08:00
|
|
|
// dvRange.SetRange(10, 20, excelize.DataValidationTypeWhole, excelize.DataValidationOperatorBetween)
|
|
|
|
// dvRange.SetError(excelize.DataValidationErrorStyleStop, "error title", "error body")
|
2019-04-20 14:57:50 +08:00
|
|
|
// err := f.AddDataValidation("Sheet1", dvRange)
|
2018-09-01 23:32:44 +08:00
|
|
|
//
|
|
|
|
// Example 2, set data validation on Sheet1!A3:B4 with validation criteria
|
|
|
|
// settings, and show input message when cell is selected:
|
|
|
|
//
|
|
|
|
// dvRange = excelize.NewDataValidation(true)
|
|
|
|
// dvRange.Sqref = "A3:B4"
|
|
|
|
// dvRange.SetRange(10, 20, excelize.DataValidationTypeWhole, excelize.DataValidationOperatorGreaterThan)
|
|
|
|
// dvRange.SetInput("input title", "input body")
|
2019-04-20 14:57:50 +08:00
|
|
|
// err = f.AddDataValidation("Sheet1", dvRange)
|
2018-09-01 23:32:44 +08:00
|
|
|
//
|
2018-09-02 01:44:32 +08:00
|
|
|
// Example 3, set data validation on Sheet1!A5:B6 with validation criteria
|
2018-09-13 10:38:01 +08:00
|
|
|
// settings, create in-cell dropdown by allowing list source:
|
2018-09-01 23:32:44 +08:00
|
|
|
//
|
|
|
|
// dvRange = excelize.NewDataValidation(true)
|
|
|
|
// dvRange.Sqref = "A5:B6"
|
|
|
|
// dvRange.SetDropList([]string{"1", "2", "3"})
|
2019-04-20 14:57:50 +08:00
|
|
|
// err = f.AddDataValidation("Sheet1", dvRange)
|
2018-09-01 23:32:44 +08:00
|
|
|
//
|
2019-04-15 11:22:57 +08:00
|
|
|
func (f *File) AddDataValidation(sheet string, dv *DataValidation) error {
|
2020-03-13 00:48:16 +08:00
|
|
|
ws, err := f.workSheetReader(sheet)
|
2019-04-15 11:22:57 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-03-13 00:48:16 +08:00
|
|
|
if nil == ws.DataValidations {
|
|
|
|
ws.DataValidations = new(xlsxDataValidations)
|
2018-07-30 22:09:41 +08:00
|
|
|
}
|
2020-03-13 00:48:16 +08:00
|
|
|
ws.DataValidations.DataValidation = append(ws.DataValidations.DataValidation, dv)
|
|
|
|
ws.DataValidations.Count = len(ws.DataValidations.DataValidation)
|
2019-04-15 11:22:57 +08:00
|
|
|
return err
|
2018-07-30 22:09:41 +08:00
|
|
|
}
|
2020-03-13 00:48:16 +08:00
|
|
|
|
|
|
|
// DeleteDataValidation delete data validation by given worksheet name and
|
|
|
|
// reference sequence.
|
|
|
|
func (f *File) DeleteDataValidation(sheet, sqref string) error {
|
|
|
|
ws, err := f.workSheetReader(sheet)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if ws.DataValidations == nil {
|
|
|
|
return nil
|
|
|
|
}
|
2021-08-06 22:44:43 +08:00
|
|
|
delCells, err := f.flatSqref(sqref)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-03-13 00:48:16 +08:00
|
|
|
dv := ws.DataValidations
|
|
|
|
for i := 0; i < len(dv.DataValidation); i++ {
|
2021-08-06 22:44:43 +08:00
|
|
|
applySqref := []string{}
|
|
|
|
colCells, err := f.flatSqref(dv.DataValidation[i].Sqref)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
for col, cells := range delCells {
|
|
|
|
for _, cell := range cells {
|
|
|
|
idx := inCoordinates(colCells[col], cell)
|
|
|
|
if idx != -1 {
|
|
|
|
colCells[col] = append(colCells[col][:idx], colCells[col][idx+1:]...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, col := range colCells {
|
|
|
|
applySqref = append(applySqref, f.squashSqref(col)...)
|
|
|
|
}
|
|
|
|
dv.DataValidation[i].Sqref = strings.Join(applySqref, " ")
|
|
|
|
if len(applySqref) == 0 {
|
2020-03-13 00:48:16 +08:00
|
|
|
dv.DataValidation = append(dv.DataValidation[:i], dv.DataValidation[i+1:]...)
|
|
|
|
i--
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dv.Count = len(dv.DataValidation)
|
|
|
|
if dv.Count == 0 {
|
|
|
|
ws.DataValidations = nil
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2021-08-06 22:44:43 +08:00
|
|
|
|
|
|
|
// squashSqref generates cell reference sequence by given cells coordinates list.
|
|
|
|
func (f *File) squashSqref(cells [][]int) []string {
|
|
|
|
if len(cells) == 1 {
|
|
|
|
cell, _ := CoordinatesToCellName(cells[0][0], cells[0][1])
|
|
|
|
return []string{cell}
|
|
|
|
} else if len(cells) == 0 {
|
|
|
|
return []string{}
|
|
|
|
}
|
|
|
|
l, r, res := 0, 0, []string{}
|
|
|
|
for i := 1; i < len(cells); i++ {
|
|
|
|
if cells[i][0] == cells[r][0] && cells[i][1]-cells[r][1] > 1 {
|
|
|
|
curr, _ := f.coordinatesToAreaRef(append(cells[l], cells[r]...))
|
|
|
|
if l == r {
|
|
|
|
curr, _ = CoordinatesToCellName(cells[l][0], cells[l][1])
|
|
|
|
}
|
|
|
|
res = append(res, curr)
|
|
|
|
l, r = i, i
|
|
|
|
} else {
|
|
|
|
r++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
curr, _ := f.coordinatesToAreaRef(append(cells[l], cells[r]...))
|
|
|
|
if l == r {
|
|
|
|
curr, _ = CoordinatesToCellName(cells[l][0], cells[l][1])
|
|
|
|
}
|
|
|
|
return append(res, curr)
|
|
|
|
}
|