/* -*-c++-*- OpenSceneGraph - Copyright (C) 1998-2006 Robert Osfield * * This library is open source and may be redistributed and/or modified under * the terms of the OpenSceneGraph Public License (OSGPL) version 0.0 or * (at your option) any later version. The full license is in LICENSE file * included with this distribution, and on the openscenegraph.org website. * * This library 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 * OpenSceneGraph Public License for more details. */ #include #include #include #include #include #include #include using namespace osg; ImageSequence::ImageData::ImageData() { } ImageSequence::ImageData::ImageData(const ImageData& id): _filename(id._filename), _image(id._image), _imageRequest(id._imageRequest) { } ImageSequence::ImageData& ImageSequence::ImageData::operator = (const ImageSequence::ImageData& rhs) { if (&rhs!=this) { _filename = rhs._filename; _image = rhs._image; _imageRequest = rhs._imageRequest; } return *this; } ImageSequence::ImageSequence() { _referenceTime = DBL_MAX; _timeMultiplier = 1.0; _mode = PRE_LOAD_ALL_IMAGES; _length = 1.0; _timePerImage = 1.0; _seekTime = 0.0; _seekTimeSet = false; _previousAppliedImageIndex = -1; } ImageSequence::ImageSequence(const ImageSequence& is,const CopyOp& copyop): osg::ImageStream(is,copyop), _referenceTime(is._referenceTime), _timeMultiplier(is._timeMultiplier), _mode(is._mode), _length(is._length), _timePerImage(is._timePerImage) { _seekTime = is._seekTime; _seekTimeSet = is._seekTimeSet; _previousAppliedImageIndex = -1; } int ImageSequence::compare(const Image& rhs) const { return ImageStream::compare(rhs); } void ImageSequence::seek(double time) { _seekTime = time; _seekTimeSet = true; } void ImageSequence::play() { _status=PLAYING; } void ImageSequence::pause() { _status=PAUSED; } void ImageSequence::rewind() { seek(0.0f); } void ImageSequence::setMode(Mode mode) { _mode = mode; } void ImageSequence::setLength(double length) { if (length<=0.0) { OSG_NOTICE<<"ImageSequence::setLength("< lock(_mutex); if (pos>=_imageDataList.size()) _imageDataList.resize(pos); _imageDataList[pos]._filename = fileName; } std::string ImageSequence::getImageFile(unsigned int pos) const { OpenThreads::ScopedLock lock(_mutex); return pos<_imageDataList.size() ? _imageDataList[pos]._filename : std::string(); } void ImageSequence::addImageFile(const std::string& fileName) { OpenThreads::ScopedLock lock(_mutex); _imageDataList.push_back(ImageData()); _imageDataList.back()._filename = fileName; computeTimePerImage(); } void ImageSequence::setImage(unsigned int pos, osg::Image* image) { OpenThreads::ScopedLock lock(_mutex); _setImage(pos,image); } void ImageSequence::_setImage(unsigned int pos, osg::Image* image) { if (pos>=_imageDataList.size()) _imageDataList.resize(pos+1); _imageDataList[pos]._image = image; _imageDataList[pos]._filename = image->getFileName(); } Image* ImageSequence::getImage(unsigned int pos) { OpenThreads::ScopedLock lock(_mutex); return pos<_imageDataList.size() ? _imageDataList[pos]._image.get() : 0; } const Image* ImageSequence::getImage(unsigned int pos) const { OpenThreads::ScopedLock lock(_mutex); return pos<_imageDataList.size() ? _imageDataList[pos]._image.get() : 0; } void ImageSequence::addImage(osg::Image* image) { if (image==0) return; OpenThreads::ScopedLock lock(_mutex); // OSG_NOTICE<<"merging image in order expected : "<getFileName()<=0 && pos(_imageDataList.size())) ? _imageDataList[pos]._image.get() : 0; if (image==0) return; // check to see if data is changing, if not don't apply if (image->data() == data()) { return; } bool discardOldImages = _mode==PAGE_AND_DISCARD_USED_IMAGES || _mode==LOAD_AND_DISCARD_IN_UPDATE_TRAVERSAL; if (discardOldImages && _previousAppliedImageIndex>=0) { if (_previousAppliedImageIndexpos) { OSG_INFO<<"Moving back from "<<_previousAppliedImageIndex<<" to "<pos) { _imageDataList[_previousAppliedImageIndex]._image = 0; OSG_INFO<<" deleting "<<_previousAppliedImageIndex<s(),image->t(),image->r(), image->getInternalTextureFormat(), image->getPixelFormat(),image->getDataType(), const_cast(image->data()), osg::Image::NO_DELETE, image->getPacking()); setMipmapLevels(image->getMipmapLevels()); } void ImageSequence::applyLoopingMode() { } int ImageSequence::imageIndex(double time) { if (getLoopingMode()==LOOPING) { double positionRatio = time/_length; time = (positionRatio - floor(positionRatio))*_length; } if (time<0.0) return 0; int index = int(time/_timePerImage); if (index>=int(_imageDataList.size())) return int(_imageDataList.size())-1; return index; } void ImageSequence::update(osg::NodeVisitor* nv) { OpenThreads::ScopedLock lock(_mutex); // if imageDataList is empty then there is nothing update can do. if (_imageDataList.empty()) return; osg::NodeVisitor::ImageRequestHandler* irh = nv->getImageRequestHandler(); const osg::FrameStamp* fs = nv->getFrameStamp(); // OSG_NOTICE<<"ImageSequence::update("<getSimulationTime(); } bool looping = getLoopingMode()==LOOPING; double time = (fs->getSimulationTime() - _referenceTime)*_timeMultiplier; bool useDirectTimeRequest = _seekTimeSet; if (_seekTimeSet || _status==PAUSED || _status==INVALID) { time = _seekTime; useDirectTimeRequest = true; _referenceTime = fs->getSimulationTime() - time/_timeMultiplier; } else { if (looping) { while (time>_length) { _referenceTime += _length/_timeMultiplier; time -= _length; } } else { if (time>_length) { _referenceTime = fs->getSimulationTime() - _length/_timeMultiplier; time = _length; } } } _seekTime = time; _seekTimeSet = false; if (irh && _mode==PRE_LOAD_ALL_IMAGES) { for(ImageDataList::iterator itr = _imageDataList.begin(); itr != _imageDataList.end(); ++itr) { if (!(itr->_image) && !(itr->_filename.empty())) { itr->_image = irh->readRefImageFile(itr->_filename, _readOptions.get()); } } } int index = int(time/_timePerImage); // OSG_NOTICE<<"time= "<=0 && !_imageDataList[index]._image.valid()) { --index; } } else if (_previousAppliedImageIndex>index) { OSG_DEBUG<<"ImageSequence::update(..) Moving back by "<<_previousAppliedImageIndex-index<(_imageDataList.size()) && !_imageDataList[index]._image.valid()) { ++index; } } } if (index>=0 && index!=_previousAppliedImageIndex) { setImageToChild(index); } } // OSG_NOTICE<<"time = "<(0, int(time/_timePerImage)); if ((i>=int(_imageDataList.size()) || !_imageDataList[i]._image)) { i = osg::minimum(i, _imageDataList.size()-1); if (loadDirectly) { OSG_NOTICE<<"Reading file, entry="<