2023-11-21 23:05:03 +08:00
|
|
|
|
using CPF.Drawing;
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using SkiaSharp;
|
|
|
|
|
using CPF.OpenGL;
|
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
|
|
|
|
|
|
namespace CPF.Skia
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
2023-11-29 16:38:58 +08:00
|
|
|
|
/// 支持OpenGL绘制的控件,在GLRender事件里绘制,开启GPU硬件加速才能使用 new SkiaDrawingFactory { UseGPU = true }
|
2023-11-21 23:05:03 +08:00
|
|
|
|
/// </summary>
|
2023-11-29 16:38:58 +08:00
|
|
|
|
public class GLView : UIElement
|
2023-11-21 23:05:03 +08:00
|
|
|
|
{
|
2023-11-29 16:38:58 +08:00
|
|
|
|
int Id;
|
|
|
|
|
int ColorBuffer;
|
|
|
|
|
int DepthRenderBuffer;
|
2023-11-21 23:05:03 +08:00
|
|
|
|
Size oldSize;
|
2023-11-29 16:38:58 +08:00
|
|
|
|
SKImage image;
|
2023-11-29 21:06:01 +08:00
|
|
|
|
SKPaint paint;
|
|
|
|
|
GRBackendTexture backendTexture;
|
2023-11-30 15:12:23 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// 支持OpenGL绘制的控件,在GLRender事件里绘制,开启GPU硬件加速才能使用 new SkiaDrawingFactory { UseGPU = true }
|
|
|
|
|
/// </summary>
|
|
|
|
|
public GLView() { }
|
2024-01-28 21:12:25 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// 绑定的FB id
|
|
|
|
|
/// </summary>
|
|
|
|
|
public int FramebufferId
|
|
|
|
|
{
|
|
|
|
|
get { return Id; }
|
|
|
|
|
}
|
2023-11-29 16:38:58 +08:00
|
|
|
|
|
|
|
|
|
//IGlContext context;
|
|
|
|
|
protected unsafe override void OnRender(DrawingContext dc)
|
2023-11-21 23:05:03 +08:00
|
|
|
|
{
|
2023-11-29 21:06:01 +08:00
|
|
|
|
var cSize = ActualSize;
|
|
|
|
|
if (cSize.Width <= 0 || cSize.Height <= 0 || DesignMode)
|
2023-11-21 23:05:03 +08:00
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-29 21:06:01 +08:00
|
|
|
|
var size = new PixelSize((int)Math.Round(cSize.Width * Root.RenderScaling), (int)Math.Round(cSize.Height * Root.RenderScaling));
|
2023-11-21 23:05:03 +08:00
|
|
|
|
var skia = dc as SkiaDrawingContext;
|
2023-11-29 16:38:58 +08:00
|
|
|
|
var _gl = skia.GlContext;
|
2023-12-03 23:47:52 +08:00
|
|
|
|
if (_gl == null)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
2023-11-29 21:06:01 +08:00
|
|
|
|
if (paint == null)
|
|
|
|
|
{
|
|
|
|
|
paint = new SKPaint();
|
|
|
|
|
}
|
|
|
|
|
paint.IsAntialias = IsAntiAlias;
|
|
|
|
|
paint.FilterQuality = IsAntiAlias ? SKFilterQuality.Medium : SKFilterQuality.None;
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
if (Id == 0)
|
2023-11-21 23:05:03 +08:00
|
|
|
|
{
|
2023-11-29 16:38:58 +08:00
|
|
|
|
Id = _gl.GenFramebuffer();
|
|
|
|
|
ColorBuffer = _gl.GenTexture();
|
|
|
|
|
DepthRenderBuffer = _gl.GenRenderbuffer();
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
_gl.BindTexture(GlConsts.GL_TEXTURE_2D, ColorBuffer);
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-12-03 23:47:52 +08:00
|
|
|
|
_gl.TexParameter(GlConsts.GL_TEXTURE_2D, GlConsts.GL_TEXTURE_MIN_FILTER, (int)GlConsts.GL_LINEAR);
|
|
|
|
|
_gl.TexParameter(GlConsts.GL_TEXTURE_2D, GlConsts.GL_TEXTURE_MAG_FILTER, GlConsts.GL_LINEAR);
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
_gl.BindTexture(GlConsts.GL_TEXTURE_2D, 0);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
OnGLLoaded(_gl);
|
|
|
|
|
}
|
2023-11-29 21:06:01 +08:00
|
|
|
|
if (cSize != oldSize)
|
2023-11-21 23:05:03 +08:00
|
|
|
|
{
|
2023-11-29 21:06:01 +08:00
|
|
|
|
oldSize = cSize;
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
_gl.BindTexture(GlConsts.GL_TEXTURE_2D, ColorBuffer);
|
|
|
|
|
_gl.TexImage2D(GlConsts.GL_TEXTURE_2D, 0, GlConsts.GL_RGBA, (int)size.Width, (int)size.Height, 0, GlConsts.GL_RGB, GlConsts.GL_UNSIGNED_BYTE, IntPtr.Zero);
|
|
|
|
|
_gl.BindTexture(GlConsts.GL_TEXTURE_2D, 0);
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
_gl.BindRenderbuffer(GlConsts.GL_RENDERBUFFER, DepthRenderBuffer);
|
|
|
|
|
_gl.RenderbufferStorage(GlConsts.GL_RENDERBUFFER, GlConsts.GL_DEPTH32F_STENCIL8, (int)size.Width, (int)size.Height);
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
_gl.BindRenderbuffer(GlConsts.GL_RENDERBUFFER, 0);
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
_gl.BindFramebuffer(GlConsts.GL_FRAMEBUFFER, Id);
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
_gl.FramebufferTexture2D(GlConsts.GL_FRAMEBUFFER, GlConsts.GL_COLOR_ATTACHMENT0, GlConsts.GL_TEXTURE_2D, ColorBuffer, 0);
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
_gl.FramebufferRenderbuffer(GlConsts.GL_FRAMEBUFFER, GlConsts.GL_DEPTH_STENCIL_ATTACHMENT, GlConsts.GL_RENDERBUFFER, DepthRenderBuffer);
|
|
|
|
|
_gl.BindFramebuffer(GlConsts.GL_FRAMEBUFFER, 0);
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
if (image != null)
|
|
|
|
|
{
|
|
|
|
|
image.Dispose();
|
|
|
|
|
}
|
2023-11-29 21:06:01 +08:00
|
|
|
|
if (backendTexture != null)
|
|
|
|
|
{
|
|
|
|
|
backendTexture.Dispose();
|
|
|
|
|
}
|
|
|
|
|
backendTexture = new GRBackendTexture((int)(size.Width), (int)(size.Height), false, new GRGlTextureInfo(0x0DE1, (uint)ColorBuffer, SKColorType.Rgba8888.ToGlSizedFormat()));
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
image = SKImage.FromTexture((GRContext)skia.GlContext.GRContext, backendTexture, GRSurfaceOrigin.BottomLeft, SKColorType.Rgba8888);
|
2023-11-21 23:05:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
_gl.BindFramebuffer(GlConsts.GL_FRAMEBUFFER, Id);
|
|
|
|
|
var vp = new float[4];
|
|
|
|
|
_gl.GetFloatv(GlConsts.GL_VIEWPORT, vp);
|
|
|
|
|
_gl.Viewport(0, 0, (int)size.Width, (int)size.Height);
|
|
|
|
|
OnGLRender(_gl);
|
|
|
|
|
_gl.Viewport((int)vp[0], (int)vp[1], (int)vp[2], (int)vp[3]);
|
2023-11-29 21:06:01 +08:00
|
|
|
|
_gl.BindFramebuffer(GlConsts.GL_FRAMEBUFFER, 0);
|
|
|
|
|
//_gl.Flush();
|
|
|
|
|
skia.SKCanvas.DrawImage(image, new SKRect(0, 0, size.Width, size.Height), new SKRect(0, 0, cSize.Width, cSize.Height), paint);
|
2023-11-29 16:38:58 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
protected virtual void OnGLRender(IGlContext gl)
|
|
|
|
|
{
|
|
|
|
|
this.RaiseEvent(new GLEventArgs(gl), nameof(GLRender));
|
2023-11-21 23:05:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
protected virtual void OnGLLoaded(IGlContext gl)
|
2023-11-21 23:05:03 +08:00
|
|
|
|
{
|
2023-11-29 16:38:58 +08:00
|
|
|
|
this.RaiseEvent(new GLEventArgs(gl), nameof(GLLoaded));
|
|
|
|
|
}
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
public event EventHandler<GLEventArgs> GLLoaded
|
|
|
|
|
{
|
|
|
|
|
add { AddHandler(value); }
|
|
|
|
|
remove { RemoveHandler(value); }
|
|
|
|
|
}
|
|
|
|
|
public event EventHandler<GLEventArgs> GLRender
|
|
|
|
|
{
|
|
|
|
|
add { AddHandler(value); }
|
|
|
|
|
remove { RemoveHandler(value); }
|
2023-11-21 23:05:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
|
2023-11-21 23:05:03 +08:00
|
|
|
|
protected override void Dispose(bool disposing)
|
|
|
|
|
{
|
2023-11-29 16:38:58 +08:00
|
|
|
|
if (Id != 0)
|
2023-11-21 23:05:03 +08:00
|
|
|
|
{
|
2023-11-29 16:38:58 +08:00
|
|
|
|
OpenglEx.DeleteFramebuffers(null, 1, new int[] { Id });
|
|
|
|
|
OpenglEx.DeleteTextures(null, 1, new int[] { ColorBuffer });
|
|
|
|
|
OpenglEx.DeleteRenderbuffers(null, 1, new int[] { DepthRenderBuffer });
|
2023-11-29 21:06:01 +08:00
|
|
|
|
Id = 0;
|
2023-11-21 23:05:03 +08:00
|
|
|
|
}
|
2023-11-29 16:38:58 +08:00
|
|
|
|
if (image != null)
|
2023-11-21 23:05:03 +08:00
|
|
|
|
{
|
2023-11-29 16:38:58 +08:00
|
|
|
|
image.Dispose();
|
|
|
|
|
image = null;
|
2023-11-21 23:05:03 +08:00
|
|
|
|
}
|
2023-11-29 21:06:01 +08:00
|
|
|
|
if (paint != null)
|
|
|
|
|
{
|
|
|
|
|
paint.Dispose();
|
|
|
|
|
paint = null;
|
|
|
|
|
}
|
|
|
|
|
if (backendTexture != null)
|
|
|
|
|
{
|
|
|
|
|
backendTexture.Dispose();
|
|
|
|
|
backendTexture = null;
|
|
|
|
|
}
|
2023-11-29 16:38:58 +08:00
|
|
|
|
base.Dispose(disposing);
|
2023-11-21 23:05:03 +08:00
|
|
|
|
}
|
2023-11-29 16:38:58 +08:00
|
|
|
|
}
|
2023-11-21 23:05:03 +08:00
|
|
|
|
|
2023-11-29 16:38:58 +08:00
|
|
|
|
public class GLEventArgs : EventArgs
|
|
|
|
|
{
|
|
|
|
|
public GLEventArgs(IGlContext gl)
|
|
|
|
|
{
|
|
|
|
|
Context = gl;
|
|
|
|
|
}
|
2023-11-29 21:06:01 +08:00
|
|
|
|
public IGlContext Context { get; private set; }
|
2023-11-21 23:05:03 +08:00
|
|
|
|
}
|
|
|
|
|
}
|