257 lines
6.6 KiB
C#
257 lines
6.6 KiB
C#
#region Copyright & License Information
|
|
/*
|
|
* Copyright 2007-2018 The OpenRA Developers (see AUTHORS)
|
|
* This file is part of OpenRA, which is free software. It is made
|
|
* available to you under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation, either version 3 of
|
|
* the License, or (at your option) any later version. For more
|
|
* information, see COPYING.
|
|
*/
|
|
#endregion
|
|
|
|
using System;
|
|
using System.Drawing;
|
|
using OpenRA.Graphics;
|
|
using SDL2;
|
|
|
|
namespace OpenRA.Platforms.Default
|
|
{
|
|
sealed class Sdl2GraphicsContext : ThreadAffine, IGraphicsContext
|
|
{
|
|
readonly Sdl2PlatformWindow window;
|
|
bool disposed;
|
|
IntPtr context;
|
|
|
|
public Sdl2GraphicsContext(Sdl2PlatformWindow window)
|
|
{
|
|
this.window = window;
|
|
context = SDL.SDL_GL_CreateContext(window.Window);
|
|
if (context == IntPtr.Zero || SDL.SDL_GL_MakeCurrent(window.Window, context) < 0)
|
|
throw new InvalidOperationException("Can not create OpenGL context. (Error: {0})".F(SDL.SDL_GetError()));
|
|
|
|
OpenGL.Initialize();
|
|
|
|
OpenGL.glEnableVertexAttribArray(Shader.VertexPosAttributeIndex);
|
|
OpenGL.CheckGLError();
|
|
OpenGL.glEnableVertexAttribArray(Shader.TexCoordAttributeIndex);
|
|
OpenGL.CheckGLError();
|
|
OpenGL.glEnableVertexAttribArray(Shader.TexMetadataAttributeIndex);
|
|
OpenGL.CheckGLError();
|
|
}
|
|
|
|
public IVertexBuffer<Vertex> CreateVertexBuffer(int size)
|
|
{
|
|
VerifyThreadAffinity();
|
|
return new VertexBuffer<Vertex>(size);
|
|
}
|
|
|
|
public ITexture CreateTexture()
|
|
{
|
|
VerifyThreadAffinity();
|
|
return new Texture();
|
|
}
|
|
|
|
public ITexture CreateTexture(Bitmap bitmap)
|
|
{
|
|
VerifyThreadAffinity();
|
|
return new Texture(bitmap);
|
|
}
|
|
|
|
public IFrameBuffer CreateFrameBuffer(Size s)
|
|
{
|
|
VerifyThreadAffinity();
|
|
return new FrameBuffer(s);
|
|
}
|
|
|
|
public IShader CreateShader(string name)
|
|
{
|
|
VerifyThreadAffinity();
|
|
return new Shader(name);
|
|
}
|
|
|
|
public void EnableScissor(int left, int top, int width, int height)
|
|
{
|
|
VerifyThreadAffinity();
|
|
|
|
if (width < 0)
|
|
width = 0;
|
|
|
|
if (height < 0)
|
|
height = 0;
|
|
|
|
var windowSize = window.WindowSize;
|
|
var windowScale = window.WindowScale;
|
|
var surfaceSize = window.SurfaceSize;
|
|
|
|
var bottom = windowSize.Height - (top + height);
|
|
if (windowSize != surfaceSize)
|
|
{
|
|
left = (int)Math.Round(windowScale * left);
|
|
bottom = (int)Math.Round(windowScale * bottom);
|
|
width = (int)Math.Round(windowScale * width);
|
|
height = (int)Math.Round(windowScale * height);
|
|
}
|
|
|
|
OpenGL.glScissor(left, bottom, width, height);
|
|
OpenGL.CheckGLError();
|
|
OpenGL.glEnable(OpenGL.GL_SCISSOR_TEST);
|
|
OpenGL.CheckGLError();
|
|
}
|
|
|
|
public void DisableScissor()
|
|
{
|
|
VerifyThreadAffinity();
|
|
OpenGL.glDisable(OpenGL.GL_SCISSOR_TEST);
|
|
OpenGL.CheckGLError();
|
|
}
|
|
|
|
public Bitmap TakeScreenshot()
|
|
{
|
|
var rect = new Rectangle(Point.Empty, window.SurfaceSize);
|
|
var bitmap = new Bitmap(rect.Width, rect.Height, System.Drawing.Imaging.PixelFormat.Format32bppArgb);
|
|
var data = bitmap.LockBits(rect,
|
|
System.Drawing.Imaging.ImageLockMode.WriteOnly, System.Drawing.Imaging.PixelFormat.Format32bppArgb);
|
|
|
|
OpenGL.glPushClientAttrib(OpenGL.GL_CLIENT_PIXEL_STORE_BIT);
|
|
|
|
OpenGL.glPixelStoref(OpenGL.GL_PACK_ROW_LENGTH, data.Stride / 4f);
|
|
OpenGL.glPixelStoref(OpenGL.GL_PACK_ALIGNMENT, 1);
|
|
|
|
OpenGL.glReadPixels(rect.X, rect.Y, rect.Width, rect.Height, OpenGL.GL_BGRA, OpenGL.GL_UNSIGNED_BYTE, data.Scan0);
|
|
OpenGL.glFinish();
|
|
|
|
OpenGL.glPopClientAttrib();
|
|
|
|
bitmap.UnlockBits(data);
|
|
|
|
// OpenGL standard defines the origin in the bottom left corner which is why this is upside-down by default.
|
|
bitmap.RotateFlip(RotateFlipType.RotateNoneFlipY);
|
|
|
|
return bitmap;
|
|
}
|
|
|
|
public void Present()
|
|
{
|
|
VerifyThreadAffinity();
|
|
SDL.SDL_GL_SwapWindow(window.Window);
|
|
}
|
|
|
|
static int ModeFromPrimitiveType(PrimitiveType pt)
|
|
{
|
|
switch (pt)
|
|
{
|
|
case PrimitiveType.PointList: return OpenGL.GL_POINTS;
|
|
case PrimitiveType.LineList: return OpenGL.GL_LINES;
|
|
case PrimitiveType.TriangleList: return OpenGL.GL_TRIANGLES;
|
|
}
|
|
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
public void DrawPrimitives(PrimitiveType pt, int firstVertex, int numVertices)
|
|
{
|
|
VerifyThreadAffinity();
|
|
OpenGL.glDrawArrays(ModeFromPrimitiveType(pt), firstVertex, numVertices);
|
|
OpenGL.CheckGLError();
|
|
}
|
|
|
|
public void Clear()
|
|
{
|
|
VerifyThreadAffinity();
|
|
OpenGL.glClearColor(0, 0, 0, 1);
|
|
OpenGL.CheckGLError();
|
|
OpenGL.glClear(OpenGL.GL_COLOR_BUFFER_BIT | OpenGL.GL_DEPTH_BUFFER_BIT);
|
|
OpenGL.CheckGLError();
|
|
}
|
|
|
|
public void EnableDepthBuffer()
|
|
{
|
|
VerifyThreadAffinity();
|
|
OpenGL.glClear(OpenGL.GL_DEPTH_BUFFER_BIT);
|
|
OpenGL.CheckGLError();
|
|
OpenGL.glEnable(OpenGL.GL_DEPTH_TEST);
|
|
OpenGL.CheckGLError();
|
|
OpenGL.glDepthFunc(OpenGL.GL_LEQUAL);
|
|
OpenGL.CheckGLError();
|
|
}
|
|
|
|
public void DisableDepthBuffer()
|
|
{
|
|
VerifyThreadAffinity();
|
|
OpenGL.glDisable(OpenGL.GL_DEPTH_TEST);
|
|
OpenGL.CheckGLError();
|
|
}
|
|
|
|
public void ClearDepthBuffer()
|
|
{
|
|
VerifyThreadAffinity();
|
|
OpenGL.glClear(OpenGL.GL_DEPTH_BUFFER_BIT);
|
|
OpenGL.CheckGLError();
|
|
}
|
|
|
|
public void SetBlendMode(BlendMode mode)
|
|
{
|
|
VerifyThreadAffinity();
|
|
OpenGL.glBlendEquation(OpenGL.GL_FUNC_ADD);
|
|
OpenGL.CheckGLError();
|
|
|
|
switch (mode)
|
|
{
|
|
case BlendMode.None:
|
|
OpenGL.glDisable(OpenGL.GL_BLEND);
|
|
break;
|
|
case BlendMode.Alpha:
|
|
OpenGL.glEnable(OpenGL.GL_BLEND);
|
|
OpenGL.CheckGLError();
|
|
OpenGL.glBlendFunc(OpenGL.GL_ONE, OpenGL.GL_ONE_MINUS_SRC_ALPHA);
|
|
break;
|
|
case BlendMode.Additive:
|
|
case BlendMode.Subtractive:
|
|
OpenGL.glEnable(OpenGL.GL_BLEND);
|
|
OpenGL.CheckGLError();
|
|
OpenGL.glBlendFunc(OpenGL.GL_ONE, OpenGL.GL_ONE);
|
|
if (mode == BlendMode.Subtractive)
|
|
{
|
|
OpenGL.CheckGLError();
|
|
OpenGL.glBlendEquation(OpenGL.GL_FUNC_REVERSE_SUBTRACT);
|
|
}
|
|
|
|
break;
|
|
case BlendMode.Multiply:
|
|
OpenGL.glEnable(OpenGL.GL_BLEND);
|
|
OpenGL.CheckGLError();
|
|
OpenGL.glBlendFunc(OpenGL.GL_DST_COLOR, OpenGL.GL_ONE_MINUS_SRC_ALPHA);
|
|
OpenGL.CheckGLError();
|
|
break;
|
|
case BlendMode.Multiplicative:
|
|
OpenGL.glEnable(OpenGL.GL_BLEND);
|
|
OpenGL.CheckGLError();
|
|
OpenGL.glBlendFunc(OpenGL.GL_ZERO, OpenGL.GL_SRC_COLOR);
|
|
break;
|
|
case BlendMode.DoubleMultiplicative:
|
|
OpenGL.glEnable(OpenGL.GL_BLEND);
|
|
OpenGL.CheckGLError();
|
|
OpenGL.glBlendFunc(OpenGL.GL_DST_COLOR, OpenGL.GL_SRC_COLOR);
|
|
break;
|
|
}
|
|
|
|
OpenGL.CheckGLError();
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
if (disposed)
|
|
return;
|
|
|
|
disposed = true;
|
|
if (context != IntPtr.Zero)
|
|
{
|
|
SDL.SDL_GL_DeleteContext(context);
|
|
context = IntPtr.Zero;
|
|
}
|
|
}
|
|
|
|
public string GLVersion { get { return OpenGL.Version; } }
|
|
}
|
|
}
|