// Copyright 2016 - 2022 The excelize Authors. All rights reserved. Use of // 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 and // read from XLAM / XLSM / XLSX / XLTM / XLTX files. Supports reading and // writing spreadsheet documents generated by Microsoft Excelâ„¢ 2007 and later. // Supports complex components by high compatibility, and provided streaming // API for generating or reading data from a worksheet with huge amounts of // data. This library needs Go version 1.15 or later. package excelize import ( "encoding/xml" "fmt" "os" "reflect" "strconv" "strings" "time" ) // CellType is the type of cell value type. type CellType byte // Cell value types enumeration. const ( CellTypeUnset CellType = iota CellTypeBool CellTypeDate CellTypeError CellTypeNumber CellTypeString ) const ( // STCellFormulaTypeArray defined the formula is an array formula. STCellFormulaTypeArray = "array" // STCellFormulaTypeDataTable defined the formula is a data table formula. STCellFormulaTypeDataTable = "dataTable" // STCellFormulaTypeNormal defined the formula is a regular cell formula. STCellFormulaTypeNormal = "normal" // STCellFormulaTypeShared defined the formula is part of a shared formula. STCellFormulaTypeShared = "shared" ) // cellTypes mapping the cell's data type and enumeration. var cellTypes = map[string]CellType{ "b": CellTypeBool, "d": CellTypeDate, "n": CellTypeNumber, "e": CellTypeError, "s": CellTypeString, "str": CellTypeString, "inlineStr": CellTypeString, } // GetCellValue provides a function to get formatted value from cell by given // worksheet name and axis in spreadsheet file. If it is possible to apply a // format to the cell value, it will do so, if not then an error will be // returned, along with the raw value of the cell. All cells' values will be // the same in a merged range. func (f *File) GetCellValue(sheet, axis string, opts ...Options) (string, error) { return f.getCellStringFunc(sheet, axis, func(x *xlsxWorksheet, c *xlsxC) (string, bool, error) { val, err := c.getValueFrom(f, f.sharedStringsReader(), parseOptions(opts...).RawCellValue) return val, true, err }) } // GetCellType provides a function to get the cell's data type by given // worksheet name and axis in spreadsheet file. func (f *File) GetCellType(sheet, axis string) (CellType, error) { var ( err error cellTypeStr string cellType CellType ) if cellTypeStr, err = f.getCellStringFunc(sheet, axis, func(x *xlsxWorksheet, c *xlsxC) (string, bool, error) { return c.T, true, nil }); err != nil { return CellTypeUnset, err } cellType = cellTypes[cellTypeStr] return cellType, err } // SetCellValue provides a function to set the value of a cell. The specified // coordinates should not be in the first row of the table, a complex number // can be set with string text. The following shows the supported data // types: // // int // int8 // int16 // int32 // int64 // uint // uint8 // uint16 // uint32 // uint64 // float32 // float64 // string // []byte // time.Duration // time.Time // bool // nil // // Note that default date format is m/d/yy h:mm of time.Time type value. You // can set numbers format by SetCellStyle() method. If you need to set the // specialized date in Excel like January 0, 1900 or February 29, 1900, these // times can not representation in Go language time.Time data type. Please set // the cell value as number 0 or 60, then create and bind the date-time number // format style for the cell. func (f *File) SetCellValue(sheet, axis string, value interface{}) error { var err error switch v := value.(type) { case int, int8, int16, int32, int64, uint, uint8, uint16, uint32, uint64: err = f.setCellIntFunc(sheet, axis, v) case float32: err = f.SetCellFloat(sheet, axis, float64(v), -1, 32) case float64: err = f.SetCellFloat(sheet, axis, v, -1, 64) case string: err = f.SetCellStr(sheet, axis, v) case []byte: err = f.SetCellStr(sheet, axis, string(v)) case time.Duration: _, d := setCellDuration(v) err = f.SetCellDefault(sheet, axis, d) if err != nil { return err } err = f.setDefaultTimeStyle(sheet, axis, 21) case time.Time: err = f.setCellTimeFunc(sheet, axis, v) case bool: err = f.SetCellBool(sheet, axis, v) case nil: err = f.SetCellDefault(sheet, axis, "") default: err = f.SetCellStr(sheet, axis, fmt.Sprint(value)) } return err } // String extracts characters from a string item. func (x xlsxSI) String() string { if len(x.R) > 0 { var rows strings.Builder for _, s := range x.R { if s.T != nil { rows.WriteString(s.T.Val) } } return bstrUnmarshal(rows.String()) } if x.T != nil { return bstrUnmarshal(x.T.Val) } return "" } // hasValue determine if cell non-blank value. func (c *xlsxC) hasValue() bool { return c.S != 0 || c.V != "" || c.F != nil || c.T != "" } // removeFormula delete formula for the cell. func (f *File) removeFormula(c *xlsxC, ws *xlsxWorksheet, sheet string) { if c.F != nil && c.Vm == nil { f.deleteCalcChain(f.getSheetID(sheet), c.R) if c.F.T == STCellFormulaTypeShared && c.F.Ref != "" { si := c.F.Si for r, row := range ws.SheetData.Row { for col, cell := range row.C { if cell.F != nil && cell.F.Si != nil && *cell.F.Si == *si { ws.SheetData.Row[r].C[col].F = nil } } } } c.F = nil } } // setCellIntFunc is a wrapper of SetCellInt. func (f *File) setCellIntFunc(sheet, axis string, value interface{}) error { var err error switch v := value.(type) { case int: err = f.SetCellInt(sheet, axis, v) case int8: err = f.SetCellInt(sheet, axis, int(v)) case int16: err = f.SetCellInt(sheet, axis, int(v)) case int32: err = f.SetCellInt(sheet, axis, int(v)) case int64: err = f.SetCellInt(sheet, axis, int(v)) case uint: err = f.SetCellInt(sheet, axis, int(v)) case uint8: err = f.SetCellInt(sheet, axis, int(v)) case uint16: err = f.SetCellInt(sheet, axis, int(v)) case uint32: err = f.SetCellInt(sheet, axis, int(v)) case uint64: err = f.SetCellInt(sheet, axis, int(v)) } return err } // setCellTimeFunc provides a method to process time type of value for // SetCellValue. func (f *File) setCellTimeFunc(sheet, axis string, value time.Time) error { ws, err := f.workSheetReader(sheet) if err != nil { return err } cellData, col, row, err := f.prepareCell(ws, axis) if err != nil { return err } ws.Lock() cellData.S = f.prepareCellStyle(ws, col, row, cellData.S) ws.Unlock() date1904, wb := false, f.workbookReader() if wb != nil && wb.WorkbookPr != nil { date1904 = wb.WorkbookPr.Date1904 } var isNum bool cellData.T, cellData.V, isNum, err = setCellTime(value, date1904) if err != nil { return err } if isNum { _ = f.setDefaultTimeStyle(sheet, axis, 22) } return err } // setCellTime prepares cell type and Excel time by given Go time.Time type // timestamp. func setCellTime(value time.Time, date1904 bool) (t string, b string, isNum bool, err error) { var excelTime float64 _, offset := value.In(value.Location()).Zone() value = value.Add(time.Duration(offset) * time.Second) if excelTime, err = timeToExcelTime(value, date1904); err != nil { return } isNum = excelTime > 0 if isNum { t, b = setCellDefault(strconv.FormatFloat(excelTime, 'f', -1, 64)) } else { t, b = setCellDefault(value.Format(time.RFC3339Nano)) } return } // setCellDuration prepares cell type and value by given Go time.Duration type // time duration. func setCellDuration(value time.Duration) (t string, v string) { v = strconv.FormatFloat(value.Seconds()/86400, 'f', -1, 32) return } // SetCellInt provides a function to set int type value of a cell by given // worksheet name, cell coordinates and cell value. func (f *File) SetCellInt(sheet, axis string, value int) error { ws, err := f.workSheetReader(sheet) if err != nil { return err } cellData, col, row, err := f.prepareCell(ws, axis) if err != nil { return err } ws.Lock() defer ws.Unlock() cellData.S = f.prepareCellStyle(ws, col, row, cellData.S) cellData.T, cellData.V = setCellInt(value) cellData.IS = nil f.removeFormula(cellData, ws, sheet) return err } // setCellInt prepares cell type and string type cell value by a given // integer. func setCellInt(value int) (t string, v string) { v = strconv.Itoa(value) return } // SetCellBool provides a function to set bool type value of a cell by given // worksheet name, cell name and cell value. func (f *File) SetCellBool(sheet, axis string, value bool) error { ws, err := f.workSheetReader(sheet) if err != nil { return err } cellData, col, row, err := f.prepareCell(ws, axis) if err != nil { return err } ws.Lock() defer ws.Unlock() cellData.S = f.prepareCellStyle(ws, col, row, cellData.S) cellData.T, cellData.V = setCellBool(value) cellData.IS = nil f.removeFormula(cellData, ws, sheet) return err } // setCellBool prepares cell type and string type cell value by a given // boolean value. func setCellBool(value bool) (t string, v string) { t = "b" if value { v = "1" } else { v = "0" } return } // SetCellFloat sets a floating point value into a cell. The precision parameter // specifies how many places after the decimal will be shown while -1 is a // special value that will use as many decimal places as necessary to // represent the number. bitSize is 32 or 64 depending on if a float32 or // float64 was originally used for the value. For Example: // // var x float32 = 1.325 // f.SetCellFloat("Sheet1", "A1", float64(x), 2, 32) func (f *File) SetCellFloat(sheet, axis string, value float64, precision, bitSize int) error { ws, err := f.workSheetReader(sheet) if err != nil { return err } cellData, col, row, err := f.prepareCell(ws, axis) if err != nil { return err } ws.Lock() defer ws.Unlock() cellData.S = f.prepareCellStyle(ws, col, row, cellData.S) cellData.T, cellData.V = setCellFloat(value, precision, bitSize) cellData.IS = nil f.removeFormula(cellData, ws, sheet) return err } // setCellFloat prepares cell type and string type cell value by a given // float value. func setCellFloat(value float64, precision, bitSize int) (t string, v string) { v = strconv.FormatFloat(value, 'f', precision, bitSize) return } // SetCellStr provides a function to set string type value of a cell. Total // number of characters that a cell can contain 32767 characters. func (f *File) SetCellStr(sheet, axis, value string) error { ws, err := f.workSheetReader(sheet) if err != nil { return err } cellData, col, row, err := f.prepareCell(ws, axis) if err != nil { return err } ws.Lock() defer ws.Unlock() cellData.S = f.prepareCellStyle(ws, col, row, cellData.S) cellData.T, cellData.V, err = f.setCellString(value) cellData.IS = nil f.removeFormula(cellData, ws, sheet) return err } // setCellString provides a function to set string type to shared string // table. func (f *File) setCellString(value string) (t, v string, err error) { if len(value) > TotalCellChars { value = value[:TotalCellChars] } t = "s" var si int if si, err = f.setSharedString(value); err != nil { return } v = strconv.Itoa(si) return } // sharedStringsLoader load shared string table from system temporary file to // memory, and reset shared string table for reader. func (f *File) sharedStringsLoader() (err error) { f.Lock() defer f.Unlock() if path, ok := f.tempFiles.Load(defaultXMLPathSharedStrings); ok { f.Pkg.Store(defaultXMLPathSharedStrings, f.readBytes(defaultXMLPathSharedStrings)) f.tempFiles.Delete(defaultXMLPathSharedStrings) if err = os.Remove(path.(string)); err != nil { return } f.SharedStrings = nil } if f.sharedStringTemp != nil { if err := f.sharedStringTemp.Close(); err != nil { return err } f.tempFiles.Delete(defaultTempFileSST) f.sharedStringItem, err = nil, os.Remove(f.sharedStringTemp.Name()) f.sharedStringTemp = nil } return } // setSharedString provides a function to add string to the share string table. func (f *File) setSharedString(val string) (int, error) { if err := f.sharedStringsLoader(); err != nil { return 0, err } sst := f.sharedStringsReader() f.Lock() defer f.Unlock() if i, ok := f.sharedStringsMap[val]; ok { return i, nil } sst.Count++ sst.UniqueCount++ t := xlsxT{Val: val} _, val, t.Space = setCellStr(val) sst.SI = append(sst.SI, xlsxSI{T: &t}) f.sharedStringsMap[val] = sst.UniqueCount - 1 return sst.UniqueCount - 1, nil } // setCellStr provides a function to set string type to cell. func setCellStr(value string) (t string, v string, ns xml.Attr) { if len(value) > TotalCellChars { value = value[:TotalCellChars] } if len(value) > 0 { prefix, suffix := value[0], value[len(value)-1] for _, ascii := range []byte{9, 10, 13, 32} { if prefix == ascii || suffix == ascii { ns = xml.Attr{ Name: xml.Name{Space: NameSpaceXML, Local: "space"}, Value: "preserve", } break } } } t, v = "str", bstrMarshal(value) return } // SetCellDefault provides a function to set string type value of a cell as // default format without escaping the cell. func (f *File) SetCellDefault(sheet, axis, value string) error { ws, err := f.workSheetReader(sheet) if err != nil { return err } cellData, col, row, err := f.prepareCell(ws, axis) if err != nil { return err } ws.Lock() defer ws.Unlock() cellData.S = f.prepareCellStyle(ws, col, row, cellData.S) cellData.T, cellData.V = setCellDefault(value) cellData.IS = nil f.removeFormula(cellData, ws, sheet) return err } // setCellDefault prepares cell type and string type cell value by a given // string. func setCellDefault(value string) (t string, v string) { if ok, _ := isNumeric(value); !ok { t = "str" } v = value return } // GetCellFormula provides a function to get formula from cell by given // worksheet name and axis in XLSX file. func (f *File) GetCellFormula(sheet, axis string) (string, error) { return f.getCellStringFunc(sheet, axis, func(x *xlsxWorksheet, c *xlsxC) (string, bool, error) { if c.F == nil { return "", false, nil } if c.F.T == STCellFormulaTypeShared && c.F.Si != nil { return getSharedFormula(x, *c.F.Si, c.R), true, nil } return c.F.Content, true, nil }) } // FormulaOpts can be passed to SetCellFormula to use other formula types. type FormulaOpts struct { Type *string // Formula type Ref *string // Shared formula ref } // SetCellFormula provides a function to set formula on the cell is taken // according to the given worksheet name and cell formula settings. The result // of the formula cell can be calculated when the worksheet is opened by the // Office Excel application or can be using the "CalcCellValue" function also // can get the calculated cell value. If the Excel application doesn't // calculate the formula automatically when the workbook has been opened, // please call "UpdateLinkedValue" after setting the cell formula functions. // // Example 1, set normal formula "=SUM(A1,B1)" for the cell "A3" on "Sheet1": // // err := f.SetCellFormula("Sheet1", "A3", "=SUM(A1,B1)") // // Example 2, set one-dimensional vertical constant array (row array) formula // "1,2,3" for the cell "A3" on "Sheet1": // // err := f.SetCellFormula("Sheet1", "A3", "={1,2,3}") // // Example 3, set one-dimensional horizontal constant array (column array) // formula '"a","b","c"' for the cell "A3" on "Sheet1": // // err := f.SetCellFormula("Sheet1", "A3", "={\"a\",\"b\",\"c\"}") // // Example 4, set two-dimensional constant array formula '{1,2,"a","b"}' for // the cell "A3" on "Sheet1": // // formulaType, ref := excelize.STCellFormulaTypeArray, "A3:A3" // err := f.SetCellFormula("Sheet1", "A3", "={1,2,\"a\",\"b\"}", // excelize.FormulaOpts{Ref: &ref, Type: &formulaType}) // // Example 5, set range array formula "A1:A2" for the cell "A3" on "Sheet1": // // formulaType, ref := excelize.STCellFormulaTypeArray, "A3:A3" // err := f.SetCellFormula("Sheet1", "A3", "=A1:A2", // excelize.FormulaOpts{Ref: &ref, Type: &formulaType}) // // Example 6, set shared formula "=A1+B1" for the cell "C1:C5" // on "Sheet1", "C1" is the master cell: // // formulaType, ref := excelize.STCellFormulaTypeShared, "C1:C5" // err := f.SetCellFormula("Sheet1", "C1", "=A1+B1", // excelize.FormulaOpts{Ref: &ref, Type: &formulaType}) // // Example 7, set table formula "=SUM(Table1[[A]:[B]])" for the cell "C2" // on "Sheet1": // // package main // // import ( // "fmt" // // "github.com/xuri/excelize/v2" // ) // // func main() { // f := excelize.NewFile() // for idx, row := range [][]interface{}{{"A", "B", "C"}, {1, 2}} { // if err := f.SetSheetRow("Sheet1", fmt.Sprintf("A%d", idx+1), &row); err != nil { // fmt.Println(err) // return // } // } // if err := f.AddTable("Sheet1", "A1", "C2", // `{"table_name":"Table1","table_style":"TableStyleMedium2"}`); err != nil { // fmt.Println(err) // return // } // formulaType := excelize.STCellFormulaTypeDataTable // if err := f.SetCellFormula("Sheet1", "C2", "=SUM(Table1[[A]:[B]])", // excelize.FormulaOpts{Type: &formulaType}); err != nil { // fmt.Println(err) // return // } // if err := f.SaveAs("Book1.xlsx"); err != nil { // fmt.Println(err) // } // } func (f *File) SetCellFormula(sheet, axis, formula string, opts ...FormulaOpts) error { ws, err := f.workSheetReader(sheet) if err != nil { return err } cellData, _, _, err := f.prepareCell(ws, axis) if err != nil { return err } if formula == "" { cellData.F = nil f.deleteCalcChain(f.getSheetID(sheet), axis) return err } if cellData.F != nil { cellData.F.Content = formula } else { cellData.F = &xlsxF{Content: formula} } for _, o := range opts { if o.Type != nil { if *o.Type == STCellFormulaTypeDataTable { return err } cellData.F.T = *o.Type if cellData.F.T == STCellFormulaTypeShared { if err = ws.setSharedFormula(*o.Ref); err != nil { return err } } } if o.Ref != nil { cellData.F.Ref = *o.Ref } } cellData.IS = nil return err } // setSharedFormula set shared formula for the cells. func (ws *xlsxWorksheet) setSharedFormula(ref string) error { coordinates, err := areaRefToCoordinates(ref) if err != nil { return err } _ = sortCoordinates(coordinates) cnt := ws.countSharedFormula() for c := coordinates[0]; c <= coordinates[2]; c++ { for r := coordinates[1]; r <= coordinates[3]; r++ { prepareSheetXML(ws, c, r) cell := &ws.SheetData.Row[r-1].C[c-1] if cell.F == nil { cell.F = &xlsxF{} } cell.F.T = STCellFormulaTypeShared cell.F.Si = &cnt } } return err } // countSharedFormula count shared formula in the given worksheet. func (ws *xlsxWorksheet) countSharedFormula() (count int) { for _, row := range ws.SheetData.Row { for _, cell := range row.C { if cell.F != nil && cell.F.Si != nil && *cell.F.Si+1 > count { count = *cell.F.Si + 1 } } } return } // GetCellHyperLink gets a cell hyperlink based on the given worksheet name and // cell coordinates. If the cell has a hyperlink, it will return 'true' and // the link address, otherwise it will return 'false' and an empty link // address. // // For example, get a hyperlink to a 'H6' cell on a worksheet named 'Sheet1': // // link, target, err := f.GetCellHyperLink("Sheet1", "H6") func (f *File) GetCellHyperLink(sheet, axis string) (bool, string, error) { // Check for correct cell name if _, _, err := SplitCellName(axis); err != nil { return false, "", err } ws, err := f.workSheetReader(sheet) if err != nil { return false, "", err } if axis, err = f.mergeCellsParser(ws, axis); err != nil { return false, "", err } if ws.Hyperlinks != nil { for _, link := range ws.Hyperlinks.Hyperlink { if link.Ref == axis { if link.RID != "" { return true, f.getSheetRelationshipsTargetByID(sheet, link.RID), err } return true, link.Location, err } } } return false, "", err } // HyperlinkOpts can be passed to SetCellHyperlink to set optional hyperlink // attributes (e.g. display value) type HyperlinkOpts struct { Display *string Tooltip *string } // SetCellHyperLink provides a function to set cell hyperlink by given // worksheet name and link URL address. LinkType defines two types of // hyperlink "External" for website or "Location" for moving to one of cell in // this workbook. Maximum limit hyperlinks in a worksheet is 65530. This // function is only used to set the hyperlink of the cell and doesn't affect // the value of the cell. If you need to set the value of the cell, please use // the other functions such as `SetCellStyle` or `SetSheetRow`. The below is // example for external link. // // display, tooltip := "https://github.com/xuri/excelize", "Excelize on GitHub" // if err := f.SetCellHyperLink("Sheet1", "A3", // "https://github.com/xuri/excelize", "External", excelize.HyperlinkOpts{ // Display: &display, // Tooltip: &tooltip, // }); err != nil { // fmt.Println(err) // } // // Set underline and font color style for the cell. // style, err := f.NewStyle(&excelize.Style{ // Font: &excelize.Font{Color: "#1265BE", Underline: "single"}, // }) // if err != nil { // fmt.Println(err) // } // err = f.SetCellStyle("Sheet1", "A3", "A3", style) // // This is another example for "Location": // // err := f.SetCellHyperLink("Sheet1", "A3", "Sheet1!A40", "Location") func (f *File) SetCellHyperLink(sheet, axis, link, linkType string, opts ...HyperlinkOpts) error { // Check for correct cell name if _, _, err := SplitCellName(axis); err != nil { return err } ws, err := f.workSheetReader(sheet) if err != nil { return err } if axis, err = f.mergeCellsParser(ws, axis); err != nil { return err } var linkData xlsxHyperlink idx := -1 if ws.Hyperlinks == nil { ws.Hyperlinks = new(xlsxHyperlinks) } for i, hyperlink := range ws.Hyperlinks.Hyperlink { if hyperlink.Ref == axis { idx = i linkData = hyperlink break } } if len(ws.Hyperlinks.Hyperlink) > TotalSheetHyperlinks { return ErrTotalSheetHyperlinks } switch linkType { case "External": sheetPath, _ := f.getSheetXMLPath(sheet) sheetRels := "xl/worksheets/_rels/" + strings.TrimPrefix(sheetPath, "xl/worksheets/") + ".rels" rID := f.setRels(linkData.RID, sheetRels, SourceRelationshipHyperLink, link, linkType) linkData = xlsxHyperlink{ Ref: axis, } linkData.RID = "rId" + strconv.Itoa(rID) f.addSheetNameSpace(sheet, SourceRelationship) case "Location": linkData = xlsxHyperlink{ Ref: axis, Location: link, } default: return fmt.Errorf("invalid link type %q", linkType) } for _, o := range opts { if o.Display != nil { linkData.Display = *o.Display } if o.Tooltip != nil { linkData.Tooltip = *o.Tooltip } } if idx == -1 { ws.Hyperlinks.Hyperlink = append(ws.Hyperlinks.Hyperlink, linkData) return err } ws.Hyperlinks.Hyperlink[idx] = linkData return err } // getCellRichText returns rich text of cell by given string item. func getCellRichText(si *xlsxSI) (runs []RichTextRun) { for _, v := range si.R { run := RichTextRun{ Text: v.T.Val, } if v.RPr != nil { font := Font{Underline: "none"} font.Bold = v.RPr.B != nil font.Italic = v.RPr.I != nil if v.RPr.U != nil { font.Underline = "single" if v.RPr.U.Val != nil { font.Underline = *v.RPr.U.Val } } if v.RPr.RFont != nil && v.RPr.RFont.Val != nil { font.Family = *v.RPr.RFont.Val } if v.RPr.Sz != nil && v.RPr.Sz.Val != nil { font.Size = *v.RPr.Sz.Val } font.Strike = v.RPr.Strike != nil if v.RPr.Color != nil { font.Color = strings.TrimPrefix(v.RPr.Color.RGB, "FF") } run.Font = &font } runs = append(runs, run) } return } // GetCellRichText provides a function to get rich text of cell by given // worksheet. func (f *File) GetCellRichText(sheet, cell string) (runs []RichTextRun, err error) { ws, err := f.workSheetReader(sheet) if err != nil { return } cellData, _, _, err := f.prepareCell(ws, cell) if err != nil { return } siIdx, err := strconv.Atoi(cellData.V) if err != nil || cellData.T != "s" { return } sst := f.sharedStringsReader() if len(sst.SI) <= siIdx || siIdx < 0 { return } runs = getCellRichText(&sst.SI[siIdx]) return } // newRpr create run properties for the rich text by given font format. func newRpr(fnt *Font) *xlsxRPr { rpr := xlsxRPr{} trueVal := "" if fnt.Bold { rpr.B = &trueVal } if fnt.Italic { rpr.I = &trueVal } if fnt.Strike { rpr.Strike = &trueVal } if fnt.Underline != "" { rpr.U = &attrValString{Val: &fnt.Underline} } if fnt.Family != "" { rpr.RFont = &attrValString{Val: &fnt.Family} } if inStrSlice([]string{"baseline", "superscript", "subscript"}, fnt.VertAlign, true) != -1 { rpr.VertAlign = &attrValString{Val: &fnt.VertAlign} } if fnt.Size > 0 { rpr.Sz = &attrValFloat{Val: &fnt.Size} } if fnt.Color != "" { rpr.Color = &xlsxColor{RGB: getPaletteColor(fnt.Color)} } return &rpr } // SetCellRichText provides a function to set cell with rich text by given // worksheet. For example, set rich text on the A1 cell of the worksheet named // Sheet1: // // package main // // import ( // "fmt" // // "github.com/xuri/excelize/v2" // ) // // func main() { // f := excelize.NewFile() // if err := f.SetRowHeight("Sheet1", 1, 35); err != nil { // fmt.Println(err) // return // } // if err := f.SetColWidth("Sheet1", "A", "A", 44); err != nil { // fmt.Println(err) // return // } // if err := f.SetCellRichText("Sheet1", "A1", []excelize.RichTextRun{ // { // Text: "bold", // Font: &excelize.Font{ // Bold: true, // Color: "2354e8", // Family: "Times New Roman", // }, // }, // { // Text: " and ", // Font: &excelize.Font{ // Family: "Times New Roman", // }, // }, // { // Text: "italic ", // Font: &excelize.Font{ // Bold: true, // Color: "e83723", // Italic: true, // Family: "Times New Roman", // }, // }, // { // Text: "text with color and font-family,", // Font: &excelize.Font{ // Bold: true, // Color: "2354e8", // Family: "Times New Roman", // }, // }, // { // Text: "\r\nlarge text with ", // Font: &excelize.Font{ // Size: 14, // Color: "ad23e8", // }, // }, // { // Text: "strike", // Font: &excelize.Font{ // Color: "e89923", // Strike: true, // }, // }, // { // Text: " superscript", // Font: &excelize.Font{ // Color: "dbc21f", // VertAlign: "superscript", // }, // }, // { // Text: " and ", // Font: &excelize.Font{ // Size: 14, // Color: "ad23e8", // VertAlign: "baseline", // }, // }, // { // Text: "underline", // Font: &excelize.Font{ // Color: "23e833", // Underline: "single", // }, // }, // { // Text: " subscript.", // Font: &excelize.Font{ // Color: "017505", // VertAlign: "subscript", // }, // }, // }); err != nil { // fmt.Println(err) // return // } // style, err := f.NewStyle(&excelize.Style{ // Alignment: &excelize.Alignment{ // WrapText: true, // }, // }) // if err != nil { // fmt.Println(err) // return // } // if err := f.SetCellStyle("Sheet1", "A1", "A1", style); err != nil { // fmt.Println(err) // return // } // if err := f.SaveAs("Book1.xlsx"); err != nil { // fmt.Println(err) // } // } func (f *File) SetCellRichText(sheet, cell string, runs []RichTextRun) error { ws, err := f.workSheetReader(sheet) if err != nil { return err } cellData, col, row, err := f.prepareCell(ws, cell) if err != nil { return err } if err := f.sharedStringsLoader(); err != nil { return err } cellData.S = f.prepareCellStyle(ws, col, row, cellData.S) si := xlsxSI{} sst := f.sharedStringsReader() var textRuns []xlsxR totalCellChars := 0 for _, textRun := range runs { totalCellChars += len(textRun.Text) if totalCellChars > TotalCellChars { return ErrCellCharsLength } run := xlsxR{T: &xlsxT{}} _, run.T.Val, run.T.Space = setCellStr(textRun.Text) fnt := textRun.Font if fnt != nil { run.RPr = newRpr(fnt) } textRuns = append(textRuns, run) } si.R = textRuns for idx, strItem := range sst.SI { if reflect.DeepEqual(strItem, si) { cellData.T, cellData.V = "s", strconv.Itoa(idx) return err } } sst.SI = append(sst.SI, si) sst.Count++ sst.UniqueCount++ cellData.T, cellData.V = "s", strconv.Itoa(len(sst.SI)-1) return err } // SetSheetRow writes an array to row by given worksheet name, starting // coordinate and a pointer to array type 'slice'. For example, writes an // array to row 6 start with the cell B6 on Sheet1: // // err := f.SetSheetRow("Sheet1", "B6", &[]interface{}{"1", nil, 2}) func (f *File) SetSheetRow(sheet, axis string, slice interface{}) error { col, row, err := CellNameToCoordinates(axis) if err != nil { return err } // Make sure 'slice' is a Ptr to Slice v := reflect.ValueOf(slice) if v.Kind() != reflect.Ptr || v.Elem().Kind() != reflect.Slice { return ErrParameterInvalid } v = v.Elem() for i := 0; i < v.Len(); i++ { cell, err := CoordinatesToCellName(col+i, row) // Error should never happen here. But keep checking to early detect regressions // if it will be introduced in the future. if err != nil { return err } if err := f.SetCellValue(sheet, cell, v.Index(i).Interface()); err != nil { return err } } return err } // getCellInfo does common preparation for all SetCell* methods. func (f *File) prepareCell(ws *xlsxWorksheet, cell string) (*xlsxC, int, int, error) { var err error cell, err = f.mergeCellsParser(ws, cell) if err != nil { return nil, 0, 0, err } col, row, err := CellNameToCoordinates(cell) if err != nil { return nil, 0, 0, err } prepareSheetXML(ws, col, row) ws.Lock() defer ws.Unlock() return &ws.SheetData.Row[row-1].C[col-1], col, row, err } // getCellStringFunc does common value extraction workflow for all GetCell* // methods. Passed function implements specific part of required logic. func (f *File) getCellStringFunc(sheet, axis string, fn func(x *xlsxWorksheet, c *xlsxC) (string, bool, error)) (string, error) { ws, err := f.workSheetReader(sheet) if err != nil { return "", err } axis, err = f.mergeCellsParser(ws, axis) if err != nil { return "", err } _, row, err := CellNameToCoordinates(axis) if err != nil { return "", err } ws.Lock() defer ws.Unlock() lastRowNum := 0 if l := len(ws.SheetData.Row); l > 0 { lastRowNum = ws.SheetData.Row[l-1].R } // keep in mind: row starts from 1 if row > lastRowNum { return "", nil } for rowIdx := range ws.SheetData.Row { rowData := &ws.SheetData.Row[rowIdx] if rowData.R != row { continue } for colIdx := range rowData.C { colData := &rowData.C[colIdx] if axis != colData.R { continue } val, ok, err := fn(ws, colData) if err != nil { return "", err } if ok { return val, nil } } } return "", nil } // formattedValue provides a function to returns a value after formatted. If // it is possible to apply a format to the cell value, it will do so, if not // then an error will be returned, along with the raw value of the cell. func (f *File) formattedValue(s int, v string, raw bool) string { if raw { return v } if s == 0 { return v } styleSheet := f.stylesReader() if s >= len(styleSheet.CellXfs.Xf) { return v } var numFmtID int if styleSheet.CellXfs.Xf[s].NumFmtID != nil { numFmtID = *styleSheet.CellXfs.Xf[s].NumFmtID } date1904, wb := false, f.workbookReader() if wb != nil && wb.WorkbookPr != nil { date1904 = wb.WorkbookPr.Date1904 } if ok := builtInNumFmtFunc[numFmtID]; ok != nil { return ok(v, builtInNumFmt[numFmtID], date1904) } if styleSheet == nil || styleSheet.NumFmts == nil { return v } for _, xlsxFmt := range styleSheet.NumFmts.NumFmt { if xlsxFmt.NumFmtID == numFmtID { return format(v, xlsxFmt.FormatCode, date1904) } } return v } // prepareCellStyle provides a function to prepare style index of cell in // worksheet by given column index and style index. func (f *File) prepareCellStyle(ws *xlsxWorksheet, col, row, style int) int { if style != 0 { return style } if row <= len(ws.SheetData.Row) { if styleID := ws.SheetData.Row[row-1].S; styleID != 0 { return styleID } } if ws.Cols != nil { for _, c := range ws.Cols.Col { if c.Min <= col && col <= c.Max && c.Style != 0 { return c.Style } } } return style } // mergeCellsParser provides a function to check merged cells in worksheet by // given axis. func (f *File) mergeCellsParser(ws *xlsxWorksheet, axis string) (string, error) { axis = strings.ToUpper(axis) if ws.MergeCells != nil { for i := 0; i < len(ws.MergeCells.Cells); i++ { if ws.MergeCells.Cells[i] == nil { ws.MergeCells.Cells = append(ws.MergeCells.Cells[:i], ws.MergeCells.Cells[i+1:]...) i-- continue } ok, err := f.checkCellInArea(axis, ws.MergeCells.Cells[i].Ref) if err != nil { return axis, err } if ok { axis = strings.Split(ws.MergeCells.Cells[i].Ref, ":")[0] } } } return axis, nil } // checkCellInArea provides a function to determine if a given coordinate is // within an area. func (f *File) checkCellInArea(cell, area string) (bool, error) { col, row, err := CellNameToCoordinates(cell) if err != nil { return false, err } if rng := strings.Split(area, ":"); len(rng) != 2 { return false, err } coordinates, err := areaRefToCoordinates(area) if err != nil { return false, err } return cellInRef([]int{col, row}, coordinates), err } // cellInRef provides a function to determine if a given range is within a // range. func cellInRef(cell, ref []int) bool { return cell[0] >= ref[0] && cell[0] <= ref[2] && cell[1] >= ref[1] && cell[1] <= ref[3] } // isOverlap find if the given two rectangles overlap or not. func isOverlap(rect1, rect2 []int) bool { return cellInRef([]int{rect1[0], rect1[1]}, rect2) || cellInRef([]int{rect1[2], rect1[1]}, rect2) || cellInRef([]int{rect1[0], rect1[3]}, rect2) || cellInRef([]int{rect1[2], rect1[3]}, rect2) || cellInRef([]int{rect2[0], rect2[1]}, rect1) || cellInRef([]int{rect2[2], rect2[1]}, rect1) || cellInRef([]int{rect2[0], rect2[3]}, rect1) || cellInRef([]int{rect2[2], rect2[3]}, rect1) } // parseSharedFormula generate dynamic part of shared formula for target cell // by given column and rows distance and origin shared formula. func parseSharedFormula(dCol, dRow int, orig []byte) (res string, start int) { var ( end int stringLiteral bool ) for end = 0; end < len(orig); end++ { c := orig[end] if c == '"' { stringLiteral = !stringLiteral } if stringLiteral { continue // Skip characters in quotes } if c >= 'A' && c <= 'Z' || c == '$' { res += string(orig[start:end]) start = end end++ foundNum := false for ; end < len(orig); end++ { idc := orig[end] if idc >= '0' && idc <= '9' || idc == '$' { foundNum = true } else if idc >= 'A' && idc <= 'Z' { if foundNum { break } } else { break } } if foundNum { cellID := string(orig[start:end]) res += shiftCell(cellID, dCol, dRow) start = end } } } return } // getSharedFormula find a cell contains the same formula as another cell, // the "shared" value can be used for the t attribute and the si attribute can // be used to refer to the cell containing the formula. Two formulas are // considered to be the same when their respective representations in // R1C1-reference notation, are the same. // // Note that this function not validate ref tag to check the cell whether in // allow area, and always return origin shared formula. func getSharedFormula(ws *xlsxWorksheet, si int, axis string) string { for _, r := range ws.SheetData.Row { for _, c := range r.C { if c.F != nil && c.F.Ref != "" && c.F.T == STCellFormulaTypeShared && c.F.Si != nil && *c.F.Si == si { col, row, _ := CellNameToCoordinates(axis) sharedCol, sharedRow, _ := CellNameToCoordinates(c.R) dCol := col - sharedCol dRow := row - sharedRow orig := []byte(c.F.Content) res, start := parseSharedFormula(dCol, dRow, orig) if start < len(orig) { res += string(orig[start:]) } return res } } } return "" } // shiftCell returns the cell shifted according to dCol and dRow taking into // consideration absolute references with dollar sign ($) func shiftCell(cellID string, dCol, dRow int) string { fCol, fRow, _ := CellNameToCoordinates(cellID) signCol, signRow := "", "" if strings.Index(cellID, "$") == 0 { signCol = "$" } else { // Shift column fCol += dCol } if strings.LastIndex(cellID, "$") > 0 { signRow = "$" } else { // Shift row fRow += dRow } colName, _ := ColumnNumberToName(fCol) return signCol + colName + signRow + strconv.Itoa(fRow) }