SkiaTextRendering.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 GrapeCity.Documents.Imaging;
  9. using GrapeCity.Documents.Text;
  10. using GrapeCity.Documents.Drawing;
  11. using GrapeCity.Documents.Imaging.Skia;
  12. using GCTEXT = GrapeCity.Documents.Text;
  13. using GCDRAW = GrapeCity.Documents.Drawing;
  14.  
  15. namespace DsImagingWeb.Demos
  16. {
  17. // This example is essentially the same as in TextRendering,
  18. // but uses DsImaging.Skia library (GcSkiaBitmap / GcSkiaGraphics)
  19. // to create and render the resulting image.
  20. public class SkiaTextRendering
  21. {
  22. public static bool IsSkiaOnly => true;
  23.  
  24. public Stream GenerateImageStream(string targetMime, Size pixelSize, float dpi, bool opaque, string[] sampleParams = null)
  25. {
  26. switch (targetMime)
  27. {
  28. case Common.Util.MimeTypes.JPEG:
  29. case Common.Util.MimeTypes.PNG:
  30. case Common.Util.MimeTypes.WEBP:
  31. break;
  32. default:
  33. throw new Exception("This sample uses Skia to create the image, and only supports JPEG, PNG and WEBP output formats.");
  34. }
  35.  
  36. using var bmp = new GcSkiaBitmap(pixelSize.Width, pixelSize.Height, opaque);
  37.  
  38. var Inch = dpi;
  39. const float fontSize = 14;
  40. using (var g = bmp.CreateGraphics(Color.White))
  41. {
  42. // TextFormat class is used throughout all DsImaging text rendering to specify
  43. // font and other character formatting:
  44. var tf = new TextFormat()
  45. {
  46. Font = GCTEXT.Font.FromFile(Path.Combine("Resources", "Fonts", "times.ttf")),
  47. FontSize = fontSize
  48. };
  49.  
  50. // 1.
  51. // The easiest way to render a short string on a page at an arbitrary location,
  52. // when you are 100% sure that the string will fit in the available space,
  53. // is to use the GcGraphics.DrawString() overload accepting just the point
  54. // at which to draw the string:
  55. g.DrawString(
  56. "1. Test string. Please read the extensive comments in this sample's code.\r\n" +
  57. "(Note that line breaks are allowed even in the simplest DrawString overload.)",
  58. tf, new PointF(Inch, Inch));
  59.  
  60. // 2.
  61. // Another overload taking a rectangle instead, plus alignment and wrapping
  62. // options, is also available and provides a bit more flexibility:
  63. g.DrawString(
  64. "2. A longer test string which will probably need more than the allocated " +
  65. "4 inches so quite possibly will wrap to show that DrawString can do that.",
  66. tf,
  67. new RectangleF(Inch, Inch * 2, Inch * 4, Inch), // the layout rectangle
  68. // The rest 3 args are optional, passing defaults here for illustration:
  69. TextAlignment.Leading, // leading (left for LTR languages) text align
  70. ParagraphAlignment.Near, // near (top for top-to-bottom flow) para align
  71. true); // word wrap
  72.  
  73. // 3.
  74. // Complementary to DrawString, a MeasureString() method is available
  75. // (with several different overloads), and can be used in pair with
  76. // DrawString when more control over text layout is needed:
  77. const string tstr3 = "3. Test string to demo MeasureString() used with DrawString().";
  78.  
  79. SizeF layoutSize = new SizeF(Inch * 3, Inch * 0.8f); // available size
  80. SizeF s = g.MeasureString(tstr3, tf, layoutSize, out int fitCharCount);
  81. // Show the passed in size in red, the measured size in blue,
  82. // and draw the string within the returned size as bounds:
  83. PointF pt = new PointF(Inch, Inch * 3);
  84. g.DrawRectangle(new RectangleF(pt, layoutSize), Color.Red);
  85. g.DrawRectangle(new RectangleF(pt, s), Color.Blue);
  86. g.DrawString(tstr3, tf, new RectangleF(pt, s));
  87.  
  88. // 4.
  89. // A much more powerful and with better performance, way to render text
  90. // is to use TextLayout. (TextLayout is used anyway by DrawString/MeasureString,
  91. // so when you use TextLayout directly, you basically cut the work in half.)
  92. // A TextLayout instance represents one or more paragraphs of text, with
  93. // the same paragraph formatting (character formats may be different,
  94. // see {MultiFormattedText}).
  95. var tl = g.CreateTextLayout();
  96. // To add text, use Append() or AppendLine() methods:
  97. tl.Append("4. First test string added to TextLayout. ", tf);
  98. tl.Append("Second test string added to TextLayout, continuing the same paragraph. ", tf);
  99. tl.AppendLine(); // Add a line break, effectively starting a new paragraph
  100. tl.Append("Third test string added to TextLayout, a new paragraph. ", tf);
  101. tl.Append("Fourth test string, with a different char formatting. ",
  102. new TextFormat(tf)
  103. {
  104. Font = GCTEXT.Font.FromFile(Path.Combine("Resources", "Fonts", "timesbi.ttf")),
  105. FontSize = fontSize,
  106. FontBold = true,
  107. FontItalic = true,
  108. ForeColor = Color.DarkSeaGreen,
  109. });
  110. // Text can be added to TextLayout without explicit TextFormat:
  111. tl.Append("Fifth test string, using the TextLayout's default format.");
  112. // ...but in that case at least the Font must be specified on the
  113. // TextLayout's DefaultFormat, otherwise PerformLayout (below) will fail:
  114. tl.DefaultFormat.Font = GCTEXT.Font.FromFile(Path.Combine("Resources", "Fonts", "timesi.ttf"));
  115. tl.DefaultFormat.FontSize = fontSize;
  116.  
  117. // Specify the layout, such as max available size etc.
  118. // Here we only provide the max width, but many more parameters can be set:
  119. tl.MaxWidth = g.Width - Inch * 2;
  120. // Paragraph formatting can also be set, here we set first line offset,
  121. // spacing between paragraphs and line spacing:
  122. tl.FirstLineIndent = Inch * 0.5f;
  123. tl.ParagraphSpacing = Inch * 0.05f;
  124. tl.LineSpacingScaleFactor = 0.8f;
  125.  
  126. // When all text has been added, and layout options specified,
  127. // the TextLayout needs to calculate the glyphs needed to render
  128. // the text, and perform the layout. This can be done with a
  129. // single call:
  130. tl.PerformLayout(true);
  131.  
  132. // Now we can draw it on the page:
  133. pt = new PointF(Inch, Inch * 4);
  134. g.DrawTextLayout(tl, pt);
  135. // TextLayout provides info about the text including the measured bounds
  136. // and much more. Here we draw the bounding box in orange red:
  137. g.DrawRectangle(new RectangleF(pt, tl.ContentRectangle.Size), Color.OrangeRed);
  138.  
  139. // 5.
  140. // TextLayout can be re-used to draw different paragraph(s), this can be useful
  141. // when you need to render a different text with the same paragraph formatting.
  142. // The Clear() call removes the text but preserves paragraph formatting:
  143. tl.Clear();
  144. tl.Append("5. This is text rendered re-using the same TextLayout. ");
  145. tl.Append("More text added to TextLayout being re-used, continuing the same paragraph. ", tf);
  146. tl.Append("And finally, some more text added.", tf);
  147. // The necessary call to calculate the glyphs and perform layout:
  148. tl.PerformLayout(true);
  149. // Render the text:
  150. g.DrawTextLayout(tl, new PointF(Inch, Inch * 5));
  151. // Draw border around the whole image:
  152. g.DrawRectangle(new RectangleF(0, 0, bmp.PixelWidth, bmp.PixelHeight), Color.DarkSlateBlue, 4);
  153. }
  154. // Done:
  155. var ms = new MemoryStream();
  156. switch (targetMime)
  157. {
  158. case Common.Util.MimeTypes.JPEG:
  159. bmp.SaveAsJpeg(ms);
  160. break;
  161. case Common.Util.MimeTypes.PNG:
  162. bmp.SaveAsPng(ms);
  163. break;
  164. case Common.Util.MimeTypes.WEBP:
  165. bmp.SaveAsWebp(ms);
  166. break;
  167. default:
  168. System.Diagnostics.Debug.Assert(false);
  169. break;
  170. }
  171. return ms;
  172. }
  173. }
  174. }
  175.