/* -*-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::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>=_fileNames.size()) _fileNames.resize(pos); _fileNames[pos] = fileName; } std::string ImageSequence::getImageFile(unsigned int pos) const { OpenThreads::ScopedLock lock(_mutex); return pos<_fileNames.size() ? _fileNames[pos] : std::string(); } void ImageSequence::addImageFile(const std::string& fileName) { OpenThreads::ScopedLock lock(_mutex); _fileNames.push_back(fileName); computeTimePerImage(); } void ImageSequence::setImage(unsigned int pos, osg::Image* image) { OpenThreads::ScopedLock lock(_mutex); OSG_INFO<<"ImageSequence::setImage("<getFileName()<<")"<=_images.size()) _images.resize(pos+1); _images[pos] = image; // prune from file requested list. FilesRequested::iterator itr = _filesRequested.find(image->getFileName()); if (itr!=_filesRequested.end()) _filesRequested.erase(itr); } Image* ImageSequence::getImage(unsigned int pos) { OpenThreads::ScopedLock lock(_mutex); return pos<_images.size() ? _images[pos].get() : 0; } const Image* ImageSequence::getImage(unsigned int pos) const { OpenThreads::ScopedLock lock(_mutex); return pos<_images.size() ? _images[pos].get() : 0; } void ImageSequence::addImage(osg::Image* image) { if (image==0) return; OpenThreads::ScopedLock lock(_mutex); // OSG_NOTICE<<"merging image in order expected : "<getFileName()<data() == data()) return; setImage(image->s(),image->t(),image->r(), image->getInternalTextureFormat(), image->getPixelFormat(),image->getDataType(), const_cast(image->data()), osg::Image::NO_DELETE, image->getPacking()); } 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(_images.size())) return int(_images.size())-1; return index; } void ImageSequence::update(osg::NodeVisitor* nv) { OpenThreads::ScopedLock lock(_mutex); 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; if (_seekTimeSet || _status==PAUSED || _status==INVALID) { time = _seekTime; _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; bool pruneOldImages = false; switch(_mode) { case(PRE_LOAD_ALL_IMAGES): { if (_fileNames.size()>_images.size()) { FileNames::iterator itr = _fileNames.begin(); for(unsigned int i=0;i<_images.size();++i) ++itr; for(; itr!=_fileNames.end(); ++itr) { osg::Image* image = irh->readImageFile(*itr); _images.push_back(image); } } irh = 0; break; } case(PAGE_AND_RETAIN_IMAGES): { break; } case(PAGE_AND_DISCARD_USED_IMAGES): { pruneOldImages = true; break; } } int index = int(time/_timePerImage); // OSG_NOTICE<<"time= "<