OpenSceneGraph/src/osg/ImageSequence.cpp

350 lines
10 KiB
C++
Raw Normal View History

2008-07-21 17:46:53 +08:00
/* -*-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 <OpenThreads/ScopedLock>
2008-07-21 17:46:53 +08:00
#include <osg/ImageSequence>
2008-07-22 01:28:22 +08:00
#include <osg/Notify>
#include <osg/Camera>
#include <osg/NodeVisitor>
#include <osg/Texture2D>
2008-07-22 01:28:22 +08:00
#include <math.h>
2008-07-21 17:46:53 +08:00
using namespace osg;
void ImageSequence::UpdateCallback::operator () (osg::StateAttribute* attr, osg::NodeVisitor* nv)
{
osg::Texture* texture = attr ? attr->asTexture() : 0;
// osg::notify(osg::NOTICE)<<"ImageSequence::UpdateCallback::"<<texture<<std::endl;
if (texture)
{
for(unsigned int i=0; i<texture->getNumImages(); ++i)
{
texture->getImage(i)->update(nv);
}
}
}
2008-07-21 17:46:53 +08:00
ImageSequence::ImageSequence()
{
2008-07-22 01:28:22 +08:00
_referenceTime = DBL_MAX;
2008-07-21 17:46:53 +08:00
_timeMultiplier = 1.0;
_mode = PRE_LOAD_ALL_IMAGES;
_duration = 1.0;
_timePerImage = 1.0;
2008-07-23 04:20:16 +08:00
_fileNamesIterator = _fileNames.end();
_fileNamesIteratorTime = DBL_MAX;
_imageIterator = _images.end();
2008-07-23 04:20:16 +08:00
_imageIteratorTime = DBL_MAX;
2008-07-21 17:46:53 +08:00
}
ImageSequence::ImageSequence(const ImageSequence& is,const CopyOp& copyop):
osg::ImageStream(is,copyop),
_referenceTime(is._referenceTime),
_timeMultiplier(is._timeMultiplier),
_mode(is._mode),
_duration(is._duration),
_timePerImage(is._timePerImage)
2008-07-21 17:46:53 +08:00
{
2008-07-23 04:20:16 +08:00
_fileNamesIterator = _fileNames.end();
_fileNamesIteratorTime = DBL_MAX;
_imageIteratorTime = DBL_MAX;
_imageIterator = _images.end();
2008-07-21 17:46:53 +08:00
}
int ImageSequence::compare(const Image& rhs) const
{
return ImageStream::compare(rhs);
}
void ImageSequence::play()
{
_status=PLAYING;
}
void ImageSequence::pause()
{
_status=PAUSED;
}
void ImageSequence::rewind()
{
_status=REWINDING;
}
2008-07-26 00:23:00 +08:00
void ImageSequence::setMode(Mode mode)
{
_mode = mode;
}
void ImageSequence::setDuration(double duration)
{
_duration = duration;
computeTimePerImage();
}
void ImageSequence::computeTimePerImage()
{
if (!_fileNames.empty()) _timePerImage = _duration / double(_fileNames.size());
else if (!_images.empty()) _timePerImage = _duration / double(_images.size());
else _timePerImage = _duration;
}
void ImageSequence::addImageFile(const std::string& fileName)
2008-07-21 17:46:53 +08:00
{
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(_mutex);
_fileNames.push_back(fileName);
computeTimePerImage();
2008-07-23 04:20:16 +08:00
if (_fileNamesIterator==_fileNames.end())
{
_fileNamesIterator = _fileNames.begin();
_fileNamesIteratorTime = _referenceTime;
}
2008-07-21 17:46:53 +08:00
}
void ImageSequence::addImage(osg::Image* image)
2008-07-21 17:46:53 +08:00
{
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(_mutex);
2008-07-22 01:28:22 +08:00
if (!_filesRequested.empty())
{
// follows is a mechanism that ensures that requested images
// get merged in the correct time order.
if (_filesRequested.front().first != image->getFileName())
{
for(FileNameImageList::iterator itr = _filesRequested.begin();
itr != _filesRequested.end();
++itr)
{
if (itr->first == image->getFileName())
{
osg::notify(osg::NOTICE)<<"inserting image into waiting stake : "<<image->getFileName()<<std::endl;
itr->second = image;
return;
}
}
// osg::notify(osg::NOTICE)<<"image not expected : "<<image->getFileName()<<std::endl;
_images.push_back(image);
}
else
{
// osg::notify(osg::NOTICE)<<"merging image in order expected : "<<image->getFileName()<<std::endl;
_images.push_back(image);
_filesRequested.pop_front();
FileNameImageList::iterator itr;
for(itr = _filesRequested.begin();
itr != _filesRequested.end() && itr->second.valid();
++itr)
{
// osg::notify(osg::NOTICE)<<" merging previously loaded, but out of order file : "<<itr->first<<std::endl;
_images.push_back(itr->second);
}
_filesRequested.erase(_filesRequested.begin(), itr);
}
}
else
{
_images.push_back(image);
}
computeTimePerImage();
if (data()==0)
2008-07-22 01:28:22 +08:00
{
_imageIterator = _images.begin();
2008-07-22 01:28:22 +08:00
_imageIteratorTime = _referenceTime;
setImageToChild(_imageIterator->get());
2008-07-22 01:28:22 +08:00
}
2008-07-21 17:46:53 +08:00
}
2008-07-22 01:28:22 +08:00
void ImageSequence::setImageToChild(const osg::Image* image)
{
// osg::notify(osg::NOTICE)<<"setImageToChild("<<image<<")"<<std::endl;
if (image==0) return;
2008-07-22 01:28:22 +08:00
setImage(image->s(),image->t(),image->r(),
image->getInternalTextureFormat(),
image->getPixelFormat(),image->getDataType(),
const_cast<unsigned char*>(image->data()),
osg::Image::NO_DELETE,
image->getPacking());
}
void ImageSequence::update(osg::NodeVisitor* nv)
2008-07-21 17:46:53 +08:00
{
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(_mutex);
osg::NodeVisitor::ImageRequestHandler* irh = nv->getImageRequestHandler();
const osg::FrameStamp* fs = nv->getFrameStamp();
// osg::notify(osg::NOTICE)<<"ImageSequence::update("<<fs<<", "<<irh<<")"<<std::endl;
2008-07-22 01:28:22 +08:00
if (_referenceTime == DBL_MAX)
{
_referenceTime = fs->getSimulationTime();
}
2008-07-23 04:20:16 +08:00
if (_fileNamesIteratorTime == DBL_MAX)
{
_fileNamesIteratorTime = _referenceTime;
}
2008-07-22 01:28:22 +08:00
if (_imageIteratorTime == DBL_MAX)
{
_imageIteratorTime = _referenceTime;
}
double time = (fs->getSimulationTime() - _referenceTime)*_timeMultiplier;
if (_status==PAUSED || _status==INVALID) time = _imageIteratorTime;
2008-07-24 03:04:46 +08:00
FileNames::iterator previous_fileNamesIterator = _fileNamesIterator;
Images::iterator previous_imageIterator = _imageIterator;
2008-07-22 01:28:22 +08:00
bool pruneOldImages = false;
bool looping = getLoopingMode()==LOOPING;
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;
}
}
2008-07-22 01:28:22 +08:00
// osg::notify(osg::NOTICE)<<"time = "<<time<<std::endl;
if (irh && _images.size()<_fileNames.size())
2008-07-22 01:28:22 +08:00
{
double preLoadTime = (time+irh->getPreLoadTime())*_timeMultiplier;
2008-07-23 04:20:16 +08:00
//
// Advance imageIterator
//
if (_fileNamesIterator!=_fileNames.end())
2008-07-22 01:28:22 +08:00
{
//osg::notify(osg::NOTICE)<<" _fileNamesIteratorTime = "<<_fileNamesIteratorTime<<" "<<_timePerImage<<std::endl;
2008-07-23 04:20:16 +08:00
while(preLoadTime > (_fileNamesIteratorTime + _timePerImage))
2008-07-22 01:28:22 +08:00
{
2008-07-23 04:20:16 +08:00
_fileNamesIteratorTime += _timePerImage;
//osg::notify(osg::NOTICE)<<" _fileNamesIteratorTime = "<<_fileNamesIteratorTime<<std::endl;
//osg::notify(osg::NOTICE)<<" need to preLoad = "<<*_fileNamesIterator<<std::endl;
2008-07-23 04:20:16 +08:00
++_fileNamesIterator;
2008-07-24 03:04:46 +08:00
if (previous_fileNamesIterator==_fileNamesIterator) break;
2008-07-23 04:20:16 +08:00
if (_fileNamesIterator ==_fileNames.end())
{
2008-07-23 04:20:16 +08:00
// return iterator to begining of set.
if (looping) _fileNamesIterator = _fileNames.begin();
else break;
}
2008-07-24 03:04:46 +08:00
_filesRequested.push_back(FileNameImagePair(*_fileNamesIterator,0));
2008-07-24 03:04:46 +08:00
irh->requestImageFile(*_fileNamesIterator, this, _fileNamesIteratorTime, fs);
2008-07-22 01:28:22 +08:00
}
}
2008-07-23 04:20:16 +08:00
if (looping && _fileNamesIterator==_fileNames.end())
2008-07-23 04:20:16 +08:00
{
_fileNamesIterator = _fileNames.begin();
}
2008-07-22 01:28:22 +08:00
}
2008-07-23 04:20:16 +08:00
2008-07-22 01:28:22 +08:00
{
2008-07-24 03:04:46 +08:00
2008-07-23 04:20:16 +08:00
//
// Advance imageIterator
//
if (_imageIterator!=_images.end())
{
// osg::notify(osg::NOTICE)<<" _imageIteratorTime = "<<_imageIteratorTime<<std::endl;
while(time > (_imageIteratorTime + _timePerImage))
{
_imageIteratorTime += _timePerImage;
// osg::notify(osg::NOTICE)<<" _imageIteratorTime = "<<_imageIteratorTime<<std::endl;
++_imageIterator;
if (_imageIterator ==_images.end())
{
if (pruneOldImages)
2008-07-23 04:20:16 +08:00
{
_images.erase(previous_imageIterator, _imageIterator);
2008-07-24 03:04:46 +08:00
previous_imageIterator = _images.begin();
2008-07-23 04:20:16 +08:00
}
if (looping)
{
// return iterator to begining of set.
_imageIterator = _images.begin();
}
else
{
break;
}
2008-07-23 04:20:16 +08:00
}
}
}
if (looping && _imageIterator==_images.end())
2008-07-23 04:20:16 +08:00
{
_imageIterator = _images.begin();
}
2008-07-22 01:28:22 +08:00
}
2008-07-23 04:20:16 +08:00
if (_imageIterator!=_images.end() && previous_imageIterator != _imageIterator)
2008-07-22 01:28:22 +08:00
{
if (pruneOldImages)
{
_images.erase(previous_imageIterator, _imageIterator);
}
setImageToChild(_imageIterator->get());
2008-07-22 01:28:22 +08:00
}
2008-07-21 17:46:53 +08:00
}