ShiftOS-C-/source/ShiftUI/Internal/XEventQueue.cs
MichaelTheShifter d40fed5ce2 Move ShiftUI source code to ShiftOS
This'll be a lot easier to work on.
2016-07-20 09:40:36 -04:00

261 lines
6.1 KiB
C#

// 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.
//
// ShiftUI.XEventQueue
//
// Authors:
// Jackson Harper (jackson@ximian.com)
// Peter Dennis Bartok (pbartok@novell.com)
//
using System;
using System.Threading;
using System.Collections;
namespace ShiftUI {
internal class XEventQueue {
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 = 100;
private static readonly int InitialLXEventSize = 10;
private static readonly int InitialPaintSize = 50;
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 {
get {
lock (lqueue) {
return xqueue.Count + lqueue.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 void EnqueueLocked (XEvent xevent)
{
lock (lqueue) {
lqueue.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 {
lock (hwnds) {
return hwnds.Count;
}
}
}
public void Enqueue (Hwnd hwnd) {
lock (hwnds) {
hwnds.Add (hwnd);
}
}
public void Remove(Hwnd hwnd) {
if (!hwnd.expose_pending && !hwnd.nc_expose_pending) {
lock (hwnds) {
hwnds.Remove (hwnd);
}
}
}
public XEvent Dequeue () {
Hwnd hwnd;
IEnumerator next;
lock (hwnds) {
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;
private int head;
private int tail;
private int size;
public XQueue (int size)
{
xevents = new XEvent [size];
}
public int Count {
get { return size; }
}
public void Enqueue (XEvent xevent)
{
if (size == xevents.Length)
Grow ();
xevents [tail] = xevent;
tail = (tail + 1) % xevents.Length;
size++;
}
public XEvent Dequeue ()
{
if (size < 1)
throw new Exception ("Attempt to dequeue empty queue.");
XEvent res = xevents [head];
head = (head + 1) % xevents.Length;
size--;
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);
XEvent [] na = new XEvent [newcap];
xevents.CopyTo (na, 0);
xevents = na;
head = 0;
tail = head + size;
}
}
}
}