2013-09-27 21:07:23 +00:00
|
|
|
#region License
|
|
|
|
//
|
|
|
|
// The Open Toolkit Library License
|
|
|
|
//
|
|
|
|
// Copyright (c) 2006 - 2013 Stefanos Apostolopoulos for the Open Toolkit library.
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights to
|
|
|
|
// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
|
|
|
// the Software, and to permit persons to whom the Software is furnished to do
|
|
|
|
// so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
|
|
// copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
|
|
|
// OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
|
|
// HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
|
|
// WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
|
|
// OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
//
|
|
|
|
#endregion
|
|
|
|
|
2013-09-27 12:41:37 +00:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Diagnostics;
|
|
|
|
using System.Drawing;
|
|
|
|
using System.Drawing.Imaging;
|
2013-09-27 21:01:46 +00:00
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
using OpenTK;
|
2013-09-27 12:41:37 +00:00
|
|
|
using OpenTK.Input;
|
|
|
|
|
|
|
|
namespace OpenTK.Platform.SDL2
|
|
|
|
{
|
|
|
|
class Sdl2NativeWindow : INativeWindow, IInputDriver
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
readonly object sync = new object();
|
|
|
|
|
2013-09-27 12:41:37 +00:00
|
|
|
Sdl2WindowInfo window;
|
2013-09-27 21:01:46 +00:00
|
|
|
uint window_id;
|
2013-09-27 12:41:37 +00:00
|
|
|
bool is_visible;
|
|
|
|
bool is_focused;
|
|
|
|
bool is_cursor_visible = true;
|
|
|
|
bool exists;
|
|
|
|
bool disposed;
|
2013-09-27 21:01:46 +00:00
|
|
|
WindowState window_state = WindowState.Normal;
|
|
|
|
WindowState previous_window_state = WindowState.Normal;
|
|
|
|
WindowBorder window_border = WindowBorder.Resizable;
|
2013-09-27 12:41:37 +00:00
|
|
|
Icon icon;
|
2013-10-01 20:01:27 +00:00
|
|
|
string window_title;
|
2013-09-27 12:41:37 +00:00
|
|
|
|
2013-10-04 08:02:19 +00:00
|
|
|
readonly IInputDriver input_driver = new Sdl2InputDriver();
|
2013-10-04 00:37:41 +00:00
|
|
|
|
2013-10-01 23:14:26 +00:00
|
|
|
readonly SDL.SDL_EventFilter EventFilterDelegate = FilterEvents;
|
2013-09-27 21:01:46 +00:00
|
|
|
|
|
|
|
static readonly Dictionary<uint, Sdl2NativeWindow> windows =
|
|
|
|
new Dictionary<uint, Sdl2NativeWindow>();
|
|
|
|
|
|
|
|
static readonly Sdl2KeyMap map = new Sdl2KeyMap();
|
|
|
|
|
2013-09-27 16:57:05 +00:00
|
|
|
public Sdl2NativeWindow(int x, int y, int width, int height,
|
|
|
|
string title, GameWindowFlags options, DisplayDevice device)
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
|
|
|
{
|
|
|
|
var bounds = device.Bounds;
|
|
|
|
var flags = TranslateFlags(options);
|
|
|
|
flags |= SDL.SDL_WindowFlags.SDL_WINDOW_OPENGL;
|
|
|
|
flags |= SDL.SDL_WindowFlags.SDL_WINDOW_RESIZABLE;
|
2013-10-03 13:11:59 +00:00
|
|
|
flags |= SDL.SDL_WindowFlags.SDL_WINDOW_HIDDEN;
|
2013-10-09 22:34:15 +00:00
|
|
|
flags |= SDL.SDL_WindowFlags.SDL_WINDOW_ALLOW_HIGHDPI;
|
2013-09-27 21:01:46 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
if ((flags & SDL.SDL_WindowFlags.SDL_WINDOW_FULLSCREEN_DESKTOP) != 0 ||
|
|
|
|
(flags & SDL.SDL_WindowFlags.SDL_WINDOW_FULLSCREEN) != 0)
|
|
|
|
window_state = WindowState.Fullscreen;
|
2013-09-27 21:01:46 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
IntPtr handle;
|
|
|
|
lock (SDL.Sync)
|
|
|
|
{
|
|
|
|
handle = SDL.SDL_CreateWindow(title, bounds.Left + x, bounds.Top + y, width, height, flags);
|
|
|
|
SDL.SDL_AddEventWatch(EventFilterDelegate, handle);
|
|
|
|
SDL.SDL_PumpEvents();
|
|
|
|
}
|
|
|
|
window = new Sdl2WindowInfo(handle, null);
|
|
|
|
window_id = SDL.SDL_GetWindowID(handle);
|
|
|
|
windows.Add(window_id, this);
|
|
|
|
window_title = title;
|
2013-09-27 12:41:37 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
exists = true;
|
2013-10-01 23:14:26 +00:00
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#region Private Members
|
|
|
|
|
|
|
|
static SDL.SDL_WindowFlags TranslateFlags(GameWindowFlags flags)
|
|
|
|
{
|
|
|
|
switch (flags)
|
|
|
|
{
|
|
|
|
case GameWindowFlags.Fullscreen:
|
2013-09-27 16:57:05 +00:00
|
|
|
return SDL.SDL_WindowFlags.SDL_WINDOW_FULLSCREEN_DESKTOP;
|
2013-09-27 12:41:37 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
return (SDL.SDL_WindowFlags)0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-30 11:51:31 +00:00
|
|
|
static Key TranslateKey(SDL.SDL_Scancode scan)
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
|
|
|
Key result = Key.Unknown;
|
2013-09-30 11:51:31 +00:00
|
|
|
if (map.ContainsKey(scan))
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
2013-09-30 11:51:31 +00:00
|
|
|
result = map[scan];
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2013-09-30 11:51:31 +00:00
|
|
|
static Key TranslateKey(SDL.SDL_Keycode key)
|
|
|
|
{
|
|
|
|
SDL.SDL_Scancode scan = SDL.SDL_GetScancodeFromKey(key);
|
|
|
|
return TranslateKey(scan);
|
|
|
|
}
|
|
|
|
|
2013-10-01 23:14:26 +00:00
|
|
|
unsafe static int FilterEvents(IntPtr user_data, IntPtr e)
|
2013-09-27 21:01:46 +00:00
|
|
|
{
|
2013-10-01 23:14:26 +00:00
|
|
|
bool processed = false;
|
|
|
|
|
|
|
|
try
|
2013-09-27 21:01:46 +00:00
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
Sdl2NativeWindow window = null;
|
|
|
|
SDL.SDL_Event ev = *(SDL.SDL_Event*)e;
|
2013-09-27 21:01:46 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
switch (ev.type)
|
|
|
|
{
|
|
|
|
case SDL.SDL_EventType.SDL_WINDOWEVENT:
|
|
|
|
if (windows.TryGetValue(ev.window.windowID, out window))
|
|
|
|
{
|
|
|
|
ProcessWindowEvent(window, ev.window);
|
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
break;
|
2013-10-01 23:14:26 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
case SDL.SDL_EventType.SDL_KEYDOWN:
|
|
|
|
case SDL.SDL_EventType.SDL_KEYUP:
|
|
|
|
if (windows.TryGetValue(ev.key.windowID, out window))
|
|
|
|
{
|
|
|
|
ProcessKeyEvent(window, ev);
|
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
break;
|
2013-10-01 23:14:26 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
case SDL.SDL_EventType.SDL_MOUSEBUTTONDOWN:
|
|
|
|
case SDL.SDL_EventType.SDL_MOUSEBUTTONUP:
|
|
|
|
if (windows.TryGetValue(ev.button.windowID, out window))
|
|
|
|
{
|
|
|
|
ProcessButtonEvent(window, ev);
|
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
break;
|
2013-10-01 23:14:26 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
case SDL.SDL_EventType.SDL_MOUSEMOTION:
|
|
|
|
if (windows.TryGetValue(ev.motion.windowID, out window))
|
|
|
|
{
|
|
|
|
ProcessMotionEvent(window, ev);
|
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
break;
|
2013-10-01 23:14:26 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
case SDL.SDL_EventType.SDL_MOUSEWHEEL:
|
|
|
|
if (windows.TryGetValue(ev.wheel.windowID, out window))
|
|
|
|
{
|
|
|
|
ProcessWheelEvent(window, ev);
|
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
break;
|
2013-10-01 23:14:26 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
case SDL.SDL_EventType.SDL_QUIT:
|
|
|
|
Debug.WriteLine("Sdl2 application quit");
|
|
|
|
break;
|
|
|
|
}
|
2013-09-27 21:01:46 +00:00
|
|
|
}
|
2013-10-01 23:14:26 +00:00
|
|
|
catch (Exception ex)
|
|
|
|
{
|
|
|
|
Debug.Print(ex.ToString());
|
|
|
|
}
|
|
|
|
|
|
|
|
return processed ? 0 : 1;
|
2013-09-27 21:01:46 +00:00
|
|
|
}
|
|
|
|
|
2013-10-01 23:14:26 +00:00
|
|
|
static void ProcessButtonEvent(Sdl2NativeWindow window, SDL.SDL_Event ev)
|
|
|
|
{
|
|
|
|
bool button_pressed = ev.button.state == SDL.SDL_PRESSED;
|
2013-10-02 15:55:30 +00:00
|
|
|
|
|
|
|
// We need MouseUp events to be reported even if they occur
|
|
|
|
// outside the window. SetWindowGrab ensures we get them.
|
|
|
|
if (window.CursorVisible)
|
|
|
|
{
|
|
|
|
SDL.SDL_SetWindowGrab(window.window.Handle,
|
|
|
|
button_pressed ? SDL.SDL_bool.SDL_TRUE : SDL.SDL_bool.SDL_FALSE);
|
|
|
|
}
|
|
|
|
|
2013-10-01 23:14:26 +00:00
|
|
|
switch (ev.button.button)
|
2013-10-04 08:02:19 +00:00
|
|
|
{/*
|
|
|
|
|
2013-10-01 23:14:26 +00:00
|
|
|
case (byte)SDL.SDL_BUTTON_LEFT:
|
|
|
|
window.mouse[MouseButton.Left] = button_pressed;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case (byte)SDL.SDL_BUTTON_MIDDLE:
|
|
|
|
window.mouse[MouseButton.Middle] = button_pressed;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case (byte)SDL.SDL_BUTTON_RIGHT:
|
|
|
|
window.mouse[MouseButton.Right] = button_pressed;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case (byte)SDL.SDL_BUTTON_X1:
|
|
|
|
window.mouse[MouseButton.Button1] = button_pressed;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case (byte)SDL.SDL_BUTTON_X2:
|
|
|
|
window.mouse[MouseButton.Button2] = button_pressed;
|
|
|
|
break;
|
2013-10-04 08:02:19 +00:00
|
|
|
*/
|
2013-10-01 23:14:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ProcessKeyEvent(Sdl2NativeWindow window, SDL.SDL_Event ev)
|
|
|
|
{
|
|
|
|
bool key_pressed = ev.key.state == SDL.SDL_PRESSED;
|
|
|
|
var key = ev.key.keysym;
|
2013-10-04 08:02:19 +00:00
|
|
|
//window.keyboard.SetKey(TranslateKey(key.scancode), (uint)key.scancode, key_pressed);
|
2013-10-01 23:14:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ProcessMotionEvent(Sdl2NativeWindow window, SDL.SDL_Event ev)
|
|
|
|
{
|
2013-10-03 23:38:19 +00:00
|
|
|
float scale = window.ClientSize.Width / (float)window.Size.Width;
|
2013-10-04 08:02:19 +00:00
|
|
|
//window.mouse.Position = new Point(
|
|
|
|
// (int)(ev.motion.x * scale), (int)(ev.motion.y * scale));
|
2013-10-01 23:14:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ProcessWheelEvent(Sdl2NativeWindow window, SDL.SDL_Event ev)
|
|
|
|
{
|
2013-10-04 08:02:19 +00:00
|
|
|
//window.mouse.Wheel += ev.wheel.y;
|
2013-10-01 23:14:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ProcessWindowEvent(Sdl2NativeWindow window, SDL.SDL_WindowEvent e)
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
|
|
|
switch (e.windowEvent)
|
|
|
|
{
|
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_CLOSE:
|
2013-10-01 23:14:26 +00:00
|
|
|
var close_args = new System.ComponentModel.CancelEventArgs();
|
|
|
|
window.Closing(window, close_args);
|
|
|
|
if (!close_args.Cancel)
|
|
|
|
{
|
|
|
|
window.Closed(window, EventArgs.Empty);
|
2013-10-03 12:59:30 +00:00
|
|
|
//window.DestroyWindow();
|
2013-10-01 23:14:26 +00:00
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_ENTER:
|
2013-10-01 23:14:26 +00:00
|
|
|
window.MouseEnter(window, EventArgs.Empty);
|
2013-09-27 12:41:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_LEAVE:
|
2013-10-01 23:14:26 +00:00
|
|
|
window.MouseLeave(window, EventArgs.Empty);
|
2013-09-27 12:41:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_EXPOSED:
|
|
|
|
// do nothing
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_FOCUS_GAINED:
|
2013-10-01 23:14:26 +00:00
|
|
|
window.is_focused = true;
|
|
|
|
window.FocusedChanged(window, EventArgs.Empty);
|
2013-09-27 12:41:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_FOCUS_LOST:
|
2013-10-01 23:14:26 +00:00
|
|
|
window.is_focused = false;
|
|
|
|
window.FocusedChanged(window, EventArgs.Empty);
|
2013-09-27 12:41:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_HIDDEN:
|
2013-10-01 23:14:26 +00:00
|
|
|
window.is_visible = false;
|
|
|
|
window.VisibleChanged(window, EventArgs.Empty);
|
2013-09-27 12:41:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_SHOWN:
|
2013-10-01 23:14:26 +00:00
|
|
|
window.is_visible = true;
|
|
|
|
window.VisibleChanged(window, EventArgs.Empty);
|
2013-09-27 12:41:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_MAXIMIZED:
|
2013-10-01 23:14:26 +00:00
|
|
|
window.previous_window_state = window.window_state;
|
|
|
|
window.window_state = OpenTK.WindowState.Maximized;
|
|
|
|
window.WindowStateChanged(window, EventArgs.Empty);
|
2013-09-27 21:01:46 +00:00
|
|
|
break;
|
|
|
|
|
2013-09-27 12:41:37 +00:00
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_MINIMIZED:
|
2013-10-01 23:14:26 +00:00
|
|
|
window.previous_window_state = window.window_state;
|
|
|
|
window.window_state = OpenTK.WindowState.Minimized;
|
|
|
|
window.WindowStateChanged(window, EventArgs.Empty);
|
2013-09-27 21:01:46 +00:00
|
|
|
break;
|
|
|
|
|
2013-09-27 12:41:37 +00:00
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_RESTORED:
|
2013-10-01 23:14:26 +00:00
|
|
|
window.window_state = window.previous_window_state;
|
|
|
|
window.WindowStateChanged(window, EventArgs.Empty);
|
2013-09-27 12:41:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_MOVED:
|
2013-10-01 23:14:26 +00:00
|
|
|
window.Move(window, EventArgs.Empty);
|
2013-09-27 12:41:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_RESIZED:
|
|
|
|
case SDL.SDL_WindowEventID.SDL_WINDOWEVENT_SIZE_CHANGED:
|
2013-10-01 23:14:26 +00:00
|
|
|
window.Resize(window, EventArgs.Empty);
|
2013-09-27 12:41:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
Debug.Print("SDL2 unhandled event: {0}", e.type);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-27 16:57:05 +00:00
|
|
|
void DestroyWindow()
|
|
|
|
{
|
|
|
|
exists = false;
|
2013-09-27 21:01:46 +00:00
|
|
|
|
2013-10-01 23:14:26 +00:00
|
|
|
if (window.Handle != IntPtr.Zero)
|
2013-09-27 21:01:46 +00:00
|
|
|
{
|
2013-10-01 23:14:26 +00:00
|
|
|
lock (SDL.Sync)
|
|
|
|
{
|
|
|
|
SDL.SDL_DelEventWatch(EventFilterDelegate, window.Handle);
|
|
|
|
if (windows.ContainsKey(window_id))
|
|
|
|
{
|
|
|
|
windows.Remove(window_id);
|
|
|
|
}
|
2013-10-03 12:59:30 +00:00
|
|
|
SDL.SDL_DestroyWindow(window.Handle);
|
2013-10-01 23:14:26 +00:00
|
|
|
}
|
2013-09-27 21:01:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
window_id = 0;
|
2013-09-27 16:57:05 +00:00
|
|
|
window.Handle = IntPtr.Zero;
|
|
|
|
}
|
|
|
|
|
2013-09-27 21:01:46 +00:00
|
|
|
void GrabCursor(bool grab)
|
|
|
|
{
|
|
|
|
SDL.SDL_ShowCursor(grab ? 0 : 1);
|
|
|
|
SDL.SDL_SetWindowGrab(window.Handle,
|
|
|
|
grab ? SDL.SDL_bool.SDL_TRUE : SDL.SDL_bool.SDL_FALSE);
|
|
|
|
SDL.SDL_SetRelativeMouseMode(
|
|
|
|
grab ? SDL.SDL_bool.SDL_TRUE : SDL.SDL_bool.SDL_FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Hack to force WindowState events to be pumped
|
|
|
|
void HideShowWindowHack()
|
|
|
|
{
|
|
|
|
SDL.SDL_HideWindow(window.Handle);
|
2013-10-01 23:14:26 +00:00
|
|
|
ProcessEvents();
|
2013-09-27 21:01:46 +00:00
|
|
|
SDL.SDL_ShowWindow(window.Handle);
|
2013-10-01 23:14:26 +00:00
|
|
|
ProcessEvents();
|
2013-09-27 21:01:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Revert to WindowState.Normal if necessary
|
|
|
|
void RestoreWindow()
|
|
|
|
{
|
|
|
|
WindowState state = WindowState;
|
|
|
|
|
|
|
|
switch (state)
|
|
|
|
{
|
|
|
|
case WindowState.Fullscreen:
|
|
|
|
SDL.SDL_SetWindowFullscreen(window.Handle, 0);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WindowState.Maximized:
|
|
|
|
SDL.SDL_RestoreWindow(window.Handle);
|
|
|
|
HideShowWindowHack();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WindowState.Minimized:
|
|
|
|
SDL.SDL_RestoreWindow(window.Handle);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-10-01 23:14:26 +00:00
|
|
|
ProcessEvents();
|
2013-09-27 21:01:46 +00:00
|
|
|
|
|
|
|
window_state = WindowState.Normal;
|
|
|
|
}
|
|
|
|
|
2013-09-27 12:41:37 +00:00
|
|
|
#endregion
|
|
|
|
|
|
|
|
#region INativeWindow Members
|
|
|
|
|
|
|
|
public event EventHandler<EventArgs> Move = delegate { };
|
|
|
|
public event EventHandler<EventArgs> Resize = delegate { };
|
|
|
|
public event EventHandler<System.ComponentModel.CancelEventArgs> Closing = delegate { };
|
|
|
|
public event EventHandler<EventArgs> Closed = delegate { };
|
|
|
|
public event EventHandler<EventArgs> Disposed = delegate { };
|
|
|
|
public event EventHandler<EventArgs> IconChanged = delegate { };
|
|
|
|
public event EventHandler<EventArgs> TitleChanged = delegate { };
|
|
|
|
public event EventHandler<EventArgs> VisibleChanged = delegate { };
|
|
|
|
public event EventHandler<EventArgs> FocusedChanged = delegate { };
|
|
|
|
public event EventHandler<EventArgs> WindowBorderChanged = delegate { };
|
|
|
|
public event EventHandler<EventArgs> WindowStateChanged = delegate { };
|
|
|
|
public event EventHandler<KeyboardKeyEventArgs> KeyDown = delegate { };
|
|
|
|
public event EventHandler<KeyPressEventArgs> KeyPress = delegate { };
|
|
|
|
public event EventHandler<KeyboardKeyEventArgs> KeyUp = delegate { };
|
|
|
|
public event EventHandler<EventArgs> MouseEnter = delegate { };
|
|
|
|
public event EventHandler<EventArgs> MouseLeave = delegate { };
|
|
|
|
|
|
|
|
public void Close()
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
if (Exists)
|
2013-10-01 23:14:26 +00:00
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
Debug.Print("SDL2 destroying window {0}", window.Handle);
|
|
|
|
|
|
|
|
SDL.SDL_Event e = new SDL.SDL_Event();
|
|
|
|
e.type = SDL.SDL_EventType.SDL_WINDOWEVENT;
|
|
|
|
e.window.windowEvent = SDL.SDL_WindowEventID.SDL_WINDOWEVENT_CLOSE;
|
|
|
|
e.window.windowID = window_id;
|
|
|
|
lock (SDL.Sync)
|
|
|
|
{
|
|
|
|
SDL.SDL_PushEvent(ref e);
|
|
|
|
}
|
2013-10-01 23:14:26 +00:00
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void ProcessEvents()
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
if (Exists)
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
2013-10-01 23:14:26 +00:00
|
|
|
SDL.SDL_PumpEvents();
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public Point PointToClient(Point point)
|
|
|
|
{
|
|
|
|
var origin = DisplayDevice.Default.Bounds.Location;
|
|
|
|
var client = Location;
|
|
|
|
return new Point(point.X + client.X - origin.X, point.Y + client.Y - origin.Y);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Point PointToScreen(Point point)
|
|
|
|
{
|
|
|
|
var origin = DisplayDevice.Default.Bounds.Location;
|
|
|
|
var client = Location;
|
|
|
|
return new Point(point.X + origin.X - client.X, point.Y + origin.Y - client.Y);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Icon Icon
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
return icon;
|
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
if (Exists)
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
// Set the new icon, if any, or clear the current
|
|
|
|
// icon if null
|
|
|
|
if (value != null)
|
|
|
|
{
|
|
|
|
using (Bitmap bmp = value.ToBitmap())
|
|
|
|
{
|
|
|
|
// Decode the icon into a SDL surface
|
|
|
|
var data = bmp.LockBits(
|
|
|
|
new Rectangle(0, 0, value.Width, value.Height),
|
|
|
|
ImageLockMode.ReadWrite,
|
|
|
|
PixelFormat.Format32bppArgb);
|
|
|
|
|
|
|
|
IntPtr surface = SDL.SDL_CreateRGBSurfaceFrom(
|
|
|
|
data.Scan0, data.Width, data.Height, 32, data.Stride,
|
|
|
|
0x00ff0000u, 0x0000ff00u, 0x000000ffu, 0xff000000u);
|
|
|
|
|
|
|
|
// Update the window icon
|
|
|
|
SDL.SDL_SetWindowIcon(window.Handle, surface);
|
|
|
|
|
|
|
|
// Free the SDL surface as it is no longer needed
|
|
|
|
SDL.SDL_FreeSurface(surface);
|
|
|
|
bmp.UnlockBits(data);
|
|
|
|
}
|
2013-09-29 21:36:28 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Clear the window icon
|
|
|
|
SDL.SDL_SetWindowIcon(window.Handle, IntPtr.Zero);
|
|
|
|
}
|
2013-09-29 21:36:28 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
icon = value;
|
|
|
|
IconChanged(this, EventArgs.Empty);
|
|
|
|
}
|
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public string Title
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
|
|
|
{
|
|
|
|
if (Exists)
|
|
|
|
{
|
|
|
|
return SDL.SDL_GetWindowTitle(window.Handle);
|
|
|
|
}
|
|
|
|
return String.Empty;
|
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
|
|
|
{
|
|
|
|
if (Exists)
|
|
|
|
{
|
|
|
|
SDL.SDL_SetWindowTitle(window.Handle, value);
|
|
|
|
window_title = value;
|
|
|
|
}
|
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool Focused
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
return is_focused;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool Visible
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
return is_visible;
|
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
|
|
|
{
|
|
|
|
if (Exists)
|
|
|
|
{
|
|
|
|
if (value)
|
|
|
|
SDL.SDL_ShowWindow(window.Handle);
|
|
|
|
else
|
|
|
|
SDL.SDL_HideWindow(window.Handle);
|
|
|
|
}
|
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool Exists
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
return exists;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public IWindowInfo WindowInfo
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
return window;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public WindowState WindowState
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-09-27 21:01:46 +00:00
|
|
|
return window_state;
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
2013-09-27 21:01:46 +00:00
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
if (Exists)
|
|
|
|
{
|
|
|
|
if (WindowState != value)
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
// Set the new WindowState
|
|
|
|
switch (value)
|
2013-09-27 16:57:05 +00:00
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
case WindowState.Fullscreen:
|
|
|
|
RestoreWindow();
|
|
|
|
if (SDL.SDL_SetWindowFullscreen(window.Handle, (uint)SDL.SDL_WindowFlags.SDL_WINDOW_FULLSCREEN_DESKTOP) < 0)
|
|
|
|
{
|
|
|
|
if (SDL.SDL_SetWindowFullscreen(window.Handle, (uint)SDL.SDL_WindowFlags.SDL_WINDOW_FULLSCREEN) < 0)
|
|
|
|
{
|
|
|
|
Debug.Print("SDL2 failed to enter fullscreen mode: {0}", SDL.SDL_GetError());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
SDL.SDL_RaiseWindow(window.Handle);
|
|
|
|
// There is no "fullscreen" message in the event loop
|
|
|
|
// so we have to mark that ourselves
|
|
|
|
window_state = WindowState.Fullscreen;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WindowState.Maximized:
|
|
|
|
RestoreWindow();
|
|
|
|
SDL.SDL_MaximizeWindow(window.Handle);
|
|
|
|
HideShowWindowHack();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WindowState.Minimized:
|
|
|
|
SDL.SDL_MinimizeWindow(window.Handle);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WindowState.Normal:
|
|
|
|
RestoreWindow();
|
|
|
|
break;
|
2013-09-27 16:57:05 +00:00
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
if (!CursorVisible)
|
|
|
|
GrabCursor(true);
|
|
|
|
}
|
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public WindowBorder WindowBorder
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-09-27 21:01:46 +00:00
|
|
|
return window_border;
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
if (WindowBorder != value)
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
2013-09-27 21:01:46 +00:00
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
if (Exists)
|
|
|
|
{
|
|
|
|
|
|
|
|
switch (value)
|
|
|
|
{
|
|
|
|
case WindowBorder.Resizable:
|
|
|
|
SDL.SDL_SetWindowBordered(window.Handle, SDL.SDL_bool.SDL_TRUE);
|
|
|
|
window_border = WindowBorder.Resizable;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WindowBorder.Hidden:
|
|
|
|
SDL.SDL_SetWindowBordered(window.Handle, SDL.SDL_bool.SDL_FALSE);
|
|
|
|
window_border = WindowBorder.Hidden;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WindowBorder.Fixed:
|
|
|
|
Debug.WriteLine("SDL2 cannot change to fixed-size windows at runtime.");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2013-09-27 21:01:46 +00:00
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
|
2013-10-03 12:59:30 +00:00
|
|
|
if (Exists)
|
|
|
|
{
|
|
|
|
WindowBorderChanged(this, EventArgs.Empty);
|
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public Rectangle Bounds
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
return new Rectangle(Location, Size);
|
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
|
|
|
Size = value.Size;
|
|
|
|
Location = value.Location;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public Point Location
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
|
|
|
{
|
|
|
|
if (Exists)
|
|
|
|
{
|
|
|
|
int x, y;
|
|
|
|
SDL.SDL_GetWindowPosition(window.Handle, out x, out y);
|
|
|
|
return new Point(x, y);
|
|
|
|
}
|
|
|
|
return new Point();
|
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
|
|
|
{
|
|
|
|
if (Exists)
|
|
|
|
{
|
|
|
|
SDL.SDL_SetWindowPosition(window.Handle, value.X, value.Y);
|
|
|
|
}
|
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public Size Size
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
|
|
|
{
|
|
|
|
if (Exists)
|
|
|
|
{
|
|
|
|
int w, h;
|
|
|
|
SDL.SDL_GetWindowSize(window.Handle, out w, out h);
|
|
|
|
return new Size(w, h);
|
|
|
|
}
|
|
|
|
return new Size();
|
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
|
|
|
{
|
|
|
|
if (Exists)
|
|
|
|
{
|
|
|
|
SDL.SDL_SetWindowSize(window.Handle, value.Width, value.Height);
|
|
|
|
}
|
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public int X
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
return Location.X;
|
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
|
|
|
Location = new Point(value, Y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public int Y
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
return Location.Y;
|
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
|
|
|
Location = new Point(X, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public int Width
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-10-03 23:38:19 +00:00
|
|
|
return ClientSize.Width;
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2013-10-03 23:38:19 +00:00
|
|
|
ClientSize = new Size(value, Height);
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public int Height
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-10-03 23:38:19 +00:00
|
|
|
return ClientSize.Height;
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2013-10-03 23:38:19 +00:00
|
|
|
ClientSize = new Size(Width, value);
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public Rectangle ClientRectangle
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-10-03 23:38:19 +00:00
|
|
|
return new Rectangle(new Point(), ClientSize);
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2013-10-03 23:38:19 +00:00
|
|
|
ClientSize = value.Size;
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public Size ClientSize
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-10-03 23:38:19 +00:00
|
|
|
int w, h;
|
2013-10-07 11:17:42 +00:00
|
|
|
if (SDL.Version.Number > 2000)
|
|
|
|
{
|
|
|
|
// SDL > 2.0.0 supports SDL_GL_GetDrawableSize for
|
|
|
|
// hidpi windows.
|
|
|
|
SDL.SDL_GL_GetDrawableSize(window.Handle, out w, out h);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
SDL.SDL_GetWindowSize(window.Handle, out w, out h);
|
|
|
|
}
|
2013-10-03 23:38:19 +00:00
|
|
|
return new Size(w, h);
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2013-10-03 23:38:19 +00:00
|
|
|
float scale = Size.Width / (float)ClientSize.Width;
|
|
|
|
Size = new Size((int)(value.Width * scale), (int)(value.Height * scale));
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public IInputDriver InputDriver
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-10-04 08:02:19 +00:00
|
|
|
return input_driver;
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool CursorVisible
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
return is_cursor_visible;
|
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2013-10-03 12:59:30 +00:00
|
|
|
lock (sync)
|
|
|
|
{
|
|
|
|
if (Exists)
|
|
|
|
{
|
|
|
|
GrabCursor(!value);
|
|
|
|
is_cursor_visible = value;
|
|
|
|
}
|
|
|
|
}
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
#region IInputDriver Members
|
|
|
|
|
|
|
|
public void Poll()
|
|
|
|
{
|
2013-10-04 08:02:19 +00:00
|
|
|
InputDriver.Poll();
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
#region IJoystickDriver Members
|
|
|
|
|
|
|
|
public IList<JoystickDevice> Joysticks
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-10-04 08:02:19 +00:00
|
|
|
return InputDriver.Joysticks;
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
#region IMouseDriver Members
|
|
|
|
|
|
|
|
public IList<MouseDevice> Mouse
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-10-04 08:02:19 +00:00
|
|
|
return InputDriver.Mouse;
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
#region IKeyboardDriver Members
|
|
|
|
|
|
|
|
public IList<KeyboardDevice> Keyboard
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2013-10-04 08:02:19 +00:00
|
|
|
return InputDriver.Keyboard;
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
#region IDisposable implementation
|
|
|
|
|
|
|
|
void Dispose(bool manual)
|
|
|
|
{
|
2013-10-01 23:14:26 +00:00
|
|
|
if (!disposed)
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
2013-10-04 08:02:19 +00:00
|
|
|
lock (sync)
|
2013-09-27 12:41:37 +00:00
|
|
|
{
|
2013-10-04 08:02:19 +00:00
|
|
|
if (manual)
|
2013-09-27 16:57:05 +00:00
|
|
|
{
|
2013-10-04 08:02:19 +00:00
|
|
|
Debug.Print("Disposing {0}", GetType());
|
|
|
|
InputDriver.Dispose();
|
2013-10-03 12:59:30 +00:00
|
|
|
if (Exists)
|
|
|
|
{
|
|
|
|
DestroyWindow();
|
|
|
|
}
|
2013-09-27 16:57:05 +00:00
|
|
|
}
|
2013-10-04 08:02:19 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
Debug.WriteLine("Sdl2NativeWindow leaked, did you forget to call Dispose()?");
|
|
|
|
}
|
|
|
|
disposed = true;
|
2013-09-27 12:41:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Dispose()
|
|
|
|
{
|
|
|
|
Dispose(true);
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
~Sdl2NativeWindow()
|
|
|
|
{
|
|
|
|
Dispose(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|