front_processing/extern/TUIO_JAVA/src/TUIO/TuioClient.java
author bastiena
Fri, 23 Mar 2012 16:24:36 +0100
changeset 10 925b7ee746e3
parent 9 0f44b7360c8d
child 28 9ccef81f02ab
permissions -rw-r--r--
Front Processing : Changed utf-8 to utf-8 without BOM

/*
    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();
            } 
        }
        
        
    }
}