如何在 Gtk3 中与 Skia 在开罗上下文中绘制

本文关键字:上下文 绘制 Skia Gtk3 中与 | 更新日期: 2023-09-27 18:36:47

我希望得到彩色矩形,但得到垃圾矩形。到目前为止,我有以下代码:

using System;
using GLib;
using SkiaSharp;
using Gtk;
namespace SkiaSharpExample {
    class CCDrawingArea : DrawingArea {
        protected override bool OnDrawn (Cairo.Context cr) {
            using (var skSurface = SKSurface.Create (100, 100, SKColorType.N_32, SKAlphaType.Premul)) {
                var canvas = skSurface.Canvas;
                var paint = new SKPaint {
                    StrokeWidth = 4, 
                    Color = new SKColor (255, 255, 255, 255)
                };
                var rect = new SKRect (10, 10, 50, 50);
                canvas.DrawRect (rect, paint);
                var image = skSurface.Snapshot ();
                Cairo.Surface surface = new Cairo.ImageSurface (
                    image.Encode ().Data,
                    Cairo.Format.Argb32,
                    image.Width, image.Height,
                    4 * image.Width * image.Height);
                surface.MarkDirty ();
                cr.SetSourceSurface (surface, 0, 0);
                cr.Paint ();
            }
            return true;
        }
    }
    class MainClass {
        public static void Main (string[] args){
            ExceptionManager.UnhandledException += delegate(UnhandledExceptionArgs expArgs) {
                Console.WriteLine (expArgs.ExceptionObject.ToString ());
                expArgs.ExitApplication = true;
            };
            Gtk.Application.Init ();
            var window = new Window ("Hello World");
            window.SetDefaultSize (1024, 800);
            window.SetPosition (WindowPosition.Center);
            window.DeleteEvent += delegate { 
                Gtk.Application.Quit (); 
            };
            var darea = new CCDrawingArea ();
            window.Add (darea);
            window.ShowAll ();
            Gtk.Application.Run ();
        }
    }
}

如何在 Gtk3 中与 Skia 在开罗上下文中绘制

我对 Skia 一无所知,也找不到任何关于其图像格式的文档,但这里的最后一个论点应该是大步前进。自然的步伐将是4*image.Width.这也是斯基亚使用的吗?(步幅是像素开头和该像素下方像素之间的字节数)

           Cairo.Surface surface = new Cairo.ImageSurface (
                image.Encode ().Data,
                Cairo.Format.Argb32,
                image.Width, image.Height,
                4 * image.Width * image.Height);

我找到了在创建SKSurface和SKCanvas之前应该使用SKBitmap的灵魂。为了获得像素数据,我应该使用SKBitmap.GetPixel方法。以下是工作示例的源代码:

    using System;
    using GLib;
    using SkiaSharp;
    using Gtk;
    namespace SkiaSharpExample
    {
        class CCDrawingArea : DrawingArea
        {
            protected override bool OnDrawn(Cairo.Context cr)
            {
                const int width = 100;
                const int height = 100;
                using (var bitmap = new SKBitmap(width, height, SKColorType.N_32, SKAlphaType.Premul))
                {
                    IntPtr len;
                    using (var skSurface = SKSurface.Create(bitmap.Info.Width, bitmap.Info.Height, SKColorType.N_32, SKAlphaType.Premul, bitmap.GetPixels(out len), bitmap.Info.RowBytes))
                    {
                        var canvas = skSurface.Canvas;
                        canvas.Clear(SKColors.White);
                        using (var paint = new SKPaint())
                        {
                            paint.StrokeWidth = 4;
                            paint.Color = new SKColor(0x2c, 0x3e, 0x50);
                            var rect = new SKRect(10, 10, 50, 50);
                            canvas.DrawRect(rect, paint);
                        }
                        Cairo.Surface surface = new Cairo.ImageSurface(
                            bitmap.GetPixels(out len),
                            Cairo.Format.Argb32,
                            bitmap.Width, bitmap.Height,
                            bitmap.Width * 4);

                        surface.MarkDirty();
                        cr.SetSourceSurface(surface, 0, 0);
                        cr.Paint();
                    }
                }
                return true;
            }
        }
        class MainClass
        {
            public static void Main(string[] args)
            {
                ExceptionManager.UnhandledException += delegate(UnhandledExceptionArgs expArgs)
                {
                    Console.WriteLine(expArgs.ExceptionObject.ToString());
                    expArgs.ExitApplication = true;
                };
                Gtk.Application.Init();
                var window = new Window("Hello Skia World");
                window.SetDefaultSize(1024, 800);
                window.SetPosition(WindowPosition.Center);
                window.DeleteEvent += delegate
                {
                    Gtk.Application.Quit();
                };
                var darea = new CCDrawingArea();
                window.Add(darea);
                window.ShowAll();
                Gtk.Application.Run();
            }
            void OnException(object o, UnhandledExceptionArgs args)
            {
            }
        }
    }