mirror of
https://github.com/ClassiCube/ClassiCube.git
synced 2025-01-24 01:52:24 -05:00
More work on OpenTK rewrite
This commit is contained in:
parent
35b27cb339
commit
5a6747084b
31 changed files with 721 additions and 1668 deletions
|
@ -251,7 +251,6 @@
|
|||
<Compile Include="Network\Utils\NetWriter.cs" />
|
||||
<Compile Include="Math\PickedPos.cs" />
|
||||
<Compile Include="Math\Picking.cs" />
|
||||
<Compile Include="Platform\DesktopWindow.cs" />
|
||||
<Compile Include="Platform\Font.cs" />
|
||||
<Compile Include="Platform\Platform.cs" />
|
||||
<Compile Include="Program.cs" />
|
||||
|
|
|
@ -155,7 +155,7 @@ namespace ClassicalSharp {
|
|||
}
|
||||
}
|
||||
|
||||
window.LoadIcon();
|
||||
LoadIcon();
|
||||
string connectString = "Connecting to " + IPAddress + ":" + Port + "..";
|
||||
if (Graphics.WarnIfNecessary(Chat)) {
|
||||
MapBordersRenderer.UseLegacyMode(true);
|
||||
|
@ -247,5 +247,19 @@ namespace ClassicalSharp {
|
|||
AddScheduledTask(defTicks, ParticleManager.Tick);
|
||||
AddScheduledTask(defTicks, Animations.Tick);
|
||||
}
|
||||
|
||||
void LoadIcon() {
|
||||
string launcherFile = "Launcher2.exe";
|
||||
if (!Platform.FileExists(launcherFile)) {
|
||||
launcherFile = "Launcher.exe";
|
||||
}
|
||||
if (!Platform.FileExists(launcherFile)) return;
|
||||
|
||||
try {
|
||||
window.Icon = Icon.ExtractAssociatedIcon(launcherFile);
|
||||
} catch (Exception ex) {
|
||||
ErrorHandler.LogError("Game.LoadIcon()", ex);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -207,7 +207,7 @@ namespace ClassicalSharp {
|
|||
}
|
||||
}
|
||||
|
||||
internal DesktopWindow window;
|
||||
internal INativeWindow window;
|
||||
|
||||
public int Width, Height;
|
||||
|
||||
|
@ -219,11 +219,6 @@ namespace ClassicalSharp {
|
|||
return window.PointToScreen(coords);
|
||||
}
|
||||
|
||||
public bool VSync {
|
||||
get { return window.VSync; }
|
||||
set { window.VSync = value; }
|
||||
}
|
||||
|
||||
bool visible = true;
|
||||
internal bool realVisible = true;
|
||||
public bool CursorVisible {
|
||||
|
|
|
@ -19,7 +19,9 @@ using ClassicalSharp.Renderers;
|
|||
using ClassicalSharp.Selections;
|
||||
using ClassicalSharp.Textures;
|
||||
using OpenTK;
|
||||
using OpenTK.Graphics;
|
||||
using OpenTK.Input;
|
||||
using OpenTK.Platform;
|
||||
#if ANDROID
|
||||
using Android.Graphics;
|
||||
#endif
|
||||
|
@ -30,9 +32,9 @@ namespace ClassicalSharp {
|
|||
|
||||
public partial class Game : IDisposable {
|
||||
|
||||
public Game(string username, string mppass, string skinServer,
|
||||
bool nullContext, int width, int height) {
|
||||
window = new DesktopWindow(this, username, nullContext, width, height);
|
||||
public Game(string username, string mppass, string skinServer, int width, int height) {
|
||||
string title = Program.AppName + " (" + username + ")";
|
||||
window = Factory.CreateWindow(width, height, title, GraphicsMode.Default, DisplayDevice.Primary);
|
||||
Username = username;
|
||||
Mppass = mppass;
|
||||
this.skinServer = skinServer;
|
||||
|
@ -56,9 +58,39 @@ namespace ClassicalSharp {
|
|||
return true;
|
||||
}
|
||||
|
||||
public void Run() { window.Run(); }
|
||||
Stopwatch render_watch = new Stopwatch();
|
||||
FrameEventArgs render_args = new FrameEventArgs();
|
||||
public void Run() {
|
||||
window.Visible = true;
|
||||
window.Closed += OnClosed;
|
||||
window.Resize += OnResize;
|
||||
|
||||
public void Exit() { window.Exit(); }
|
||||
OnLoad();
|
||||
OnResize(null, null);
|
||||
Utils.LogDebug("Entering main loop.");
|
||||
render_watch.Start();
|
||||
|
||||
while (true) {
|
||||
window.ProcessEvents();
|
||||
if (window.Exists && !isExiting) {
|
||||
// Cap the maximum time drift to 1 second (e.g. when the process is suspended).
|
||||
double time = render_watch.Elapsed.TotalSeconds;
|
||||
if (time > 1.0) time = 1.0;
|
||||
if (time <= 0) continue;
|
||||
|
||||
render_watch.Reset();
|
||||
render_watch.Start();
|
||||
RenderFrame(time);
|
||||
} else { return; }
|
||||
}
|
||||
}
|
||||
|
||||
bool isExiting;
|
||||
public void Exit() {
|
||||
isExiting = true;
|
||||
// TODO: is isExiting right
|
||||
window.Close();
|
||||
}
|
||||
|
||||
public void SetViewDistance(int distance, bool userDist) {
|
||||
if (userDist) {
|
||||
|
@ -211,7 +243,7 @@ namespace ClassicalSharp {
|
|||
return -1;
|
||||
}
|
||||
|
||||
internal void OnResize() {
|
||||
void OnResize(object sender, EventArgs e) {
|
||||
Size size = window.ClientSize;
|
||||
Width = size.Width; Height = size.Height;
|
||||
|
||||
|
@ -220,6 +252,10 @@ namespace ClassicalSharp {
|
|||
Gui.OnResize();
|
||||
}
|
||||
|
||||
void OnClosed(object sender, EventArgs e) {
|
||||
Dispose();
|
||||
}
|
||||
|
||||
void OnNewMapCore(object sender, EventArgs e) {
|
||||
for (int i = 0; i < Components.Count; i++)
|
||||
Components[i].OnNewMap(this);
|
||||
|
@ -408,6 +444,8 @@ namespace ClassicalSharp {
|
|||
|
||||
Drawer2D.DisposeInstance();
|
||||
Graphics.Dispose();
|
||||
// TODO: is this needed
|
||||
//window.Dispose();
|
||||
|
||||
if (!Options.HasChanged()) return;
|
||||
Options.Load();
|
||||
|
|
|
@ -441,7 +441,6 @@ namespace ClassicalSharp.GraphicsAPI {
|
|||
|
||||
bool vsync = false;
|
||||
public override void SetVSync(Game game, bool value) {
|
||||
game.VSync = value;
|
||||
if (vsync == value) return;
|
||||
vsync = value;
|
||||
|
||||
|
@ -500,10 +499,10 @@ namespace ClassicalSharp.GraphicsAPI {
|
|||
args.BackBufferHeight = height;
|
||||
args.BackBufferFormat = viewFormat;
|
||||
args.BackBufferCount = 1;
|
||||
args.EnableAutoDepthStencil = true;
|
||||
args.EnableAutoDepthStencil = 1;
|
||||
args.PresentationInterval = vsync ? PresentInterval.One : PresentInterval.Immediate;
|
||||
args.SwapEffect = SwapEffect.Discard;
|
||||
args.Windowed = true;
|
||||
args.Windowed = 1;
|
||||
return args;
|
||||
}
|
||||
|
||||
|
|
|
@ -6,7 +6,9 @@ using System.Drawing.Imaging;
|
|||
using System.IO;
|
||||
using System.Runtime.InteropServices;
|
||||
using OpenTK;
|
||||
using OpenTK.Graphics;
|
||||
using OpenTK.Graphics.OpenGL;
|
||||
using OpenTK.Platform;
|
||||
using BmpPixelFormat = System.Drawing.Imaging.PixelFormat;
|
||||
using GlPixelFormat = OpenTK.Graphics.OpenGL.PixelFormat;
|
||||
|
||||
|
@ -21,8 +23,13 @@ namespace ClassicalSharp.GraphicsAPI {
|
|||
const int dynamicListId = 1234567891;
|
||||
IntPtr dynamicListData;
|
||||
#endif
|
||||
IGraphicsContext glContext;
|
||||
|
||||
public OpenGLApi(IWindowInfo window) {
|
||||
glContext = Factory.Default.CreateGLContext(GraphicsMode.Default, window);
|
||||
glContext.MakeCurrent(window);
|
||||
glContext.LoadAll();
|
||||
|
||||
public OpenGLApi() {
|
||||
MinZNear = 0.1f;
|
||||
InitFields();
|
||||
int texDims;
|
||||
|
@ -529,14 +536,14 @@ namespace ClassicalSharp.GraphicsAPI {
|
|||
}
|
||||
|
||||
public override void EndFrame(Game game) {
|
||||
game.window.SwapBuffers();
|
||||
glContext.SwapBuffers();
|
||||
#if GL11
|
||||
activeList = -1;
|
||||
#endif
|
||||
}
|
||||
|
||||
public override void SetVSync(Game game, bool value) {
|
||||
game.VSync = value;
|
||||
glContext.VSync = value;
|
||||
}
|
||||
|
||||
bool isIntelRenderer;
|
||||
|
@ -587,6 +594,12 @@ namespace ClassicalSharp.GraphicsAPI {
|
|||
|
||||
public override void OnWindowResize(Game game) {
|
||||
GL.Viewport(0, 0, game.Width, game.Height);
|
||||
glContext.Update(game.window);
|
||||
}
|
||||
|
||||
public override void Dispose() {
|
||||
base.Dispose();
|
||||
glContext.Dispose();
|
||||
}
|
||||
|
||||
void InitFields() {
|
||||
|
|
|
@ -1,52 +0,0 @@
|
|||
// Copyright 2014-2017 ClassicalSharp | Licensed under BSD-3
|
||||
using System;
|
||||
using System.Drawing;
|
||||
using OpenTK;
|
||||
using OpenTK.Graphics;
|
||||
|
||||
namespace ClassicalSharp {
|
||||
|
||||
/// <summary> Implementation of a native window and native input handling mechanism on Windows, OSX, and Linux. </summary>
|
||||
public sealed class DesktopWindow : GameWindow {
|
||||
|
||||
Game game;
|
||||
public DesktopWindow(Game game, string username, bool nullContext, int width, int height) :
|
||||
base(width, height, GraphicsMode.Default, Program.AppName + " (" + username + ")", nullContext, 0, DisplayDevice.Primary) {
|
||||
this.game = game;
|
||||
}
|
||||
|
||||
protected override void OnLoad(EventArgs e) {
|
||||
game.OnLoad();
|
||||
base.OnLoad(e);
|
||||
}
|
||||
|
||||
public override void Dispose() {
|
||||
game.Dispose();
|
||||
base.Dispose();
|
||||
}
|
||||
|
||||
protected override void OnRenderFrame(FrameEventArgs e) {
|
||||
game.RenderFrame(e.Time);
|
||||
base.OnRenderFrame(e);
|
||||
}
|
||||
|
||||
protected override void OnResize(object sender, EventArgs e) {
|
||||
game.OnResize();
|
||||
base.OnResize(sender, e);
|
||||
}
|
||||
|
||||
public void LoadIcon() {
|
||||
string launcherFile = "Launcher2.exe";
|
||||
if (!Platform.FileExists(launcherFile)) {
|
||||
launcherFile = "Launcher.exe";
|
||||
}
|
||||
if (!Platform.FileExists(launcherFile)) return;
|
||||
|
||||
try {
|
||||
Icon = Icon.ExtractAssociatedIcon(launcherFile);
|
||||
} catch (Exception ex) {
|
||||
ErrorHandler.LogError("DesktopWindow.LoadIcon()", ex);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -38,11 +38,6 @@ namespace ClassicalSharp {
|
|||
OpenTK.Configuration.SkipPerfCountersHack();
|
||||
Utils.LogDebug("Starting " + AppName + "..");
|
||||
|
||||
bool nullContext = true;
|
||||
#if !USE_DX
|
||||
nullContext = false;
|
||||
#endif
|
||||
|
||||
Options.Load();
|
||||
DisplayDevice device = DisplayDevice.Primary;
|
||||
int width = Options.GetInt(OptionsKey.WindowWidth, 0, device.Width, 0);
|
||||
|
@ -57,17 +52,17 @@ namespace ClassicalSharp {
|
|||
if (args.Length == 0 || args.Length == 1) {
|
||||
const string skinServer = "http://static.classicube.net/skins/";
|
||||
string user = args.Length > 0 ? args[0] : "Singleplayer";
|
||||
using (Game game = new Game(user, null, skinServer, nullContext, width, height))
|
||||
using (Game game = new Game(user, null, skinServer, width, height))
|
||||
game.Run();
|
||||
} else if (args.Length < 4) {
|
||||
Utils.LogDebug("ClassicalSharp.exe is only the raw client. You must either use the launcher or"
|
||||
+ " provide command line arguments to start the client.");
|
||||
} else {
|
||||
RunMultiplayer(args, nullContext, width, height);
|
||||
RunMultiplayer(args, width, height);
|
||||
}
|
||||
}
|
||||
|
||||
static void RunMultiplayer(string[] args, bool nullContext, int width, int height) {
|
||||
static void RunMultiplayer(string[] args, int width, int height) {
|
||||
IPAddress ip = null;
|
||||
if (!IPAddress.TryParse(args[2], out ip)) {
|
||||
Utils.LogDebug("Invalid IP \"" + args[2] + '"'); return;
|
||||
|
@ -83,7 +78,7 @@ namespace ClassicalSharp {
|
|||
}
|
||||
|
||||
string skinServer = args.Length >= 5 ? args[4] : "http://static.classicube.net/skins/";
|
||||
using (Game game = new Game(args[0], args[1], skinServer, nullContext, width, height)) {
|
||||
using (Game game = new Game(args[0], args[1], skinServer, width, height)) {
|
||||
game.IPAddress = ip;
|
||||
game.Port = port;
|
||||
game.Run();
|
||||
|
|
|
@ -15,13 +15,14 @@ using Launcher.Web;
|
|||
using OpenTK;
|
||||
using OpenTK.Graphics;
|
||||
using OpenTK.Input;
|
||||
using OpenTK.Platform;
|
||||
|
||||
namespace Launcher {
|
||||
|
||||
public sealed partial class LauncherWindow {
|
||||
|
||||
/// <summary> Underlying native window instance. </summary>
|
||||
public NativeWindow Window;
|
||||
public INativeWindow Window;
|
||||
|
||||
/// <summary> Platform specific class used to draw 2D elements,
|
||||
/// such as text, rounded rectangles and lines. </summary>
|
||||
|
@ -169,7 +170,7 @@ namespace Launcher {
|
|||
}
|
||||
|
||||
public void Run() {
|
||||
Window = new NativeWindow(640, 400, Program.AppName,
|
||||
Window = Factory.CreateWindow(640, 400, Program.AppName,
|
||||
GraphicsMode.Default, DisplayDevice.Primary);
|
||||
Window.Visible = true;
|
||||
Drawer = new GdiPlusDrawer2D();
|
||||
|
|
|
@ -39,227 +39,11 @@ namespace OpenTK
|
|||
{
|
||||
public enum GameWindowFlags { Default = 0 };
|
||||
|
||||
/// <summary>
|
||||
/// The GameWindow class contains cross-platform methods to create and render on an OpenGL
|
||||
/// window, handle input and load resources.
|
||||
/// </summary>
|
||||
/// <remarks>
|
||||
/// GameWindow contains several events you can hook or override to add your custom logic:
|
||||
/// <list>
|
||||
/// <item>
|
||||
/// OnLoad: Occurs after creating the OpenGL context, but before entering the main loop.
|
||||
/// Override to load resources.
|
||||
/// </item>
|
||||
/// <item>
|
||||
/// OnUnload: Occurs after exiting the main loop, but before deleting the OpenGL context.
|
||||
/// Override to unload resources.
|
||||
/// </item>
|
||||
/// <item>
|
||||
/// OnResize: Occurs whenever GameWindow is resized. You should update the OpenGL Viewport
|
||||
/// and Projection Matrix here.
|
||||
/// </item>
|
||||
/// <item>
|
||||
/// OnUpdateFrame: Occurs at the specified logic update rate. Override to add your game
|
||||
/// logic.
|
||||
/// </item>
|
||||
/// <item>
|
||||
/// OnRenderFrame: Occurs at the specified frame render rate. Override to add your
|
||||
/// rendering code.
|
||||
/// </item>
|
||||
/// </list>
|
||||
/// Call the Run() method to start the application's main loop. Run(double, double) takes two
|
||||
/// parameters that
|
||||
/// specify the logic update rate, and the render update rate.
|
||||
/// </remarks>
|
||||
public class GameWindow : NativeWindow, IGameWindow, IDisposable {
|
||||
|
||||
IGraphicsContext glContext;
|
||||
bool isExiting = false;
|
||||
double render_period;
|
||||
// TODO: Implement these:
|
||||
double render_time;
|
||||
bool vsync;
|
||||
|
||||
Stopwatch render_watch = new Stopwatch();
|
||||
double next_render = 0.0;
|
||||
FrameEventArgs render_args = new FrameEventArgs();
|
||||
|
||||
/// <summary>Constructs a new GameWindow with the specified attributes.</summary>
|
||||
/// <param name="width">The width of the GameWindow in pixels.</param>
|
||||
/// <param name="height">The height of the GameWindow in pixels.</param>
|
||||
/// <param name="mode">The OpenTK.Graphics.GraphicsMode of the GameWindow.</param>
|
||||
/// <param name="title">The title of the GameWindow.</param>
|
||||
/// <param name="options">GameWindow options regarding window appearance and behavior.</param>
|
||||
/// <param name="device">The OpenTK.Graphics.DisplayDevice to construct the GameWindow in.</param>
|
||||
public GameWindow(int width, int height, GraphicsMode mode, string title, bool nullContext,
|
||||
GameWindowFlags options, DisplayDevice device) : base(width, height, title, mode, device) {
|
||||
try {
|
||||
glContext = nullContext ? new NullContext() :
|
||||
Factory.Default.CreateGLContext(mode, WindowInfo);
|
||||
glContext.MakeCurrent(WindowInfo);
|
||||
glContext.LoadAll();
|
||||
VSync = true;
|
||||
} catch (Exception e) {
|
||||
Debug.Print(e.ToString());
|
||||
base.Dispose();
|
||||
throw;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Disposes of the GameWindow, releasing all resources consumed by it. </summary>
|
||||
public override void Dispose() {
|
||||
try {
|
||||
Dispose(true);
|
||||
} finally {
|
||||
try {
|
||||
if (glContext != null) {
|
||||
glContext.Dispose();
|
||||
glContext = null;
|
||||
}
|
||||
} finally {
|
||||
base.Dispose();
|
||||
}
|
||||
}
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
/// <summary> Closes the GameWindow. Equivalent to <see cref="NativeWindow.Close"/> method. </summary>
|
||||
/// <remarks> <para>Override if you are not using <see cref="GameWindow.Run()"/>.</para>
|
||||
/// <para>If you override this method, place a call to base.Exit(), to ensure proper OpenTK shutdown.</para> </remarks>
|
||||
public virtual void Exit() {
|
||||
Close();
|
||||
}
|
||||
|
||||
/// <summary> Called when the NativeWindow is about to close. </summary>
|
||||
/// <param name="e">
|
||||
/// The <see cref="System.ComponentModel.CancelEventArgs" /> for this event.
|
||||
/// Set e.Cancel to true in order to stop the GameWindow from closing.</param>
|
||||
protected override void OnClosing(object sender, System.ComponentModel.CancelEventArgs e) {
|
||||
base.OnClosing(sender, e);
|
||||
if (!e.Cancel) {
|
||||
isExiting = true;
|
||||
OnUnload(EventArgs.Empty);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Called after an OpenGL context has been established, but before entering the main loop. </summary>
|
||||
/// <param name="e">Not used.</param>
|
||||
protected virtual void OnLoad(EventArgs e) {
|
||||
if (Load != null) Load(this, e);
|
||||
}
|
||||
|
||||
/// <summary> Called after GameWindow.Exit was called, but before destroying the OpenGL context. </summary>
|
||||
/// <param name="e">Not used.</param>
|
||||
protected virtual void OnUnload(EventArgs e) {
|
||||
if (Unload != null) Unload(this, e);
|
||||
}
|
||||
|
||||
/// <summary> Enters the game loop of the GameWindow updating and rendering at maximum frequency. </summary>
|
||||
/// <remarks> When overriding the default game loop you should call ProcessEvents()
|
||||
/// to ensure that your GameWindow responds to operating system events.
|
||||
/// <para> Once ProcessEvents() returns, it is time to call update and render the next frame. </para> </remarks>
|
||||
/// <param name="frames_per_second">The frequency of RenderFrame events.</param>
|
||||
public void Run() {
|
||||
EnsureUndisposed();
|
||||
try {
|
||||
Visible = true; // Make sure the GameWindow is visible.
|
||||
OnLoad(EventArgs.Empty);
|
||||
OnResize(null, EventArgs.Empty);
|
||||
Debug.Print("Entering main loop.");
|
||||
render_watch.Start();
|
||||
|
||||
while (true) {
|
||||
ProcessEvents();
|
||||
if (Exists && !IsExiting)
|
||||
RaiseRenderFrame(render_watch, ref next_render, render_args);
|
||||
else
|
||||
return;
|
||||
}
|
||||
} finally {
|
||||
if (Exists) {
|
||||
// TODO: Should similar behaviour be retained, possibly on native window level?
|
||||
//while (this.Exists)
|
||||
// ProcessEvents(false);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void RaiseRenderFrame(Stopwatch render_watch, ref double next_render, FrameEventArgs render_args) {
|
||||
// Cap the maximum time drift to 1 second (e.g. when the process is suspended).
|
||||
double time = render_watch.Elapsed.TotalSeconds;
|
||||
if (time > 1.0)
|
||||
time = 1.0;
|
||||
if (time <= 0)
|
||||
return;
|
||||
double time_left = next_render - time;
|
||||
|
||||
if (time_left <= 0.0) {
|
||||
// Schedule next render event. The 1 second cap ensures the process does not appear to hang.
|
||||
next_render = time_left;
|
||||
if (next_render < -1.0)
|
||||
next_render = -1.0;
|
||||
|
||||
render_watch.Reset();
|
||||
render_watch.Start();
|
||||
|
||||
if (time > 0) {
|
||||
render_period = render_args.Time = time;
|
||||
OnRenderFrameInternal(render_args);
|
||||
render_time = render_watch.Elapsed.TotalSeconds;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Swaps the front and back buffer, presenting the rendered scene to the user. </summary>
|
||||
public void SwapBuffers() {
|
||||
EnsureUndisposed();
|
||||
this.Context.SwapBuffers();
|
||||
}
|
||||
|
||||
/// <summary> Returns the opengl IGraphicsContext associated with the current GameWindow. </summary>
|
||||
public IGraphicsContext Context {
|
||||
get { EnsureUndisposed(); return glContext; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a value indicating whether the shutdown sequence has been initiated
|
||||
/// for this window, by calling GameWindow.Exit() or hitting the 'close' button.
|
||||
/// If this property is true, it is no longer safe to use any OpenTK.Input or
|
||||
/// OpenTK.Graphics.OpenGL functions or properties.
|
||||
/// </summary>
|
||||
public bool IsExiting {
|
||||
get { EnsureUndisposed(); return isExiting; }
|
||||
}
|
||||
|
||||
/// <summary> Gets a double representing the actual frequency of RenderFrame events, in hertz (i.e. fps or frames per second). </summary>
|
||||
public double RenderFrequency {
|
||||
get {
|
||||
EnsureUndisposed();
|
||||
if (render_period == 0.0)
|
||||
return 1.0;
|
||||
return 1.0 / render_period;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Gets a double representing the period of RenderFrame events, in seconds. </summary>
|
||||
public double RenderPeriod {
|
||||
get { EnsureUndisposed(); return render_period; }
|
||||
}
|
||||
|
||||
/// <summary> Gets a double representing the time spent in the RenderFrame function, in seconds. </summary>
|
||||
public double RenderTime {
|
||||
get { EnsureUndisposed(); return render_time; }
|
||||
protected set { EnsureUndisposed(); render_time = value; }
|
||||
}
|
||||
|
||||
/// <summary> Gets or sets the VSyncMode. </summary>
|
||||
public bool VSync {
|
||||
get { EnsureUndisposed(); return vsync; }
|
||||
set { EnsureUndisposed(); Context.VSync = (vsync = value); }
|
||||
}
|
||||
public class GameWindow {
|
||||
|
||||
// TODO:
|
||||
/// <summary> Gets or states the state of the NativeWindow. </summary>
|
||||
public override WindowState WindowState {
|
||||
/*public override WindowState WindowState {
|
||||
get { return base.WindowState; }
|
||||
set {
|
||||
base.WindowState = value;
|
||||
|
@ -268,33 +52,6 @@ namespace OpenTK
|
|||
if (Context != null)
|
||||
Context.Update(WindowInfo);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Occurs before the window is displayed for the first time. </summary>
|
||||
public event EventHandler Load;
|
||||
|
||||
/// <summary> Occurs when it is time to render a frame. </summary>
|
||||
public event EventHandler<FrameEventArgs> RenderFrame;
|
||||
|
||||
/// <summary> Occurs before the window is destroyed. </summary>
|
||||
public event EventHandler Unload;
|
||||
|
||||
/// <summary> Override to add custom cleanup logic. </summary>
|
||||
/// <param name="manual">True, if this method was called by the application; false if this was called by the finalizer thread.</param>
|
||||
protected virtual void Dispose(bool manual) { }
|
||||
|
||||
/// <summary> Called when the frame is rendered. </summary>
|
||||
/// <param name="e">Contains information necessary for frame rendering.</param>
|
||||
/// <remarks> Subscribe to the <see cref="RenderFrame"/> event instead of overriding this method. </remarks>
|
||||
protected virtual void OnRenderFrame(FrameEventArgs e) {
|
||||
if (RenderFrame != null) RenderFrame(this, e);
|
||||
}
|
||||
|
||||
protected override void OnResize(object sender, EventArgs e) {
|
||||
base.OnResize(sender, e);
|
||||
glContext.Update(base.WindowInfo);
|
||||
}
|
||||
|
||||
private void OnRenderFrameInternal(FrameEventArgs e) { if (Exists && !isExiting) OnRenderFrame(e); }
|
||||
}*/
|
||||
}
|
||||
}
|
|
@ -1,73 +0,0 @@
|
|||
#region License
|
||||
//
|
||||
// The Open Toolkit Library License
|
||||
//
|
||||
// Copyright (c) 2006 - 2009 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
|
||||
|
||||
using System;
|
||||
using OpenTK.Platform;
|
||||
|
||||
namespace OpenTK.Graphics {
|
||||
|
||||
// Provides the foundation for all IGraphicsContext implementations.
|
||||
abstract class GraphicsContextBase : IGraphicsContext {
|
||||
|
||||
bool disposed;
|
||||
|
||||
protected IntPtr ContextHandle;
|
||||
protected GraphicsMode Mode;
|
||||
|
||||
public abstract void SwapBuffers();
|
||||
|
||||
public abstract void MakeCurrent(IWindowInfo window);
|
||||
|
||||
public abstract bool IsCurrent { get; }
|
||||
|
||||
public bool IsDisposed {
|
||||
get { return disposed; }
|
||||
protected set { disposed = value; }
|
||||
}
|
||||
|
||||
public abstract bool VSync { get; set; }
|
||||
|
||||
public virtual void Update(IWindowInfo window) { }
|
||||
|
||||
public GraphicsMode GraphicsMode { get { return Mode; } }
|
||||
|
||||
public abstract void LoadAll();
|
||||
|
||||
public IntPtr Context { get { return ContextHandle; } }
|
||||
|
||||
public abstract IntPtr GetAddress(string function);
|
||||
|
||||
public void Dispose() {
|
||||
Dispose(true);
|
||||
GC.SuppressFinalize( this );
|
||||
}
|
||||
|
||||
protected abstract void Dispose(bool calledManually);
|
||||
|
||||
~GraphicsContextBase() { Dispose(false); }
|
||||
}
|
||||
}
|
|
@ -1,21 +0,0 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Text;
|
||||
|
||||
namespace OpenTK.Graphics
|
||||
{
|
||||
/// <summary>
|
||||
/// Represents errors related to a GraphicsContext.
|
||||
/// </summary>
|
||||
public class GraphicsContextException : Exception
|
||||
{
|
||||
/// <summary>
|
||||
/// Constructs a new GraphicsContextException.
|
||||
/// </summary>
|
||||
public GraphicsContextException() : base() { }
|
||||
/// <summary>
|
||||
/// Constructs a new GraphicsContextException with the given error message.
|
||||
/// </summary>
|
||||
public GraphicsContextException(string message) : base(message) { }
|
||||
}
|
||||
}
|
|
@ -55,4 +55,9 @@ namespace OpenTK.Graphics {
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
public class GraphicsModeException : Exception {
|
||||
public GraphicsModeException() : base() { }
|
||||
public GraphicsModeException(string message) : base(message) { }
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,14 +0,0 @@
|
|||
using System;
|
||||
|
||||
namespace OpenTK.Graphics {
|
||||
|
||||
/// <summary> Represents errors related to unavailable graphics parameters. </summary>
|
||||
public class GraphicsModeException : Exception {
|
||||
|
||||
/// <summary> Constructs a new GraphicsModeException. </summary>
|
||||
public GraphicsModeException() : base() { }
|
||||
|
||||
/// <summary> Constructs a new GraphicsModeException with the given error message. </summary>
|
||||
public GraphicsModeException(string message) : base(message) { }
|
||||
}
|
||||
}
|
|
@ -12,10 +12,10 @@ using OpenTK.Platform;
|
|||
namespace OpenTK.Graphics {
|
||||
|
||||
/// <summary> Provides methods for creating and interacting with an OpenGL context. </summary>
|
||||
public interface IGraphicsContext : IDisposable {
|
||||
public abstract class IGraphicsContext : IDisposable {
|
||||
|
||||
/// <summary> Swaps buffers, presenting the rendered scene to the user. </summary>
|
||||
void SwapBuffers();
|
||||
public abstract void SwapBuffers();
|
||||
|
||||
/// <summary> Makes the GraphicsContext current in the calling thread. </summary>
|
||||
/// <param name="window">An OpenTK.Platform.IWindowInfo structure that points to a valid window.</param>
|
||||
|
@ -23,36 +23,45 @@ namespace OpenTK.Graphics {
|
|||
/// <para>OpenGL commands in one thread, affect the GraphicsContext which is current in that thread.</para>
|
||||
/// <para>It is an error to issue an OpenGL command in a thread without a current GraphicsContext.</para>
|
||||
/// </remarks>
|
||||
void MakeCurrent(IWindowInfo window);
|
||||
public abstract void MakeCurrent(IWindowInfo window);
|
||||
|
||||
/// <summary> Gets a <see cref="System.Boolean"/> indicating whether this instance is current in the calling thread. </summary>
|
||||
bool IsCurrent { get; }
|
||||
|
||||
/// <summary> Gets a <see cref="System.Boolean"/> indicating whether this instance has been disposed.
|
||||
/// It is an error to access any instance methods if this property returns true. </summary>
|
||||
bool IsDisposed { get; }
|
||||
public abstract bool IsCurrent { get; }
|
||||
|
||||
/// <summary> Gets or sets a value indicating whether VSyncing is enabled. </summary>
|
||||
bool VSync { get; set; }
|
||||
public abstract bool VSync { get; set; }
|
||||
|
||||
/// <summary> Updates the graphics context. This must be called when the region the graphics context
|
||||
/// is drawn to is resized. </summary>
|
||||
/// <param name="window"></param>
|
||||
void Update(IWindowInfo window);
|
||||
public virtual void Update(IWindowInfo window) { }
|
||||
|
||||
/// <summary> Gets the GraphicsMode of this instance. </summary>
|
||||
GraphicsMode GraphicsMode { get; }
|
||||
public GraphicsMode Mode;
|
||||
|
||||
/// <summary> Loads all OpenGL entry points. Requires this instance to be current on the calling thread. </summary>
|
||||
void LoadAll();
|
||||
public abstract void LoadAll();
|
||||
|
||||
/// <summary> Gets a handle to the OpenGL rendering context. </summary>
|
||||
IntPtr Context { get; }
|
||||
/// <summary> Handle to the OpenGL rendering context. </summary>
|
||||
public IntPtr ContextHandle;
|
||||
|
||||
/// <summary> Gets the address of an OpenGL extension function. </summary>
|
||||
/// <param name="function">The name of the OpenGL function (e.g. "glGetString")</param>
|
||||
/// <returns> A pointer to the specified function or IntPtr.Zero if the function isn't
|
||||
/// available in the current opengl context. </returns>
|
||||
IntPtr GetAddress(string function);
|
||||
public abstract IntPtr GetAddress(string function);
|
||||
|
||||
public void Dispose() {
|
||||
Dispose(true);
|
||||
GC.SuppressFinalize( this );
|
||||
}
|
||||
|
||||
protected abstract void Dispose(bool calledManually);
|
||||
|
||||
~IGraphicsContext() { Dispose(false); }
|
||||
}
|
||||
|
||||
public class GraphicsContextException : Exception {
|
||||
public GraphicsContextException() : base() { }
|
||||
public GraphicsContextException(string message) : base(message) { }
|
||||
}
|
||||
}
|
||||
|
|
|
@ -15,12 +15,12 @@ namespace OpenTK.Graphics.OpenGL {
|
|||
|
||||
static GL() { }
|
||||
|
||||
GraphicsContextBase context;
|
||||
IGraphicsContext context;
|
||||
protected override IntPtr GetAddress( string funcname ) {
|
||||
return context.GetAddress( funcname );
|
||||
}
|
||||
|
||||
internal void LoadEntryPoints( GraphicsContextBase context ) {
|
||||
internal void LoadEntryPoints( IGraphicsContext context ) {
|
||||
this.context = context;
|
||||
Debug.Print("Loading OpenGL function pointers... ");
|
||||
AlphaFuncAddress = GetAddress( "glAlphaFunc" );
|
||||
|
|
|
@ -1,50 +0,0 @@
|
|||
#region License
|
||||
//
|
||||
// The Open Toolkit Library License
|
||||
//
|
||||
// Copyright (c) 2006 - 2009 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
|
||||
|
||||
using System;
|
||||
|
||||
namespace OpenTK.Platform {
|
||||
|
||||
/// <summary> Defines the interface for a GameWindow. </summary>
|
||||
public interface IGameWindow : INativeWindow {
|
||||
|
||||
/// <summary> Enters the game loop of the GameWindow using the maximum update rate. </summary>
|
||||
void Run();
|
||||
|
||||
/// <summary> Swaps the front and back buffers of the current GraphicsContext, presenting the rendered scene to the user. </summary>
|
||||
void SwapBuffers();
|
||||
|
||||
/// <summary> Occurs before the window is displayed for the first time. </summary>
|
||||
event EventHandler Load;
|
||||
|
||||
/// <summary> Occurs before the window is destroyed. </summary>
|
||||
event EventHandler Unload;
|
||||
|
||||
/// <summary> Occurs when it is time to render a frame. </summary>
|
||||
event EventHandler<FrameEventArgs> RenderFrame;
|
||||
}
|
||||
}
|
|
@ -34,96 +34,136 @@ using OpenTK.Platform;
|
|||
namespace OpenTK {
|
||||
|
||||
/// <summary> Defines the interface for a native window. </summary>
|
||||
public interface INativeWindow : IDisposable {
|
||||
public abstract class INativeWindow : IDisposable {
|
||||
|
||||
/// <summary> Gets the current contents of the clipboard. </summary>
|
||||
string GetClipboardText();
|
||||
public abstract string GetClipboardText();
|
||||
|
||||
/// <summary> Sets the current contents of the clipboard. </summary>
|
||||
void SetClipboardText( string value );
|
||||
public abstract void SetClipboardText(string value);
|
||||
|
||||
/// <summary> Gets or sets the <see cref="System.Drawing.Icon"/> of the window. </summary>
|
||||
Icon Icon { get; set; }
|
||||
public abstract Icon Icon { get; set; }
|
||||
|
||||
/// <summary> Gets a System.Boolean that indicates whether this window has input focus. </summary>
|
||||
bool Focused { get; }
|
||||
public abstract bool Focused { get; }
|
||||
|
||||
/// <summary> Gets or sets a System.Boolean that indicates whether the window is visible. </summary>
|
||||
bool Visible { get; set; }
|
||||
public abstract bool Visible { get; set; }
|
||||
|
||||
/// <summary> Gets a System.Boolean that indicates whether the window has been created and has not been destroyed. </summary>
|
||||
bool Exists { get; }
|
||||
public abstract bool Exists { get; }
|
||||
|
||||
/// <summary> Gets the <see cref="OpenTK.Platform.IWindowInfo"/> for this window. </summary>
|
||||
IWindowInfo WindowInfo { get; }
|
||||
public abstract IWindowInfo WindowInfo { get; }
|
||||
|
||||
/// <summary> Gets or sets the <see cref="OpenTK.WindowState"/> for this window. </summary>
|
||||
WindowState WindowState { get; set; }
|
||||
public abstract WindowState WindowState { get; set; }
|
||||
|
||||
/// <summary> Gets or sets a <see cref="System.Drawing.Rectangle"/> structure the contains the external bounds of this window, in screen coordinates.
|
||||
/// External bounds include the title bar, borders and drawing area of the window. </summary>
|
||||
Rectangle Bounds { get; set; }
|
||||
public abstract Rectangle Bounds { get; set; }
|
||||
|
||||
/// <summary> Gets or sets a <see cref="System.Drawing.Point"/> structure that contains the location of this window on the desktop. </summary>
|
||||
Point Location { get; set; }
|
||||
public abstract Point Location { get; set; }
|
||||
|
||||
/// <summary> Gets or sets a <see cref="System.Drawing.Size"/> structure that contains the external size of this window. </summary>
|
||||
Size Size { get; set; }
|
||||
public abstract Size Size { get; set; }
|
||||
|
||||
/// <summary> Gets or sets a <see cref="System.Drawing.Rectangle"/> structure that contains the internal bounds of this window, in client coordinates.
|
||||
/// The internal bounds include the drawing area of the window, but exclude the titlebar and window borders. </summary>
|
||||
Rectangle ClientRectangle { get; set; }
|
||||
public abstract Rectangle ClientRectangle { get; set; }
|
||||
|
||||
/// <summary> Gets or sets a <see cref="System.Drawing.Size"/> structure that contains the internal size this window. </summary>
|
||||
Size ClientSize { get; set; }
|
||||
public abstract Size ClientSize { get; set; }
|
||||
|
||||
/// <summary> Closes this window. </summary>
|
||||
void Close();
|
||||
public abstract void Close();
|
||||
|
||||
/// <summary> Processes pending window events. </summary>
|
||||
void ProcessEvents();
|
||||
public abstract void ProcessEvents();
|
||||
|
||||
/// <summary> Transforms the specified point from screen to client coordinates. </summary>
|
||||
/// <param name="point"> A <see cref="System.Drawing.Point"/> to transform. </param>
|
||||
/// <returns> The point transformed to client coordinates. </returns>
|
||||
Point PointToClient(Point point);
|
||||
public abstract Point PointToClient(Point point);
|
||||
|
||||
/// <summary> Transforms the specified point from client to screen coordinates. </summary>
|
||||
/// <param name="point"> A <see cref="System.Drawing.Point"/> to transform. </param>
|
||||
/// <returns> The point transformed to screen coordinates. </returns>
|
||||
Point PointToScreen(Point point);
|
||||
public abstract Point PointToScreen(Point point);
|
||||
/*public virtual Point PointToScreen(Point point) {
|
||||
// Here we use the fact that PointToClient just translates the point, and PointToScreen
|
||||
// should perform the inverse operation.
|
||||
Point trans = PointToClient(Point.Empty);
|
||||
point.X -= trans.X;
|
||||
point.Y -= trans.Y;
|
||||
return point;
|
||||
}*/
|
||||
|
||||
/// <summary> Gets or sets the cursor position in screen coordinates. </summary>
|
||||
Point DesktopCursorPos { get; set; }
|
||||
public abstract Point DesktopCursorPos { get; set; }
|
||||
|
||||
/// <summary> Gets or sets whether the cursor is visible in the window. </summary>
|
||||
bool CursorVisible { get; set; }
|
||||
public abstract bool CursorVisible { get; set; }
|
||||
|
||||
/// <summary> Occurs whenever the window is moved. </summary>
|
||||
event EventHandler Move;
|
||||
public event EventHandler Move;
|
||||
protected void RaiseMove() {
|
||||
if (Move != null) Move(this, EventArgs.Empty);
|
||||
}
|
||||
|
||||
/// <summary> Occurs whenever the window is resized. </summary>
|
||||
event EventHandler Resize;
|
||||
public event EventHandler Resize;
|
||||
protected void RaiseResize() {
|
||||
if (Resize != null) Resize(this, EventArgs.Empty);
|
||||
}
|
||||
|
||||
/// <summary> Occurs when the window is about to close. </summary>
|
||||
event EventHandler<CancelEventArgs> Closing;
|
||||
public event EventHandler Closing;
|
||||
protected void RaiseClosing() {
|
||||
if (Closing != null) Closing(this, EventArgs.Empty);
|
||||
}
|
||||
|
||||
/// <summary> Occurs after the window has closed. </summary>
|
||||
event EventHandler Closed;
|
||||
|
||||
/// <summary> Occurs when the window is disposed. </summary>
|
||||
event EventHandler Disposed;
|
||||
public event EventHandler Closed;
|
||||
protected void RaiseClosed() {
|
||||
if (Closed != null) Closed(this, EventArgs.Empty);
|
||||
}
|
||||
|
||||
/// <summary> Occurs when the <see cref="Visible"/> property of the window changes. </summary>
|
||||
event EventHandler VisibleChanged;
|
||||
public event EventHandler VisibleChanged;
|
||||
protected void RaiseVisibleChanged() {
|
||||
if (VisibleChanged != null) VisibleChanged(this, EventArgs.Empty);
|
||||
}
|
||||
|
||||
/// <summary> Occurs when the <see cref="Focused"/> property of the window changes. </summary>
|
||||
event EventHandler FocusedChanged;
|
||||
public event EventHandler FocusedChanged;
|
||||
protected void RaiseFocusedChanged() {
|
||||
if (FocusedChanged != null) FocusedChanged(this, EventArgs.Empty);
|
||||
}
|
||||
|
||||
/// <summary> Occurs when the <see cref="WindowState"/> property of the window changes. </summary>
|
||||
event EventHandler WindowStateChanged;
|
||||
public event EventHandler WindowStateChanged;
|
||||
protected void RaiseWindowStateChanged() {
|
||||
if (WindowStateChanged != null) WindowStateChanged(this, EventArgs.Empty);
|
||||
}
|
||||
|
||||
/// <summary> Occurs whenever a character is typed. </summary>
|
||||
event EventHandler<KeyPressEventArgs> KeyPress;
|
||||
public event EventHandler<KeyPressEventArgs> KeyPress;
|
||||
KeyPressEventArgs pressArgs = new KeyPressEventArgs();
|
||||
protected void RaiseKeyPress(char key) {
|
||||
pressArgs.KeyChar = key;
|
||||
if (KeyPress != null) KeyPress(this, pressArgs);
|
||||
}
|
||||
|
||||
public void Dispose() {
|
||||
Dispose(true);
|
||||
GC.SuppressFinalize( this );
|
||||
}
|
||||
|
||||
protected abstract void Dispose(bool calledManually);
|
||||
|
||||
~INativeWindow() { Dispose(false); }
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,348 +0,0 @@
|
|||
#region License
|
||||
//
|
||||
// The Open Toolkit Library License
|
||||
//
|
||||
// Copyright (c) 2006 - 2009 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
|
||||
|
||||
using System;
|
||||
using System.ComponentModel;
|
||||
using System.Drawing;
|
||||
using OpenTK.Graphics;
|
||||
using OpenTK.Input;
|
||||
using OpenTK.Platform;
|
||||
|
||||
namespace OpenTK {
|
||||
|
||||
/// <summary> Instances of this class implement the <see cref="OpenTK.INativeWindow"/> interface on the current platform. </summary>
|
||||
public class NativeWindow : INativeWindow {
|
||||
|
||||
private readonly INativeWindow implementation;
|
||||
private bool disposed, events;
|
||||
|
||||
/// <summary>Constructs a new centered NativeWindow with the specified attributes.</summary>
|
||||
/// <param name="width">The width of the NativeWindow in pixels.</param>
|
||||
/// <param name="height">The height of the NativeWindow in pixels.</param>
|
||||
/// <param name="title">The title of the NativeWindow.</param>
|
||||
/// <param name="options">GameWindow options specifying window appearance and behavior.</param>
|
||||
/// <param name="mode">The OpenTK.Graphics.GraphicsMode of the NativeWindow.</param>
|
||||
/// <param name="device">The OpenTK.Graphics.DisplayDevice to construct the NativeWindow in.</param>
|
||||
/// <exception cref="System.ArgumentOutOfRangeException">If width or height is less than 1.</exception>
|
||||
/// <exception cref="System.ArgumentNullException">If mode or device is null.</exception>
|
||||
public NativeWindow(int width, int height, string title, GraphicsMode mode, DisplayDevice device)
|
||||
: this(device.Bounds.Left + (device.Bounds.Width - width) / 2,
|
||||
device.Bounds.Top + (device.Bounds.Height - height) / 2,
|
||||
width, height, title, mode, device) { }
|
||||
|
||||
public NativeWindow(int width, int height, string title, GameWindowFlags flags, GraphicsMode mode, DisplayDevice device) : this(width, height, title, mode, device) {}
|
||||
|
||||
/// <summary>Constructs a new NativeWindow with the specified attributes.</summary>
|
||||
/// <param name="x">Horizontal screen space coordinate of the NativeWindow's origin.</param>
|
||||
/// <param name="y">Vertical screen space coordinate of the NativeWindow's origin.</param>
|
||||
/// <param name="width">The width of the NativeWindow in pixels.</param>
|
||||
/// <param name="height">The height of the NativeWindow in pixels.</param>
|
||||
/// <param name="title">The title of the NativeWindow.</param>
|
||||
/// <param name="options">GameWindow options specifying window appearance and behavior.</param>
|
||||
/// <param name="mode">The OpenTK.Graphics.GraphicsMode of the NativeWindow.</param>
|
||||
/// <param name="device">The OpenTK.Graphics.DisplayDevice to construct the NativeWindow in.</param>
|
||||
/// <exception cref="System.ArgumentOutOfRangeException">If width or height is less than 1.</exception>
|
||||
/// <exception cref="System.ArgumentNullException">If mode or device is null.</exception>
|
||||
public NativeWindow(int x, int y, int width, int height, string title, GraphicsMode mode, DisplayDevice device) {
|
||||
// TODO: Should a constraint be added for the position?
|
||||
if (width < 1)
|
||||
throw new ArgumentOutOfRangeException("width", "Must be greater than zero.");
|
||||
if (height < 1)
|
||||
throw new ArgumentOutOfRangeException("height", "Must be greater than zero.");
|
||||
if (mode == null)
|
||||
throw new ArgumentNullException("mode");
|
||||
if (device == null)
|
||||
throw new ArgumentNullException("device");
|
||||
|
||||
implementation = Factory.Default.CreateNativeWindow(x, y, width, height, title, mode, device);
|
||||
}
|
||||
|
||||
/// <summary> Closes the NativeWindow. </summary>
|
||||
public void Close() {
|
||||
EnsureUndisposed();
|
||||
implementation.Close();
|
||||
}
|
||||
|
||||
/// <summary> Gets the current contents of the clipboard. </summary>
|
||||
public string GetClipboardText() {
|
||||
EnsureUndisposed(); return implementation.GetClipboardText();
|
||||
}
|
||||
|
||||
/// <summary> Sets the current contents of the clipboard. </summary>
|
||||
public void SetClipboardText(string value) {
|
||||
EnsureUndisposed(); implementation.SetClipboardText(value);
|
||||
}
|
||||
|
||||
/// <summary> Transforms the specified point from screen to client coordinates. </summary>
|
||||
/// <param name="point"> A <see cref="System.Drawing.Point"/> to transform. </param>
|
||||
/// <returns> The point transformed to client coordinates. </returns>
|
||||
public Point PointToClient(Point point) {
|
||||
return implementation.PointToClient(point);
|
||||
}
|
||||
|
||||
/// <summary> Transforms the specified point from client to screen coordinates. </summary>
|
||||
/// <param name="point"> A <see cref="System.Drawing.Point"/> to transform. </param>
|
||||
/// <returns> The point transformed to screen coordinates. </returns>
|
||||
public Point PointToScreen(Point point) {
|
||||
// Here we use the fact that PointToClient just translates the point, and PointToScreen
|
||||
// should perform the inverse operation.
|
||||
Point trans = PointToClient(Point.Empty);
|
||||
point.X -= trans.X;
|
||||
point.Y -= trans.Y;
|
||||
return point;
|
||||
}
|
||||
|
||||
/// <summary> Processes operating system events until the NativeWindow becomes idle. </summary>
|
||||
public void ProcessEvents() {
|
||||
ProcessEvents(false);
|
||||
}
|
||||
|
||||
/// <summary> Gets or sets a <see cref="System.Drawing.Rectangle"/> structure that contains the external bounds of this window, in screen coordinates.
|
||||
/// External bounds include the title bar, borders and drawing area of the window. </summary>
|
||||
public Rectangle Bounds {
|
||||
get { EnsureUndisposed(); return implementation.Bounds; }
|
||||
set { EnsureUndisposed(); implementation.Bounds = value; }
|
||||
}
|
||||
|
||||
/// <summary> Gets or sets a <see cref="System.Drawing.Rectangle"/> structure that contains the internal bounds of this window, in client coordinates.
|
||||
/// The internal bounds include the drawing area of the window, but exclude the titlebar and window borders. </summary>
|
||||
public Rectangle ClientRectangle {
|
||||
get { EnsureUndisposed(); return implementation.ClientRectangle; }
|
||||
set { EnsureUndisposed(); implementation.ClientRectangle = value; }
|
||||
}
|
||||
|
||||
/// <summary> Gets or sets a <see cref="System.Drawing.Size"/> structure that contains the internal size this window. </summary>
|
||||
public Size ClientSize {
|
||||
get { EnsureUndisposed(); return implementation.ClientSize; }
|
||||
set { EnsureUndisposed(); implementation.ClientSize = value;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Gets a value indicating whether a render window exists. </summary>
|
||||
public bool Exists {
|
||||
get {
|
||||
return IsDisposed ? false : implementation.Exists; // TODO: Should disposed be ignored instead?
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Gets a System.Boolean that indicates whether this NativeWindow has input focus. </summary>
|
||||
public bool Focused {
|
||||
get { EnsureUndisposed(); return implementation.Focused; }
|
||||
}
|
||||
|
||||
/// <summary> Gets or sets the System.Drawing.Icon for this GameWindow. </summary>
|
||||
public Icon Icon {
|
||||
get { EnsureUndisposed(); return implementation.Icon; }
|
||||
set { EnsureUndisposed(); implementation.Icon = value; }
|
||||
}
|
||||
|
||||
/// <summary> Gets or sets a <see cref="System.Drawing.Point"/> structure that contains the location of this window on the desktop. </summary>
|
||||
public Point Location {
|
||||
get { EnsureUndisposed(); return implementation.Location; }
|
||||
set { EnsureUndisposed(); implementation.Location = value; }
|
||||
}
|
||||
|
||||
/// <summary> Gets or sets a <see cref="System.Drawing.Size"/> structure that contains the external size of this window. </summary>
|
||||
public Size Size {
|
||||
get { EnsureUndisposed(); return implementation.Size; }
|
||||
set { EnsureUndisposed(); implementation.Size = value; }
|
||||
}
|
||||
|
||||
/// <summary> Gets or sets a System.Boolean that indicates whether this NativeWindow is visible. </summary>
|
||||
public bool Visible {
|
||||
get { EnsureUndisposed(); return implementation.Visible; }
|
||||
set { EnsureUndisposed(); implementation.Visible = value; }
|
||||
}
|
||||
|
||||
/// <summary> Gets the <see cref="OpenTK.Platform.IWindowInfo"/> of this window. </summary>
|
||||
public IWindowInfo WindowInfo {
|
||||
get { EnsureUndisposed(); return implementation.WindowInfo; }
|
||||
}
|
||||
|
||||
/// <summary> Gets or states the state of the NativeWindow. </summary>
|
||||
public virtual WindowState WindowState {
|
||||
get { return implementation.WindowState; }
|
||||
set { implementation.WindowState = value; }
|
||||
}
|
||||
|
||||
/// <summary> Sets whether the cursor is visible in the window. </summary>
|
||||
public bool CursorVisible {
|
||||
get { return implementation.CursorVisible; }
|
||||
set { implementation.CursorVisible = value; }
|
||||
}
|
||||
|
||||
/// <summary> Gets or sets the cursor position in screen coordinates. </summary>
|
||||
public Point DesktopCursorPos {
|
||||
get { return implementation.DesktopCursorPos; }
|
||||
set { implementation.DesktopCursorPos = value; }
|
||||
}
|
||||
|
||||
/// <summary> Occurs after the window has closed. </summary>
|
||||
public event EventHandler Closed;
|
||||
|
||||
/// <summary> Occurs when the window is about to close. </summary>
|
||||
public event EventHandler<CancelEventArgs> Closing;
|
||||
|
||||
/// <summary> Occurs when the window is disposed. </summary>
|
||||
public event EventHandler Disposed;
|
||||
|
||||
/// <summary> Occurs when the <see cref="Focused"/> property of the window changes. </summary>
|
||||
public event EventHandler FocusedChanged;
|
||||
|
||||
/// <summary> Occurs whenever a character is typed. </summary>
|
||||
public event EventHandler<KeyPressEventArgs> KeyPress;
|
||||
|
||||
/// <summary> Occurs whenever the window is moved. </summary>
|
||||
public event EventHandler Move;
|
||||
|
||||
/// <summary> Occurs whenever the window is resized. </summary>
|
||||
public event EventHandler Resize;
|
||||
|
||||
/// <summary> Occurs when the <see cref="Visible"/> property of the window changes. </summary>
|
||||
public event EventHandler VisibleChanged;
|
||||
|
||||
/// <summary> Occurs when the <see cref="WindowState"/> property of the window changes. </summary>
|
||||
public event EventHandler WindowStateChanged;
|
||||
|
||||
/// <summary> Releases all non-managed resources belonging to this NativeWindow. </summary>
|
||||
public virtual void Dispose() {
|
||||
if (!IsDisposed) {
|
||||
implementation.Dispose();
|
||||
GC.SuppressFinalize(this);
|
||||
IsDisposed = true;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Ensures that this NativeWindow has not been disposed. </summary>
|
||||
/// <exception cref="System.ObjectDisposedException"> If this NativeWindow has been disposed. </exception>
|
||||
protected void EnsureUndisposed() {
|
||||
if (disposed) throw new ObjectDisposedException(GetType().Name);
|
||||
}
|
||||
|
||||
/// <summary> Gets or sets a <see cref="System.Boolean"/>, which indicates whether this instance has been disposed. </summary>
|
||||
protected bool IsDisposed {
|
||||
get { return disposed; }
|
||||
set { disposed = value; }
|
||||
}
|
||||
|
||||
/// <summary> Called when the NativeWindow has closed. </summary>
|
||||
/// <param name="e">Not used.</param>
|
||||
protected virtual void OnClosed(object sender, EventArgs e) {
|
||||
if (Closed != null) Closed(this, e);
|
||||
}
|
||||
|
||||
/// <summary> Called when the NativeWindow is about to close. </summary>
|
||||
/// <param name="e"> The <see cref="System.ComponentModel.CancelEventArgs" /> for this event.
|
||||
/// Set e.Cancel to true in order to stop the NativeWindow from closing.</param>
|
||||
protected virtual void OnClosing(object sender, CancelEventArgs e) {
|
||||
if (Closing != null) Closing(this, e);
|
||||
}
|
||||
|
||||
/// <summary> Called when the NativeWindow is disposed. </summary>
|
||||
protected virtual void OnDisposed(object sender, EventArgs e) {
|
||||
if (Disposed != null) Disposed(this, e);
|
||||
}
|
||||
|
||||
/// <summary> Called when the <see cref="OpenTK.INativeWindow.Focused"/> property of the NativeWindow has changed. </summary>
|
||||
protected virtual void OnFocusedChanged(object sender, EventArgs e) {
|
||||
if (FocusedChanged != null) FocusedChanged(this, e);
|
||||
}
|
||||
|
||||
/// <summary> Called when a character is typed. </summary>
|
||||
/// <param name="e">The <see cref="OpenTK.KeyPressEventArgs"/> for this event.</param>
|
||||
protected virtual void OnKeyPress(object sender, KeyPressEventArgs e) {
|
||||
if (KeyPress != null) KeyPress(this, e);
|
||||
}
|
||||
|
||||
/// <summary> Called when the NativeWindow is moved. </summary>
|
||||
protected virtual void OnMove(object sender, EventArgs e) {
|
||||
if (Move != null) Move(this, e);
|
||||
}
|
||||
|
||||
/// <summary> Called when the NativeWindow is resized. </summary>
|
||||
/// <param name="e">Not used.</param>
|
||||
protected virtual void OnResize(object sender, EventArgs e) {
|
||||
if (Resize != null) Resize(this, e);
|
||||
}
|
||||
|
||||
/// <summary> Called when the <see cref="OpenTK.INativeWindow.Visible"/> property of the NativeWindow has changed. </summary>
|
||||
protected virtual void OnVisibleChanged(object sender, EventArgs e) {
|
||||
if (VisibleChanged != null) VisibleChanged(this, e);
|
||||
}
|
||||
|
||||
/// <summary> Called when the WindowState of this NativeWindow has changed. </summary>
|
||||
protected virtual void OnWindowStateChanged(object sender, EventArgs e) {
|
||||
if (WindowStateChanged != null) WindowStateChanged(this, e);
|
||||
}
|
||||
|
||||
/// <summary> Processes operating system events until the NativeWindow becomes idle. </summary>
|
||||
/// <param name="retainEvents">If true, the state of underlying system event propagation will be preserved, otherwise event propagation will be enabled if it has not been already.</param>
|
||||
protected void ProcessEvents(bool retainEvents) {
|
||||
EnsureUndisposed();
|
||||
if (!retainEvents && !events) Events = true;
|
||||
implementation.ProcessEvents();
|
||||
}
|
||||
|
||||
private void OnClosedInternal(object sender, EventArgs e) {
|
||||
OnClosed(null, e);
|
||||
Events = false;
|
||||
}
|
||||
|
||||
private bool Events {
|
||||
set {
|
||||
if (value) {
|
||||
if (events) {
|
||||
throw new InvalidOperationException("Event propagation is already enabled.");
|
||||
}
|
||||
implementation.Closed += OnClosed;
|
||||
implementation.Closing += OnClosing;
|
||||
implementation.Disposed += OnDisposed;
|
||||
implementation.FocusedChanged += OnFocusedChanged;
|
||||
implementation.KeyPress += OnKeyPress;
|
||||
implementation.Move += OnMove;
|
||||
implementation.Resize += OnResize;
|
||||
implementation.VisibleChanged += OnVisibleChanged;
|
||||
implementation.WindowStateChanged += OnWindowStateChanged;
|
||||
events = true;
|
||||
} else if (events) {
|
||||
implementation.Closed -= OnClosed;
|
||||
implementation.Closing -= OnClosing;
|
||||
implementation.Disposed -= OnDisposed;
|
||||
implementation.FocusedChanged -= OnFocusedChanged;
|
||||
implementation.KeyPress -= OnKeyPress;
|
||||
implementation.Move -= OnMove;
|
||||
implementation.Resize -= OnResize;
|
||||
implementation.VisibleChanged -= OnVisibleChanged;
|
||||
implementation.WindowStateChanged -= OnWindowStateChanged;
|
||||
events = false;
|
||||
} else {
|
||||
throw new InvalidOperationException("Event propagation is already disabled.");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -55,17 +55,12 @@
|
|||
<Compile Include="FrameEventArgs.cs" />
|
||||
<Compile Include="GameWindow.cs" />
|
||||
<Compile Include="Graphics\OpenGL\GL.cs" />
|
||||
<Compile Include="IGameWindow.cs" />
|
||||
<Compile Include="INativeWindow.cs" />
|
||||
<Compile Include="Interop.cs" />
|
||||
<Compile Include="KeyPressEventArgs.cs" />
|
||||
<Compile Include="MemUtils.cs" />
|
||||
<Compile Include="NativeWindow.cs" />
|
||||
<Compile Include="Graphics\ColorFormat.cs" />
|
||||
<Compile Include="Graphics\GraphicsContextBase.cs" />
|
||||
<Compile Include="Graphics\GraphicsContextException.cs" />
|
||||
<Compile Include="Graphics\GraphicsMode.cs" />
|
||||
<Compile Include="Graphics\GraphicsModeException.cs" />
|
||||
<Compile Include="Graphics\IGraphicsContext.cs" />
|
||||
<Compile Include="Graphics\OpenGL\GLEnums.cs" />
|
||||
<Compile Include="Graphics\OpenGL\GLHelper.cs" />
|
||||
|
@ -100,7 +95,6 @@
|
|||
<Compile Include="Platform\Windows\WinWindow.cs" />
|
||||
<Compile Include="Platform\Windows\WinKeyMap.cs" />
|
||||
<Compile Include="Platform\Windows\WinWindowInfo.cs" />
|
||||
<Compile Include="Platform\NullContext.cs" />
|
||||
<Compile Include="Platform\X11\API.cs" />
|
||||
<Compile Include="Platform\X11\Glx.cs" />
|
||||
<Compile Include="Platform\X11\Structs.cs" />
|
||||
|
@ -117,7 +111,6 @@
|
|||
<Compile Include="SharpDX.Direct3D\Resources.cs" />
|
||||
<Compile Include="SharpDX.Direct3D\Structures.cs" />
|
||||
<Compile Include="SharpDX\ComObject.cs" />
|
||||
<Compile Include="SharpDX\Raw.cs" />
|
||||
<Compile Include="SharpDX\Result.cs" />
|
||||
<Compile Include="WindowState.cs" />
|
||||
</ItemGroup>
|
||||
|
|
|
@ -30,17 +30,23 @@ using OpenTK.Graphics;
|
|||
|
||||
namespace OpenTK.Platform {
|
||||
|
||||
interface IPlatformFactory {
|
||||
INativeWindow CreateNativeWindow(int x, int y, int width, int height, string title, GraphicsMode mode, DisplayDevice device);
|
||||
public interface IPlatformFactory {
|
||||
INativeWindow CreateWindow(int x, int y, int width, int height, string title, GraphicsMode mode, DisplayDevice device);
|
||||
|
||||
void InitDisplayDeviceDriver();
|
||||
|
||||
IGraphicsContext CreateGLContext(GraphicsMode mode, IWindowInfo window);
|
||||
}
|
||||
|
||||
internal static class Factory {
|
||||
public static class Factory {
|
||||
public static readonly IPlatformFactory Default;
|
||||
|
||||
public static INativeWindow CreateWindow(int width, int height, string title, GraphicsMode mode, DisplayDevice device) {
|
||||
int x = device.Bounds.Left + (device.Bounds.Width - width) / 2;
|
||||
int y = device.Bounds.Top + (device.Bounds.Height - height) / 2;
|
||||
return Default.CreateWindow(x, y, width, height, title, mode, device);
|
||||
}
|
||||
|
||||
static Factory() {
|
||||
if (Configuration.RunningOnWindows) Default = new Windows.WinFactory();
|
||||
else if (Configuration.RunningOnMacOS) Default = new MacOS.MacOSFactory();
|
||||
|
@ -53,7 +59,7 @@ namespace OpenTK.Platform {
|
|||
namespace OpenTK.Platform.MacOS {
|
||||
class MacOSFactory : IPlatformFactory {
|
||||
|
||||
public INativeWindow CreateNativeWindow(int x, int y, int width, int height, string title, GraphicsMode mode, DisplayDevice device) {
|
||||
public INativeWindow CreateWindow(int x, int y, int width, int height, string title, GraphicsMode mode, DisplayDevice device) {
|
||||
return new CarbonWindow(x, y, width, height, title, device);
|
||||
}
|
||||
|
||||
|
@ -70,7 +76,7 @@ namespace OpenTK.Platform.MacOS {
|
|||
namespace OpenTK.Platform.Windows {
|
||||
class WinFactory : IPlatformFactory {
|
||||
|
||||
public INativeWindow CreateNativeWindow(int x, int y, int width, int height, string title, GraphicsMode mode, DisplayDevice device) {
|
||||
public INativeWindow CreateWindow(int x, int y, int width, int height, string title, GraphicsMode mode, DisplayDevice device) {
|
||||
return new WinWindow(x, y, width, height, title, device);
|
||||
}
|
||||
|
||||
|
@ -87,7 +93,7 @@ namespace OpenTK.Platform.Windows {
|
|||
namespace OpenTK.Platform.X11 {
|
||||
class X11Factory : IPlatformFactory {
|
||||
|
||||
public INativeWindow CreateNativeWindow(int x, int y, int width, int height, string title, GraphicsMode mode, DisplayDevice device) {
|
||||
public INativeWindow CreateWindow(int x, int y, int width, int height, string title, GraphicsMode mode, DisplayDevice device) {
|
||||
return new X11Window(x, y, width, height, title, mode, device);
|
||||
}
|
||||
|
||||
|
|
|
@ -10,7 +10,7 @@ using AGLPixelFormat = System.IntPtr;
|
|||
|
||||
namespace OpenTK.Platform.MacOS {
|
||||
|
||||
class AglContext : GraphicsContextBase {
|
||||
class AglContext : IGraphicsContext {
|
||||
|
||||
bool mVSync = false;
|
||||
// Todo: keep track of which display adapter was specified when the context was created.
|
||||
|
@ -253,10 +253,9 @@ namespace OpenTK.Platform.MacOS {
|
|||
|
||||
#region IDisposable Members
|
||||
|
||||
protected override void Dispose(bool disposing)
|
||||
{
|
||||
if (IsDisposed || ContextHandle == IntPtr.Zero)
|
||||
return;
|
||||
bool IsDisposed;
|
||||
protected override void Dispose(bool disposing) {
|
||||
if (IsDisposed || ContextHandle == IntPtr.Zero) return;
|
||||
|
||||
Debug.Print("Disposing of AGL context.");
|
||||
Agl.aglSetCurrentContext(IntPtr.Zero);
|
||||
|
|
|
@ -54,7 +54,6 @@ namespace OpenTK.Platform.MacOS
|
|||
int mTitlebarHeight;
|
||||
WindowState windowState = WindowState.Normal;
|
||||
internal static Dictionary<IntPtr, WeakReference> WindowRefs = new Dictionary<IntPtr, WeakReference>();
|
||||
KeyPressEventArgs mKeyPressArgs = new KeyPressEventArgs();
|
||||
bool mIsActive = false;
|
||||
Icon mIcon;
|
||||
|
||||
|
@ -73,17 +72,8 @@ namespace OpenTK.Platform.MacOS
|
|||
TargetDisplayDevice = device;
|
||||
}
|
||||
|
||||
#region IDisposable
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
Dispose(true);
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
protected virtual void Dispose(bool disposing) {
|
||||
if (mIsDisposed)
|
||||
return;
|
||||
protected override void Dispose(bool disposing) {
|
||||
if (mIsDisposed) return;
|
||||
|
||||
Debug.Print("Disposing of CarbonGLNative window.");
|
||||
API.DisposeWindow(window.WindowRef);
|
||||
|
@ -97,12 +87,6 @@ namespace OpenTK.Platform.MacOS
|
|||
DisposeUPP();
|
||||
}
|
||||
|
||||
~CarbonWindow() {
|
||||
Dispose(false);
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region Private Members
|
||||
|
||||
void DisposeUPP()
|
||||
|
@ -260,7 +244,6 @@ namespace OpenTK.Platform.MacOS
|
|||
case KeyboardEventKind.RawKeyRepeat:
|
||||
case KeyboardEventKind.RawKeyUp:
|
||||
GetCharCodes(inEvent, out code, out charCode);
|
||||
mKeyPressArgs.KeyChar = charCode;
|
||||
break;
|
||||
}
|
||||
|
||||
|
@ -278,7 +261,7 @@ namespace OpenTK.Platform.MacOS
|
|||
|
||||
case KeyboardEventKind.RawKeyDown:
|
||||
Keyboard.Set(tkKey, true);
|
||||
OnKeyPress(mKeyPressArgs);
|
||||
RaiseKeyPress(charCode);
|
||||
return OSStatus.NoError;
|
||||
|
||||
case KeyboardEventKind.RawKeyUp:
|
||||
|
@ -298,24 +281,17 @@ namespace OpenTK.Platform.MacOS
|
|||
switch ((WindowEventKind)evt.EventKind)
|
||||
{
|
||||
case WindowEventKind.WindowClose:
|
||||
CancelEventArgs cancel = new CancelEventArgs();
|
||||
OnClosing(cancel);
|
||||
|
||||
if (cancel.Cancel)
|
||||
return OSStatus.NoError;
|
||||
else
|
||||
RaiseClosing();
|
||||
return OSStatus.EventNotHandled;
|
||||
|
||||
case WindowEventKind.WindowClosed:
|
||||
mExists = false;
|
||||
OnClosed();
|
||||
|
||||
RaiseClosed();
|
||||
return OSStatus.NoError;
|
||||
|
||||
case WindowEventKind.WindowBoundsChanged:
|
||||
int thisWidth = ClientRectangle.Width;
|
||||
int thisHeight = ClientRectangle.Height;
|
||||
|
||||
LoadSize();
|
||||
|
||||
if (thisWidth != ClientRectangle.Width || thisHeight != ClientRectangle.Height)
|
||||
|
@ -501,9 +477,7 @@ namespace OpenTK.Platform.MacOS
|
|||
|
||||
protected void OnResize() {
|
||||
LoadSize();
|
||||
if (Resize != null) {
|
||||
Resize(this, EventArgs.Empty);
|
||||
}
|
||||
RaiseResize();
|
||||
}
|
||||
|
||||
private void LoadSize() {
|
||||
|
@ -522,7 +496,7 @@ namespace OpenTK.Platform.MacOS
|
|||
#region INativeWindow Members
|
||||
|
||||
IntPtr pbStr, utf16, utf8;
|
||||
public string GetClipboardText() {
|
||||
public override string GetClipboardText() {
|
||||
IntPtr pbRef = GetPasteboard();
|
||||
API.PasteboardSynchronize( pbRef );
|
||||
|
||||
|
@ -564,7 +538,7 @@ namespace OpenTK.Platform.MacOS
|
|||
return Encoding.UTF8.GetString( text );
|
||||
}
|
||||
|
||||
public void SetClipboardText( string value ) {
|
||||
public override void SetClipboardText( string value ) {
|
||||
IntPtr pbRef = GetPasteboard();
|
||||
OSStatus err = API.PasteboardClear( pbRef );
|
||||
if( err != OSStatus.NoError )
|
||||
|
@ -595,31 +569,31 @@ namespace OpenTK.Platform.MacOS
|
|||
return pbRef;
|
||||
}
|
||||
|
||||
public void ProcessEvents() {
|
||||
public override void ProcessEvents() {
|
||||
API.ProcessEvents();
|
||||
}
|
||||
|
||||
public Point PointToClient(Point point) {
|
||||
public override Point PointToClient(Point point) {
|
||||
IntPtr handle = window.WindowRef;
|
||||
Rect r = Carbon.API.GetWindowBounds(window.WindowRef, WindowRegionCode.ContentRegion);
|
||||
return new Point(point.X - r.X, point.Y - r.Y);
|
||||
}
|
||||
|
||||
public Point PointToScreen(Point point) {
|
||||
public override Point PointToScreen(Point point) {
|
||||
IntPtr handle = window.WindowRef;
|
||||
Rect r = Carbon.API.GetWindowBounds(window.WindowRef, WindowRegionCode.ContentRegion);
|
||||
return new Point(point.X + r.X, point.Y + r.Y);
|
||||
}
|
||||
|
||||
public bool Exists {
|
||||
public override bool Exists {
|
||||
get { return mExists; }
|
||||
}
|
||||
|
||||
public IWindowInfo WindowInfo {
|
||||
public override IWindowInfo WindowInfo {
|
||||
get { return window; }
|
||||
}
|
||||
|
||||
public Icon Icon {
|
||||
public override Icon Icon {
|
||||
get { return mIcon; }
|
||||
set { SetIcon(value); }
|
||||
}
|
||||
|
@ -677,7 +651,7 @@ namespace OpenTK.Platform.MacOS
|
|||
}
|
||||
}
|
||||
|
||||
public bool Visible {
|
||||
public override bool Visible {
|
||||
get { return API.IsWindowVisible(window.WindowRef); }
|
||||
set {
|
||||
if (value && Visible == false)
|
||||
|
@ -687,11 +661,11 @@ namespace OpenTK.Platform.MacOS
|
|||
}
|
||||
}
|
||||
|
||||
public bool Focused {
|
||||
public override bool Focused {
|
||||
get { return this.mIsActive; }
|
||||
}
|
||||
|
||||
public Rectangle Bounds {
|
||||
public override Rectangle Bounds {
|
||||
get { return bounds; }
|
||||
set {
|
||||
Location = value.Location;
|
||||
|
@ -699,17 +673,17 @@ namespace OpenTK.Platform.MacOS
|
|||
}
|
||||
}
|
||||
|
||||
public Point Location {
|
||||
public override Point Location {
|
||||
get { return Bounds.Location; }
|
||||
set { SetLocation((short)value.X, (short)value.Y); }
|
||||
}
|
||||
|
||||
public Size Size {
|
||||
public override Size Size {
|
||||
get { return bounds.Size; }
|
||||
set { SetSize((short)value.Width, (short)value.Height); }
|
||||
}
|
||||
|
||||
public Rectangle ClientRectangle {
|
||||
public override Rectangle ClientRectangle {
|
||||
get { return clientRectangle; }
|
||||
set {
|
||||
// just set the size, and ignore the location value.
|
||||
|
@ -718,7 +692,7 @@ namespace OpenTK.Platform.MacOS
|
|||
}
|
||||
}
|
||||
|
||||
public Size ClientSize {
|
||||
public override Size ClientSize {
|
||||
get { return clientRectangle.Size; }
|
||||
set {
|
||||
API.SizeWindow(window.WindowRef, (short)value.Width, (short)value.Height, true);
|
||||
|
@ -726,36 +700,28 @@ namespace OpenTK.Platform.MacOS
|
|||
}
|
||||
}
|
||||
|
||||
public void Close() {
|
||||
CancelEventArgs e = new CancelEventArgs();
|
||||
OnClosing(e);
|
||||
if (e.Cancel) return;
|
||||
|
||||
OnClosed();
|
||||
public override void Close() {
|
||||
RaiseClosed();
|
||||
Dispose();
|
||||
}
|
||||
|
||||
public WindowState WindowState {
|
||||
get
|
||||
{
|
||||
public override WindowState WindowState {
|
||||
get {
|
||||
if (windowState == WindowState.Fullscreen)
|
||||
return WindowState.Fullscreen;
|
||||
|
||||
if (Carbon.API.IsWindowCollapsed(window.WindowRef))
|
||||
return WindowState.Minimized;
|
||||
|
||||
if (Carbon.API.IsWindowInStandardState(window.WindowRef))
|
||||
{
|
||||
if (Carbon.API.IsWindowInStandardState(window.WindowRef)) {
|
||||
return WindowState.Maximized;
|
||||
}
|
||||
|
||||
return WindowState.Normal;
|
||||
}
|
||||
set
|
||||
{
|
||||
if (value == WindowState)
|
||||
return;
|
||||
|
||||
set {
|
||||
if (value == WindowState) return;
|
||||
Debug.Print("Switching window state from {0} to {1}", WindowState, value);
|
||||
WindowState oldState = WindowState;
|
||||
|
||||
|
@ -813,71 +779,29 @@ namespace OpenTK.Platform.MacOS
|
|||
break;
|
||||
}
|
||||
|
||||
OnWindowStateChanged();
|
||||
RaiseWindowStateChanged();
|
||||
OnResize();
|
||||
}
|
||||
|
||||
#region --- Event wrappers ---
|
||||
|
||||
private void OnKeyPress(KeyPressEventArgs keyPressArgs)
|
||||
{
|
||||
if (KeyPress != null)
|
||||
KeyPress(this, keyPressArgs);
|
||||
}
|
||||
|
||||
|
||||
private void OnWindowStateChanged()
|
||||
{
|
||||
if (WindowStateChanged != null)
|
||||
WindowStateChanged(this, EventArgs.Empty);
|
||||
}
|
||||
|
||||
protected virtual void OnClosing(CancelEventArgs e)
|
||||
{
|
||||
if (Closing != null)
|
||||
Closing(this, e);
|
||||
}
|
||||
|
||||
protected virtual void OnClosed()
|
||||
{
|
||||
if (Closed != null)
|
||||
Closed(this, EventArgs.Empty);
|
||||
}
|
||||
|
||||
|
||||
private void OnActivate()
|
||||
{
|
||||
private void OnActivate() {
|
||||
mIsActive = true;
|
||||
if (FocusedChanged != null)
|
||||
FocusedChanged(this, EventArgs.Empty);
|
||||
RaiseFocusedChanged();
|
||||
}
|
||||
private void OnDeactivate()
|
||||
{
|
||||
|
||||
private void OnDeactivate() {
|
||||
mIsActive = false;
|
||||
if (FocusedChanged != null)
|
||||
FocusedChanged(this, EventArgs.Empty);
|
||||
RaiseFocusedChanged();
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
public event EventHandler Load;
|
||||
public event EventHandler Unload;
|
||||
public event EventHandler Move;
|
||||
public event EventHandler Resize;
|
||||
public event EventHandler<CancelEventArgs> Closing;
|
||||
public event EventHandler Closed;
|
||||
public event EventHandler Disposed;
|
||||
public event EventHandler ClientSizeChanged;
|
||||
public event EventHandler VisibleChanged;
|
||||
public event EventHandler FocusedChanged;
|
||||
public event EventHandler WindowStateChanged;
|
||||
public event EventHandler<KeyPressEventArgs> KeyPress;
|
||||
|
||||
#endregion
|
||||
|
||||
#region IInputDriver Members
|
||||
|
||||
public Point DesktopCursorPos {
|
||||
public override Point DesktopCursorPos {
|
||||
get {
|
||||
HIPoint point = default( HIPoint );
|
||||
// NOTE: HIGetMousePosition is only available on OSX 10.5 or later
|
||||
|
@ -894,7 +818,7 @@ namespace OpenTK.Platform.MacOS
|
|||
}
|
||||
|
||||
bool visible = true;
|
||||
public bool CursorVisible {
|
||||
public override bool CursorVisible {
|
||||
get { return visible; }
|
||||
set {
|
||||
visible = value;
|
||||
|
|
|
@ -1,41 +0,0 @@
|
|||
using System;
|
||||
using OpenTK.Graphics;
|
||||
|
||||
namespace OpenTK.Platform {
|
||||
|
||||
internal sealed class NullContext : GraphicsContextBase {
|
||||
|
||||
public NullContext() {
|
||||
Debug.Print( "doing nothing" );
|
||||
}
|
||||
|
||||
public override void SwapBuffers() {
|
||||
}
|
||||
|
||||
public override void MakeCurrent( IWindowInfo window ) {
|
||||
}
|
||||
|
||||
public override bool IsCurrent {
|
||||
get { return true; }
|
||||
}
|
||||
|
||||
public override bool VSync {
|
||||
get { return true; }
|
||||
set { }
|
||||
}
|
||||
|
||||
public override void LoadAll() {
|
||||
}
|
||||
|
||||
public override IntPtr GetAddress(string funcName) {
|
||||
return IntPtr.Zero;
|
||||
}
|
||||
|
||||
public override string ToString() {
|
||||
return "(empty)";
|
||||
}
|
||||
|
||||
protected override void Dispose(bool calledManually) {
|
||||
}
|
||||
}
|
||||
}
|
|
@ -101,6 +101,10 @@ namespace OpenTK.Platform.Windows {
|
|||
//internal static extern bool ScreenToClient(IntPtr hWnd, ref POINT point);
|
||||
internal static extern bool ScreenToClient(IntPtr hWnd, ref Point point);
|
||||
|
||||
[DllImport("user32.dll", SetLastError = true), SuppressUnmanagedCodeSecurity]
|
||||
//internal static extern bool ClientToScreen(IntPtr hWnd, ref POINT point);
|
||||
internal static extern bool ClientToScreen(IntPtr hWnd, ref Point point);
|
||||
|
||||
[DllImport("user32.dll", SetLastError = true), SuppressUnmanagedCodeSecurity]
|
||||
internal extern static bool GetClientRect(IntPtr windowHandle, out Win32Rectangle clientRectangle);
|
||||
|
||||
|
|
|
@ -16,7 +16,7 @@ namespace OpenTK.Platform.Windows {
|
|||
/// Provides methods to create and control an opengl context on the Windows platform.
|
||||
/// This class supports OpenTK, and is not intended for use by OpenTK programs.
|
||||
/// </summary>
|
||||
internal sealed class WinGLContext : GraphicsContextBase
|
||||
internal sealed class WinGLContext : IGraphicsContext
|
||||
{
|
||||
static IntPtr opengl32Handle;
|
||||
const string opengl32Name = "OPENGL32.DLL";
|
||||
|
@ -162,28 +162,13 @@ namespace OpenTK.Platform.Windows {
|
|||
}
|
||||
|
||||
protected override void Dispose(bool calledManually) {
|
||||
if (IsDisposed) return;
|
||||
|
||||
if (calledManually) {
|
||||
DestroyContext();
|
||||
} else {
|
||||
Debug.Print("[Warning] OpenGL context {0} leaked. Did you forget to call IGraphicsContext.Dispose()?", ContextHandle);
|
||||
}
|
||||
IsDisposed = true;
|
||||
}
|
||||
|
||||
private void DestroyContext() {
|
||||
if (ContextHandle == IntPtr.Zero) return;
|
||||
|
||||
try {
|
||||
if (calledManually) {
|
||||
// This will fail if the user calls Dispose() on thread X when the context is current on thread Y.
|
||||
if (!Wgl.wglDeleteContext(ContextHandle))
|
||||
Debug.Print("Failed to destroy OpenGL context {0}. Error: {1}",
|
||||
ContextHandle.ToString(), Marshal.GetLastWin32Error());
|
||||
} catch (AccessViolationException e) {
|
||||
Debug.Print("An access violation occured while destroying the OpenGL context. Please report at http://www.opentk.com.");
|
||||
Debug.Print("Marshal.GetLastWin32Error(): {0}", Marshal.GetLastWin32Error().ToString());
|
||||
Debug.Print(e.ToString());
|
||||
Wgl.wglDeleteContext(ContextHandle);
|
||||
} else {
|
||||
Debug.Print("=== [Warning] OpenGL context leaked ===");
|
||||
}
|
||||
ContextHandle = IntPtr.Zero;
|
||||
}
|
||||
|
|
|
@ -58,7 +58,6 @@ namespace OpenTK.Platform.Windows
|
|||
|
||||
static readonly WinKeyMap KeyMap = new WinKeyMap();
|
||||
const long ExtendedBit = 1 << 24; // Used to distinguish left and right control, alt and enter keys.
|
||||
KeyPressEventArgs key_press = new KeyPressEventArgs();
|
||||
|
||||
public WinWindow(int x, int y, int width, int height, string title, DisplayDevice device) {
|
||||
WindowProcedureDelegate = WindowProcedure;
|
||||
|
@ -85,8 +84,8 @@ namespace OpenTK.Platform.Windows
|
|||
bool new_focused_state = Focused;
|
||||
focused = (wParam.ToInt64() & 0xFFFF) != 0;
|
||||
|
||||
if (new_focused_state != Focused && FocusedChanged != null)
|
||||
FocusedChanged(this, EventArgs.Empty);
|
||||
if (new_focused_state != Focused)
|
||||
RaiseFocusedChanged();
|
||||
break;
|
||||
|
||||
case WindowMessage.ENTERMENULOOP:
|
||||
|
@ -104,8 +103,7 @@ namespace OpenTK.Platform.Windows
|
|||
Point new_location = new Point(pos->x, pos->y);
|
||||
if (Location != new_location) {
|
||||
bounds.Location = new_location;
|
||||
if (Move != null)
|
||||
Move(this, EventArgs.Empty);
|
||||
RaiseMove();
|
||||
}
|
||||
|
||||
Size new_size = new Size(pos->cx, pos->cy);
|
||||
|
@ -121,8 +119,8 @@ namespace OpenTK.Platform.Windows
|
|||
bounds.X, bounds.Y, bounds.Width, bounds.Height,
|
||||
SetWindowPosFlags.NOZORDER | SetWindowPosFlags.NOOWNERZORDER |
|
||||
SetWindowPosFlags.NOACTIVATE | SetWindowPosFlags.NOSENDCHANGING);
|
||||
if (suppress_resize <= 0 && Resize != null)
|
||||
Resize(this, EventArgs.Empty);
|
||||
|
||||
if (suppress_resize <= 0) RaiseResize();
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
@ -150,8 +148,7 @@ namespace OpenTK.Platform.Windows
|
|||
|
||||
if (new_state != windowState) {
|
||||
windowState = new_state;
|
||||
if (WindowStateChanged != null)
|
||||
WindowStateChanged(this, EventArgs.Empty);
|
||||
RaiseWindowStateChanged();
|
||||
}
|
||||
break;
|
||||
|
||||
|
@ -160,10 +157,7 @@ namespace OpenTK.Platform.Windows
|
|||
#region Input events
|
||||
|
||||
case WindowMessage.CHAR:
|
||||
key_press.KeyChar = (char)wParam.ToInt64();
|
||||
|
||||
if (KeyPress != null)
|
||||
KeyPress(this, key_press);
|
||||
RaiseKeyPress((char)wParam.ToInt64());
|
||||
break;
|
||||
|
||||
case WindowMessage.MOUSEMOVE:
|
||||
|
@ -309,27 +303,16 @@ namespace OpenTK.Platform.Windows
|
|||
break;
|
||||
|
||||
case WindowMessage.CLOSE:
|
||||
System.ComponentModel.CancelEventArgs e = new System.ComponentModel.CancelEventArgs();
|
||||
|
||||
if (Closing != null)
|
||||
Closing(this, e);
|
||||
|
||||
if (!e.Cancel) {
|
||||
RaiseClosing();
|
||||
DestroyWindow();
|
||||
break;
|
||||
}
|
||||
|
||||
return IntPtr.Zero;
|
||||
|
||||
case WindowMessage.DESTROY:
|
||||
exists = false;
|
||||
|
||||
API.UnregisterClass(ClassName, Instance);
|
||||
window.Dispose();
|
||||
|
||||
if (Closed != null)
|
||||
Closed(this, EventArgs.Empty);
|
||||
|
||||
RaiseClosed();
|
||||
break;
|
||||
|
||||
#endregion
|
||||
|
@ -409,7 +392,7 @@ namespace OpenTK.Platform.Windows
|
|||
|
||||
const uint GMEM_MOVEABLE = 2;
|
||||
const uint CF_UNICODETEXT = 13, CF_TEXT = 1;
|
||||
public unsafe string GetClipboardText() {
|
||||
public override unsafe string GetClipboardText() {
|
||||
// retry up to 10 times
|
||||
for (int i = 0; i < 10; i++) {
|
||||
if (!API.OpenClipboard(window.winHandle)) {
|
||||
|
@ -435,7 +418,7 @@ namespace OpenTK.Platform.Windows
|
|||
return "";
|
||||
}
|
||||
|
||||
public unsafe void SetClipboardText( string value ) {
|
||||
public override unsafe void SetClipboardText( string value ) {
|
||||
UIntPtr dstSize = (UIntPtr)((value.Length + 1) * Marshal.SystemDefaultCharSize);
|
||||
// retry up to 10 times
|
||||
for (int i = 0; i < 10; i++) {
|
||||
|
@ -466,7 +449,7 @@ namespace OpenTK.Platform.Windows
|
|||
dst2[numChars] = '\0';
|
||||
}
|
||||
|
||||
public Rectangle Bounds {
|
||||
public override Rectangle Bounds {
|
||||
get { return bounds; }
|
||||
set {
|
||||
// Note: the bounds variable is updated when the resize/move message arrives.
|
||||
|
@ -474,7 +457,7 @@ namespace OpenTK.Platform.Windows
|
|||
}
|
||||
}
|
||||
|
||||
public Point Location {
|
||||
public override Point Location {
|
||||
get { return Bounds.Location; }
|
||||
set {
|
||||
// Note: the bounds variable is updated when the resize/move message arrives.
|
||||
|
@ -482,7 +465,7 @@ namespace OpenTK.Platform.Windows
|
|||
}
|
||||
}
|
||||
|
||||
public Size Size {
|
||||
public override Size Size {
|
||||
get { return Bounds.Size; }
|
||||
set {
|
||||
// Note: the bounds variable is updated when the resize/move message arrives.
|
||||
|
@ -490,7 +473,7 @@ namespace OpenTK.Platform.Windows
|
|||
}
|
||||
}
|
||||
|
||||
public Rectangle ClientRectangle {
|
||||
public override Rectangle ClientRectangle {
|
||||
get {
|
||||
if (client_rectangle.Width == 0)
|
||||
client_rectangle.Width = 1;
|
||||
|
@ -502,7 +485,7 @@ namespace OpenTK.Platform.Windows
|
|||
}
|
||||
}
|
||||
|
||||
public Size ClientSize {
|
||||
public override Size ClientSize {
|
||||
get { return ClientRectangle.Size; }
|
||||
set {
|
||||
WindowStyle style = (WindowStyle)API.GetWindowLong(window.winHandle, GetWindowLongOffsets.STYLE);
|
||||
|
@ -512,7 +495,7 @@ namespace OpenTK.Platform.Windows
|
|||
}
|
||||
}
|
||||
|
||||
public Icon Icon {
|
||||
public override Icon Icon {
|
||||
get { return icon; }
|
||||
set {
|
||||
icon = value;
|
||||
|
@ -526,11 +509,11 @@ namespace OpenTK.Platform.Windows
|
|||
}
|
||||
}
|
||||
|
||||
public bool Focused {
|
||||
public override bool Focused {
|
||||
get { return focused; }
|
||||
}
|
||||
|
||||
public bool Visible {
|
||||
public override bool Visible {
|
||||
get { return API.IsWindowVisible(window.winHandle); }
|
||||
set {
|
||||
if (value) {
|
||||
|
@ -545,13 +528,13 @@ namespace OpenTK.Platform.Windows
|
|||
}
|
||||
}
|
||||
|
||||
public bool Exists { get { return exists; } }
|
||||
public override bool Exists { get { return exists; } }
|
||||
|
||||
public void Close() {
|
||||
public override void Close() {
|
||||
API.PostMessage(window.winHandle, WindowMessage.CLOSE, IntPtr.Zero, IntPtr.Zero);
|
||||
}
|
||||
|
||||
public WindowState WindowState {
|
||||
public override WindowState WindowState {
|
||||
get { return windowState; }
|
||||
set {
|
||||
if (WindowState == value)
|
||||
|
@ -648,7 +631,7 @@ namespace OpenTK.Platform.Windows
|
|||
}
|
||||
}
|
||||
|
||||
public Point PointToClient(Point point) {
|
||||
public override Point PointToClient(Point point) {
|
||||
if (!API.ScreenToClient(window.winHandle, ref point))
|
||||
throw new InvalidOperationException(String.Format(
|
||||
"Could not convert point {0} from client to screen coordinates. Windows error: {1}",
|
||||
|
@ -657,24 +640,17 @@ namespace OpenTK.Platform.Windows
|
|||
return point;
|
||||
}
|
||||
|
||||
public Point PointToScreen(Point p) {
|
||||
throw new NotImplementedException();
|
||||
public override Point PointToScreen(Point point) {
|
||||
if (!API.ClientToScreen(window.winHandle, ref point))
|
||||
throw new InvalidOperationException(String.Format(
|
||||
"Could not convert point {0} from client to screen coordinates. Windows error: {1}",
|
||||
point.ToString(), Marshal.GetLastWin32Error()));
|
||||
|
||||
return point;
|
||||
}
|
||||
|
||||
public event EventHandler Move;
|
||||
public event EventHandler Resize;
|
||||
public event EventHandler<System.ComponentModel.CancelEventArgs> Closing;
|
||||
public event EventHandler Closed;
|
||||
public event EventHandler Disposed;
|
||||
public event EventHandler IconChanged;
|
||||
public event EventHandler ClientSizeChanged;
|
||||
public event EventHandler VisibleChanged;
|
||||
public event EventHandler FocusedChanged;
|
||||
public event EventHandler WindowStateChanged;
|
||||
public event EventHandler<KeyPressEventArgs> KeyPress;
|
||||
|
||||
MSG msg;
|
||||
public void ProcessEvents() {
|
||||
public override void ProcessEvents() {
|
||||
while( API.PeekMessage(ref msg, IntPtr.Zero, 0, 0, 1) ) {
|
||||
API.TranslateMessage(ref msg);
|
||||
API.DispatchMessage(ref msg);
|
||||
|
@ -684,11 +660,11 @@ namespace OpenTK.Platform.Windows
|
|||
focused = foreground == window.winHandle;
|
||||
}
|
||||
|
||||
public IWindowInfo WindowInfo {
|
||||
public override IWindowInfo WindowInfo {
|
||||
get { return window; }
|
||||
}
|
||||
|
||||
public Point DesktopCursorPos {
|
||||
public override Point DesktopCursorPos {
|
||||
get {
|
||||
POINT pos = default( POINT );
|
||||
API.GetCursorPos( ref pos );
|
||||
|
@ -698,7 +674,7 @@ namespace OpenTK.Platform.Windows
|
|||
}
|
||||
|
||||
bool cursorVisible = true;
|
||||
public bool CursorVisible {
|
||||
public override bool CursorVisible {
|
||||
get { return cursorVisible; }
|
||||
set {
|
||||
cursorVisible = value;
|
||||
|
@ -706,27 +682,17 @@ namespace OpenTK.Platform.Windows
|
|||
}
|
||||
}
|
||||
|
||||
public void Dispose() {
|
||||
Dispose(true);
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
protected override void Dispose(bool calledManually) {
|
||||
if (disposed) return;
|
||||
|
||||
void Dispose(bool calledManually) {
|
||||
if (!disposed) {
|
||||
if (calledManually) {
|
||||
// Safe to clean managed resources
|
||||
DestroyWindow();
|
||||
if (Icon != null)
|
||||
Icon.Dispose();
|
||||
if (Icon != null) Icon.Dispose();
|
||||
} else {
|
||||
Debug.Print("[Warning] INativeWindow leaked ({0}). Did you forget to call INativeWindow.Dispose()?", this);
|
||||
Debug.Print("=== [Warning] INativeWindow leaked ===");
|
||||
}
|
||||
disposed = true;
|
||||
}
|
||||
}
|
||||
|
||||
~WinWindow() {
|
||||
Dispose(false);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -13,7 +13,7 @@ namespace OpenTK.Platform.X11 {
|
|||
/// \internal
|
||||
/// <summary> Provides methods to create and control an opengl context on the X11 platform.
|
||||
/// This class supports OpenTK, and is not intended for use by OpenTK programs. </summary>
|
||||
internal sealed class X11GLContext : GraphicsContextBase {
|
||||
internal sealed class X11GLContext : IGraphicsContext {
|
||||
|
||||
IntPtr Display;
|
||||
X11WindowInfo currentWindow;
|
||||
|
@ -105,19 +105,18 @@ namespace OpenTK.Platform.X11 {
|
|||
}
|
||||
|
||||
protected override void Dispose(bool manuallyCalled) {
|
||||
if (!IsDisposed) {
|
||||
if (ContextHandle == IntPtr.Zero) return;
|
||||
|
||||
if (manuallyCalled) {
|
||||
IntPtr display = Display;
|
||||
|
||||
if (IsCurrent) {
|
||||
Glx.glXMakeCurrent(display, IntPtr.Zero, IntPtr.Zero);
|
||||
}
|
||||
Glx.glXDestroyContext(display, ContextHandle);
|
||||
}
|
||||
} else {
|
||||
Debug.Print("[Warning] {0} leaked.", this.GetType().Name);
|
||||
Debug.Print("=== [Warning] OpenGL context leaked ===");
|
||||
}
|
||||
IsDisposed = true;
|
||||
ContextHandle = IntPtr.Zero;
|
||||
}
|
||||
|
||||
internal static GraphicsMode SelectGraphicsMode( GraphicsMode template, out XVisualInfo info ) {
|
||||
|
|
|
@ -77,7 +77,6 @@ namespace OpenTK.Platform.X11 {
|
|||
// Keyboard input
|
||||
readonly byte[] ascii = new byte[16];
|
||||
readonly char[] chars = new char[16];
|
||||
readonly KeyPressEventArgs KPEventArgs = new KeyPressEventArgs();
|
||||
|
||||
X11KeyMap keymap = new X11KeyMap();
|
||||
int firstKeyCode, lastKeyCode; // The smallest and largest KeyCode supported by the X server.
|
||||
|
@ -145,7 +144,7 @@ namespace OpenTK.Platform.X11 {
|
|||
e.ConfigureEvent.height = height;
|
||||
RefreshWindowBounds(ref e);
|
||||
|
||||
Debug.Print("X11GLNative window created successfully (id: {0}).", Handle);
|
||||
Debug.Print("X11GLNative window created successfully (id: {0}).", window.WindowHandle);
|
||||
SetupInput();
|
||||
exists = true;
|
||||
}
|
||||
|
@ -279,8 +278,7 @@ namespace OpenTK.Platform.X11 {
|
|||
Point newLoc = new Point(e.ConfigureEvent.x - borderLeft, e.ConfigureEvent.y - borderTop);
|
||||
if (Location != newLoc) {
|
||||
bounds.Location = newLoc;
|
||||
if (Move != null)
|
||||
Move(this, EventArgs.Empty);
|
||||
RaiseMove();
|
||||
}
|
||||
|
||||
// Note: width and height denote the internal (client) size.
|
||||
|
@ -288,13 +286,11 @@ namespace OpenTK.Platform.X11 {
|
|||
Size newSize = new Size(
|
||||
e.ConfigureEvent.width + borderLeft + borderRight,
|
||||
e.ConfigureEvent.height + borderTop + borderBottom);
|
||||
|
||||
if (bounds.Size != newSize) {
|
||||
bounds.Size = newSize;
|
||||
client_rectangle.Size = new Size(e.ConfigureEvent.width, e.ConfigureEvent.height);
|
||||
|
||||
if (this.Resize != null) {
|
||||
Resize(this, EventArgs.Empty);
|
||||
}
|
||||
RaiseResize();
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -305,7 +301,7 @@ namespace OpenTK.Platform.X11 {
|
|||
API.XCheckTypedWindowEvent(window.Display, window.WindowHandle, XEventName.SelectionRequest, ref e);
|
||||
}
|
||||
|
||||
public unsafe void ProcessEvents() {
|
||||
public override unsafe void ProcessEvents() {
|
||||
// Process all pending events
|
||||
while (Exists && window != null) {
|
||||
if (!GetPendingEvent ()) break;
|
||||
|
@ -318,8 +314,7 @@ namespace OpenTK.Platform.X11 {
|
|||
bool previous_visible = visible;
|
||||
visible = e.type == XEventName.MapNotify;
|
||||
if (visible != previous_visible)
|
||||
if (VisibleChanged != null)
|
||||
VisibleChanged(this, EventArgs.Empty);
|
||||
RaiseVisibleChanged();
|
||||
} break;
|
||||
|
||||
case XEventName.CreateNotify:
|
||||
|
@ -329,17 +324,11 @@ namespace OpenTK.Platform.X11 {
|
|||
case XEventName.ClientMessage:
|
||||
if (!isExiting && e.ClientMessageEvent.ptr1 == wm_destroy) {
|
||||
Debug.Print("Exit message received.");
|
||||
CancelEventArgs ce = new CancelEventArgs();
|
||||
if (Closing != null)
|
||||
Closing(this, ce);
|
||||
RaiseClosing();
|
||||
|
||||
if (!ce.Cancel) {
|
||||
isExiting = true;
|
||||
|
||||
DestroyWindow();
|
||||
if (Closed != null)
|
||||
Closed(this, EventArgs.Empty);
|
||||
}
|
||||
RaiseClosed();
|
||||
} break;
|
||||
|
||||
case XEventName.DestroyNotify:
|
||||
|
@ -357,15 +346,10 @@ namespace OpenTK.Platform.X11 {
|
|||
int status = API.XLookupString(ref e.KeyEvent, ascii, ascii.Length, null, IntPtr.Zero);
|
||||
Encoding.Default.GetChars(ascii, 0, status, chars, 0);
|
||||
|
||||
EventHandler<KeyPressEventArgs> key_press = KeyPress;
|
||||
if (key_press != null) {
|
||||
for (int i = 0; i < status; i++) {
|
||||
// ignore NULL char after non-ASCII input char, like ä or å on Finnish keyboard layout
|
||||
if (chars[i] == '\0') continue;
|
||||
|
||||
KPEventArgs.KeyChar = chars[i];
|
||||
key_press(this, KPEventArgs);
|
||||
}
|
||||
RaiseKeyPress(chars[i]);
|
||||
}
|
||||
break;
|
||||
|
||||
|
@ -402,8 +386,8 @@ namespace OpenTK.Platform.X11 {
|
|||
{
|
||||
bool previous_focus = has_focus;
|
||||
has_focus = e.type == XEventName.FocusIn;
|
||||
if (has_focus != previous_focus && FocusedChanged != null)
|
||||
FocusedChanged(this, EventArgs.Empty);
|
||||
if (has_focus != previous_focus)
|
||||
RaiseFocusedChanged();
|
||||
} break;
|
||||
|
||||
case XEventName.MappingNotify:
|
||||
|
@ -416,8 +400,7 @@ namespace OpenTK.Platform.X11 {
|
|||
|
||||
case XEventName.PropertyNotify:
|
||||
if (e.PropertyEvent.atom == net_wm_state) {
|
||||
if (WindowStateChanged != null)
|
||||
WindowStateChanged (this, EventArgs.Empty);
|
||||
RaiseWindowStateChanged();
|
||||
}
|
||||
|
||||
//if (e.PropertyEvent.atom == net_frame_extents) {
|
||||
|
@ -494,7 +477,7 @@ namespace OpenTK.Platform.X11 {
|
|||
return e.SelectionRequestEvent.target;
|
||||
}
|
||||
|
||||
public string GetClipboardText() {
|
||||
public override string GetClipboardText() {
|
||||
IntPtr owner = API.XGetSelectionOwner(window.Display, xa_clipboard);
|
||||
if (owner == IntPtr.Zero) return ""; // no window owner
|
||||
|
||||
|
@ -510,12 +493,12 @@ namespace OpenTK.Platform.X11 {
|
|||
return "";
|
||||
}
|
||||
|
||||
public void SetClipboardText(string value) {
|
||||
public override void SetClipboardText(string value) {
|
||||
clipboard_copy_text = value;
|
||||
API.XSetSelectionOwner(window.Display, xa_clipboard, window.WinHandle, IntPtr.Zero);
|
||||
}
|
||||
|
||||
public Rectangle Bounds {
|
||||
public override Rectangle Bounds {
|
||||
get { return bounds; }
|
||||
set {
|
||||
API.XMoveResizeWindow(
|
||||
|
@ -525,7 +508,7 @@ namespace OpenTK.Platform.X11 {
|
|||
}
|
||||
}
|
||||
|
||||
public Point Location {
|
||||
public override Point Location {
|
||||
get { return Bounds.Location; }
|
||||
set {
|
||||
API.XMoveWindow(window.Display, window.WindowHandle, value.X, value.Y);
|
||||
|
@ -533,7 +516,7 @@ namespace OpenTK.Platform.X11 {
|
|||
}
|
||||
}
|
||||
|
||||
public Size Size {
|
||||
public override Size Size {
|
||||
get { return Bounds.Size; }
|
||||
set {
|
||||
int width = value.Width - borderLeft - borderRight;
|
||||
|
@ -545,7 +528,7 @@ namespace OpenTK.Platform.X11 {
|
|||
}
|
||||
}
|
||||
|
||||
public Rectangle ClientRectangle {
|
||||
public override Rectangle ClientRectangle {
|
||||
get {
|
||||
if (client_rectangle.Width == 0)
|
||||
client_rectangle.Width = 1;
|
||||
|
@ -559,12 +542,12 @@ namespace OpenTK.Platform.X11 {
|
|||
}
|
||||
}
|
||||
|
||||
public Size ClientSize {
|
||||
public override Size ClientSize {
|
||||
get { return ClientRectangle.Size; }
|
||||
set { ClientRectangle = new Rectangle(Point.Empty, value); }
|
||||
}
|
||||
|
||||
public Icon Icon {
|
||||
public override Icon Icon {
|
||||
get { return icon; }
|
||||
set {
|
||||
if (value == icon)
|
||||
|
@ -615,11 +598,11 @@ namespace OpenTK.Platform.X11 {
|
|||
}
|
||||
}
|
||||
|
||||
public bool Focused {
|
||||
public override bool Focused {
|
||||
get { return has_focus; }
|
||||
}
|
||||
|
||||
public WindowState WindowState {
|
||||
public override WindowState WindowState {
|
||||
get {
|
||||
IntPtr actual_atom, nitems, bytes_after, prop = IntPtr.Zero;
|
||||
int actual_format;
|
||||
|
@ -709,19 +692,7 @@ namespace OpenTK.Platform.X11 {
|
|||
}
|
||||
}
|
||||
|
||||
public event EventHandler Load;
|
||||
public event EventHandler Unload;
|
||||
public event EventHandler Move;
|
||||
public event EventHandler Resize;
|
||||
public event EventHandler<CancelEventArgs> Closing;
|
||||
public event EventHandler Closed;
|
||||
public event EventHandler Disposed;
|
||||
public event EventHandler VisibleChanged;
|
||||
public event EventHandler FocusedChanged;
|
||||
public event EventHandler WindowStateChanged;
|
||||
public event EventHandler<KeyPressEventArgs> KeyPress;
|
||||
|
||||
public Point DesktopCursorPos {
|
||||
public override Point DesktopCursorPos {
|
||||
get {
|
||||
IntPtr root, child;
|
||||
int rootX, rootY, childX, childY, mask;
|
||||
|
@ -735,7 +706,7 @@ namespace OpenTK.Platform.X11 {
|
|||
}
|
||||
|
||||
bool cursorVisible = true;
|
||||
public bool CursorVisible {
|
||||
public override bool CursorVisible {
|
||||
get { return cursorVisible; }
|
||||
set {
|
||||
cursorVisible = value;
|
||||
|
@ -758,16 +729,11 @@ namespace OpenTK.Platform.X11 {
|
|||
}
|
||||
|
||||
/// <summary> Returns true if a render window/context exists. </summary>
|
||||
public bool Exists {
|
||||
public override bool Exists {
|
||||
get { return exists; }
|
||||
}
|
||||
|
||||
/// <summary> Gets the current window handle. </summary>
|
||||
public IntPtr Handle {
|
||||
get { return window.WindowHandle; }
|
||||
}
|
||||
|
||||
public bool Visible {
|
||||
public override bool Visible {
|
||||
get { return visible; }
|
||||
set {
|
||||
if (value && !visible) {
|
||||
|
@ -778,11 +744,11 @@ namespace OpenTK.Platform.X11 {
|
|||
}
|
||||
}
|
||||
|
||||
public IWindowInfo WindowInfo {
|
||||
public override IWindowInfo WindowInfo {
|
||||
get { return window; }
|
||||
}
|
||||
|
||||
public void Close() {
|
||||
public override void Close() {
|
||||
XEvent ev = new XEvent();
|
||||
ev.type = XEventName.ClientMessage;
|
||||
ev.ClientMessageEvent.format = 32;
|
||||
|
@ -794,33 +760,28 @@ namespace OpenTK.Platform.X11 {
|
|||
API.XFlush(window.Display);
|
||||
}
|
||||
|
||||
public void DestroyWindow() {
|
||||
void DestroyWindow() {
|
||||
Debug.Print("X11GLNative shutdown sequence initiated.");
|
||||
API.XSync(window.Display, true);
|
||||
API.XDestroyWindow(window.Display, window.WindowHandle);
|
||||
exists = false;
|
||||
}
|
||||
|
||||
public Point PointToClient(Point point) {
|
||||
public override Point PointToClient(Point point) {
|
||||
int ox, oy;
|
||||
IntPtr child;
|
||||
API.XTranslateCoordinates(window.Display, window.RootWindow, window.WindowHandle, point.X, point.Y, out ox, out oy, out child);
|
||||
return new Point( ox, oy );
|
||||
}
|
||||
|
||||
public Point PointToScreen(Point point) {
|
||||
public override Point PointToScreen(Point point) {
|
||||
int ox, oy;
|
||||
IntPtr child;
|
||||
API.XTranslateCoordinates(window.Display, window.WindowHandle, window.RootWindow, point.X, point.Y, out ox, out oy, out child);
|
||||
return new Point( ox, oy );
|
||||
}
|
||||
|
||||
public void Dispose() {
|
||||
this.Dispose(true);
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
private void Dispose(bool manuallyCalled) {
|
||||
protected override void Dispose(bool manuallyCalled) {
|
||||
if (disposed) return;
|
||||
if (manuallyCalled) {
|
||||
if (window != null && window.WindowHandle != IntPtr.Zero) {
|
||||
|
@ -831,13 +792,9 @@ namespace OpenTK.Platform.X11 {
|
|||
window = null;
|
||||
}
|
||||
} else {
|
||||
Debug.Print("[Warning] {0} leaked.", this.GetType().Name);
|
||||
Debug.Print("=== [Warning] INativeWindow leaked ===");
|
||||
}
|
||||
disposed = true;
|
||||
}
|
||||
|
||||
~X11Window() {
|
||||
this.Dispose(false);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -23,6 +23,20 @@ using System.Runtime.InteropServices;
|
|||
|
||||
namespace SharpDX.Direct3D9 {
|
||||
|
||||
[StructLayout(LayoutKind.Sequential)]
|
||||
public struct LockedRectangle {
|
||||
public int Pitch;
|
||||
public IntPtr DataPointer;
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Sequential)]
|
||||
public struct D3DRect {
|
||||
public int Left;
|
||||
public int Top;
|
||||
public int Right;
|
||||
public int Bottom;
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Sequential)]
|
||||
public struct PresentParameters {
|
||||
public int BackBufferWidth;
|
||||
|
@ -33,8 +47,8 @@ namespace SharpDX.Direct3D9 {
|
|||
public int MultiSampleQuality;
|
||||
public SwapEffect SwapEffect;
|
||||
public IntPtr DeviceWindowHandle;
|
||||
public RawBool Windowed;
|
||||
public RawBool EnableAutoDepthStencil;
|
||||
public int Windowed;
|
||||
public int EnableAutoDepthStencil;
|
||||
public Format AutoDepthStencilFormat;
|
||||
public PresentFlags PresentFlags;
|
||||
public int FullScreenRefreshRateInHz;
|
||||
|
|
|
@ -1,60 +0,0 @@
|
|||
// Copyright (c) 2010-2013 SharpDX - Alexandre Mutel
|
||||
//
|
||||
// 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.
|
||||
|
||||
using System;
|
||||
using System.Runtime.InteropServices;
|
||||
|
||||
namespace SharpDX {
|
||||
|
||||
[StructLayout(LayoutKind.Sequential, Size = 4)]
|
||||
public struct RawBool {
|
||||
private int boolValue;
|
||||
|
||||
public RawBool(bool boolValue) {
|
||||
this.boolValue = boolValue ? 1 : 0;
|
||||
}
|
||||
|
||||
public static implicit operator bool(RawBool booleanValue) {
|
||||
return booleanValue.boolValue != 0;
|
||||
}
|
||||
|
||||
public static implicit operator RawBool(bool boolValue) {
|
||||
return new RawBool(boolValue);
|
||||
}
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Sequential)]
|
||||
public struct LockedRectangle {
|
||||
|
||||
/// <summary> Gets the number of bytes per row. </summary>
|
||||
public int Pitch;
|
||||
|
||||
/// <summary> Pointer to the data. </summary>
|
||||
public IntPtr DataPointer;
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Sequential)]
|
||||
public struct D3DRect {
|
||||
public int Left;
|
||||
public int Top;
|
||||
public int Right;
|
||||
public int Bottom;
|
||||
}
|
||||
}
|
Loading…
Add table
Reference in a new issue