// Copyright 2016 - 2021 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 XLSX / XLSM / XLTM 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 ( "fmt" "strings" ) // MergeCell provides a function to merge cells by given coordinate area and // sheet name. Merging cells only keeps the upper-left cell value, and // discards the other values. For example create a merged cell of D3:E9 on // Sheet1: // // err := f.MergeCell("Sheet1", "D3", "E9") // // If you create a merged cell that overlaps with another existing merged cell, // those merged cells that already exist will be removed. // // B1(x1,y1) D1(x2,y1) // +------------------------+ // | | // A4(x3,y3) | C4(x4,y3) | // +------------------------+ | // | | | | // | |B5(x1,y2) | D5(x2,y2)| // | +------------------------+ // | | // |A8(x3,y4) C8(x4,y4)| // +------------------------+ // func (f *File) MergeCell(sheet, hcell, vcell string) error { rect1, err := f.areaRefToCoordinates(hcell + ":" + vcell) if err != nil { return err } // Correct the coordinate area, such correct C1:B3 to B1:C3. _ = sortCoordinates(rect1) hcell, _ = CoordinatesToCellName(rect1[0], rect1[1]) vcell, _ = CoordinatesToCellName(rect1[2], rect1[3]) ws, err := f.workSheetReader(sheet) if err != nil { return err } ref := hcell + ":" + vcell if ws.MergeCells != nil { for i := 0; i < len(ws.MergeCells.Cells); i++ { cellData := ws.MergeCells.Cells[i] if cellData == nil { continue } cc := strings.Split(cellData.Ref, ":") if len(cc) != 2 { return fmt.Errorf("invalid area %q", cellData.Ref) } rect2, err := f.areaRefToCoordinates(cellData.Ref) if err != nil { return err } // Delete the merged cells of the overlapping area. if isOverlap(rect1, rect2) { ws.MergeCells.Cells = append(ws.MergeCells.Cells[:i], ws.MergeCells.Cells[i+1:]...) i-- if rect1[0] > rect2[0] { rect1[0], rect2[0] = rect2[0], rect1[0] } if rect1[2] < rect2[2] { rect1[2], rect2[2] = rect2[2], rect1[2] } if rect1[1] > rect2[1] { rect1[1], rect2[1] = rect2[1], rect1[1] } if rect1[3] < rect2[3] { rect1[3], rect2[3] = rect2[3], rect1[3] } hcell, _ = CoordinatesToCellName(rect1[0], rect1[1]) vcell, _ = CoordinatesToCellName(rect1[2], rect1[3]) ref = hcell + ":" + vcell } } ws.MergeCells.Cells = append(ws.MergeCells.Cells, &xlsxMergeCell{Ref: ref}) } else { ws.MergeCells = &xlsxMergeCells{Cells: []*xlsxMergeCell{{Ref: ref}}} } ws.MergeCells.Count = len(ws.MergeCells.Cells) return err } // UnmergeCell provides a function to unmerge a given coordinate area. // For example unmerge area D3:E9 on Sheet1: // // err := f.UnmergeCell("Sheet1", "D3", "E9") // // Attention: overlapped areas will also be unmerged. func (f *File) UnmergeCell(sheet string, hcell, vcell string) error { ws, err := f.workSheetReader(sheet) if err != nil { return err } rect1, err := f.areaRefToCoordinates(hcell + ":" + vcell) if err != nil { return err } // Correct the coordinate area, such correct C1:B3 to B1:C3. _ = sortCoordinates(rect1) // return nil since no MergeCells in the sheet if ws.MergeCells == nil { return nil } i := 0 for _, cellData := range ws.MergeCells.Cells { if cellData == nil { continue } cc := strings.Split(cellData.Ref, ":") if len(cc) != 2 { return fmt.Errorf("invalid area %q", cellData.Ref) } rect2, err := f.areaRefToCoordinates(cellData.Ref) if err != nil { return err } if isOverlap(rect1, rect2) { continue } ws.MergeCells.Cells[i] = cellData i++ } ws.MergeCells.Cells = ws.MergeCells.Cells[:i] ws.MergeCells.Count = len(ws.MergeCells.Cells) if ws.MergeCells.Count == 0 { ws.MergeCells = nil } return nil } // GetMergeCells provides a function to get all merged cells from a worksheet // currently. func (f *File) GetMergeCells(sheet string) ([]MergeCell, error) { var mergeCells []MergeCell ws, err := f.workSheetReader(sheet) if err != nil { return mergeCells, err } if ws.MergeCells != nil { mergeCells = make([]MergeCell, 0, len(ws.MergeCells.Cells)) for i := range ws.MergeCells.Cells { ref := ws.MergeCells.Cells[i].Ref axis := strings.Split(ref, ":")[0] val, _ := f.GetCellValue(sheet, axis) mergeCells = append(mergeCells, []string{ref, val}) } } return mergeCells, err } // MergeCell define a merged cell data. // It consists of the following structure. // example: []string{"D4:E10", "cell value"} type MergeCell []string // GetCellValue returns merged cell value. func (m *MergeCell) GetCellValue() string { return (*m)[1] } // GetStartAxis returns the merge start axis. // example: "C2" func (m *MergeCell) GetStartAxis() string { axis := strings.Split((*m)[0], ":") return axis[0] } // GetEndAxis returns the merge end axis. // example: "D4" func (m *MergeCell) GetEndAxis() string { axis := strings.Split((*m)[0], ":") return axis[1] }