SkiaAntialiasing.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.Collections.Generic;
  9. using System.Linq;
  10. using System.Numerics;
  11. using GrapeCity.Documents.Drawing;
  12. using GrapeCity.Documents.Text;
  13. using GrapeCity.Documents.Imaging;
  14. using GrapeCity.Documents.Imaging.Skia;
  15. using GCTEXT = GrapeCity.Documents.Text;
  16. using GCDRAW = GrapeCity.Documents.Drawing;
  17.  
  18. namespace DsImagingWeb.Demos
  19. {
  20. // Text is always rendered with anti-aliasing in Skia.
  21. // Graphics rendering on the other hand can be controlled by
  22. // setting the GcSkiaGraphics.Aliased property.
  23. // This example shows the difference between shapes drawn
  24. // with that property set to false (which is the default)
  25. // and true. Note the more jagged appearance of some lines
  26. // in the lower part of the image (drawn with Aliased set
  27. // to false).
  28. public class SkiaAntialiasing
  29. {
  30. public static bool IsSkiaOnly => true;
  31.  
  32. private GCTEXT.Font _font = null;
  33.  
  34. // Helper method to draw a polygon and a caption beneath it.
  35. // Can also be used to just calculate the points without actual drawing.
  36. // startAngle is for the first point, clockwise from (1,0).
  37. private PointF[] DrawPolygon(GcGraphics g, PointF center, float r, int n, float startAngle, GCDRAW.Pen pen, string caption = null)
  38. {
  39. PointF[] pts = new PointF[n];
  40. for (int i = 0; i < n; ++i)
  41. pts[i] = new PointF(center.X + (float)(r * Math.Cos(startAngle + 2 * Math.PI * i / n)), center.Y + (float)(r * Math.Sin(startAngle + 2 * Math.PI * i / n)));
  42. if (pen != null)
  43. g.DrawPolygon(pts, pen);
  44. if (!string.IsNullOrEmpty(caption))
  45. DrawCaption(g, center, r, caption);
  46. return pts;
  47. }
  48.  
  49. // Helper method to draw a caption beneath a shape:
  50. private void DrawCaption(GcGraphics g, PointF center, float r, string caption)
  51. {
  52. g.DrawString(caption,
  53. new TextFormat()
  54. {
  55. Font = _font,
  56. FontSize = 12,
  57. },
  58. new RectangleF(center.X - r, center.Y - r, r * 2, r * 2),
  59. TextAlignment.Center, ParagraphAlignment.Center, false);
  60. }
  61.  
  62. // Main entry point.
  63. public Stream GenerateImageStream(string targetMime, Size pixelSize, float dpi, bool opaque, string[] sampleParams = null)
  64. {
  65. switch (targetMime)
  66. {
  67. case Common.Util.MimeTypes.JPEG:
  68. case Common.Util.MimeTypes.PNG:
  69. case Common.Util.MimeTypes.WEBP:
  70. break;
  71. default:
  72. throw new Exception("This sample uses Skia to create the image, and only supports JPEG, PNG and WEBP output formats.");
  73. }
  74.  
  75. _font = GCTEXT.Font.FromFile(Path.Combine("Resources", "Fonts", "times.ttf"));
  76. var Inch = dpi;
  77. var bmp = new GcSkiaBitmap(pixelSize.Width, pixelSize.Height, opaque);
  78. using (var g = bmp.CreateGraphics(Color.White))
  79. {
  80. DrawShapes(g, bmp.PixelWidth, bmp.PixelHeight, false);
  81. DrawShapes(g, bmp.PixelWidth, bmp.PixelHeight, true);
  82. }
  83. // Done:
  84. var ms = new MemoryStream();
  85. switch (targetMime)
  86. {
  87. case Common.Util.MimeTypes.JPEG:
  88. bmp.SaveAsJpeg(ms);
  89. break;
  90. case Common.Util.MimeTypes.PNG:
  91. bmp.SaveAsPng(ms);
  92. break;
  93. case Common.Util.MimeTypes.WEBP:
  94. bmp.SaveAsWebp(ms);
  95. break;
  96. default:
  97. System.Diagnostics.Debug.Assert(false);
  98. break;
  99. }
  100. return ms;
  101. }
  102.  
  103. void DrawShapes(GcGraphics g, int pixelWidth, int pixelHeight, bool aliased)
  104. {
  105. if (g is GcSkiaGraphics gskia)
  106. gskia.Aliased = aliased;
  107.  
  108. // Set up the helper layout grid:
  109. var Inch = g.Resolution;
  110. var grid = new
  111. {
  112. Cols = 4,
  113. Rows = 3,
  114. MarginX = Inch / 6,
  115. MarginY = Inch / 4,
  116. Radius = Inch * 0.6f,
  117. StepX = (pixelWidth - Inch / 2) / 4,
  118. StepY = (pixelHeight - Inch / 4) / 7f,
  119. };
  120.  
  121. // Insertion point of the next figure's center:
  122. PointF startIp = new PointF(grid.MarginX + grid.StepX / 2, grid.MarginY + grid.StepY / 2 + 10);
  123. if (aliased)
  124. startIp.Y += pixelHeight / 2;
  125.  
  126. PointF ip = startIp;
  127.  
  128. // Header:
  129. var pad = 8;
  130. var bord = new RectangleF(pad, pad + (aliased ? pixelHeight / 2 : 0), pixelWidth - pad * 2, pixelHeight / 2 - pad * 2);
  131. g.DrawString($"Shapes (Skia) - Aliased is {aliased}",
  132. new TextFormat()
  133. {
  134. Font = _font,
  135. FontSize = 14,
  136. },
  137. bord,
  138. TextAlignment.Center, ParagraphAlignment.Near);
  139. g.DrawRoundRect(bord, pad, Color.MediumPurple, 2, DashStyle.DashDot);
  140.  
  141. // Pen used to draw shapes:
  142. var pen = new GCDRAW.Pen(Color.Orange, 1);
  143. pen.LineJoin = PenLineJoin.Round;
  144. int fill = 100; // Surfaces fill alpha
  145.  
  146. // Circle:
  147. g.DrawEllipse(new RectangleF(ip.X - grid.Radius, ip.Y - grid.Radius, grid.Radius * 2, grid.Radius * 2), pen);
  148. DrawCaption(g, ip, grid.Radius, "Circle");
  149. ip.X += grid.StepX;
  150.  
  151. // Ellipse:
  152. g.DrawEllipse(new RectangleF(ip.X - grid.Radius * 1.4f, ip.Y - grid.Radius / 2, grid.Radius * 2 * 1.4f, grid.Radius), pen);
  153. DrawCaption(g, ip, grid.Radius, "Ellipse");
  154. ip.X += grid.StepX;
  155.  
  156. // Cylinder:
  157. float radX = grid.Radius / 1.4f;
  158. float radY = grid.Radius / 6;
  159. float height = grid.Radius * 1.8f;
  160. g.DrawEllipse(new RectangleF(ip.X - radX, ip.Y - height / 2, radX * 2, radY * 2), pen);
  161. g.FillEllipse(new RectangleF(ip.X - radX, ip.Y + height / 2 - radY * 2, radX * 2, radY * 2), Color.FromArgb(fill, pen.Color));
  162. g.DrawEllipse(new RectangleF(ip.X - radX, ip.Y + height / 2 - radY * 2, radX * 2, radY * 2), pen);
  163. g.DrawLine(new PointF(ip.X - radX, ip.Y - height / 2 + radY), new PointF(ip.X - radX, ip.Y + height / 2 - radY), pen);
  164. g.DrawLine(new PointF(ip.X + radX, ip.Y - height / 2 + radY), new PointF(ip.X + radX, ip.Y + height / 2 - radY), pen);
  165. DrawCaption(g, ip, grid.Radius, "Cylinder");
  166. pen.Color = Color.Indigo;
  167. ip.X += grid.StepX;
  168.  
  169. // Cube:
  170. float cubex = 6;
  171. var cubePtsFar = DrawPolygon(g, new PointF(ip.X - cubex, ip.Y - cubex), grid.Radius, 4, (float)-Math.PI / 4, pen);
  172. var cubePtsNear = DrawPolygon(g, new PointF(ip.X + cubex, ip.Y + cubex), grid.Radius, 4, (float)-Math.PI / 4, pen);
  173. g.DrawLine(cubePtsFar[0], cubePtsNear[0], pen);
  174. g.DrawLine(cubePtsFar[1], cubePtsNear[1], pen);
  175. g.DrawLine(cubePtsFar[2], cubePtsNear[2], pen);
  176. g.DrawLine(cubePtsFar[3], cubePtsNear[3], pen);
  177. g.FillPolygon(new PointF[] { cubePtsFar[1], cubePtsFar[2], cubePtsNear[2], cubePtsNear[1], }, Color.FromArgb(fill, pen.Color));
  178. DrawCaption(g, ip, grid.Radius, "Cube");
  179. pen.Color = Color.DarkGreen;
  180. ip.X = startIp.X;
  181. ip.Y += grid.StepY;
  182.  
  183. // Pentagon:
  184. DrawPolygon(g, ip, grid.Radius, 5, (float)-Math.PI / 2, pen, "Pentagon");
  185. ip.X += grid.StepX;
  186.  
  187. // Hexagon:
  188. // For sample sake, we apply a transform to make the hexagon wider and shorter:
  189. g.Transform = Matrix3x2.CreateScale(1.4f, 0.8f) * Matrix3x2.CreateTranslation(ip.X, ip.Y);
  190. DrawPolygon(g, PointF.Empty, grid.Radius, 6, 0, pen, null);
  191. g.Transform = Matrix3x2.Identity;
  192. DrawCaption(g, ip, grid.Radius, "Hexagon");
  193. ip.X += grid.StepX;
  194.  
  195. // Octagon:
  196. DrawPolygon(g, ip, grid.Radius, 8, (float)-Math.PI / 8, pen, "Octagon");
  197. pen.Color = Color.DarkRed;
  198. ip.X += grid.StepX;
  199.  
  200. // Triangle:
  201. DrawPolygon(g, ip, grid.Radius, 3, (float)-Math.PI / 2, pen, "Triangle");
  202. ip.X = startIp.X;
  203. ip.Y += grid.StepY;
  204.  
  205. // Filled pentagram:
  206. var pts = DrawPolygon(g, ip, grid.Radius, 5, (float)-Math.PI / 2, pen, "Pentagram");
  207. pts = new PointF[] { pts[0], pts[2], pts[4], pts[1], pts[3], };
  208. g.FillPolygon(pts, Color.FromArgb(fill, pen.Color));
  209. g.DrawPolygon(pts, pen);
  210. ip.X += grid.StepX;
  211.  
  212. // Set up a simple kind of oblique projection to draw a pyramid:
  213. var angle = Math.PI / 6;
  214. float s = (float)Math.Sin(angle);
  215. float c = (float)Math.Cos(angle);
  216. Func<float, float, float, PointF> project = (x_, y_, z_) => new PointF(x_ - c * y_ * 0.5f, -(z_ - s * y_ * 0.5f));
  217. Func<Vector3, PointF> p3d = v_ => project(v_.X, v_.Y, v_.Z);
  218. float hedge = grid.Radius; // 1/2 edge
  219. // Debug - draw the 3 axis:
  220. // g.DrawLine(project(0, 0, 0), project(100, 0, 0), Color.Red);
  221. // g.DrawLine(project(0, 0, 0), project(0, 100, 0), Color.Green);
  222. // g.DrawLine(project(0, 0, 0), project(0, 0, 100), Color.Blue);
  223. // 3d points forming a square pyramid:
  224. var pts3d = new Vector3[]
  225. {
  226. new Vector3(-hedge, -hedge, 0),
  227. new Vector3(hedge, -hedge, 0),
  228. new Vector3(hedge, hedge, 0),
  229. new Vector3(-hedge, hedge, 0),
  230. new Vector3(0, 0, hedge * 2),
  231. };
  232. // project the points to draw the pyramid:
  233. pts = pts3d.Select(v_ => p3d(v_)).ToArray();
  234. g.Transform = Matrix3x2.CreateTranslation(ip.X, ip.Y + hedge * 0.7f);
  235. // Visible edges:
  236. g.DrawPolygon(new PointF[] { pts[4], pts[1], pts[2], pts[3], pts[4], pts[2] }, pen);
  237. // Invisible edges:
  238. pen.Width /= 2;
  239. pen.Color = Color.FromArgb(fill, pen.Color);
  240. g.DrawLine(pts[0], pts[4], pen);
  241. g.DrawLine(pts[0], pts[1], pen);
  242. g.DrawLine(pts[0], pts[3], pen);
  243. g.FillPolygon(pts.Take(4).ToArray(), pen.Color);
  244. //
  245. g.Transform = Matrix3x2.Identity;
  246. DrawCaption(g, ip, grid.Radius, "Pyramid");
  247. ip.X += grid.StepX;
  248. pen.Width *= 2;
  249. pen.Color = Color.Green;
  250.  
  251. // Cone:
  252. float baseh = grid.Radius * 0.3f;
  253. pts = DrawPolygon(g, ip, grid.Radius, 3, (float)-Math.PI / 2, null, "Cone");
  254. g.DrawLines(new PointF[] { pts[2], pts[0], pts[1] }, pen);
  255. var rect = new RectangleF(pts[2].X, pts[2].Y - baseh / 2, pts[1].X - pts[2].X, baseh);
  256. g.FillEllipse(rect, Color.FromArgb(fill, pen.Color));
  257. g.DrawEllipse(rect, pen);
  258. ip.X += grid.StepX;
  259.  
  260. // Trapezoid (use DrawPolygon to just get the points of the square):
  261. float dx = 10;
  262. pts = DrawPolygon(g, ip, grid.Radius, 4, (float)-Math.PI / 4, null, "Trapezoid");
  263. pts[0].X -= dx;
  264. pts[1].X += dx;
  265. pts[2].X -= dx;
  266. pts[3].X += dx;
  267. g.DrawPolygon(pts, pen);
  268. }
  269. }
  270. }
  271.