+// 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.
+//
+// Copyright (c) 2004-2006 Novell, Inc.
//
// System.Windows.Forms.XEventQueue
//
-// Author(s):
+// Authors:
// Jackson Harper (jackson@ximian.com)
-//
-// Copyright (C) Novell, Inc (http://www.novell.com)
+// Peter Dennis Bartok (pbartok@novell.com)
//
using System;
using System.Threading;
using System.Collections;
-
namespace System.Windows.Forms {
internal class XEventQueue {
- private XQueue xqueue;
- private XQueue lqueue; // Events inserted from threads other then the main X thread
+ private XQueue xqueue;
+ private XQueue lqueue; // Events inserted from threads other then the main X thread
+ private PaintQueue paint; // Paint-only queue
+ internal ArrayList timer_list;
+ private Thread thread;
+ private bool dispatch_idle;
- private static readonly int InitialXEventSize = 50;
+ private static readonly int InitialXEventSize = 100;
private static readonly int InitialLXEventSize = 10;
+ private static readonly int InitialPaintSize = 50;
- public XEventQueue ()
- {
+ public XEventQueue (Thread thread) {
xqueue = new XQueue (InitialXEventSize);
lqueue = new XQueue (InitialLXEventSize);
+ paint = new PaintQueue(InitialPaintSize);
+ timer_list = new ArrayList ();
+ this.thread = thread;
+ this.dispatch_idle = true;
}
public int Count {
}
}
+ public PaintQueue Paint {
+ get {
+ return paint;
+ }
+ }
+
+ public Thread Thread {
+ get {
+ return thread;
+ }
+ }
+
public void Enqueue (XEvent xevent)
{
+ if (Thread.CurrentThread != thread) {
+ Console.WriteLine ("Hwnd.Queue.Enqueue called from a different thread without locking.");
+ Console.WriteLine (Environment.StackTrace);
+ }
+
xqueue.Enqueue (xevent);
}
public XEvent Dequeue ()
{
+ if (Thread.CurrentThread != thread) {
+ Console.WriteLine ("Hwnd.Queue.Dequeue called from a different thread without locking.");
+ Console.WriteLine (Environment.StackTrace);
+ }
+
if (xqueue.Count == 0) {
lock (lqueue) {
return lqueue.Dequeue ();
return xqueue.Dequeue ();
}
+ public XEvent Peek()
+ {
+ if (Thread.CurrentThread != thread) {
+ Console.WriteLine ("Hwnd.Queue.Peek called from a different thread without locking.");
+ Console.WriteLine (Environment.StackTrace);
+ }
+
+ if (xqueue.Count == 0) {
+ lock (lqueue) {
+ return lqueue.Peek ();
+ }
+ }
+ return xqueue.Peek();
+ }
+
+ public bool DispatchIdle {
+ get {
+ return dispatch_idle;
+ }
+ set {
+ dispatch_idle = value;
+ }
+ }
+
+ public class PaintQueue {
+
+ private ArrayList hwnds;
+ private XEvent xevent;
+
+ public PaintQueue (int size) {
+ hwnds = new ArrayList(size);
+ xevent = new XEvent();
+ xevent.AnyEvent.type = XEventName.Expose;
+ }
+
+ public int Count {
+ get { return hwnds.Count; }
+ }
+
+ public void Enqueue (Hwnd hwnd) {
+ hwnds.Add(hwnd);
+ }
+
+ public void Remove(Hwnd hwnd) {
+ if (!hwnd.expose_pending && !hwnd.nc_expose_pending) {
+ hwnds.Remove(hwnd);
+ }
+ }
+
+ public XEvent Dequeue () {
+ Hwnd hwnd;
+ IEnumerator next;
+
+ if (hwnds.Count == 0) {
+ xevent.ExposeEvent.window = IntPtr.Zero;
+ return xevent;
+ }
+
+ next = hwnds.GetEnumerator();
+ next.MoveNext();
+ hwnd = (Hwnd)next.Current;
+
+ // We only remove the event from the queue if we have one expose left since
+ // a single 'entry in our queue may be for both NC and Client exposed
+ if ( !(hwnd.nc_expose_pending && hwnd.expose_pending)) {
+ hwnds.Remove(hwnd);
+ }
+ if (hwnd.expose_pending) {
+ xevent.ExposeEvent.window = hwnd.client_window;
+#if not
+ xevent.ExposeEvent.x = hwnd.invalid.X;
+ xevent.ExposeEvent.y = hwnd.invalid.Y;
+ xevent.ExposeEvent.width = hwnd.invalid.Width;
+ xevent.ExposeEvent.height = hwnd.invalid.Height;
+#endif
+ return xevent;
+ } else {
+ xevent.ExposeEvent.window = hwnd.whole_window;
+ xevent.ExposeEvent.x = hwnd.nc_invalid.X;
+ xevent.ExposeEvent.y = hwnd.nc_invalid.Y;
+ xevent.ExposeEvent.width = hwnd.nc_invalid.Width;
+ xevent.ExposeEvent.height = hwnd.nc_invalid.Height;
+ return xevent;
+ }
+ }
+ }
+
private class XQueue {
private XEvent [] xevents;
return res;
}
+ public XEvent Peek() {
+ if (size < 1) {
+ throw new Exception ("Attempt to peek at empty queue");
+ }
+ return xevents[head];
+ }
+
private void Grow ()
{
int newcap = (xevents.Length * 2);