#region --- License --- /* Copyright (c) 2007 Stefanos Apostolopoulos * See license.txt for license info */ #endregion #region --- Using directives --- using System; using System.Collections.Generic; using System.Text; using System.Runtime.InteropServices; using System.Windows.Forms; using System.Diagnostics; #endregion namespace OpenTK.Platform.Windows { /// /// Drives GameWindow on Windows. /// This class supports OpenTK, and is not intended for use by OpenTK programs. /// sealed class WinGLNative : NativeWindow, OpenTK.Platform.INativeGLWindow, IDisposable { #region --- Fields --- private WinGLContext glContext; private DisplayMode mode = new DisplayMode(); private bool fullscreen = false; private bool disposed; private bool quit; private bool created; private WindowInfo info; #endregion #region --- Contructors --- /// /// Constructs a new WinGLNative class. Call CreateWindow to create the /// actual render window. /// public WinGLNative() { Debug.Print("Native window driver: {0}", this.ToString()); } #endregion #region protected override void WndProc(ref Message m) /// /// For use in WndProc only. /// private int width, height; /// /// Processes incoming WM_* messages. /// /// Reference to the incoming Windows Message. protected override void WndProc(ref Message m) { switch (m.Msg) { case API.Constants.WM_WINDOWPOSCHANGED: // Get window size width = Marshal.ReadInt32(m.LParam, (int)Marshal.OffsetOf(typeof(API.WindowPosition), "cx")); height = Marshal.ReadInt32(m.LParam, (int)Marshal.OffsetOf(typeof(API.WindowPosition), "cy")); //if (resizeEventArgs.Width != width || resizeEventArgs.Height != height) if (mode.Width != width || mode.Height != height) { // If the size has changed, raise the ResizeEvent. resizeEventArgs.Width = width; resizeEventArgs.Height = height; this.OnResize(resizeEventArgs); // The message was processed. return; } // If the message was not a resize notification, send it to the default WndProc. break; case API.Constants.WM_CREATE: // Set the window width and height: mode.Width = Marshal.ReadInt32(m.LParam, (int)Marshal.OffsetOf(typeof(API.CreateStruct), "cx")); mode.Height = Marshal.ReadInt32(m.LParam, (int)Marshal.OffsetOf(typeof(API.CreateStruct), "cy")); info = new WindowInfo(); info.Handle = this.Handle; info.Parent = null; Debug.Print("Window handle: {0}", this.Handle); // Raise the Create event this.OnCreate(EventArgs.Empty); // Raise the resize event: //resizeEventArgs.Width = width; //resizeEventArgs.Height = height; //this.OnResize(resizeEventArgs); return; //case API.Constants.WM_KEYDOWN: // Legacy input events //case API.Constants.WM_KEYUP: // break; case API.Constants.WM_CLOSE: //this.Exit(); //return; break; case API.Constants.WM_DESTROY: if (this.Handle != IntPtr.Zero) { Debug.Print("Window handle {0} destroyed.", this.Handle); this.DestroyHandle(); } API.PostQuitMessage(0); return; case API.Constants.WM_QUIT: quit = true; Debug.WriteLine("Application quit."); return; } //base.WndProc(ref m); DefWndProc(ref m); } #endregion #region --- INativeGLWindow Members --- #region private void CreateWindow(DisplayMode mode) public void CreateWindow(DisplayMode mode) { Debug.Print("Creating native window with mode: {0}", mode.ToString()); Debug.Indent(); CreateParams cp = new CreateParams(); cp.ClassStyle = (int)API.WindowClassStyle.OwnDC | (int)API.WindowClassStyle.VRedraw | (int)API.WindowClassStyle.HRedraw | (int)API.WindowClassStyle.Ime; cp.Style = (int)API.WindowStyle.Visible | (int)API.WindowStyle.ClipChildren | (int)API.WindowStyle.ClipSiblings | (int)API.WindowStyle.OverlappedWindow; cp.Width = mode.Width; cp.Height = mode.Height; cp.Caption = "OpenTK Game Window"; // Keep in mind that some construction code runs in WM_CREATE, // which is raised CreateHandle() CreateHandle(cp); glContext = new WinGLContext( this.Handle, new DisplayMode( width, height, new ColorDepth(32), 16, 0, 0, 2, fullscreen, false, false, 0.0f ) ); if (this.Handle != IntPtr.Zero && glContext != null) { Debug.WriteLine("Window creation was succesful."); created = true; } else { throw new ApplicationException(String.Format( "Could not create native window and/or context. Handle: {0}, Context {1}", this.Handle, this.Context.ToString())); } Debug.Unindent(); } /* private void CreateWindow() { WinApi.WindowClass wc = new WinApi.WindowClass(); wc.style = WinApi.WindowClassStyle.HRedraw | WinApi.WindowClassStyle.VRedraw | WinApi.WindowClassStyle.OwnDC; wc.WindowProcedure = new WinApi.WindowProcedureEventHandler(WndProc); wc.Instance = instance; //wc.ClassName = Marshal.StringToHGlobalAuto(className); wc.ClassName = className; classAtom = WinApi.RegisterClass(wc); if (classAtom == 0) { throw new Exception("Could not register class, error: " + Marshal.GetLastWin32Error()); } // Change for fullscreen! handle = WinApi.CreateWindowEx( WinApi.ExtendedWindowStyle.ApplicationWindow | WinApi.ExtendedWindowStyle.OverlappedWindow | WinApi.ExtendedWindowStyle.Topmost, className, //Marshal.StringToHGlobalAuto("OpenTK Game Window"), "OpenTK Game Window", WinApi.WindowStyle.OverlappedWindow | WinApi.WindowStyle.ClipChildren | WinApi.WindowStyle.ClipSiblings, 0, 0, 640, 480, IntPtr.Zero, IntPtr.Zero, instance, IntPtr.Zero ); if (handle == IntPtr.Zero) { throw new Exception("Could not create window, error: " + Marshal.GetLastWin32Error()); } } */ #endregion #region public void Exit() /// /// Starts the teardown sequence for the current window. /// public void Exit() { API.PostMessage(this.Handle, API.Constants.WM_DESTROY, IntPtr.Zero, IntPtr.Zero); } #endregion #region public void ProcessEvents() private int ret; System.Windows.Forms.Message msg; public void ProcessEvents() { while (!IsIdle) { ret = API.GetMessage(out msg, Handle, 0, 0); if (ret == -1) { throw new ApplicationException(String.Format( "An error happened while processing the message queue. Windows error: {0}", Marshal.GetLastWin32Error())); } API.DispatchMessage(ref msg); WndProc(ref msg); } } #endregion #region public event CreateEvent Create; public event CreateEvent Create; private void OnCreate(EventArgs e) { if (this.Create != null) { this.Create(this, e); } } #endregion #region public bool Created /// /// Returns true if a render window/context exists. /// public bool Created { get { return created; } } #endregion #region public bool Quit public bool Quit { get { return quit; } } #endregion #region public IGLContext Context public IGLContext Context { get { return glContext; } } #endregion #region public bool Fullscreen public bool Fullscreen { get { return false; //throw new NotImplementedException(); } set { fullscreen = false; throw new NotImplementedException(); } } #endregion #region public bool IsIdle public bool IsIdle { get { //return !API.PeekMessage(out msg, IntPtr.Zero, 0, 0, 0); return !API.PeekMessage(out msg, this.Handle, 0, 0, 0); //return API.GetQueueStatus(API.QueueStatusFlags.ALLEVENTS) == 0; } } #endregion #region public IWindowInfo WindowInfo public IWindowInfo WindowInfo { get { return info; } } #endregion #endregion #region --- IDisposable Members --- public void Dispose() { this.Dispose(true); GC.SuppressFinalize(this); } private void Dispose(bool calledManually) { if (!disposed) { // Clean unmanaged resources here: if (calledManually) { // Safe to clean managed resources glContext.Dispose(); base.DestroyHandle(); } disposed = true; } } ~WinGLNative() { Dispose(false); } #endregion #region --- IResizable Members --- #region public int Width public int Width { get { return mode.Width; } set { throw new NotImplementedException(); //WinApi.PostMessage( // this.Handle, // WinApi.Constants.WM_WINDOWPOSCHANGING, //mode.Width = value; } } #endregion #region public int Height public int Height { get { return mode.Height; } set { throw new NotImplementedException(); //WinApi.PostMessage( // this.Handle, // WinApi.Constants.WM_WINDOWPOSCHANGING, //mode.Height = value; } } #endregion #region public event ResizeEvent Resize public event ResizeEvent Resize; private ResizeEventArgs resizeEventArgs = new ResizeEventArgs(); public void OnResize(ResizeEventArgs e) { mode.Width = e.Width; mode.Height = e.Height; if (this.Resize != null) this.Resize(this, e); } #endregion #endregion } #region class WindowHandle : Microsoft.Win32.SafeHandles.SafeHandleZeroOrMinusOneIsInvalid /* class WindowHandle : Microsoft.Win32.SafeHandles.SafeHandleZeroOrMinusOneIsInvalid { protected override bool ReleaseHandle() { throw new Exception("The method or operation is not implemented."); } public override bool IsInvalid { get { return base.IsInvalid; } } } */ #endregion }