front_processing/extern/TUIO_JAVA/src/TUIO/TuioClient.java
changeset 10 925b7ee746e3
parent 9 0f44b7360c8d
child 28 9ccef81f02ab
--- a/front_processing/extern/TUIO_JAVA/src/TUIO/TuioClient.java	Thu Mar 22 18:15:53 2012 +0100
+++ b/front_processing/extern/TUIO_JAVA/src/TUIO/TuioClient.java	Fri Mar 23 16:24:36 2012 +0100
@@ -1,643 +1,643 @@
-/*
-    TUIO Java backend - part of the reacTIVision project
-    http://reactivision.sourceforge.net/
-
-    Copyright (c) 2005-2009 Martin Kaltenbrunner <mkalten@iua.upf.edu>
-
-    This program is free software; you can redistribute it and/or modify
-    it under the terms of the GNU General Public License as published by
-    the Free Software Foundation; either version 2 of the License, or
-    (at your option) any later version.
-
-    This program is distributed in the hope that it will be useful,
-    but WITHOUT ANY WARRANTY; without even the implied warranty of
-    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
-    GNU General Public License for more details.
-
-    You should have received a copy of the GNU General Public License
-    along with this program; if not, write to the Free Software
-    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
-*/
-package TUIO;
-
-import com.illposed.osc.*;
-import java.util.*;
-
-/**
- * The TuioClient class is the central TUIO protocol decoder component. It provides a simple callback infrastructure using the {@link TuioListener} interface.
- * In order to receive and decode TUIO messages an instance of TuioClient needs to be created. The TuioClient instance then generates TUIO events
- * which are broadcasted to all registered classes that implement the {@link TuioListener} interface.<P> 
- * <code>
- * TuioClient client = new TuioClient();<br/>
- * client.addTuioListener(myTuioListener);<br/>
- * client.connect();<br/>
- * </code>
- *
- * @author Martin Kaltenbrunner
- * @version 1.4
- */ 
-public class TuioClient implements OSCListener {
-    
-    public String comm;
-    
-    private int port = 3333;
-    private OSCPortIn oscPort;
-    private boolean connected = false;
-    private Hashtable<Long,TuioObject> objectList = new Hashtable<Long,TuioObject>();
-    private Vector<Long> aliveObjectList = new Vector<Long>();
-    private Vector<Long> newObjectList = new Vector<Long>();
-    private Hashtable<Long,TuioCursor> cursorList = new Hashtable<Long,TuioCursor>();
-    private Vector<Long> aliveCursorList = new Vector<Long>();
-    private Vector<Long> newCursorList = new Vector<Long>();
-    private Hashtable<Long,TuioString> stringList = new Hashtable<Long,TuioString>();
-    private Vector<Long> aliveStringList = new Vector<Long>();
-    private Vector<Long> newStringList = new Vector<Long>();
-
-    private Vector<TuioObject> frameObjects = new Vector<TuioObject>();
-    private Vector<TuioCursor> frameCursors = new Vector<TuioCursor>();
-    private Vector<TuioString> frameStrings = new Vector<TuioString>();
-
-    private Vector<TuioCursor> freeCursorList = new Vector<TuioCursor>();
-    private int maxCursorID = -1;
-    
-    private Vector<TuioString> freeStringList = new Vector<TuioString>();
-    private int maxStringID = -1;
-    
-    private long currentFrame = 0;
-    private TuioTime currentTime;
-
-    private Vector<TuioListener> listenerList = new Vector<TuioListener>();
-    
-    /**
-     * The default constructor creates a client that listens to the default TUIO port 3333
-     */
-    public TuioClient() {}
-
-    /**
-     * This constructor creates a client that listens to the provided port
-     *
-     * @param  port  the listening port number
-     */
-    public TuioClient(int port) {
-        this.port = port;
-    }
-        
-    /**
-     * The TuioClient starts listening to TUIO messages on the configured UDP port
-     * All reveived TUIO messages are decoded and the resulting TUIO events are broadcasted to all registered TuioListeners
-     */
-    public void connect() {
-
-        TuioTime.initSession();
-        currentTime = new TuioTime();
-        currentTime.reset();
-        
-        try {
-            oscPort = new OSCPortIn(port);
-            oscPort.addListener("/tuio/2Dobj",this);
-            oscPort.addListener("/tuio/3Dcur",this);
-            oscPort.addListener("/tuio/_siP",this);
-            oscPort.startListening();
-            connected = true;
-        } catch (Exception e) {
-            System.out.println("TuioClient: failed to connect to port "+port);
-            connected = false;
-        }        
-    }
-    
-    /**
-     * The TuioClient stops listening to TUIO messages on the configured UDP port
-     */
-    public void disconnect() {
-        oscPort.stopListening();
-        try { Thread.sleep(100); }
-        catch (Exception e) {};
-        oscPort.close();
-        connected = false;
-    }
-
-    /**
-     * Returns true if this TuioClient is currently connected.
-     * @return    true if this TuioClient is currently connected
-     */
-    public boolean isConnected() { return connected; }
-
-    /**
-     * Adds the provided TuioListener to the list of registered TUIO event listeners
-     *
-     * @param  listener  the TuioListener to add
-     */
-    public void addTuioListener(TuioListener listener) {
-        listenerList.addElement(listener);
-    }
-    
-    /**
-     * Removes the provided TuioListener from the list of registered TUIO event listeners
-     *
-     * @param  listener  the TuioListener to remove
-     */
-    public void removeTuioListener(TuioListener listener) {    
-        listenerList.removeElement(listener);
-    }
-
-    /**
-     * Removes all TuioListener from the list of registered TUIO event listeners
-     */
-    public void removeAllTuioListeners() {    
-        listenerList.clear();
-    }
-    
-    /**
-     * Returns a Vector of all currently active TuioObjects
-     *
-     * @return  a Vector of all currently active TuioObjects
-     */
-    public Vector<TuioObject> getTuioObjects() {
-        return new Vector<TuioObject>(objectList.values());
-    }
-    
-    /**
-     * Returns a Vector of all currently active TuioCursors
-     *
-     * @return  a Vector of all currently active TuioCursors
-     */
-    public Vector<TuioCursor> getTuioCursors() {
-        return new Vector<TuioCursor>(cursorList.values());
-    }
-    
-    /**
-     * Returns a Vector of all currently active TuioStrings
-     *
-     * @return  a Vector of all currently active TuioStrings
-     */
-    public Vector<TuioString> getTuioStrings() {
-        return new Vector<TuioString>(stringList.values());
-    }
-
-    /**
-     * Returns the TuioObject corresponding to the provided Session ID
-     * or NULL if the Session ID does not refer to an active TuioObject
-     *
-     * @return  an active TuioObject corresponding to the provided Session ID or NULL
-     */
-    public TuioObject getTuioObject(long s_id) {
-        return objectList.get(s_id);
-    }
-    
-    /**
-     * Returns the TuioCursor corresponding to the provided Session ID
-     * or NULL if the Session ID does not refer to an active TuioCursor
-     *
-     * @return  an active TuioCursor corresponding to the provided Session ID or NULL
-     */
-    public TuioCursor getTuioCursor(long s_id) {
-        return cursorList.get(s_id);
-    }
-
-    /**
-     * Returns the TuioString corresponding to the provided Session ID
-     * or NULL if the Session ID does not refer to an active TuioString
-     *
-     * @return  an active TuioString corresponding to the provided Session ID or NULL
-     */
-    public TuioString getTuioString(long s_id) {
-        return stringList.get(s_id);
-    }    
-
-    /**
-     * The OSC callback method where all TUIO messages are received and decoded
-     * and where the TUIO event callbacks are dispatched
-     *
-     * @param  date    the time stamp of the OSC bundle
-     * @param  message    the received OSC message
-     */
-    public void acceptMessage(Date date, OSCMessage message) {
-    
-        Object[] args = message.getArguments();
-        String command = (String)args[0];
-        String address = message.getAddress();
-
-        if (address.equals("/tuio/2Dobj")) {
-
-            if (command.equals("set")) {
-                
-                long s_id  = ((Integer)args[1]).longValue();
-                int c_id  = ((Integer)args[2]).intValue();
-                float xpos = ((Float)args[3]).floatValue();
-                float ypos = ((Float)args[4]).floatValue();
-                float angle = ((Float)args[5]).floatValue();
-                float xspeed = ((Float)args[6]).floatValue();
-                float yspeed = ((Float)args[7]).floatValue();
-                float rspeed = ((Float)args[8]).floatValue();
-                float maccel = ((Float)args[9]).floatValue();
-                float raccel = ((Float)args[10]).floatValue();
-                
-                if (objectList.get(s_id) == null) {
-                
-                    TuioObject addObject = new TuioObject(s_id,c_id,xpos,ypos,angle);
-                    frameObjects.addElement(addObject);
-                    
-                } else {
-                
-                    TuioObject tobj = objectList.get(s_id);
-                    if (tobj==null) return;
-                    if ((tobj.xpos!=xpos) || (tobj.ypos!=ypos) || (tobj.angle!=angle) || (tobj.x_speed!=xspeed) || (tobj.y_speed!=yspeed) || (tobj.rotation_speed!=rspeed) || (tobj.motion_accel!=maccel) || (tobj.rotation_accel!=raccel)) {
-                        
-                        TuioObject updateObject = new TuioObject(s_id,c_id,xpos,ypos,angle);
-                        updateObject.update(xpos,ypos,angle,xspeed,yspeed,rspeed,maccel,raccel);
-                        frameObjects.addElement(updateObject);
-                    }
-                
-                }
-                
-            } else if (command.equals("alive")) {
-    
-                newObjectList.clear();
-                for (int i=1;i<args.length;i++) {
-                    // get the message content
-                    long s_id = ((Integer)args[i]).longValue();
-                    newObjectList.addElement(s_id);
-                    // reduce the object list to the lost objects
-                    if (aliveObjectList.contains(s_id))
-                         aliveObjectList.removeElement(s_id);
-                }
-                
-                // remove the remaining objects
-                for (int i=0;i<aliveObjectList.size();i++) {
-                    TuioObject removeObject = objectList.get(aliveObjectList.elementAt(i));
-                    if (removeObject==null) continue;
-                    removeObject.remove(currentTime);
-                    frameObjects.addElement(removeObject);
-                }
-                    
-            } else if (command.equals("fseq")) {
-                
-                long fseq = ((Integer)args[1]).longValue();
-                boolean lateFrame = false;
-                
-                if (fseq>0) {
-                    if (fseq>currentFrame) currentTime = TuioTime.getSessionTime();
-                    if ((fseq>=currentFrame) || ((currentFrame-fseq)>100)) currentFrame=fseq;
-                    else lateFrame = true;
-                } else if (TuioTime.getSessionTime().subtract(currentTime).getTotalMilliseconds()>100) {
-                    currentTime = TuioTime.getSessionTime();
-                }
-                
-                if (!lateFrame) {
-                    Enumeration<TuioObject> frameEnum = frameObjects.elements();
-                    while(frameEnum.hasMoreElements()) {
-                        TuioObject tobj = frameEnum.nextElement();
-                        
-                        switch (tobj.getTuioState()) {
-                            case TuioObject.TUIO_REMOVED:
-                                TuioObject removeObject = tobj;
-                                removeObject.remove(currentTime);
-                                for (int i=0;i<listenerList.size();i++) {
-                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
-                                    if (listener!=null) listener.removeTuioObject(removeObject);
-                                }                                
-                                objectList.remove(removeObject.getSessionID());
-                                break;
-
-                            case TuioObject.TUIO_ADDED:
-                                TuioObject addObject = new TuioObject(currentTime,tobj.getSessionID(),tobj.getSymbolID(),tobj.getX(),tobj.getY(),tobj.getAngle());
-                                objectList.put(addObject.getSessionID(),addObject);
-                                for (int i=0;i<listenerList.size();i++) {
-                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
-                                    if (listener!=null) listener.addTuioObject(addObject);
-                                }
-                                break;
-                                                                
-                            default:
-                                TuioObject updateObject = objectList.get(tobj.getSessionID());
-                                if ( (tobj.getX()!=updateObject.getX() && tobj.getXSpeed()==0) || (tobj.getY()!=updateObject.getY() && tobj.getYSpeed()==0) )
-                                    updateObject.update(currentTime,tobj.getX(),tobj.getY(),tobj.getAngle());
-                                else
-                                    updateObject.update(currentTime,tobj.getX(),tobj.getY(),tobj.getAngle(),tobj.getXSpeed(),tobj.getYSpeed(),tobj.getRotationSpeed(),tobj.getMotionAccel(),tobj.getRotationAccel());
-
-                                for (int i=0;i<listenerList.size();i++) {
-                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
-                                    if (listener!=null) listener.updateTuioObject(updateObject);
-                                }
-                        }
-                    }
-                    
-                    for (int i=0;i<listenerList.size();i++) {
-                        TuioListener listener = (TuioListener)listenerList.elementAt(i);
-                        if (listener!=null) listener.refresh(new TuioTime(currentTime));
-                    }
-                    
-                    Vector<Long> buffer = aliveObjectList;
-                    aliveObjectList = newObjectList;
-                    // recycling the vector
-                    newObjectList = buffer;                    
-                }
-                frameObjects.clear();
-            }
-        } else if (address.equals("/tuio/3Dcur")) {
-
-            if (command.equals("set")) {
-
-                long s_id  = ((Integer)args[1]).longValue();
-                float xpos = ((Float)args[2]).floatValue();
-                float ypos = ((Float)args[3]).floatValue();
-                float zpos = ((Float)args[4]).floatValue();
-                float xspeed = ((Float)args[5]).floatValue();
-                float yspeed = ((Float)args[6]).floatValue();
-                float maccel = ((Float)args[7]).floatValue();
-                
-                if (cursorList.get(s_id) == null) {
-                                    
-                    TuioCursor addCursor = new TuioCursor(s_id, -1 ,xpos,ypos,zpos);
-                    frameCursors.addElement(addCursor);
-                    
-                } else {
-                
-                    TuioCursor tcur = cursorList.get(s_id);
-                    if (tcur==null) return;
-                    if ((tcur.xpos!=xpos) || (tcur.ypos!=ypos) || (tcur.zpos!=zpos) || (tcur.x_speed!=xspeed) || (tcur.y_speed!=yspeed) || (tcur.motion_accel!=maccel)) {
-
-                        TuioCursor updateCursor = new TuioCursor(s_id,tcur.getCursorID(),xpos,ypos,zpos);
-                        updateCursor.update(xpos,ypos,zpos,xspeed,yspeed,maccel);
-                        frameCursors.addElement(updateCursor);
-                    }
-                }
-                
-                //System.out.println("set cur " + s_id+" "+xpos+" "+ypos+" "+xspeed+" "+yspeed+" "+maccel);
-                
-            } else if (command.equals("alive")) {
-    
-                newCursorList.clear();
-                for (int i=1;i<args.length;i++) {
-                    // get the message content
-                    long s_id = ((Integer)args[i]).longValue();
-                    newCursorList.addElement(s_id);
-                    // reduce the cursor list to the lost cursors
-                    if (aliveCursorList.contains(s_id)) 
-                        aliveCursorList.removeElement(s_id);
-                }
-                
-                // remove the remaining cursors
-                for (int i=0;i<aliveCursorList.size();i++) {
-                    TuioCursor removeCursor = cursorList.get(aliveCursorList.elementAt(i));
-                    if (removeCursor==null) continue;
-                    removeCursor.remove(currentTime);
-                    frameCursors.addElement(removeCursor);
-                }
-                                
-            } else if (command.equals("fseq")) {
-                long fseq = ((Integer)args[1]).longValue();
-                boolean lateFrame = false;
-                
-                if (fseq>0) {
-                    if (fseq>currentFrame) currentTime = TuioTime.getSessionTime();
-                    if ((fseq>=currentFrame) || ((currentFrame-fseq)>100)) currentFrame = fseq;
-                    else lateFrame = true;
-                } else if (TuioTime.getSessionTime().subtract(currentTime).getTotalMilliseconds()>100) {
-                    currentTime = TuioTime.getSessionTime();
-                }
-                if (!lateFrame) {
-
-                    Enumeration<TuioCursor> frameEnum = frameCursors.elements();
-                    while(frameEnum.hasMoreElements()) {
-                        TuioCursor tcur = frameEnum.nextElement();
-                        
-                        switch (tcur.getTuioState()) {
-                            case TuioCursor.TUIO_REMOVED:
-                            
-                                TuioCursor removeCursor = tcur;
-                                removeCursor.remove(currentTime);
-                                
-                                for (int i=0;i<listenerList.size();i++) {
-                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
-                                    if (listener!=null) listener.removeTuioCursor(removeCursor);
-                                }
-
-                                cursorList.remove(removeCursor.getSessionID());
-
-                                if (removeCursor.getCursorID()==maxCursorID) {
-                                    maxCursorID = -1;
-                                    if (cursorList.size()>0) {
-                                        Enumeration<TuioCursor> clist = cursorList.elements();
-                                        while (clist.hasMoreElements()) {
-                                            int c_id = clist.nextElement().getCursorID();
-                                            if (c_id>maxCursorID) maxCursorID=c_id;
-                                        }
-                                        
-                                        Enumeration<TuioCursor> flist = freeCursorList.elements();
-                                        while (flist.hasMoreElements()) {
-                                            int c_id = flist.nextElement().getCursorID();
-                                            if (c_id>=maxCursorID) freeCursorList.removeElement(c_id);
-                                        }
-                                    } else freeCursorList.clear();
-                                } else if (removeCursor.getCursorID()<maxCursorID) {
-                                    freeCursorList.addElement(removeCursor);
-                                }
-                                
-                                break;
-
-                            case TuioCursor.TUIO_ADDED:
-
-                                int c_id = cursorList.size();
-                                if ((cursorList.size()<=maxCursorID) && (freeCursorList.size()>0)) {
-                                    TuioCursor closestCursor = freeCursorList.firstElement();
-                                    Enumeration<TuioCursor> testList = freeCursorList.elements();
-                                    while (testList.hasMoreElements()) {
-                                        TuioCursor testCursor = testList.nextElement();
-                                        if (testCursor.getDistance(tcur)<closestCursor.getDistance(tcur)) closestCursor = testCursor;
-                                    }
-                                    c_id = closestCursor.getCursorID();
-                                    freeCursorList.removeElement(closestCursor);
-                                } else maxCursorID = c_id;        
-                                
-                                TuioCursor addCursor = new TuioCursor(currentTime,tcur.getSessionID(),c_id,tcur.getX(),tcur.getY(),tcur.getZ());
-                                cursorList.put(addCursor.getSessionID(),addCursor);
-                                
-                                for (int i=0;i<listenerList.size();i++) {
-                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
-                                    if (listener!=null) listener.addTuioCursor(addCursor);
-                                }
-                                break;
-                                
-                            default:
-                                
-                                TuioCursor updateCursor = cursorList.get(tcur.getSessionID());
-                                if ( (tcur.getX()!=updateCursor.getX() && tcur.getXSpeed()==0) || (tcur.getY()!=updateCursor.getY() && tcur.getYSpeed()==0) || (tcur.getZ()!=updateCursor.getZ()) )
-                                    updateCursor.update(currentTime,tcur.getX(),tcur.getY(),tcur.getZ());
-                                else 
-                                    updateCursor.update(currentTime,tcur.getX(),tcur.getY(),tcur.getZ(),tcur.getXSpeed(),tcur.getYSpeed(),tcur.getMotionAccel());
-                                    
-                                for (int i=0;i<listenerList.size();i++) {
-                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
-                                    if (listener!=null) listener.updateTuioCursor(updateCursor);
-                                }
-                        }
-                    }
-                    
-                    for (int i=0;i<listenerList.size();i++) {
-                        TuioListener listener = (TuioListener)listenerList.elementAt(i);
-                        if (listener!=null) listener.refresh(new TuioTime(currentTime));
-                    }
-                    
-                    Vector<Long> buffer = aliveCursorList;
-                    aliveCursorList = newCursorList;
-                    // recycling the vector
-                    newCursorList = buffer;                
-                }
-                
-                frameCursors.clear();
-            } 
-        } else if (address.equals("/tuio/_siP")) {
-
-            if (command.equals("set")) {
-
-                long s_id  = ((Integer)args[1]).longValue();
-                String msg = args[2].toString();
-                
-                if (stringList.get(s_id) == null) {
-                                    
-                    TuioString addString = new TuioString(s_id, -1, msg);
-                    frameStrings.addElement(addString);
-                    
-                } else {
-                
-                    TuioString tstr = stringList.get(s_id);
-                    if (tstr==null) return;
-                    if (!tstr.getMessage().equals(msg)) {
-
-                        TuioString updateString = new TuioString(s_id,tstr.getStringID(),msg);
-                        updateString.update(msg);
-                        frameStrings.addElement(updateString);
-                    }
-                }
-                
-                //System.out.println("set cur " + s_id+" "+xpos+" "+ypos+" "+xspeed+" "+yspeed+" "+maccel);
-                
-            } else if (command.equals("alive")) {
-    
-                newStringList.clear();
-                for (int i=1;i<args.length;i++) {
-                    // get the message content
-                    long s_id = ((Integer)args[i]).longValue();
-                    newStringList.addElement(s_id);
-                    // reduce the cursor list to the lost cursors
-                    if (aliveStringList.contains(s_id)) 
-                        aliveStringList.removeElement(s_id);
-                }
-                
-                // remove the remaining cursors
-                for (int i=0;i<aliveStringList.size();i++) {
-                    TuioString removeString = stringList.get(aliveStringList.elementAt(i));
-                    if (removeString==null) continue;
-                    removeString.remove(currentTime);
-                    frameStrings.addElement(removeString);
-                }
-                                
-            } else if (command.equals("fseq")) {
-                long fseq = ((Integer)args[1]).longValue();
-                boolean lateFrame = false;
-                
-                if (fseq>0) {
-                    if (fseq>currentFrame) currentTime = TuioTime.getSessionTime();
-                    if ((fseq>=currentFrame) || ((currentFrame-fseq)>100)) currentFrame = fseq;
-                    else lateFrame = true;
-                } else if (TuioTime.getSessionTime().subtract(currentTime).getTotalMilliseconds()>100) {
-                    currentTime = TuioTime.getSessionTime();
-                }
-                if (!lateFrame) {
-
-                    Enumeration<TuioString> frameEnum = frameStrings.elements();
-                    while(frameEnum.hasMoreElements()) {
-                        TuioString tstr = frameEnum.nextElement();
-                        
-                        switch (tstr.getTuioState()) {
-                            case TuioString.TUIO_REMOVED:
-                            
-                                TuioString removeString = tstr;
-                                removeString.remove(currentTime);
-                                
-                                for (int i=0;i<listenerList.size();i++) {
-                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
-                                    if (listener!=null) listener.removeTuioString(removeString);
-                                }
-
-                                stringList.remove(removeString.getSessionID());
-
-                                if (removeString.getStringID()==maxStringID) {
-                                    maxStringID = -1;
-                                    if (stringList.size()>0) {
-                                        Enumeration<TuioString> slist = stringList.elements();
-                                        while (slist.hasMoreElements()) {
-                                            int sl_id = slist.nextElement().getStringID();
-                                            if (sl_id>maxStringID) maxStringID=sl_id;
-                                        }
-                                        
-                                        Enumeration<TuioString> flist = freeStringList.elements();
-                                        while (flist.hasMoreElements()) {
-                                            int sl_id = flist.nextElement().getStringID();
-                                            if (sl_id>=maxStringID) freeStringList.removeElement(sl_id);
-                                        }
-                                    } else freeStringList.clear();
-                                } else if (removeString.getStringID()<maxStringID) {
-                                    freeStringList.addElement(removeString);
-                                }
-                                
-                                break;
-
-                            case TuioString.TUIO_ADDED:
-
-                                int sl_id = stringList.size();
-                                if ((stringList.size()<=maxStringID) && (freeStringList.size()>0)) {
-                                    TuioString closestString = freeStringList.firstElement();
-                                    Enumeration<TuioString> testList = freeStringList.elements();
-                                    while (testList.hasMoreElements()) {
-                                        TuioString testString = testList.nextElement();
-                                        //if (testString.getDistance(tstr)<closestString.getDistance(tstr))
-                                        closestString = testString;
-                                    }
-                                    sl_id = closestString.getStringID();
-                                    freeStringList.removeElement(closestString);
-                                } else maxStringID = sl_id;        
-                                
-                                TuioString addString = new TuioString(currentTime,tstr.getSessionID(),sl_id,tstr.getMessage());
-                                stringList.put(addString.getSessionID(),addString);
-                                
-                                for (int i=0;i<listenerList.size();i++) {
-                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
-                                    if (listener!=null) listener.addTuioString(addString);
-                                }
-                                break;
-                                
-                            default:
-                                
-                                TuioString updateString = stringList.get(tstr.getSessionID());
-                                if ( tstr.getMessage()!=updateString.getMessage() )
-                                    updateString.update(currentTime,tstr.getMessage());
-                                else 
-                                    updateString.update(currentTime,tstr.getMessage());
-                                    
-                                for (int i=0;i<listenerList.size();i++) {
-                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
-                                    if (listener!=null) listener.updateTuioString(updateString);
-                                }
-                        }
-                    }
-                    
-                    for (int i=0;i<listenerList.size();i++) {
-                        TuioListener listener = (TuioListener)listenerList.elementAt(i);
-                        if (listener!=null) listener.refresh(new TuioTime(currentTime));
-                    }
-                    
-                    Vector<Long> buffer = aliveStringList;
-                    aliveStringList = newStringList;
-                    // recycling the vector
-                    newStringList = buffer;                
-                }
-                
-                frameStrings.clear();
-            } 
-        }
-        
-        
-    }
-}
+/*
+    TUIO Java backend - part of the reacTIVision project
+    http://reactivision.sourceforge.net/
+
+    Copyright (c) 2005-2009 Martin Kaltenbrunner <mkalten@iua.upf.edu>
+
+    This program is free software; you can redistribute it and/or modify
+    it under the terms of the GNU General Public License as published by
+    the Free Software Foundation; either version 2 of the License, or
+    (at your option) any later version.
+
+    This program is distributed in the hope that it will be useful,
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+    GNU General Public License for more details.
+
+    You should have received a copy of the GNU General Public License
+    along with this program; if not, write to the Free Software
+    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+*/
+package TUIO;
+
+import com.illposed.osc.*;
+import java.util.*;
+
+/**
+ * The TuioClient class is the central TUIO protocol decoder component. It provides a simple callback infrastructure using the {@link TuioListener} interface.
+ * In order to receive and decode TUIO messages an instance of TuioClient needs to be created. The TuioClient instance then generates TUIO events
+ * which are broadcasted to all registered classes that implement the {@link TuioListener} interface.<P> 
+ * <code>
+ * TuioClient client = new TuioClient();<br/>
+ * client.addTuioListener(myTuioListener);<br/>
+ * client.connect();<br/>
+ * </code>
+ *
+ * @author Martin Kaltenbrunner
+ * @version 1.4
+ */ 
+public class TuioClient implements OSCListener {
+    
+    public String comm;
+    
+    private int port = 3333;
+    private OSCPortIn oscPort;
+    private boolean connected = false;
+    private Hashtable<Long,TuioObject> objectList = new Hashtable<Long,TuioObject>();
+    private Vector<Long> aliveObjectList = new Vector<Long>();
+    private Vector<Long> newObjectList = new Vector<Long>();
+    private Hashtable<Long,TuioCursor> cursorList = new Hashtable<Long,TuioCursor>();
+    private Vector<Long> aliveCursorList = new Vector<Long>();
+    private Vector<Long> newCursorList = new Vector<Long>();
+    private Hashtable<Long,TuioString> stringList = new Hashtable<Long,TuioString>();
+    private Vector<Long> aliveStringList = new Vector<Long>();
+    private Vector<Long> newStringList = new Vector<Long>();
+
+    private Vector<TuioObject> frameObjects = new Vector<TuioObject>();
+    private Vector<TuioCursor> frameCursors = new Vector<TuioCursor>();
+    private Vector<TuioString> frameStrings = new Vector<TuioString>();
+
+    private Vector<TuioCursor> freeCursorList = new Vector<TuioCursor>();
+    private int maxCursorID = -1;
+    
+    private Vector<TuioString> freeStringList = new Vector<TuioString>();
+    private int maxStringID = -1;
+    
+    private long currentFrame = 0;
+    private TuioTime currentTime;
+
+    private Vector<TuioListener> listenerList = new Vector<TuioListener>();
+    
+    /**
+     * The default constructor creates a client that listens to the default TUIO port 3333
+     */
+    public TuioClient() {}
+
+    /**
+     * This constructor creates a client that listens to the provided port
+     *
+     * @param  port  the listening port number
+     */
+    public TuioClient(int port) {
+        this.port = port;
+    }
+        
+    /**
+     * The TuioClient starts listening to TUIO messages on the configured UDP port
+     * All reveived TUIO messages are decoded and the resulting TUIO events are broadcasted to all registered TuioListeners
+     */
+    public void connect() {
+
+        TuioTime.initSession();
+        currentTime = new TuioTime();
+        currentTime.reset();
+        
+        try {
+            oscPort = new OSCPortIn(port);
+            oscPort.addListener("/tuio/2Dobj",this);
+            oscPort.addListener("/tuio/3Dcur",this);
+            oscPort.addListener("/tuio/_siP",this);
+            oscPort.startListening();
+            connected = true;
+        } catch (Exception e) {
+            System.out.println("TuioClient: failed to connect to port "+port);
+            connected = false;
+        }        
+    }
+    
+    /**
+     * The TuioClient stops listening to TUIO messages on the configured UDP port
+     */
+    public void disconnect() {
+        oscPort.stopListening();
+        try { Thread.sleep(100); }
+        catch (Exception e) {};
+        oscPort.close();
+        connected = false;
+    }
+
+    /**
+     * Returns true if this TuioClient is currently connected.
+     * @return    true if this TuioClient is currently connected
+     */
+    public boolean isConnected() { return connected; }
+
+    /**
+     * Adds the provided TuioListener to the list of registered TUIO event listeners
+     *
+     * @param  listener  the TuioListener to add
+     */
+    public void addTuioListener(TuioListener listener) {
+        listenerList.addElement(listener);
+    }
+    
+    /**
+     * Removes the provided TuioListener from the list of registered TUIO event listeners
+     *
+     * @param  listener  the TuioListener to remove
+     */
+    public void removeTuioListener(TuioListener listener) {    
+        listenerList.removeElement(listener);
+    }
+
+    /**
+     * Removes all TuioListener from the list of registered TUIO event listeners
+     */
+    public void removeAllTuioListeners() {    
+        listenerList.clear();
+    }
+    
+    /**
+     * Returns a Vector of all currently active TuioObjects
+     *
+     * @return  a Vector of all currently active TuioObjects
+     */
+    public Vector<TuioObject> getTuioObjects() {
+        return new Vector<TuioObject>(objectList.values());
+    }
+    
+    /**
+     * Returns a Vector of all currently active TuioCursors
+     *
+     * @return  a Vector of all currently active TuioCursors
+     */
+    public Vector<TuioCursor> getTuioCursors() {
+        return new Vector<TuioCursor>(cursorList.values());
+    }
+    
+    /**
+     * Returns a Vector of all currently active TuioStrings
+     *
+     * @return  a Vector of all currently active TuioStrings
+     */
+    public Vector<TuioString> getTuioStrings() {
+        return new Vector<TuioString>(stringList.values());
+    }
+
+    /**
+     * Returns the TuioObject corresponding to the provided Session ID
+     * or NULL if the Session ID does not refer to an active TuioObject
+     *
+     * @return  an active TuioObject corresponding to the provided Session ID or NULL
+     */
+    public TuioObject getTuioObject(long s_id) {
+        return objectList.get(s_id);
+    }
+    
+    /**
+     * Returns the TuioCursor corresponding to the provided Session ID
+     * or NULL if the Session ID does not refer to an active TuioCursor
+     *
+     * @return  an active TuioCursor corresponding to the provided Session ID or NULL
+     */
+    public TuioCursor getTuioCursor(long s_id) {
+        return cursorList.get(s_id);
+    }
+
+    /**
+     * Returns the TuioString corresponding to the provided Session ID
+     * or NULL if the Session ID does not refer to an active TuioString
+     *
+     * @return  an active TuioString corresponding to the provided Session ID or NULL
+     */
+    public TuioString getTuioString(long s_id) {
+        return stringList.get(s_id);
+    }    
+
+    /**
+     * The OSC callback method where all TUIO messages are received and decoded
+     * and where the TUIO event callbacks are dispatched
+     *
+     * @param  date    the time stamp of the OSC bundle
+     * @param  message    the received OSC message
+     */
+    public void acceptMessage(Date date, OSCMessage message) {
+    
+        Object[] args = message.getArguments();
+        String command = (String)args[0];
+        String address = message.getAddress();
+
+        if (address.equals("/tuio/2Dobj")) {
+
+            if (command.equals("set")) {
+                
+                long s_id  = ((Integer)args[1]).longValue();
+                int c_id  = ((Integer)args[2]).intValue();
+                float xpos = ((Float)args[3]).floatValue();
+                float ypos = ((Float)args[4]).floatValue();
+                float angle = ((Float)args[5]).floatValue();
+                float xspeed = ((Float)args[6]).floatValue();
+                float yspeed = ((Float)args[7]).floatValue();
+                float rspeed = ((Float)args[8]).floatValue();
+                float maccel = ((Float)args[9]).floatValue();
+                float raccel = ((Float)args[10]).floatValue();
+                
+                if (objectList.get(s_id) == null) {
+                
+                    TuioObject addObject = new TuioObject(s_id,c_id,xpos,ypos,angle);
+                    frameObjects.addElement(addObject);
+                    
+                } else {
+                
+                    TuioObject tobj = objectList.get(s_id);
+                    if (tobj==null) return;
+                    if ((tobj.xpos!=xpos) || (tobj.ypos!=ypos) || (tobj.angle!=angle) || (tobj.x_speed!=xspeed) || (tobj.y_speed!=yspeed) || (tobj.rotation_speed!=rspeed) || (tobj.motion_accel!=maccel) || (tobj.rotation_accel!=raccel)) {
+                        
+                        TuioObject updateObject = new TuioObject(s_id,c_id,xpos,ypos,angle);
+                        updateObject.update(xpos,ypos,angle,xspeed,yspeed,rspeed,maccel,raccel);
+                        frameObjects.addElement(updateObject);
+                    }
+                
+                }
+                
+            } else if (command.equals("alive")) {
+    
+                newObjectList.clear();
+                for (int i=1;i<args.length;i++) {
+                    // get the message content
+                    long s_id = ((Integer)args[i]).longValue();
+                    newObjectList.addElement(s_id);
+                    // reduce the object list to the lost objects
+                    if (aliveObjectList.contains(s_id))
+                         aliveObjectList.removeElement(s_id);
+                }
+                
+                // remove the remaining objects
+                for (int i=0;i<aliveObjectList.size();i++) {
+                    TuioObject removeObject = objectList.get(aliveObjectList.elementAt(i));
+                    if (removeObject==null) continue;
+                    removeObject.remove(currentTime);
+                    frameObjects.addElement(removeObject);
+                }
+                    
+            } else if (command.equals("fseq")) {
+                
+                long fseq = ((Integer)args[1]).longValue();
+                boolean lateFrame = false;
+                
+                if (fseq>0) {
+                    if (fseq>currentFrame) currentTime = TuioTime.getSessionTime();
+                    if ((fseq>=currentFrame) || ((currentFrame-fseq)>100)) currentFrame=fseq;
+                    else lateFrame = true;
+                } else if (TuioTime.getSessionTime().subtract(currentTime).getTotalMilliseconds()>100) {
+                    currentTime = TuioTime.getSessionTime();
+                }
+                
+                if (!lateFrame) {
+                    Enumeration<TuioObject> frameEnum = frameObjects.elements();
+                    while(frameEnum.hasMoreElements()) {
+                        TuioObject tobj = frameEnum.nextElement();
+                        
+                        switch (tobj.getTuioState()) {
+                            case TuioObject.TUIO_REMOVED:
+                                TuioObject removeObject = tobj;
+                                removeObject.remove(currentTime);
+                                for (int i=0;i<listenerList.size();i++) {
+                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
+                                    if (listener!=null) listener.removeTuioObject(removeObject);
+                                }                                
+                                objectList.remove(removeObject.getSessionID());
+                                break;
+
+                            case TuioObject.TUIO_ADDED:
+                                TuioObject addObject = new TuioObject(currentTime,tobj.getSessionID(),tobj.getSymbolID(),tobj.getX(),tobj.getY(),tobj.getAngle());
+                                objectList.put(addObject.getSessionID(),addObject);
+                                for (int i=0;i<listenerList.size();i++) {
+                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
+                                    if (listener!=null) listener.addTuioObject(addObject);
+                                }
+                                break;
+                                                                
+                            default:
+                                TuioObject updateObject = objectList.get(tobj.getSessionID());
+                                if ( (tobj.getX()!=updateObject.getX() && tobj.getXSpeed()==0) || (tobj.getY()!=updateObject.getY() && tobj.getYSpeed()==0) )
+                                    updateObject.update(currentTime,tobj.getX(),tobj.getY(),tobj.getAngle());
+                                else
+                                    updateObject.update(currentTime,tobj.getX(),tobj.getY(),tobj.getAngle(),tobj.getXSpeed(),tobj.getYSpeed(),tobj.getRotationSpeed(),tobj.getMotionAccel(),tobj.getRotationAccel());
+
+                                for (int i=0;i<listenerList.size();i++) {
+                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
+                                    if (listener!=null) listener.updateTuioObject(updateObject);
+                                }
+                        }
+                    }
+                    
+                    for (int i=0;i<listenerList.size();i++) {
+                        TuioListener listener = (TuioListener)listenerList.elementAt(i);
+                        if (listener!=null) listener.refresh(new TuioTime(currentTime));
+                    }
+                    
+                    Vector<Long> buffer = aliveObjectList;
+                    aliveObjectList = newObjectList;
+                    // recycling the vector
+                    newObjectList = buffer;                    
+                }
+                frameObjects.clear();
+            }
+        } else if (address.equals("/tuio/3Dcur")) {
+
+            if (command.equals("set")) {
+
+                long s_id  = ((Integer)args[1]).longValue();
+                float xpos = ((Float)args[2]).floatValue();
+                float ypos = ((Float)args[3]).floatValue();
+                float zpos = ((Float)args[4]).floatValue();
+                float xspeed = ((Float)args[5]).floatValue();
+                float yspeed = ((Float)args[6]).floatValue();
+                float maccel = ((Float)args[7]).floatValue();
+                
+                if (cursorList.get(s_id) == null) {
+                                    
+                    TuioCursor addCursor = new TuioCursor(s_id, -1 ,xpos,ypos,zpos);
+                    frameCursors.addElement(addCursor);
+                    
+                } else {
+                
+                    TuioCursor tcur = cursorList.get(s_id);
+                    if (tcur==null) return;
+                    if ((tcur.xpos!=xpos) || (tcur.ypos!=ypos) || (tcur.zpos!=zpos) || (tcur.x_speed!=xspeed) || (tcur.y_speed!=yspeed) || (tcur.motion_accel!=maccel)) {
+
+                        TuioCursor updateCursor = new TuioCursor(s_id,tcur.getCursorID(),xpos,ypos,zpos);
+                        updateCursor.update(xpos,ypos,zpos,xspeed,yspeed,maccel);
+                        frameCursors.addElement(updateCursor);
+                    }
+                }
+                
+                //System.out.println("set cur " + s_id+" "+xpos+" "+ypos+" "+xspeed+" "+yspeed+" "+maccel);
+                
+            } else if (command.equals("alive")) {
+    
+                newCursorList.clear();
+                for (int i=1;i<args.length;i++) {
+                    // get the message content
+                    long s_id = ((Integer)args[i]).longValue();
+                    newCursorList.addElement(s_id);
+                    // reduce the cursor list to the lost cursors
+                    if (aliveCursorList.contains(s_id)) 
+                        aliveCursorList.removeElement(s_id);
+                }
+                
+                // remove the remaining cursors
+                for (int i=0;i<aliveCursorList.size();i++) {
+                    TuioCursor removeCursor = cursorList.get(aliveCursorList.elementAt(i));
+                    if (removeCursor==null) continue;
+                    removeCursor.remove(currentTime);
+                    frameCursors.addElement(removeCursor);
+                }
+                                
+            } else if (command.equals("fseq")) {
+                long fseq = ((Integer)args[1]).longValue();
+                boolean lateFrame = false;
+                
+                if (fseq>0) {
+                    if (fseq>currentFrame) currentTime = TuioTime.getSessionTime();
+                    if ((fseq>=currentFrame) || ((currentFrame-fseq)>100)) currentFrame = fseq;
+                    else lateFrame = true;
+                } else if (TuioTime.getSessionTime().subtract(currentTime).getTotalMilliseconds()>100) {
+                    currentTime = TuioTime.getSessionTime();
+                }
+                if (!lateFrame) {
+
+                    Enumeration<TuioCursor> frameEnum = frameCursors.elements();
+                    while(frameEnum.hasMoreElements()) {
+                        TuioCursor tcur = frameEnum.nextElement();
+                        
+                        switch (tcur.getTuioState()) {
+                            case TuioCursor.TUIO_REMOVED:
+                            
+                                TuioCursor removeCursor = tcur;
+                                removeCursor.remove(currentTime);
+                                
+                                for (int i=0;i<listenerList.size();i++) {
+                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
+                                    if (listener!=null) listener.removeTuioCursor(removeCursor);
+                                }
+
+                                cursorList.remove(removeCursor.getSessionID());
+
+                                if (removeCursor.getCursorID()==maxCursorID) {
+                                    maxCursorID = -1;
+                                    if (cursorList.size()>0) {
+                                        Enumeration<TuioCursor> clist = cursorList.elements();
+                                        while (clist.hasMoreElements()) {
+                                            int c_id = clist.nextElement().getCursorID();
+                                            if (c_id>maxCursorID) maxCursorID=c_id;
+                                        }
+                                        
+                                        Enumeration<TuioCursor> flist = freeCursorList.elements();
+                                        while (flist.hasMoreElements()) {
+                                            int c_id = flist.nextElement().getCursorID();
+                                            if (c_id>=maxCursorID) freeCursorList.removeElement(c_id);
+                                        }
+                                    } else freeCursorList.clear();
+                                } else if (removeCursor.getCursorID()<maxCursorID) {
+                                    freeCursorList.addElement(removeCursor);
+                                }
+                                
+                                break;
+
+                            case TuioCursor.TUIO_ADDED:
+
+                                int c_id = cursorList.size();
+                                if ((cursorList.size()<=maxCursorID) && (freeCursorList.size()>0)) {
+                                    TuioCursor closestCursor = freeCursorList.firstElement();
+                                    Enumeration<TuioCursor> testList = freeCursorList.elements();
+                                    while (testList.hasMoreElements()) {
+                                        TuioCursor testCursor = testList.nextElement();
+                                        if (testCursor.getDistance(tcur)<closestCursor.getDistance(tcur)) closestCursor = testCursor;
+                                    }
+                                    c_id = closestCursor.getCursorID();
+                                    freeCursorList.removeElement(closestCursor);
+                                } else maxCursorID = c_id;        
+                                
+                                TuioCursor addCursor = new TuioCursor(currentTime,tcur.getSessionID(),c_id,tcur.getX(),tcur.getY(),tcur.getZ());
+                                cursorList.put(addCursor.getSessionID(),addCursor);
+                                
+                                for (int i=0;i<listenerList.size();i++) {
+                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
+                                    if (listener!=null) listener.addTuioCursor(addCursor);
+                                }
+                                break;
+                                
+                            default:
+                                
+                                TuioCursor updateCursor = cursorList.get(tcur.getSessionID());
+                                if ( (tcur.getX()!=updateCursor.getX() && tcur.getXSpeed()==0) || (tcur.getY()!=updateCursor.getY() && tcur.getYSpeed()==0) || (tcur.getZ()!=updateCursor.getZ()) )
+                                    updateCursor.update(currentTime,tcur.getX(),tcur.getY(),tcur.getZ());
+                                else 
+                                    updateCursor.update(currentTime,tcur.getX(),tcur.getY(),tcur.getZ(),tcur.getXSpeed(),tcur.getYSpeed(),tcur.getMotionAccel());
+                                    
+                                for (int i=0;i<listenerList.size();i++) {
+                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
+                                    if (listener!=null) listener.updateTuioCursor(updateCursor);
+                                }
+                        }
+                    }
+                    
+                    for (int i=0;i<listenerList.size();i++) {
+                        TuioListener listener = (TuioListener)listenerList.elementAt(i);
+                        if (listener!=null) listener.refresh(new TuioTime(currentTime));
+                    }
+                    
+                    Vector<Long> buffer = aliveCursorList;
+                    aliveCursorList = newCursorList;
+                    // recycling the vector
+                    newCursorList = buffer;                
+                }
+                
+                frameCursors.clear();
+            } 
+        } else if (address.equals("/tuio/_siP")) {
+
+            if (command.equals("set")) {
+
+                long s_id  = ((Integer)args[1]).longValue();
+                String msg = args[2].toString();
+                
+                if (stringList.get(s_id) == null) {
+                                    
+                    TuioString addString = new TuioString(s_id, -1, msg);
+                    frameStrings.addElement(addString);
+                    
+                } else {
+                
+                    TuioString tstr = stringList.get(s_id);
+                    if (tstr==null) return;
+                    if (!tstr.getMessage().equals(msg)) {
+
+                        TuioString updateString = new TuioString(s_id,tstr.getStringID(),msg);
+                        updateString.update(msg);
+                        frameStrings.addElement(updateString);
+                    }
+                }
+                
+                //System.out.println("set cur " + s_id+" "+xpos+" "+ypos+" "+xspeed+" "+yspeed+" "+maccel);
+                
+            } else if (command.equals("alive")) {
+    
+                newStringList.clear();
+                for (int i=1;i<args.length;i++) {
+                    // get the message content
+                    long s_id = ((Integer)args[i]).longValue();
+                    newStringList.addElement(s_id);
+                    // reduce the cursor list to the lost cursors
+                    if (aliveStringList.contains(s_id)) 
+                        aliveStringList.removeElement(s_id);
+                }
+                
+                // remove the remaining cursors
+                for (int i=0;i<aliveStringList.size();i++) {
+                    TuioString removeString = stringList.get(aliveStringList.elementAt(i));
+                    if (removeString==null) continue;
+                    removeString.remove(currentTime);
+                    frameStrings.addElement(removeString);
+                }
+                                
+            } else if (command.equals("fseq")) {
+                long fseq = ((Integer)args[1]).longValue();
+                boolean lateFrame = false;
+                
+                if (fseq>0) {
+                    if (fseq>currentFrame) currentTime = TuioTime.getSessionTime();
+                    if ((fseq>=currentFrame) || ((currentFrame-fseq)>100)) currentFrame = fseq;
+                    else lateFrame = true;
+                } else if (TuioTime.getSessionTime().subtract(currentTime).getTotalMilliseconds()>100) {
+                    currentTime = TuioTime.getSessionTime();
+                }
+                if (!lateFrame) {
+
+                    Enumeration<TuioString> frameEnum = frameStrings.elements();
+                    while(frameEnum.hasMoreElements()) {
+                        TuioString tstr = frameEnum.nextElement();
+                        
+                        switch (tstr.getTuioState()) {
+                            case TuioString.TUIO_REMOVED:
+                            
+                                TuioString removeString = tstr;
+                                removeString.remove(currentTime);
+                                
+                                for (int i=0;i<listenerList.size();i++) {
+                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
+                                    if (listener!=null) listener.removeTuioString(removeString);
+                                }
+
+                                stringList.remove(removeString.getSessionID());
+
+                                if (removeString.getStringID()==maxStringID) {
+                                    maxStringID = -1;
+                                    if (stringList.size()>0) {
+                                        Enumeration<TuioString> slist = stringList.elements();
+                                        while (slist.hasMoreElements()) {
+                                            int sl_id = slist.nextElement().getStringID();
+                                            if (sl_id>maxStringID) maxStringID=sl_id;
+                                        }
+                                        
+                                        Enumeration<TuioString> flist = freeStringList.elements();
+                                        while (flist.hasMoreElements()) {
+                                            int sl_id = flist.nextElement().getStringID();
+                                            if (sl_id>=maxStringID) freeStringList.removeElement(sl_id);
+                                        }
+                                    } else freeStringList.clear();
+                                } else if (removeString.getStringID()<maxStringID) {
+                                    freeStringList.addElement(removeString);
+                                }
+                                
+                                break;
+
+                            case TuioString.TUIO_ADDED:
+
+                                int sl_id = stringList.size();
+                                if ((stringList.size()<=maxStringID) && (freeStringList.size()>0)) {
+                                    TuioString closestString = freeStringList.firstElement();
+                                    Enumeration<TuioString> testList = freeStringList.elements();
+                                    while (testList.hasMoreElements()) {
+                                        TuioString testString = testList.nextElement();
+                                        //if (testString.getDistance(tstr)<closestString.getDistance(tstr))
+                                        closestString = testString;
+                                    }
+                                    sl_id = closestString.getStringID();
+                                    freeStringList.removeElement(closestString);
+                                } else maxStringID = sl_id;        
+                                
+                                TuioString addString = new TuioString(currentTime,tstr.getSessionID(),sl_id,tstr.getMessage());
+                                stringList.put(addString.getSessionID(),addString);
+                                
+                                for (int i=0;i<listenerList.size();i++) {
+                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
+                                    if (listener!=null) listener.addTuioString(addString);
+                                }
+                                break;
+                                
+                            default:
+                                
+                                TuioString updateString = stringList.get(tstr.getSessionID());
+                                if ( tstr.getMessage()!=updateString.getMessage() )
+                                    updateString.update(currentTime,tstr.getMessage());
+                                else 
+                                    updateString.update(currentTime,tstr.getMessage());
+                                    
+                                for (int i=0;i<listenerList.size();i++) {
+                                    TuioListener listener = (TuioListener)listenerList.elementAt(i);
+                                    if (listener!=null) listener.updateTuioString(updateString);
+                                }
+                        }
+                    }
+                    
+                    for (int i=0;i<listenerList.size();i++) {
+                        TuioListener listener = (TuioListener)listenerList.elementAt(i);
+                        if (listener!=null) listener.refresh(new TuioTime(currentTime));
+                    }
+                    
+                    Vector<Long> buffer = aliveStringList;
+                    aliveStringList = newStringList;
+                    // recycling the vector
+                    newStringList = buffer;                
+                }
+                
+                frameStrings.clear();
+            } 
+        }
+        
+        
+    }
+}