1
1
mirror of https://github.com/OpenDiablo2/OpenDiablo2 synced 2024-09-30 15:15:56 -04:00
OpenDiablo2/OpenDiablo2.SDL2/SDL2RenderWindow.cs

353 lines
12 KiB
C#
Raw Normal View History

2018-11-22 00:18:42 -05:00
using OpenDiablo2.Common.Interfaces;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using SDL2;
using System.IO;
using System.Drawing;
using OpenDiablo2.Common.Models;
using Autofac;
using System.Runtime.InteropServices;
using OpenDiablo2.Common.Enums;
2018-11-22 00:18:42 -05:00
namespace OpenDiablo2.SDL2_
{
public sealed class SDL2RenderWindow : IRenderWindow, IRenderTarget, IMouseInfoProvider, IKeyboardInfoProvider
2018-11-22 00:18:42 -05:00
{
private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
2018-11-22 00:18:42 -05:00
private IntPtr window, renderer;
public bool IsRunning { get; private set; }
public int MouseX { get; internal set; } = 0;
public int MouseY { get; internal set; } = 0;
public bool LeftMouseDown { get; internal set; } = false;
public bool RightMouseDown { get; internal set; } = false;
public bool ReserveMouse { get; set; } = false;
public OnKeyPressed KeyPressCallback { get; set; }
private readonly IMPQProvider mpqProvider;
private readonly IPaletteProvider paletteProvider;
private readonly IResourceManager resourceManager;
2018-11-22 00:18:42 -05:00
2018-11-24 23:49:56 -05:00
private IntPtr cellTexture;
public SDL2RenderWindow(
IMPQProvider mpqProvider,
IPaletteProvider paletteProvider,
IResourceManager resourceManager
)
2018-11-22 00:18:42 -05:00
{
this.mpqProvider = mpqProvider;
this.paletteProvider = paletteProvider;
this.resourceManager = resourceManager;
2018-11-22 00:18:42 -05:00
SDL.SDL_Init(SDL.SDL_INIT_EVERYTHING);
if (SDL.SDL_SetHint(SDL.SDL_HINT_RENDER_SCALE_QUALITY, "0") == SDL.SDL_bool.SDL_FALSE)
throw new ApplicationException($"Unable to Init hinting: {SDL.SDL_GetError()}");
window = SDL.SDL_CreateWindow("OpenDiablo2", SDL.SDL_WINDOWPOS_UNDEFINED, SDL.SDL_WINDOWPOS_UNDEFINED, 800, 600, SDL.SDL_WindowFlags.SDL_WINDOW_SHOWN);
if (window == IntPtr.Zero)
throw new ApplicationException($"Unable to create SDL Window: {SDL.SDL_GetError()}");
renderer = SDL.SDL_CreateRenderer(window, -1, SDL.SDL_RendererFlags.SDL_RENDERER_ACCELERATED);
if (renderer == IntPtr.Zero)
throw new ApplicationException($"Unable to create SDL Window: {SDL.SDL_GetError()}");
SDL.SDL_SetRenderDrawBlendMode(renderer, SDL.SDL_BlendMode.SDL_BLENDMODE_BLEND);
SDL.SDL_ShowCursor(0);
2018-11-24 23:49:56 -05:00
cellTexture = SDL.SDL_CreateTexture(renderer, SDL.SDL_PIXELFORMAT_ARGB8888, (int)SDL.SDL_TextureAccess.SDL_TEXTUREACCESS_STREAMING, 256, 256);
SDL.SDL_SetTextureBlendMode(cellTexture, SDL.SDL_BlendMode.SDL_BLENDMODE_BLEND);
IsRunning = true;
2018-11-22 00:18:42 -05:00
}
public void Quit() => IsRunning = false;
2018-11-22 00:18:42 -05:00
public void Dispose()
{
SDL.SDL_DestroyRenderer(renderer);
SDL.SDL_DestroyWindow(window);
SDL.SDL_Quit();
}
public unsafe bool KeyIsPressed(int scancode)
{
int numKeys;
byte* keys = (byte*)SDL.SDL_GetKeyboardState(out numKeys);
return keys[scancode] > 0;
}
2018-11-22 00:18:42 -05:00
public void Clear()
{
2018-11-23 15:22:35 -05:00
SDL.SDL_SetRenderTarget(renderer, IntPtr.Zero);
SDL.SDL_SetRenderDrawColor(renderer, 0, 0, 0, 255);
2018-11-22 00:18:42 -05:00
SDL.SDL_RenderClear(renderer);
}
public void Sync()
{
SDL.SDL_RenderPresent(renderer);
}
public unsafe void Update()
2018-11-22 00:18:42 -05:00
{
while (SDL.SDL_PollEvent(out SDL.SDL_Event evt) != 0)
{
if (evt.type == SDL.SDL_EventType.SDL_MOUSEMOTION)
{
MouseX = evt.motion.x;
MouseY = evt.motion.y;
continue;
}
else if (evt.type == SDL.SDL_EventType.SDL_MOUSEBUTTONDOWN)
{
2018-11-24 23:49:56 -05:00
switch ((uint)evt.button.button)
{
case SDL.SDL_BUTTON_LEFT:
LeftMouseDown = true;
break;
case SDL.SDL_BUTTON_RIGHT:
RightMouseDown = true;
break;
}
}
else if (evt.type == SDL.SDL_EventType.SDL_MOUSEBUTTONUP)
{
switch ((uint)evt.button.button)
{
case SDL.SDL_BUTTON_LEFT:
LeftMouseDown = false;
break;
case SDL.SDL_BUTTON_RIGHT:
RightMouseDown = false;
break;
}
}
else if (evt.type == SDL.SDL_EventType.SDL_KEYDOWN)
{
if (evt.key.keysym.sym == SDL.SDL_Keycode.SDLK_BACKSPACE && KeyPressCallback != null)
KeyPressCallback('\b');
}
else if (evt.type == SDL.SDL_EventType.SDL_TEXTINPUT)
{
KeyPressCallback?.Invoke(Marshal.PtrToStringAnsi((IntPtr)evt.text.text)[0]);
continue;
}
else if (evt.type == SDL.SDL_EventType.SDL_QUIT)
{
IsRunning = false;
continue;
}
2018-11-22 00:18:42 -05:00
}
}
2018-11-22 16:22:39 -05:00
public void Draw(ISprite sprite, Point location)
{
sprite.Location = location;
Draw(sprite);
}
2018-11-22 00:18:42 -05:00
public void Draw(ISprite sprite, int frame, Point location)
{
sprite.Location = location;
sprite.Frame = frame;
Draw(sprite);
}
public void Draw(ISprite sprite, int frame)
{
sprite.Frame = frame;
Draw(sprite);
}
2018-11-22 00:18:42 -05:00
public void Draw(ISprite sprite)
{
var spr = sprite as SDL2Sprite;
2018-11-24 17:54:15 -05:00
if (spr.texture == IntPtr.Zero)
return;
2018-11-22 00:18:42 -05:00
var loc = spr.GetRenderPoint();
var destRect = new SDL.SDL_Rect
{
x = loc.X,
y = loc.Y,
w = spr.FrameSize.Width,
h = spr.FrameSize.Height
};
2018-11-23 15:22:35 -05:00
SDL.SDL_RenderCopy(renderer, spr.texture, IntPtr.Zero, ref destRect);
2018-11-22 00:18:42 -05:00
}
public void Draw(ISprite sprite, int xSegments, int ySegments, int offset)
{
var spr = sprite as SDL2Sprite;
var segSize = xSegments * ySegments;
for (var y = 0; y < ySegments; y++)
{
for (var x = 0; x < xSegments; x++)
{
var textureIndex = x + (y * xSegments) + (offset * xSegments * ySegments);
2018-11-23 15:22:35 -05:00
spr.Frame = Math.Min(spr.TotalFrames - 1, Math.Max(0, textureIndex));
var destRect = new SDL.SDL_Rect
{
x = sprite.Location.X + (x * 256),
y = sprite.Location.Y + (y * 256) - (int)(spr.FrameSize.Height - spr.source.Frames[textureIndex].Height),
w = spr.FrameSize.Width,
h = spr.FrameSize.Height
};
2018-11-23 15:22:35 -05:00
SDL.SDL_RenderCopy(renderer, spr.texture, IntPtr.Zero, ref destRect);
}
}
}
public ISprite LoadSprite(string resourcePath, string palette) => LoadSprite(resourcePath, palette, Point.Empty);
public ISprite LoadSprite(string resourcePath, string palette, Point location)
{
var result = new SDL2Sprite(resourceManager.GetImageSet(resourcePath), renderer)
2018-11-22 16:22:39 -05:00
{
CurrentPalette = paletteProvider.PaletteTable[palette],
Location = location
};
return result;
}
public IFont LoadFont(string resourcePath, string palette)
{
var result = new SDL2Font(resourceManager.GetMPQFont(resourcePath), renderer)
2018-11-22 16:22:39 -05:00
{
CurrentPalette = paletteProvider.PaletteTable[palette]
};
return result;
}
public ILabel CreateLabel(IFont font)
{
var result = new SDL2Label(font, renderer);
return result;
}
public ILabel CreateLabel(IFont font, string text)
{
var result = CreateLabel(font);
result.Text = text;
return result;
}
public ILabel CreateLabel(IFont font, Point position, string text)
{
var result = new SDL2Label(font, renderer)
{
Text = text,
Location = position
2018-11-22 16:22:39 -05:00
};
return result;
}
2018-11-22 16:22:39 -05:00
public void Draw(ILabel label)
{
var lbl = label as SDL2Label;
var loc = lbl.Location;
2018-11-22 16:22:39 -05:00
var destRect = new SDL.SDL_Rect
{
x = loc.X,
y = loc.Y,
w = lbl.textureSize.Width,
h = lbl.textureSize.Height
};
2018-11-22 16:22:39 -05:00
SDL.SDL_RenderCopy(renderer, lbl.texture, IntPtr.Zero, ref destRect);
}
2018-11-25 09:42:39 -05:00
public unsafe void DrawMapCell(int xCell, int yCell, int xPixel, int yPixel, MPQDS1 mapData, int main_index, int sub_index, Palette palette)
2018-11-24 23:49:56 -05:00
{
MPQDT1Tile tile = null;
for (int i = 0; i < mapData.DT1s.Count(); i++)
{
if (mapData.DT1s[i] == null)
continue;
tile = mapData.DT1s[i].Tiles.FirstOrDefault(z => z.MainIndex == main_index && z.SubIndex == sub_index);
if (tile != null)
break;
}
if (tile == null)
throw new ApplicationException("Could not locate tile!");
var frameSize = new Size(tile.Width, Math.Abs(tile.Height));
var srcRect = new SDL.SDL_Rect { x = 0, y = 0, w = frameSize.Width, h = frameSize.Height };
var frameSizeMax = frameSize.Width * frameSize.Height;
2018-11-25 09:42:39 -05:00
if (SDL.SDL_LockTexture(cellTexture, IntPtr.Zero, out IntPtr pixels, out int pitch) != 0)
{
log.Error("Could not lock texture for map rendering");
return;
}
2018-11-24 23:49:56 -05:00
try
{
UInt32* data = (UInt32*)pixels;
for (var i = 0; i < frameSizeMax; i++)
data[i] = 0x0;
var pitchChange = (pitch / 4);
foreach (var block in tile.Blocks)
{
2018-11-25 09:42:39 -05:00
var index = block.PositionX + ((block.PositionY) * pitchChange);
var xx = 0;
foreach (var colorIndex in block.PixelData)
2018-11-24 23:49:56 -05:00
{
2018-11-25 09:42:39 -05:00
try
2018-11-24 23:49:56 -05:00
{
2018-11-25 09:42:39 -05:00
var color = palette.Colors[colorIndex];
2018-11-24 23:49:56 -05:00
if ((color & 0xFFFFFF) > 0)
data[index] = color;
2018-11-25 09:42:39 -05:00
} finally
{
index++;
xx++;
if (xx == 32)
{
index -= 32;
index += pitchChange;
xx = 0;
}
2018-11-24 23:49:56 -05:00
}
}
}
}
finally
{
SDL.SDL_UnlockTexture(cellTexture);
}
var dstRect = new SDL.SDL_Rect { x = xPixel, y = yPixel, w = frameSize.Width, h = frameSize.Height };
SDL.SDL_RenderCopy(renderer, cellTexture, ref srcRect, ref dstRect);
}
2018-11-22 00:18:42 -05:00
}
}