sheet_test.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  1. package xlsx
  2. import (
  3. "bytes"
  4. "encoding/xml"
  5. . "gopkg.in/check.v1"
  6. )
  7. type SheetSuite struct{}
  8. var _ = Suite(&SheetSuite{})
  9. // Test we can add a Row to a Sheet
  10. func (s *SheetSuite) TestAddRow(c *C) {
  11. var f *File
  12. f = NewFile()
  13. sheet := f.AddSheet("MySheet")
  14. row := sheet.AddRow()
  15. c.Assert(row, NotNil)
  16. c.Assert(len(sheet.Rows), Equals, 1)
  17. }
  18. func (s *SheetSuite) TestMakeXLSXSheetFromRows(c *C) {
  19. file := NewFile()
  20. sheet := file.AddSheet("Sheet1")
  21. row := sheet.AddRow()
  22. cell := row.AddCell()
  23. cell.Value = "A cell!"
  24. refTable := NewSharedStringRefTable()
  25. styles := newXlsxStyleSheet(nil)
  26. xSheet := sheet.makeXLSXSheet(refTable, styles)
  27. c.Assert(xSheet.Dimension.Ref, Equals, "A1")
  28. c.Assert(xSheet.SheetData.Row, HasLen, 1)
  29. xRow := xSheet.SheetData.Row[0]
  30. c.Assert(xRow.R, Equals, 1)
  31. c.Assert(xRow.Spans, Equals, "")
  32. c.Assert(xRow.C, HasLen, 1)
  33. xC := xRow.C[0]
  34. c.Assert(xC.R, Equals, "A1")
  35. c.Assert(xC.S, Equals, 0)
  36. c.Assert(xC.T, Equals, "s") // Shared string type
  37. c.Assert(xC.V, Equals, "0") // reference to shared string
  38. xSST := refTable.makeXLSXSST()
  39. c.Assert(xSST.Count, Equals, 1)
  40. c.Assert(xSST.UniqueCount, Equals, 1)
  41. c.Assert(xSST.SI, HasLen, 1)
  42. xSI := xSST.SI[0]
  43. c.Assert(xSI.T, Equals, "A cell!")
  44. }
  45. // Test if the NumFmts assigned properly according the FormatCode in cell.
  46. func (s *SheetSuite) TestMakeXLSXSheetWithNumFormats(c *C) {
  47. file := NewFile()
  48. sheet := file.AddSheet("Sheet1")
  49. row := sheet.AddRow()
  50. cell1 := row.AddCell()
  51. cell1.Value = "A cell!"
  52. cell1.numFmt = "general"
  53. cell2 := row.AddCell()
  54. cell2.Value = "37947.7500001"
  55. cell2.numFmt = "0"
  56. cell3 := row.AddCell()
  57. cell3.Value = "37947.7500001"
  58. cell3.numFmt = "mm-dd-yy"
  59. cell4 := row.AddCell()
  60. cell4.Value = "37947.7500001"
  61. cell4.numFmt = "hh:mm:ss"
  62. refTable := NewSharedStringRefTable()
  63. styles := newXlsxStyleSheet(nil)
  64. worksheet := sheet.makeXLSXSheet(refTable, styles)
  65. c.Assert(styles.CellStyleXfs.Count, Equals, 1)
  66. // The 0th CellStyleXf could just be getting the zero values by default
  67. c.Assert(styles.CellStyleXfs.Xf[0].FontId, Equals, 0)
  68. c.Assert(styles.CellStyleXfs.Xf[0].FillId, Equals, 0)
  69. c.Assert(styles.CellStyleXfs.Xf[0].BorderId, Equals, 0)
  70. c.Assert(styles.CellXfs.Count, Equals, 4)
  71. c.Assert(styles.CellXfs.Xf[0].NumFmtId, Equals, 0)
  72. c.Assert(styles.CellXfs.Xf[1].NumFmtId, Equals, 1)
  73. c.Assert(styles.CellXfs.Xf[2].NumFmtId, Equals, 14)
  74. c.Assert(styles.CellXfs.Xf[3].NumFmtId, Equals, 164)
  75. c.Assert(styles.NumFmts.Count, Equals, 1)
  76. c.Assert(styles.NumFmts.NumFmt[0].NumFmtId, Equals, 164)
  77. c.Assert(styles.NumFmts.NumFmt[0].FormatCode, Equals, "hh:mm:ss")
  78. // Finally we check that the cell points to the right CellXf /
  79. // CellStyleXf.
  80. c.Assert(worksheet.SheetData.Row[0].C[0].S, Equals, 0)
  81. c.Assert(worksheet.SheetData.Row[0].C[1].S, Equals, 1)
  82. }
  83. // When we create the xlsxSheet we also populate the xlsxStyles struct
  84. // with style information.
  85. func (s *SheetSuite) TestMakeXLSXSheetAlsoPopulatesXLSXSTyles(c *C) {
  86. file := NewFile()
  87. sheet := file.AddSheet("Sheet1")
  88. row := sheet.AddRow()
  89. cell1 := row.AddCell()
  90. cell1.Value = "A cell!"
  91. style1 := NewStyle()
  92. style1.Font = *NewFont(10, "Verdana")
  93. style1.Fill = *NewFill("solid", "FFFFFFFF", "00000000")
  94. style1.Border = *NewBorder("none", "thin", "none", "thin")
  95. cell1.SetStyle(style1)
  96. // We need a second style to check that Xfs are populated correctly.
  97. cell2 := row.AddCell()
  98. cell2.Value = "Another cell!"
  99. style2 := NewStyle()
  100. style2.Font = *NewFont(10, "Verdana")
  101. style2.Fill = *NewFill("solid", "FFFFFFFF", "00000000")
  102. style2.Border = *NewBorder("none", "thin", "none", "thin")
  103. cell2.SetStyle(style2)
  104. refTable := NewSharedStringRefTable()
  105. styles := newXlsxStyleSheet(nil)
  106. worksheet := sheet.makeXLSXSheet(refTable, styles)
  107. c.Assert(styles.Fonts.Count, Equals, 2)
  108. c.Assert(styles.Fonts.Font[0].Sz.Val, Equals, "12")
  109. c.Assert(styles.Fonts.Font[0].Name.Val, Equals, "Verdana")
  110. c.Assert(styles.Fonts.Font[1].Sz.Val, Equals, "10")
  111. c.Assert(styles.Fonts.Font[1].Name.Val, Equals, "Verdana")
  112. c.Assert(styles.Fills.Count, Equals, 3)
  113. c.Assert(styles.Fills.Fill[0].PatternFill.PatternType, Equals, "none")
  114. c.Assert(styles.Fills.Fill[0].PatternFill.FgColor.RGB, Equals, "FFFFFFFF")
  115. c.Assert(styles.Fills.Fill[0].PatternFill.BgColor.RGB, Equals, "00000000")
  116. c.Assert(styles.Borders.Count, Equals, 2)
  117. c.Assert(styles.Borders.Border[1].Left.Style, Equals, "none")
  118. c.Assert(styles.Borders.Border[1].Right.Style, Equals, "thin")
  119. c.Assert(styles.Borders.Border[1].Top.Style, Equals, "none")
  120. c.Assert(styles.Borders.Border[1].Bottom.Style, Equals, "thin")
  121. c.Assert(styles.CellStyleXfs.Count, Equals, 2)
  122. // The 0th CellStyleXf could just be getting the zero values by default
  123. c.Assert(styles.CellStyleXfs.Xf[0].FontId, Equals, 0)
  124. c.Assert(styles.CellStyleXfs.Xf[0].FillId, Equals, 0)
  125. c.Assert(styles.CellStyleXfs.Xf[0].BorderId, Equals, 0)
  126. c.Assert(styles.CellXfs.Count, Equals, 2)
  127. c.Assert(styles.CellXfs.Xf[0].FontId, Equals, 0)
  128. c.Assert(styles.CellXfs.Xf[0].FillId, Equals, 0)
  129. c.Assert(styles.CellXfs.Xf[0].BorderId, Equals, 0)
  130. // Finally we check that the cell points to the right CellXf /
  131. // CellStyleXf.
  132. c.Assert(worksheet.SheetData.Row[0].C[0].S, Equals, 1)
  133. c.Assert(worksheet.SheetData.Row[0].C[1].S, Equals, 1)
  134. }
  135. // If the column width is not customised, the xslxCol.CustomWidth field is set to 0.
  136. func (s *SheetSuite) TestMakeXLSXSheetDefaultsCustomColWidth(c *C) {
  137. file := NewFile()
  138. sheet := file.AddSheet("Sheet1")
  139. row := sheet.AddRow()
  140. cell1 := row.AddCell()
  141. cell1.Value = "A cell!"
  142. refTable := NewSharedStringRefTable()
  143. styles := newXlsxStyleSheet(nil)
  144. worksheet := sheet.makeXLSXSheet(refTable, styles)
  145. c.Assert(worksheet.Cols.Col[0].CustomWidth, Equals, 0)
  146. }
  147. // If the column width is customised, the xslxCol.CustomWidth field is set to 1.
  148. func (s *SheetSuite) TestMakeXLSXSheetSetsCustomColWidth(c *C) {
  149. file := NewFile()
  150. sheet := file.AddSheet("Sheet1")
  151. err := sheet.SetColWidth(0, 0, 10.5)
  152. c.Assert(err, IsNil)
  153. refTable := NewSharedStringRefTable()
  154. styles := newXlsxStyleSheet(nil)
  155. worksheet := sheet.makeXLSXSheet(refTable, styles)
  156. c.Assert(worksheet.Cols.Col[0].CustomWidth, Equals, 1)
  157. }
  158. func (s *SheetSuite) TestMarshalSheet(c *C) {
  159. file := NewFile()
  160. sheet := file.AddSheet("Sheet1")
  161. row := sheet.AddRow()
  162. cell := row.AddCell()
  163. cell.Value = "A cell!"
  164. refTable := NewSharedStringRefTable()
  165. styles := newXlsxStyleSheet(nil)
  166. xSheet := sheet.makeXLSXSheet(refTable, styles)
  167. output := bytes.NewBufferString(xml.Header)
  168. body, err := xml.Marshal(xSheet)
  169. c.Assert(err, IsNil)
  170. c.Assert(body, NotNil)
  171. _, err = output.Write(body)
  172. c.Assert(err, IsNil)
  173. expectedXLSXSheet := `<?xml version="1.0" encoding="UTF-8"?>
  174. <worksheet xmlns="http://schemas.openxmlformats.org/spreadsheetml/2006/main"><sheetPr filterMode="false"><pageSetUpPr fitToPage="false"></pageSetUpPr></sheetPr><dimension ref="A1"></dimension><sheetViews><sheetView windowProtection="false" showFormulas="false" showGridLines="true" showRowColHeaders="true" showZeros="true" rightToLeft="false" tabSelected="true" showOutlineSymbols="true" defaultGridColor="true" view="normal" topLeftCell="A1" colorId="64" zoomScale="100" zoomScaleNormal="100" zoomScalePageLayoutView="100" workbookViewId="0"><selection pane="topLeft" activeCell="A1" activeCellId="0" sqref="A1"></selection></sheetView></sheetViews><sheetFormatPr defaultRowHeight="12.85"></sheetFormatPr><cols><col collapsed="false" hidden="false" max="1" min="1" style="0" width="9.5"></col></cols><sheetData><row r="1"><c r="A1" t="s"><v>0</v></c></row></sheetData><printOptions headings="false" gridLines="false" gridLinesSet="true" horizontalCentered="false" verticalCentered="false"></printOptions><pageMargins left="0.7875" right="0.7875" top="1.05277777777778" bottom="1.05277777777778" header="0.7875" footer="0.7875"></pageMargins><pageSetup paperSize="9" scale="100" firstPageNumber="1" fitToWidth="1" fitToHeight="1" pageOrder="downThenOver" orientation="portrait" usePrinterDefaults="false" blackAndWhite="false" draft="false" cellComments="none" useFirstPageNumber="true" horizontalDpi="300" verticalDpi="300" copies="1"></pageSetup><headerFooter differentFirst="false" differentOddEven="false"><oddHeader>&amp;C&amp;&#34;Times New Roman,Regular&#34;&amp;12&amp;A</oddHeader><oddFooter>&amp;C&amp;&#34;Times New Roman,Regular&#34;&amp;12Page &amp;P</oddFooter></headerFooter></worksheet>`
  175. c.Assert(output.String(), Equals, expectedXLSXSheet)
  176. }
  177. func (s *SheetSuite) TestMarshalSheetWithMultipleCells(c *C) {
  178. file := NewFile()
  179. sheet := file.AddSheet("Sheet1")
  180. row := sheet.AddRow()
  181. cell := row.AddCell()
  182. cell.Value = "A cell (with value 1)!"
  183. cell = row.AddCell()
  184. cell.Value = "A cell (with value 2)!"
  185. refTable := NewSharedStringRefTable()
  186. styles := newXlsxStyleSheet(nil)
  187. xSheet := sheet.makeXLSXSheet(refTable, styles)
  188. output := bytes.NewBufferString(xml.Header)
  189. body, err := xml.Marshal(xSheet)
  190. c.Assert(err, IsNil)
  191. c.Assert(body, NotNil)
  192. _, err = output.Write(body)
  193. c.Assert(err, IsNil)
  194. expectedXLSXSheet := `<?xml version="1.0" encoding="UTF-8"?>
  195. <worksheet xmlns="http://schemas.openxmlformats.org/spreadsheetml/2006/main"><sheetPr filterMode="false"><pageSetUpPr fitToPage="false"></pageSetUpPr></sheetPr><dimension ref="A1:B1"></dimension><sheetViews><sheetView windowProtection="false" showFormulas="false" showGridLines="true" showRowColHeaders="true" showZeros="true" rightToLeft="false" tabSelected="true" showOutlineSymbols="true" defaultGridColor="true" view="normal" topLeftCell="A1" colorId="64" zoomScale="100" zoomScaleNormal="100" zoomScalePageLayoutView="100" workbookViewId="0"><selection pane="topLeft" activeCell="A1" activeCellId="0" sqref="A1"></selection></sheetView></sheetViews><sheetFormatPr defaultRowHeight="12.85"></sheetFormatPr><cols><col collapsed="false" hidden="false" max="1" min="1" style="0" width="9.5"></col><col collapsed="false" hidden="false" max="2" min="2" style="0" width="9.5"></col></cols><sheetData><row r="1"><c r="A1" t="s"><v>0</v></c><c r="B1" t="s"><v>1</v></c></row></sheetData><printOptions headings="false" gridLines="false" gridLinesSet="true" horizontalCentered="false" verticalCentered="false"></printOptions><pageMargins left="0.7875" right="0.7875" top="1.05277777777778" bottom="1.05277777777778" header="0.7875" footer="0.7875"></pageMargins><pageSetup paperSize="9" scale="100" firstPageNumber="1" fitToWidth="1" fitToHeight="1" pageOrder="downThenOver" orientation="portrait" usePrinterDefaults="false" blackAndWhite="false" draft="false" cellComments="none" useFirstPageNumber="true" horizontalDpi="300" verticalDpi="300" copies="1"></pageSetup><headerFooter differentFirst="false" differentOddEven="false"><oddHeader>&amp;C&amp;&#34;Times New Roman,Regular&#34;&amp;12&amp;A</oddHeader><oddFooter>&amp;C&amp;&#34;Times New Roman,Regular&#34;&amp;12Page &amp;P</oddFooter></headerFooter></worksheet>`
  196. c.Assert(output.String(), Equals, expectedXLSXSheet)
  197. }
  198. func (s *SheetSuite) TestSetColWidth(c *C) {
  199. file := NewFile()
  200. sheet := file.AddSheet("Sheet1")
  201. _ = sheet.SetColWidth(0, 0, 10.5)
  202. _ = sheet.SetColWidth(1, 5, 11)
  203. c.Assert(sheet.Cols[0].Width, Equals, 10.5)
  204. c.Assert(sheet.Cols[0].Max, Equals, 1)
  205. c.Assert(sheet.Cols[0].Min, Equals, 1)
  206. c.Assert(sheet.Cols[1].Width, Equals, float64(11))
  207. c.Assert(sheet.Cols[1].Max, Equals, 6)
  208. c.Assert(sheet.Cols[1].Min, Equals, 2)
  209. }
  210. func (s *SheetSuite) TestSetRowHeightCM(c *C) {
  211. file := NewFile()
  212. sheet := file.AddSheet("Sheet1")
  213. row := sheet.AddRow()
  214. row.SetHeightCM(1.5)
  215. c.Assert(row.Height, Equals, 42.51968505)
  216. }
  217. func (s *SheetSuite) TestAlignment(c *C) {
  218. leftalign := Alignment{Horizontal: "left"}
  219. centerHalign := Alignment{Horizontal: "center"}
  220. rightalign := Alignment{Horizontal: "right"}
  221. file := NewFile()
  222. sheet := file.AddSheet("Sheet1")
  223. style := NewStyle()
  224. hrow := sheet.AddRow()
  225. // Horizontals
  226. cell := hrow.AddCell()
  227. cell.Value = "left"
  228. style.Alignment = leftalign
  229. style.ApplyAlignment = true
  230. cell.SetStyle(style)
  231. style = NewStyle()
  232. cell = hrow.AddCell()
  233. cell.Value = "centerH"
  234. style.Alignment = centerHalign
  235. style.ApplyAlignment = true
  236. cell.SetStyle(style)
  237. style = NewStyle()
  238. cell = hrow.AddCell()
  239. cell.Value = "right"
  240. style.Alignment = rightalign
  241. style.ApplyAlignment = true
  242. cell.SetStyle(style)
  243. // Verticals
  244. topalign := Alignment{Vertical: "top"}
  245. centerValign := Alignment{Vertical: "center"}
  246. bottomalign := Alignment{Vertical: "bottom"}
  247. style = NewStyle()
  248. vrow := sheet.AddRow()
  249. cell = vrow.AddCell()
  250. cell.Value = "top"
  251. style.Alignment = topalign
  252. style.ApplyAlignment = true
  253. cell.SetStyle(style)
  254. style = NewStyle()
  255. cell = vrow.AddCell()
  256. cell.Value = "centerV"
  257. style.Alignment = centerValign
  258. style.ApplyAlignment = true
  259. cell.SetStyle(style)
  260. style = NewStyle()
  261. cell = vrow.AddCell()
  262. cell.Value = "bottom"
  263. style.Alignment = bottomalign
  264. style.ApplyAlignment = true
  265. cell.SetStyle(style)
  266. parts, err := file.MarshallParts()
  267. c.Assert(err, IsNil)
  268. stylepart := parts["xl/styles.xml"]
  269. shouldbe := `<?xml version="1.0" encoding="UTF-8"?>
  270. <styleSheet xmlns="http://schemas.openxmlformats.org/spreadsheetml/2006/main"><fonts count="1"><font><sz val="12"/><name val="Verdana"/><family val="0"/><charset val="0"/></font></fonts><fills count="2"><fill><patternFill patternType="none"><fgColor rgb="FFFFFFFF"/><bgColor rgb="00000000"/></patternFill></fill><fill><patternFill patternType="lightGray"/></fill></fills><borders count="1"><border><left style="none"></left><right style="none"></right><top style="none"></top><bottom style="none"></bottom></border></borders><cellStyleXfs count="7"><xf applyAlignment="0" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="" indent="0" shrinkToFit="0" textRotation="0" vertical="" wrapText="0"/></xf><xf applyAlignment="1" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="left" indent="0" shrinkToFit="0" textRotation="0" vertical="" wrapText="0"/></xf><xf applyAlignment="1" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="center" indent="0" shrinkToFit="0" textRotation="0" vertical="" wrapText="0"/></xf><xf applyAlignment="1" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="right" indent="0" shrinkToFit="0" textRotation="0" vertical="" wrapText="0"/></xf><xf applyAlignment="1" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="" indent="0" shrinkToFit="0" textRotation="0" vertical="top" wrapText="0"/></xf><xf applyAlignment="1" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="" indent="0" shrinkToFit="0" textRotation="0" vertical="center" wrapText="0"/></xf><xf applyAlignment="1" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="" indent="0" shrinkToFit="0" textRotation="0" vertical="bottom" wrapText="0"/></xf></cellStyleXfs><cellXfs count="7"><xf applyAlignment="0" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="" indent="0" shrinkToFit="0" textRotation="0" vertical="" wrapText="0"/></xf><xf applyAlignment="1" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="left" indent="0" shrinkToFit="0" textRotation="0" vertical="" wrapText="0"/></xf><xf applyAlignment="1" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="center" indent="0" shrinkToFit="0" textRotation="0" vertical="" wrapText="0"/></xf><xf applyAlignment="1" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="right" indent="0" shrinkToFit="0" textRotation="0" vertical="" wrapText="0"/></xf><xf applyAlignment="1" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="" indent="0" shrinkToFit="0" textRotation="0" vertical="top" wrapText="0"/></xf><xf applyAlignment="1" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="" indent="0" shrinkToFit="0" textRotation="0" vertical="center" wrapText="0"/></xf><xf applyAlignment="1" applyBorder="0" applyFont="0" applyFill="0" applyNumberFormat="0" applyProtection="0" borderId="0" fillId="0" fontId="0" numFmtId="0"><alignment horizontal="" indent="0" shrinkToFit="0" textRotation="0" vertical="bottom" wrapText="0"/></xf></cellXfs></styleSheet>`
  271. output := bytes.NewBufferString(shouldbe)
  272. c.Assert(output.String(), Equals, stylepart)
  273. }