CPF/CPF.Skia/GlView.cs

179 lines
6.1 KiB
C#
Raw Normal View History

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>
/// 支持OpenGL绘制的控件在GLRender事件里绘制开启GPU硬件加速才能使用 new SkiaDrawingFactory { UseGPU = true }
2023-11-21 23:05:03 +08:00
/// </summary>
public class GLView : UIElement
2023-11-21 23:05:03 +08:00
{
int Id;
int ColorBuffer;
int DepthRenderBuffer;
2023-11-21 23:05:03 +08:00
Size oldSize;
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; }
}
//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;
var _gl = skia.GlContext;
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
if (Id == 0)
2023-11-21 23:05:03 +08:00
{
Id = _gl.GenFramebuffer();
ColorBuffer = _gl.GenTexture();
DepthRenderBuffer = _gl.GenRenderbuffer();
2023-11-21 23:05:03 +08:00
_gl.BindTexture(GlConsts.GL_TEXTURE_2D, ColorBuffer);
2023-11-21 23:05:03 +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
_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
_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
_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
_gl.BindRenderbuffer(GlConsts.GL_RENDERBUFFER, 0);
2023-11-21 23:05:03 +08:00
_gl.BindFramebuffer(GlConsts.GL_FRAMEBUFFER, Id);
2023-11-21 23:05:03 +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
_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
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
image = SKImage.FromTexture((GRContext)skia.GlContext.GRContext, backendTexture, GRSurfaceOrigin.BottomLeft, SKColorType.Rgba8888);
2023-11-21 23:05:03 +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);
}
protected virtual void OnGLRender(IGlContext gl)
{
this.RaiseEvent(new GLEventArgs(gl), nameof(GLRender));
2023-11-21 23:05:03 +08:00
}
protected virtual void OnGLLoaded(IGlContext gl)
2023-11-21 23:05:03 +08:00
{
this.RaiseEvent(new GLEventArgs(gl), nameof(GLLoaded));
}
2023-11-21 23:05:03 +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-21 23:05:03 +08:00
protected override void Dispose(bool disposing)
{
if (Id != 0)
2023-11-21 23:05:03 +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
}
if (image != null)
2023-11-21 23:05:03 +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;
}
base.Dispose(disposing);
2023-11-21 23:05:03 +08:00
}
}
2023-11-21 23:05:03 +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
}
}