AvaloniaUI - 如何直接在画布上绘制

问题描述

在 GDI+ 中,可以直接在画布上绘制(创建一个内存位图,然后在那里做任何需要做的事情)。

我需要相同的 Avalonia 用于“自定义控件”,我被告知这是可能的,因为 SkiaSharp.Canvas 可以访问。任何人都可以提供一些有关如何执行此操作的线索吗?

一个例子是连续变化的曲线,例如语音频率。如果您不直接在画布(或 Xaml 世界中的任何名称)上执行此操作,则它会太慢并且占用太多资源,尤其是当您需要在一个屏幕上显示 10 到 20 个时。

我有 GDI+、JavaFX、QML 方面的背景,但我对 Xaml 领域还很陌生。我已经阅读了完整的 Avalonia 文档,但在这个方向上没有任何内容。我知道在 Avalonia 项目的现阶段还有其他优先事项。

解决方法

我从评论中提到的“RenderDemo”frankenapps 中获取了“LineBoundsDemoControl”作为我问题的答案。

实际绘制发生在“drawingContext”的“Render”方法中。用笔和画笔看起来很像 GDI+。

如果您像我一样不熟悉 Avalonia/xaml,那么带有“AffectsRender”的静态构造函数可能是最奇怪的。根据 Avalonia 源代码,此方法表明属性更改应导致控件失效(重绘)。一直在学习...

这个方法应该在控件的静态构造函数中调用 控件上的每个属性在更改时都会导致重绘。 这类似于 WPF 的 FrameworkPropertyMetadata.AffectsRender 标志。

using System;
using Avalonia;
using Avalonia.Controls;
using Avalonia.Media;
using Avalonia.Rendering.SceneGraph;
using Avalonia.Threading;

namespace RenderDemo.Controls
{
    public class LineBoundsDemoControl : Control
    {
        static LineBoundsDemoControl()
        {
            AffectsRender<LineBoundsDemoControl>(AngleProperty);
        }

        public LineBoundsDemoControl()
        {
            var timer = new DispatcherTimer();
            timer.Interval = TimeSpan.FromSeconds(1 / 60.0);
            timer.Tick += (sender,e) => Angle += Math.PI / 360;
            timer.Start();
        }

        public static readonly StyledProperty<double> AngleProperty =
            AvaloniaProperty.Register<LineBoundsDemoControl,double>(nameof(Angle));        

        public double Angle
        {
            get => GetValue(AngleProperty);
            set => SetValue(AngleProperty,value);
        }

        public override void Render(DrawingContext drawingContext)
        {
            var lineLength = Math.Sqrt((100 * 100) + (100 * 100));

            var diffX = LineBoundsHelper.CalculateAdjSide(Angle,lineLength);
            var diffY = LineBoundsHelper.CalculateOppSide(Angle,lineLength);


            var p1 = new Point(200,200);
            var p2 = new Point(p1.X + diffX,p1.Y + diffY);

            var pen = new Pen(Brushes.Green,20,lineCap: PenLineCap.Square);
            var boundPen = new Pen(Brushes.Black);

            drawingContext.DrawLine(pen,p1,p2);

            drawingContext.DrawRectangle(boundPen,LineBoundsHelper.CalculateBounds(p1,p2,pen));
        }
    }
}

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...