More work on OpenTK rewrite

This commit is contained in:
UnknownShadow200 2018-07-12 03:16:18 +10:00
parent 35b27cb339
commit 5a6747084b
31 changed files with 721 additions and 1668 deletions

View file

@ -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" />

View file

@ -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);
}
}
}
}

View file

@ -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 {

View file

@ -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();

View file

@ -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;
}

View file

@ -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() {

View file

@ -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);
}
}
}
}

View file

@ -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();

View file

@ -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();

View file

@ -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); }
}*/
}
}

View file

@ -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); }
}
}

View file

@ -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) { }
}
}

View file

@ -55,4 +55,9 @@ namespace OpenTK.Graphics {
}
}
}
public class GraphicsModeException : Exception {
public GraphicsModeException() : base() { }
public GraphicsModeException(string message) : base(message) { }
}
}

View file

@ -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) { }
}
}

View file

@ -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) { }
}
}

View file

@ -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" );

View file

@ -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;
}
}

View file

@ -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); }
}
}

View file

@ -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.");
}
}
}
}
}

View file

@ -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>

View file

@ -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);
}

View file

@ -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);

View file

@ -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;

View file

@ -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) {
}
}
}

View file

@ -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);

View file

@ -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;
}

View file

@ -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);
}
}
}

View file

@ -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 ) {

View file

@ -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);
}
}
}

View file

@ -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;

View file

@ -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;
}
}