InteractiveCanvas.cs 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300
  1. using Avalonia;
  2. using Avalonia.Media;
  3. using Avalonia.Platform;
  4. using Avalonia.Rendering.SceneGraph;
  5. using Avalonia.Skia;
  6. using SkiaSharp;
  7. namespace QuestPDF.Previewer;
  8. class InteractiveCanvas : ICustomDrawOperation
  9. {
  10. public Rect Bounds { get; set; }
  11. public float RenderingScale { get; set; }
  12. private List<DocumentStructure.PageSize> PageSizes { get; set; } = new();
  13. private List<RenderedPageSnapshot> PageSnapshotCache { get; set; } = new();
  14. private float Width => (float)Bounds.Width;
  15. private float Height => (float)Bounds.Height;
  16. public float Scale { get; private set; } = 1;
  17. public float TranslateX { get; set; }
  18. public float TranslateY { get; set; }
  19. private const float MinScale = 1 / 8f;
  20. private const float MaxScale = 8f;
  21. private const float PageSpacing = 25f;
  22. private const float SafeZone = 25f;
  23. public float TotalPagesHeight => PageSizes.Sum(x => x.Height) + (PageSizes.Count - 1) * PageSpacing;
  24. public float TotalHeight => TotalPagesHeight + SafeZone * 2 / Scale;
  25. public float MaxWidth => PageSizes.Any() ? PageSizes.Max(x => x.Width) : 0;
  26. public float MaxTranslateY => TotalHeight - Height / Scale;
  27. public float ScrollPercentY
  28. {
  29. get
  30. {
  31. return TranslateY / MaxTranslateY;
  32. }
  33. set
  34. {
  35. TranslateY = value * MaxTranslateY;
  36. }
  37. }
  38. public float ScrollViewportSizeY
  39. {
  40. get
  41. {
  42. var viewPortSize = Height / Scale / TotalHeight;
  43. return Math.Clamp(viewPortSize, 0, 1);
  44. }
  45. }
  46. #region interaction
  47. public void SetNewDocumentStructure(DocumentStructure document)
  48. {
  49. foreach (var renderedSnapshot in PageSnapshotCache)
  50. renderedSnapshot.Image.Dispose();
  51. PageSnapshotCache.Clear();
  52. PageSizes = document.Pages.ToList();
  53. }
  54. public ICollection<PageSnapshotIndex> GetMissingSnapshots()
  55. {
  56. var requiredKeys = GetVisiblePages(padding: 500).Select(x => (x.pageIndex, PreferredZoomLevel)).ToList();
  57. var availableKeys = PageSnapshotCache.Select(x => (x.PageIndex, x.ZoomLevel)).ToList();
  58. var missingKeys = requiredKeys.Except(availableKeys).ToArray();
  59. return missingKeys
  60. .Select(x => new PageSnapshotIndex
  61. {
  62. PageIndex = x.Item1,
  63. ZoomLevel = x.Item2
  64. })
  65. .ToList();
  66. }
  67. public void AddSnapshots(ICollection<RenderedPageSnapshot> snapshots)
  68. {
  69. PageSnapshotCache.AddRange(snapshots);
  70. }
  71. #endregion
  72. #region transformations
  73. private void LimitScale()
  74. {
  75. Scale = Math.Max(Scale, MinScale);
  76. Scale = Math.Min(Scale, MaxScale);
  77. }
  78. private void LimitTranslate()
  79. {
  80. if (TotalPagesHeight > Height / Scale)
  81. {
  82. TranslateY = Math.Min(TranslateY, MaxTranslateY);
  83. TranslateY = Math.Max(TranslateY, 0);
  84. }
  85. else
  86. {
  87. TranslateY = (TotalPagesHeight - Height / Scale) / 2;
  88. }
  89. if (Width / Scale < MaxWidth)
  90. {
  91. var maxTranslateX = (Width / 2 - SafeZone) / Scale - MaxWidth / 2;
  92. TranslateX = Math.Min(TranslateX, -maxTranslateX);
  93. TranslateX = Math.Max(TranslateX, maxTranslateX);
  94. }
  95. else
  96. {
  97. TranslateX = 0;
  98. }
  99. }
  100. public void TranslateWithCurrentScale(float x, float y)
  101. {
  102. TranslateX += x / Scale;
  103. TranslateY += y / Scale;
  104. LimitTranslate();
  105. }
  106. public void ZoomToPoint(float x, float y, float factor)
  107. {
  108. var oldScale = Scale;
  109. Scale *= factor;
  110. LimitScale();
  111. TranslateX -= x / Scale - x / oldScale;
  112. TranslateY -= y / Scale - y / oldScale;
  113. LimitTranslate();
  114. }
  115. #endregion
  116. #region rendering
  117. private int PreferredZoomLevel => (int)Math.Clamp(Math.Ceiling(Math.Log2(Scale * RenderingScale)), -2, 2);
  118. private IEnumerable<(int pageIndex, float verticalOffset)> GetVisiblePages(float padding = 100)
  119. {
  120. padding /= Scale;
  121. var visibleOffsetFrom = TranslateY - padding;
  122. var visibleOffsetTo = TranslateY + Height / Scale + padding;
  123. var topOffset = 0f;
  124. foreach (var pageIndex in Enumerable.Range(0, PageSizes.Count))
  125. {
  126. var page = PageSizes.ElementAt(pageIndex);
  127. if (topOffset + page.Height > visibleOffsetFrom)
  128. yield return (pageIndex, topOffset);
  129. topOffset += page.Height + PageSpacing;
  130. if (topOffset > visibleOffsetTo)
  131. yield break;
  132. }
  133. }
  134. public void Render(ImmediateDrawingContext context)
  135. {
  136. // get SkiaSharp canvas
  137. var leaseFeature = context.TryGetFeature<ISkiaSharpApiLeaseFeature>();
  138. using var lease = leaseFeature.Lease();
  139. var canvas = lease.SkCanvas;
  140. if (canvas == null)
  141. return;
  142. // draw document
  143. if (PageSizes.Count <= 0)
  144. return;
  145. LimitScale();
  146. LimitTranslate();
  147. var originalMatrix = canvas.TotalMatrix;
  148. canvas.Translate(Width / 2, 0);
  149. canvas.Scale(Scale);
  150. canvas.Translate(TranslateX, -TranslateY + SafeZone / Scale);
  151. foreach (var (pageIndex, offset) in GetVisiblePages())
  152. {
  153. var pageSize = PageSizes.ElementAt(pageIndex);
  154. canvas.Save();
  155. canvas.Translate(-pageSize.Width / 2f, offset);
  156. DrawBlankPage(canvas, pageSize.Width, pageSize.Height);
  157. DrawPageSnapshot(canvas, pageIndex);
  158. canvas.Restore();
  159. }
  160. canvas.SetMatrix(originalMatrix);
  161. DrawInnerGradient(canvas);
  162. }
  163. private void DrawPageSnapshot(SKCanvas canvas, int pageIndex)
  164. {
  165. var page = PageSizes.ElementAt(pageIndex);
  166. var renderedSnapshot = PageSnapshotCache
  167. .Where(x => x.PageIndex == pageIndex)
  168. .OrderBy(x => Math.Abs(PreferredZoomLevel - x.ZoomLevel))
  169. .ThenByDescending(x => x.ZoomLevel)
  170. .FirstOrDefault();
  171. if (renderedSnapshot == null)
  172. return;
  173. using var drawImagePaint = new SKPaint
  174. {
  175. FilterQuality = SKFilterQuality.High
  176. };
  177. var renderingScale = (float)Math.Pow(2, renderedSnapshot.ZoomLevel);
  178. canvas.Save();
  179. canvas.ClipRect(new SKRect(0, 0, page.Width, page.Height));
  180. canvas.Scale(1 / renderingScale);
  181. canvas.DrawImage(renderedSnapshot.Image, SKPoint.Empty, drawImagePaint);
  182. canvas.Restore();
  183. }
  184. public void Dispose() { }
  185. public bool Equals(ICustomDrawOperation? other) => false;
  186. public bool HitTest(Point p) => true;
  187. #endregion
  188. #region blank page
  189. private static SKPaint BlankPagePaint = new SKPaint
  190. {
  191. Color = SKColors.White
  192. };
  193. private static SKPaint BlankPageShadowPaint = new SKPaint
  194. {
  195. ImageFilter = SKImageFilter.CreateBlendMode(
  196. SKBlendMode.Overlay,
  197. SKImageFilter.CreateDropShadowOnly(0, 6, 6, 6, SKColors.Black.WithAlpha(64)),
  198. SKImageFilter.CreateDropShadowOnly(0, 10, 14, 14, SKColors.Black.WithAlpha(32)))
  199. };
  200. private static void DrawBlankPage(SKCanvas canvas, float width, float height)
  201. {
  202. canvas.DrawRect(0, 0, width, height, BlankPageShadowPaint);
  203. canvas.DrawRect(0, 0, width, height, BlankPagePaint);
  204. }
  205. #endregion
  206. #region inner viewport gradient
  207. private const int InnerGradientSize = (int)SafeZone;
  208. private static readonly SKColor InnerGradientColor = SKColor.Parse("#666");
  209. private void DrawInnerGradient(SKCanvas canvas)
  210. {
  211. // gamma correction
  212. var colors = Enumerable
  213. .Range(0, InnerGradientSize)
  214. .Select(x => 1f - x / (float) InnerGradientSize)
  215. .Select(x => Math.Pow(x, 2f))
  216. .Select(x => (byte)(x * 255))
  217. .Select(x => InnerGradientColor.WithAlpha(x))
  218. .ToArray();
  219. using var fogPaint = new SKPaint
  220. {
  221. Shader = SKShader.CreateLinearGradient(
  222. new SKPoint(0, 0),
  223. new SKPoint(0, InnerGradientSize),
  224. colors,
  225. SKShaderTileMode.Clamp)
  226. };
  227. canvas.DrawRect(0, 0, Width, InnerGradientSize, fogPaint);
  228. }
  229. #endregion
  230. }