StatementTable.cs
  1. //
  2. // This code is part of Document Solutions for Imaging demos.
  3. // Copyright (c) MESCIUS inc. All rights reserved.
  4. //
  5. using System;
  6. using System.IO;
  7. using System.Drawing;
  8. using System.Numerics;
  9. using System.Collections.Generic;
  10. using GrapeCity.Documents.Drawing;
  11. using GrapeCity.Documents.Text;
  12. using GrapeCity.Documents.Imaging;
  13. using GrapeCity.Documents.Layout;
  14. using GCTEXT = GrapeCity.Documents.Text;
  15. using GCDRAW = GrapeCity.Documents.Drawing;
  16.  
  17. namespace DsImagingWeb.Demos
  18. {
  19. // This example shows how to use the GrapeCity.Documents.Drawing.TableRenderer
  20. // class to render a document with a layout typically used in account statements or invoices.
  21. public class StatementTable
  22. {
  23. public GcBitmap GenerateImage(Size pixelSize, float dpi, bool opaque, string[] sampleParams = null)
  24. {
  25. var bmp = new GcBitmap(pixelSize.Width, pixelSize.Height, opaque, dpi, dpi);
  26. using var g = bmp.CreateGraphics(Color.White);
  27. DrawTable(g, pixelSize.Width, pixelSize.Height);
  28. return bmp;
  29. }
  30.  
  31. class InfoLine
  32. {
  33. public InfoLine(int indent, string text, int sum2021, int sum2020, int sum2019, bool boldText = false,
  34. bool captionOnly = false, bool drawSign = false, bool thinLine = false, bool boldLine = false)
  35. {
  36. Indent = indent;
  37. Text = text;
  38. Sum2021 = sum2021;
  39. Sum2020 = sum2020;
  40. Sum2019 = sum2019;
  41. BoldText = boldText;
  42. CaptionOnly = captionOnly;
  43. DrawSign = drawSign;
  44. ThinLine = thinLine;
  45. BoldLine = boldLine;
  46. }
  47. public int Indent { get; }
  48. public string Text { get; }
  49. public int Sum2021 { get; }
  50. public int Sum2020 { get; }
  51. public int Sum2019 { get; }
  52. public bool BoldText { get; }
  53. public bool CaptionOnly { get; }
  54. public bool DrawSign { get; }
  55. public bool ThinLine { get; }
  56. public bool BoldLine { get; }
  57. }
  58.  
  59. static void DrawTable(GcGraphics g, int pageWidth, int pageHeight)
  60. {
  61. var items = new InfoLine[]
  62. {
  63. new InfoLine(0, "Net sales:", 0, 0, 0, captionOnly: true),
  64. new InfoLine(1, "Products", 297392, 220747, 213883, drawSign: true),
  65. new InfoLine(1, "Services", 68425, 53768, 46291, thinLine: true),
  66. new InfoLine(2, "Total net sales", 365817, 274515, 260174, boldText: true),
  67. new InfoLine(0, "Cost of sales:", 0, 0, 0, captionOnly: true),
  68. new InfoLine(1, "Products", 192266, 151286, 144996),
  69. new InfoLine(1, "Services", 20715, 18273, 16786, thinLine: true),
  70. new InfoLine(2, "Total cost of sales", 212981, 169559, 161782, boldText: true, thinLine: true),
  71. new InfoLine(3, "Gross margin", 152836, 104956, 98392, boldText: true, thinLine: true),
  72. new InfoLine(0, "Operating expenses:", 0, 0, 0, captionOnly: true),
  73. new InfoLine(1, "Research and development", 21914, 18752, 16217),
  74. new InfoLine(1, "Selling, general and administrative", 21973, 19916, 18245, thinLine: true),
  75. new InfoLine(2, "Total operating expenses", 43887, 38668, 34462, boldText: true, thinLine: true),
  76. new InfoLine(0, " ", 0, 0, 0, captionOnly: true),
  77. new InfoLine(0, "Operating income", 108949, 66288, 63930),
  78. new InfoLine(0, "Other income/(expense), net", 258, 803, 1807, thinLine: true),
  79. new InfoLine(0, "Income before provision for income taxes", 109207, 67091, 65737, boldText: true),
  80. new InfoLine(0, "Provision for income taxes", 14527, 9680, 10481, thinLine: true),
  81. new InfoLine(0, "Net income", 94680, 57411, 55256, boldText: true, drawSign: true, boldLine: true),
  82. };
  83.  
  84. var host = new LayoutHost();
  85. var view = host.CreateView(pageWidth, pageHeight);
  86.  
  87. var rt = view.CreateRect();
  88. // rt.AnchorTopLeft(null, 30, 50);
  89. rt.AnchorTopLeft(null, 64, 64);
  90.  
  91. var ta = new TableRenderer(g,
  92. rt, FixedTableSides.TopLeft,
  93. rowCount: items.Length + 3,
  94. columnCount: 7,
  95. gridLineColor: Color.Transparent,
  96. gridLineWidth: 0);
  97.  
  98. var columns = ta.ColumnRects;
  99. columns[0].SetWidth(420);
  100. columns[1].SetWidth(25);
  101. columns[2].SetWidth(100);
  102. columns[3].SetWidth(25);
  103. columns[4].SetWidth(100);
  104. columns[5].SetWidth(25);
  105. columns[6].SetWidth(100);
  106.  
  107. var fmt = new TextFormat
  108. {
  109. Font = GCTEXT.Font.FromFile(Path.Combine("Resources", "Fonts", "consola.ttf")),
  110. ForeColor = Color.FromArgb(38, 38, 38),
  111. FontSize = 14,
  112. FontSizeInGraphicUnits = true,
  113. };
  114. var fmtCaptionBold = new TextFormat(fmt)
  115. {
  116. FontBold = true,
  117. FontSize = 15
  118. };
  119. var fmtCaptionLight = new TextFormat(fmt)
  120. {
  121. ForeColor = Color.FromArgb(142, 142, 142),
  122. FontSize = 15
  123. };
  124. var fmtBold = new TextFormat(fmt)
  125. {
  126. FontBold = true
  127. };
  128.  
  129. var csHeader1 = new CellStyle
  130. {
  131. TextAlignment = TextAlignment.Center,
  132. PaddingBottom = 25,
  133. CreateTextLayout = (g, cs, data) =>
  134. {
  135. var tl = g.CreateTextLayout();
  136. tl.AppendLine("ACME Inc.\n\nCONSOLIDATED STATEMENTS OF OPERATIONS", fmtCaptionBold);
  137. tl.Append("(in millions, except number of shares which are reflected in thousands and per share amounts)", fmtCaptionLight);
  138. return tl;
  139. }
  140. };
  141. ta.AddCell(csHeader1, 0, 0, 1, 7, null);
  142.  
  143. var csThinLine = new CellStyle
  144. {
  145. Background = true,
  146. Borders = FrameBorders.BottomBorder,
  147. LinePaddingBottom = -1,
  148. LineWidth = 1
  149. };
  150. var csBoldLine = new CellStyle(csThinLine)
  151. {
  152. LinePaddingBottom = -3,
  153. LineWidth = 3
  154. };
  155.  
  156. var csHeader2 = new CellStyle(csThinLine)
  157. {
  158. PaddingBottom = 5,
  159. TextAlignment = TextAlignment.Center,
  160. TextFormat = fmtBold
  161. };
  162. ta.AddCell(csHeader2, 1, 1, 1, 6, "Years ended");
  163. ta.SetHorizontalGridLineWidth(2, 1);
  164.  
  165. var csHeader3 = new CellStyle
  166. {
  167. PaddingTopBottom = 5,
  168. TextAlignment = TextAlignment.Center,
  169. TextFormat = fmtBold
  170. };
  171. ta.AddCell(csHeader3, 2, 1, 1, 2, "September 25,\n2021");
  172. ta.AddCell(csHeader3, 2, 3, 1, 2, "September 26,\n2020");
  173. ta.AddCell(csHeader3, 2, 5, 1, 2, "September 28,\n2019");
  174. ta.AddCell(csThinLine, 2, 1, 1, 6);
  175. ta.SetHorizontalGridLineWidth(3, 1);
  176.  
  177. var csText = new CellStyle
  178. {
  179. TextFormat = fmt,
  180. TextAlignment = TextAlignment.Leading,
  181. PaddingTopBottom = 4
  182. };
  183. var csBoldText = new CellStyle(csText)
  184. {
  185. TextFormat = fmtBold
  186. };
  187. var csNum = new CellStyle
  188. {
  189. TextFormat = fmt,
  190. TextAlignment = TextAlignment.Trailing,
  191. PaddingTopBottom = 4,
  192. PaddingRight = 10
  193. };
  194. var csBoldNum = new CellStyle(csNum)
  195. {
  196. TextFormat = fmtBold
  197. };
  198. var csBand = new CellStyle
  199. {
  200. FillColor = Color.FromArgb(245, 245, 245),
  201. Background = true
  202. };
  203.  
  204. for (int i = 0; i < items.Length; i++)
  205. {
  206. var item = items[i];
  207. int rowIndex = i + 3;
  208.  
  209. if ((i & 1) != 0)
  210. {
  211. ta.AddCell(csBand, rowIndex, 0, 1, 7);
  212. }
  213.  
  214. var tc = ta.AddCell(item.BoldText ? csBoldText : csText, rowIndex, 0, item.Text);
  215. tc.TextLayout.FirstLineIndent = item.Indent * 8 + 10;
  216.  
  217. var csCurrNum = item.BoldText ? csBoldNum : csNum;
  218. if (item.DrawSign)
  219. {
  220. ta.AddCell(csCurrNum, rowIndex, 1, "$");
  221. ta.AddCell(csCurrNum, rowIndex, 3, "$");
  222. ta.AddCell(csCurrNum, rowIndex, 5, "$");
  223. }
  224. if (!item.CaptionOnly)
  225. {
  226. ta.AddCell(csCurrNum, rowIndex, 2, item.Sum2021.ToString("n0"));
  227. ta.AddCell(csCurrNum, rowIndex, 4, item.Sum2020.ToString("n0"));
  228. ta.AddCell(csCurrNum, rowIndex, 6, item.Sum2019.ToString("n0"));
  229. }
  230.  
  231. if (item.ThinLine)
  232. {
  233. ta.AddCell(csThinLine, rowIndex, 1, 1, 6);
  234. ta.SetHorizontalGridLineWidth(rowIndex + 1, 1);
  235. }
  236. else if (item.BoldLine)
  237. {
  238. ta.AddCell(csBoldLine, rowIndex, 1, 1, 6);
  239. ta.SetHorizontalGridLineWidth(rowIndex + 1, 3);
  240. }
  241. }
  242.  
  243. ta.Render();
  244. }
  245. }
  246. }
  247.