001: /*
002: * Copyright 2000,2005 wingS development team.
003: *
004: * This file is part of wingS (http://wingsframework.org).
005: *
006: * wingS is free software; you can redistribute it and/or modify
007: * it under the terms of the GNU Lesser General Public License
008: * as published by the Free Software Foundation; either version 2.1
009: * of the License, or (at your option) any later version.
010: *
011: * Please see COPYING for the complete licence.
012: */
013: package org.wings.util;
014:
015: /*
016: * Die Klasse ist praktisch von der Swing Implementierung
017: * abgeleitet. Leider brauch ich einen Timer, der auch innerhalb des
018: * Swing Event Threads Impulse gibt. Deshalb angepasst. Ich hoffe das
019: * gibt keine Problemen mit dem Swing Team.
020: */
021:
022: /**
023: * @author <a href="mailto:haaf@mercatis.de">Armin Haaf</a>
024: */
025: public final class TimerQueue implements Runnable {
026: /*
027: * Es gibt nur eine Queue pro Application.
028: */
029: private static TimerQueue sharedInstance = new TimerQueue();
030:
031: private Timer firstTimer;
032: private boolean running;
033:
034: /*
035: * Es gibt nur eine TimerQueue pro Application.
036: */
037: private TimerQueue() {
038: super ();
039:
040: // Now start the TimerQueue thread.
041: start();
042: }
043:
044: /*
045: * Die eine Queue!!
046: */
047: public static TimerQueue sharedInstance() {
048: return sharedInstance;
049: }
050:
051: private synchronized void start() {
052: System.err.println("Timer started");
053: if (running) {
054: throw new RuntimeException(
055: "Can't start a TimerQueue that is already running");
056: } else {
057: Thread timerThread = new Thread(this , "TimerQueue");
058:
059: try {
060: timerThread.setDaemon(true);
061: } catch (SecurityException e) {
062: }
063: timerThread.start();
064: running = true;
065: }
066: }
067:
068: private synchronized void stop() {
069: running = false;
070: notifyAll();
071: }
072:
073: public synchronized void addTimer(Timer timer, long expirationTime) {
074: Timer previousTimer, nextTimer;
075:
076: // If the Timer is already in the queue, then ignore the add.
077: if (timer.running) {
078: return;
079: }
080:
081: previousTimer = null;
082: nextTimer = firstTimer;
083:
084: // Insert the Timer into the linked list in the order they will
085: // expire. If two timers expire at the same time, put the newer entry
086: // later so they expire in the order they came in.
087:
088: while (nextTimer != null) {
089: if (nextTimer.expirationTime > expirationTime)
090: break;
091:
092: previousTimer = nextTimer;
093: nextTimer = nextTimer.nextTimer;
094: }
095:
096: if (previousTimer == null)
097: firstTimer = timer;
098: else
099: previousTimer.nextTimer = timer;
100:
101: timer.expirationTime = expirationTime;
102: timer.nextTimer = nextTimer;
103: timer.running = true;
104: notifyAll();
105: }
106:
107: public synchronized void removeTimer(Timer timer) {
108: boolean found;
109: Timer previousTimer, nextTimer;
110:
111: if (!timer.running)
112: return;
113:
114: previousTimer = null;
115: nextTimer = firstTimer;
116: found = false;
117:
118: while (nextTimer != null) {
119: if (nextTimer == timer) {
120: found = true;
121: break;
122: }
123:
124: previousTimer = nextTimer;
125: nextTimer = nextTimer.nextTimer;
126: }
127:
128: if (!found)
129: return;
130:
131: if (previousTimer == null)
132: firstTimer = timer.nextTimer;
133: else
134: previousTimer.nextTimer = timer.nextTimer;
135:
136: timer.expirationTime = 0;
137: timer.nextTimer = null;
138: timer.running = false;
139: }
140:
141: public synchronized boolean containsTimer(Timer timer) {
142: return timer.running;
143: }
144:
145: // If there are a ton of timers, this method may never return. It loops
146: // checking to see if the head of the Timer list has expired. If it has,
147: // it posts the Timer and reschedules it if necessary.
148:
149: public synchronized long postExpiredTimers() {
150: long currentTime, timeToWait;
151: Timer timer;
152:
153: // The timeToWait we return should never be negative and only be zero
154: // when we have no Timers to wait for.
155:
156: do {
157: timer = firstTimer;
158: if (timer == null)
159: return 0;
160:
161: currentTime = System.currentTimeMillis();
162: timeToWait = timer.expirationTime - currentTime;
163:
164: if (timeToWait <= 0) {
165: try {
166: timer.post(); // have timer post an event
167: } catch (SecurityException e) {
168: }
169:
170: // remove the timer from the queue
171: removeTimer(timer);
172:
173: // This tries to keep the interval uniform at the cost of
174: // drift.
175: if (timer.isRepeats()) {
176: addTimer(timer, currentTime + timer.getDelay());
177: }
178: }
179:
180: // Allow other threads to call addTimer() and removeTimer()
181: // even when we are posting Timers like mad. Since the wait()
182: // releases the lock, be sure not to maintain any state
183: // between iterations of the loop.
184:
185: try {
186: wait(1);
187: } catch (InterruptedException e) {
188: }
189: } while (timeToWait <= 0);
190:
191: return timeToWait;
192: }
193:
194: public synchronized void run() {
195: long timeToWait;
196:
197: try {
198: while (running) {
199: timeToWait = postExpiredTimers();
200: try {
201: wait(timeToWait);
202: } catch (InterruptedException e) {
203: }
204: }
205: } catch (ThreadDeath td) {
206: running = false;
207: // Mark all the timers we contain as not being queued.
208: Timer timer = firstTimer;
209: while (timer != null) {
210: timer.eventQueued = false;
211: timer = timer.nextTimer;
212: }
213: sharedInstance.start();
214: throw td;
215: }
216: }
217:
218: public synchronized String toString() {
219: SStringBuilder buf;
220: Timer nextTimer;
221:
222: buf = new SStringBuilder();
223: buf.append("TimerQueue (");
224:
225: nextTimer = firstTimer;
226: while (nextTimer != null) {
227: buf.append(nextTimer.toString());
228:
229: nextTimer = nextTimer.nextTimer;
230: if (nextTimer != null)
231: buf.append(", ");
232: }
233:
234: buf.append(")");
235: return buf.toString();
236: }
237: }
|