Source

Coin / src / vrml97 / ProximitySensor.cpp

Full commit
Peder Blekken a78b2c9 
Marius Kintel 3ba6b41 




























Peder Blekken a78b2c9 

Lars J. Aas 94c1d19 





Peder Blekken a78b2c9 


Peder Blekken b092bcb 



























































































Peder Blekken a78b2c9 

















































Jostein 62cd825 
Peder Blekken a78b2c9 
Peder Blekken 9e0a986 








Lars J. Aas 92263e4 

Peder Blekken a78b2c9 

Peder Blekken 9e0a986 













Peder Blekken a78b2c9 











kristian a3a5361 
Peder Blekken a78b2c9 









Peder Blekken 9e0a986 




Peder Blekken a78b2c9 



















Peder Blekken 9e0a986 





































Peder Blekken 629ed76 
Peder Blekken 9e0a986 
Peder Blekken 629ed76 
Peder Blekken 9e0a986 








Peder Blekken a78b2c9 










Jostein 62cd825 
Peder Blekken a78b2c9 




Jostein 62cd825 
Peder Blekken a78b2c9 




Jostein 62cd825 
Peder Blekken a78b2c9 








Lars J. Aas 94c1d19 
/**************************************************************************\
 * Copyright (c) Kongsberg Oil & Gas Technologies AS
 * All rights reserved.
 * 
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are
 * met:
 * 
 * Redistributions of source code must retain the above copyright notice,
 * this list of conditions and the following disclaimer.
 * 
 * Redistributions in binary form must reproduce the above copyright
 * notice, this list of conditions and the following disclaimer in the
 * documentation and/or other materials provided with the distribution.
 * 
 * Neither the name of the copyright holder nor the names of its
 * contributors may be used to endorse or promote products derived from
 * this software without specific prior written permission.
 * 
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 * HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
\**************************************************************************/

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif // HAVE_CONFIG_H

#ifdef HAVE_VRML97

/*!
  \class SoVRMLProximitySensor SoVRMLProximitySensor.h Inventor/VRMLnodes/SoVRMLProximitySensor.h
  \brief The SoVRMLProximitySensor class is used to generate events when the viewer enters or exits a region.
  \ingroup VRMLnodes

  \WEB3DCOPYRIGHT

  \verbatim
  ProximitySensor {
    exposedField SFVec3f    center      0 0 0    # (-,)
    exposedField SFVec3f    size        0 0 0    # [0,)
    exposedField SFBool     enabled     TRUE
    eventOut     SFBool     isActive
    eventOut     SFVec3f    position_changed
    eventOut     SFRotation orientation_changed
    eventOut     SFTime     enterTime
    eventOut     SFTime     exitTime
  }
  \endverbatim
  
  The ProximitySensor node generates events when the viewer enters,
  exits, and moves within a region in space (defined by a box). 
  
  A proximity sensor is enabled or disabled by sending it an enabled
  event with a value of TRUE or FALSE. A disabled sensor does not send
  events.

  A ProximitySensor node generates isActive TRUE/FALSE events as the
  viewer enters and exits the rectangular box defined by its center
  and size fields. Browsers shall interpolate viewer positions and
  timestamp the isActive events with the exact time the viewer first
  intersected the proximity region.

  The \e center field defines the centre point of the proximity region
  in object space. The \e size field specifies a vector which defines
  the width (x), height (y), and depth (z) of the box bounding the
  region. The components of the size field shall be greater than or
  equal to zero. ProximitySensor nodes are affected by the
  hierarchical transformations of their parents.

  The \e enterTime event is generated whenever the isActive TRUE event
  is generated (user enters the box), and \e exitTime events are
  generated whenever an isActive FALSE event is generated (user exits
  the box).

  The \e position_changed and \e orientation_changed eventOuts send
  events whenever the user is contained within the proximity region
  and the position and orientation of the viewer changes with respect
  to the ProximitySensor node's coordinate system including enter and
  exit times.
  
  The viewer movement may be a result of a variety of circumstances
  resulting from browser navigation, ProximitySensor node's coordinate
  system changes, or bound Viewpoint node's position or orientation
  changes.

  Each ProximitySensor node behaves independently of all other
  ProximitySensor nodes. Every enabled ProximitySensor node that is
  affected by the viewer's movement receives and sends events,
  possibly resulting in multiple

  ProximitySensor nodes receiving and sending events
  simultaneously. Unlike TouchSensor nodes, there is no notion of a
  ProximitySensor node lower in the scene graph "grabbing" events.

  Instanced (DEF/USE) ProximitySensor nodes use the union of all the
  boxes to check for enter and exit. A multiply instanced
  ProximitySensor node will detect enter and exit for all instances of
  the box and send enter/exit events appropriately. However, the
  results are undefined if the any of the boxes of a multiply
  instanced ProximitySensor node overlap.

  A ProximitySensor node that surrounds the entire world has an
  enterTime equal to the time that the world was entered and can be
  used to start up animations or behaviours as soon as a world is
  loaded.

  A ProximitySensor node with a box containing zero volume (i.e., any
  size field element of 0.0) cannot generate events. This is
  equivalent to setting the enabled field to FALSE.

  A ProximitySensor read from a VRML file shall generate isActive
  TRUE, position_changed, orientation_changed and enterTime events if
  the sensor is enabled and the viewer is inside the proximity region.

  A ProximitySensor inserted into the transformation hierarchy shall
  generate isActive TRUE, position_changed, orientation_changed and
  enterTime events if the sensor is enabled and the viewer is inside
  the proximity region.
  
  A ProximitySensor removed from the transformation hierarchy shall
  generate isActive FALSE, position_changed, orientation_changed and
  exitTime events if the sensor is enabled and the viewer is inside
  the proximity region.

*/

/*!
  \var SoSFVec3f SoVRMLProximitySensor::center
  The center of the region. Default value is (0,0,0).
*/

/*!
  \var SoSFVec3f SoVRMLProximitySensor::size
  The region size. Default value is (0, 0, 0).
*/

/*!
  \var SoSFBool SoVRMLProximitySensor::enabled
  Enable/disable sensor. Default value is TRUE.
*/


/*!
  \var SoSFBool SoVRMLProximitySensor::isActive
  An eventOut that is sent every time the sensor changes state.
*/

/*!
  \var SoSFVec3f SoVRMLProximitySensor::position_changed

  An eventOut that is sent when the viewer is inside the region and
  the viewer position or orientation is changed.

*/

/*!
  \var SoSFRotation SoVRMLProximitySensor::orientation_changed

  An eventOut that is sent when the viewer is inside the region and
  the viewer position or orientation is changed.

*/

/*!
  \var SoSFTime SoVRMLProximitySensor::enterTime
  An eventOut that is sent when the viewer enters the region.
*/

/*!
  \var SoSFTime SoVRMLProximitySensor::exitTime
  An eventOut that is sent when the viewer exits the region.
*/

#include <Inventor/VRMLnodes/SoVRMLProximitySensor.h>
#include "coindefs.h"

#include <Inventor/VRMLnodes/SoVRMLMacros.h>
#include <Inventor/actions/SoAction.h>
#include <Inventor/elements/SoViewVolumeElement.h>
#include <Inventor/elements/SoViewingMatrixElement.h>
#include <Inventor/elements/SoModelMatrixElement.h>
#include <Inventor/SoDB.h>
#include <Inventor/SbBox3f.h>
#include <Inventor/misc/SoState.h>

#include "nodes/SoSubNodeP.h"

SO_NODE_SOURCE(SoVRMLProximitySensor);

//
// returns the current time. First tries the realTime field, then
// SbTime::getTimeOfDay() if field is not found.
//
static SbTime
prox_get_current_time(void)
{
  SoField * realtime = SoDB::getGlobalField("realTime");
  if (realtime && realtime->isOfType(SoSFTime::getClassTypeId())) {
    return ((SoSFTime*)realtime)->getValue();
  }
  return SbTime::getTimeOfDay();
}

// Doc in parent
void
SoVRMLProximitySensor::initClass(void)
{
  SO_NODE_INTERNAL_INIT_CLASS(SoVRMLProximitySensor, SO_VRML97_NODE_TYPE);
}

/*!
  Constructor.
*/
SoVRMLProximitySensor::SoVRMLProximitySensor(void)
{
  SO_VRMLNODE_INTERNAL_CONSTRUCTOR(SoVRMLProximitySensor);

  SO_VRMLNODE_ADD_EXPOSED_FIELD(center, (0.0f, 0.0f, 0.0f));
  SO_VRMLNODE_ADD_EXPOSED_FIELD(size, (0.0f, 0.0f, 0.0f));
  SO_VRMLNODE_ADD_EXPOSED_FIELD(enabled, (TRUE));

  SO_VRMLNODE_ADD_EVENT_OUT(isActive);
  SO_VRMLNODE_ADD_EVENT_OUT(position_changed);
  SO_VRMLNODE_ADD_EVENT_OUT(orientation_changed);
  SO_VRMLNODE_ADD_EVENT_OUT(enterTime);
  SO_VRMLNODE_ADD_EVENT_OUT(exitTime);

  // initialize eventOut values that we might read
  this->isActive = FALSE;
  this->position_changed = SbVec3f(0.0f, 0.0f, 0.0f);
  this->orientation_changed = SbRotation();
}

/*!
  Destructor.
*/
SoVRMLProximitySensor::~SoVRMLProximitySensor()
{
}

// Doc in parent
SbBool
SoVRMLProximitySensor::affectsState(void) const
{
  return FALSE;
}

// Doc in parent
void
SoVRMLProximitySensor::doAction(SoAction * action)
{
  if (!this->enabled.getValue()) return;

  SbBool wasactive = this->isActive.getValue();
  SbVec3f s = this->size.getValue() * 0.5f;
  SbTime currtime = prox_get_current_time();

  if (s[0] <= 0.0f || s[1] <= 0.0f || s[2] <= 0.0f) {
    if (wasactive) {
      this->isActive = FALSE;
      this->exitTime = currtime;
    }
    return;
  }

  SoState * state = action->getState();
  const SbViewVolume & vv = SoViewVolumeElement::get(state);
  const SbMatrix & mm = SoModelMatrixElement::get(state);
  const SbMatrix & vm = SoViewingMatrixElement::get(state);

  // FIXME: if it's not possible to invert the matrix, move proximity
  // area to world space and do the testing there.

  SbVec3f viewer = vv.getProjectionPoint(); // world space
  mm.inverse().multVecMatrix(viewer, viewer); // object space

  SbVec3f c = this->center.getValue();
  SbBox3f box(c[0]-s[0], c[1]-s[1], c[2]-s[2],
              c[0]+s[0], c[1]+s[1], c[2]+s[2]);
  SbBool inside = box.intersect(viewer);

  SbRotation oldrot = this->orientation_changed.getValue();
  SbVec3f oldpos = this->position_changed.getValue();

  if (inside) {
    if (!wasactive) {
      this->isActive = TRUE;
      this->enterTime = currtime;
    }
    SbRotation newrot = SbRotation(vm).inverse();
    if (!wasactive || newrot != oldrot) {
      this->orientation_changed = newrot;
    }
    if (!wasactive || viewer != oldpos) {
      this->position_changed = viewer;
    }
  }
  else if (!inside && wasactive) {
    this->isActive = FALSE;
    this->exitTime = currtime;
  }  
}

// Doc in parent
void
SoVRMLProximitySensor::GLRender(SoGLRenderAction * action)
{
  SoVRMLProximitySensor::doAction((SoAction*) action);
}

// Doc in parent
void
SoVRMLProximitySensor::callback(SoCallbackAction * COIN_UNUSED_ARG(action))
{
}

// Doc in parent
void
SoVRMLProximitySensor::rayPick(SoRayPickAction * COIN_UNUSED_ARG(action))
{
}

// Doc in parent
void
SoVRMLProximitySensor::getBoundingBox(SoGetBoundingBoxAction * COIN_UNUSED_ARG(action))
{
}

// Doc in parent
void
SoVRMLProximitySensor::notify(SoNotList * list)
{
  inherited::notify(list);
}

#endif // HAVE_VRML97