#include #include #include #include #include #include #include #include #include using namespace osg; using namespace osgUtil; Hit::Hit() { } Hit::Hit(const Hit& hit) { // copy data across. _ratio = hit._ratio; _originalLineSegment = hit._originalLineSegment; _localLineSegment = hit._localLineSegment; _nodePath = hit._nodePath; _geode = hit._geode; _drawable = hit._drawable; _matrix = hit._matrix; _inverse = hit._inverse; _vecIndexList = hit._vecIndexList; _primitiveIndex = hit._primitiveIndex; _intersectPoint = hit._intersectPoint; _intersectNormal = hit._intersectNormal; } Hit::~Hit() { } Hit& Hit::operator = (const Hit& hit) { if (&hit==this) return *this; _matrix = hit._matrix; _inverse = hit._inverse; _originalLineSegment = hit._originalLineSegment; _localLineSegment = hit._localLineSegment; // copy data across. _ratio = hit._ratio; _originalLineSegment = hit._originalLineSegment; _localLineSegment = hit._localLineSegment; _nodePath = hit._nodePath; _geode = hit._geode; _drawable = hit._drawable; _vecIndexList = hit._vecIndexList; _primitiveIndex = hit._primitiveIndex; _intersectPoint = hit._intersectPoint; _intersectNormal = hit._intersectNormal; return *this; } const osg::Vec3 Hit::getWorldIntersectNormal() const { if (_inverse.valid()) { osg::Vec3 norm = osg::Matrix::transform3x3(*_inverse,_intersectNormal); norm.normalize(); return norm; } else return _intersectNormal; } IntersectVisitor::IntersectState::IntersectState() { _segmentMaskStack.push_back(0xffffffff); } IntersectVisitor::IntersectState::~IntersectState() { } bool IntersectVisitor::IntersectState::isCulled(const BoundingSphere& bs,LineSegmentmentMask& segMaskOut) { bool hit = false; LineSegmentmentMask mask = 0x00000001; segMaskOut = 0x00000000; LineSegmentmentMask segMaskIn = _segmentMaskStack.back(); // notify(INFO) << << "IntersectState::isCulled() mask in "<second)->intersect(bs)) { // notify(INFO) << << "Hit "; segMaskOut = segMaskOut| mask; hit = true; } mask = mask << 1; } // notify(INFO) << << "mask = "<second)->intersect(bb)) { segMaskOut = segMaskOut| mask; hit = true; } mask = mask << 1; } return !hit; } IntersectVisitor::IntersectVisitor() { // overide the default node visitor mode. setTraversalMode(NodeVisitor::TRAVERSE_ACTIVE_CHILDREN); reset(); } IntersectVisitor::~IntersectVisitor() { _hitReportingMode = ONLY_NEAREST_HIT; } void IntersectVisitor::reset() { _intersectStateStack.clear(); // create a empty IntersectState on the the intersectStateStack. IntersectState* nis = osgNew IntersectState; _intersectStateStack.push_back(nis); } bool IntersectVisitor::hits() { for(LineSegmentHitListMap::iterator itr = _segHitList.begin(); itr != _segHitList.end(); ++itr) { if (!(itr->second.empty())) return true; } return false; } void IntersectVisitor::addLineSegment(LineSegment* seg) { if (!seg) return; if (!seg->valid()) { notify(WARN)<<"Warning: invalid line segment passed to IntersectVisitor::addLineSegment(..)"<start()<<" "<end()<<" segment ignored.."<< std::endl; return; } // first check to see if segment has already been added. for(LineSegmentHitListMap::iterator itr = _segHitList.begin(); itr != _segHitList.end(); ++itr) { if (itr->first == seg) return; } // create a new segment transformed to local coordintes. IntersectState* cis = _intersectStateStack.back().get(); LineSegment* ns = osgNew LineSegment; if (cis->_inverse.valid()) ns->mult(*seg,*(cis->_inverse)); else *ns = *seg; cis->addLineSegmentPair(seg,ns); } void IntersectVisitor::pushMatrix(const Matrix& matrix) { IntersectState* nis = osgNew IntersectState; IntersectState* cis = _intersectStateStack.back().get(); if (cis->_matrix.valid()) { nis->_matrix = osgNew Matrix; nis->_matrix->mult(matrix,*(cis->_matrix)); } else { nis->_matrix = osgNew Matrix(matrix); } Matrix* inverse_world = osgNew Matrix; inverse_world->invert(*(nis->_matrix)); nis->_inverse = inverse_world; IntersectState::LineSegmentmentMask segMaskIn = cis->_segmentMaskStack.back(); IntersectState::LineSegmentmentMask mask = 0x00000001; for(IntersectState::LineSegmentList::iterator sitr=cis->_segList.begin(); sitr!=cis->_segList.end(); ++sitr) { if ((segMaskIn & mask)) { LineSegment* seg = osgNew LineSegment; seg->mult(*(sitr->first),*inverse_world); nis->addLineSegmentPair(sitr->first.get(),seg); } mask = mask << 1; } _intersectStateStack.push_back(nis); } void IntersectVisitor::popMatrix() { if (!_intersectStateStack.empty()) { _intersectStateStack.pop_back(); } } bool IntersectVisitor::enterNode(Node& node) { const BoundingSphere& bs = node.getBound(); if (bs.valid()) { IntersectState* cis = _intersectStateStack.back().get(); IntersectState::LineSegmentmentMask sm=0xffffffff; if (cis->isCulled(bs,sm)) return false; cis->_segmentMaskStack.push_back(sm); _nodePath.push_back(&node); return true; } else { return false; } } void IntersectVisitor::leaveNode() { IntersectState* cis = _intersectStateStack.back().get(); cis->_segmentMaskStack.pop_back(); _nodePath.pop_back(); } void IntersectVisitor::apply(Node& node) { if (!enterNode(node)) return; traverse(node); leaveNode(); } struct TriangleIntersect { osg::ref_ptr _seg; Vec3 _s; Vec3 _d; float _length; int _index; float _ratio; bool _hit; typedef std::multimap > TriangleHitList; TriangleHitList _thl; TriangleIntersect() { } TriangleIntersect(const LineSegment& seg,float ratio=FLT_MAX) { set(seg,ratio); } void set(const LineSegment& seg,float ratio=FLT_MAX) { _seg=new LineSegment(seg); _hit=false; _index = 0; _ratio = ratio; _s = _seg->start(); _d = _seg->end()-_seg->start(); _length = _d.length(); _d /= _length; } // bool intersect(const Vec3& v1,const Vec3& v2,const Vec3& v3,float& r) inline void operator () (const Vec3& v1,const Vec3& v2,const Vec3& v3) { ++_index; if (v1==v2 || v2==v3 || v1==v3) return; Vec3 v12 = v2-v1; Vec3 n12 = v12^_d; float ds12 = (_s-v1)*n12; float d312 = (v3-v1)*n12; if (d312>=0.0f) { if (ds12<0.0f) return; if (ds12>d312) return; } else // d312 < 0 { if (ds12>0.0f) return; if (ds12=0.0f) { if (ds23<0.0f) return; if (ds23>d123) return; } else // d123 < 0 { if (ds23>0.0f) return; if (ds23=0.0f) { if (ds31<0.0f) return; if (ds31>d231) return; } else // d231 < 0 { if (ds31>0.0f) return; if (ds31_length) return; osg::Vec3 normal = v12^v23; normal.normalize(); float r = d/_length; if (!in.valid()) { osg::notify(WARN)<<"Warning:: Picked up error in TriangleIntersect"< > (r,std::pair(_index-1,normal))); _hit = true; } }; bool IntersectVisitor::intersect(Drawable& drawable) { bool hitFlag = false; IntersectState* cis = _intersectStateStack.back().get(); const BoundingBox& bb = drawable.getBound(); for(IntersectState::LineSegmentList::iterator sitr=cis->_segList.begin(); sitr!=cis->_segList.end(); ++sitr) { if (sitr->second->intersect(bb)) { TriangleFunctor ti; ti.set(*sitr->second); drawable.applyPrimitiveOperation(ti); if (ti._hit) { for(TriangleIntersect::TriangleHitList::iterator thitr=ti._thl.begin(); thitr!=ti._thl.end(); ++thitr) { Hit hit; hit._nodePath = _nodePath; hit._matrix = cis->_matrix; hit._inverse = cis->_inverse; hit._drawable = &drawable; if (_nodePath.empty()) hit._geode = NULL; else hit._geode = dynamic_cast(_nodePath.back()); hit._ratio = thitr->first; hit._primitiveIndex = thitr->second.first; hit._originalLineSegment = sitr->first; hit._localLineSegment = sitr->second; hit._intersectPoint = sitr->second->start()*(1.0f-hit._ratio)+ sitr->second->end()*hit._ratio; hit._intersectNormal = thitr->second.second; _segHitList[sitr->first.get()].push_back(hit); std::sort(_segHitList[sitr->first.get()].begin(),_segHitList[sitr->first.get()].end()); hitFlag = true; } } } } return hitFlag; } void IntersectVisitor::apply(Geode& geode) { if (!enterNode(geode)) return; for(int i = 0; i < geode.getNumDrawables(); i++ ) { intersect(*geode.getDrawable(i)); } leaveNode(); } void IntersectVisitor::apply(Billboard& node) { if (!enterNode(node)) return; leaveNode(); } void IntersectVisitor::apply(Group& node) { if (!enterNode(node)) return; traverse(node); leaveNode(); } void IntersectVisitor::apply(Transform& node) { if (!enterNode(node)) return; osg::ref_ptr matrix = osgNew Matrix; node.getLocalToWorldMatrix(*matrix,this); pushMatrix(*matrix); traverse(node); popMatrix(); leaveNode(); } void IntersectVisitor::apply(Switch& node) { apply((Group&)node); } void IntersectVisitor::apply(LOD& node) { apply((Group&)node); }