2003-01-22 00:45:36 +08:00
|
|
|
/* -*-c++-*- OpenSceneGraph - Copyright (C) 1998-2003 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.
|
|
|
|
*/
|
2001-10-04 05:44:07 +08:00
|
|
|
#include <osg/GL>
|
|
|
|
#include <osg/GLU>
|
|
|
|
|
|
|
|
#include <osg/Image>
|
|
|
|
#include <osg/Notify>
|
2001-01-11 00:32:10 +08:00
|
|
|
|
|
|
|
#include <osg/Geode>
|
2002-06-26 04:27:51 +08:00
|
|
|
#include <osg/Geometry>
|
2001-09-20 05:08:56 +08:00
|
|
|
#include <osg/StateSet>
|
2002-08-25 03:39:39 +08:00
|
|
|
#include <osg/Texture2D>
|
2004-09-29 18:01:46 +08:00
|
|
|
#include <osg/Texture3D>
|
2001-01-11 00:32:10 +08:00
|
|
|
|
2004-08-17 04:57:24 +08:00
|
|
|
#include "dxtctool.h"
|
2001-09-20 05:08:56 +08:00
|
|
|
|
2001-01-11 00:32:10 +08:00
|
|
|
using namespace osg;
|
2002-02-23 01:12:10 +08:00
|
|
|
using namespace std;
|
2001-01-11 00:32:10 +08:00
|
|
|
|
|
|
|
Image::Image()
|
|
|
|
{
|
2004-07-27 18:11:45 +08:00
|
|
|
setDataVariance(STATIC);
|
|
|
|
|
2002-04-12 01:15:07 +08:00
|
|
|
_fileName = "";
|
|
|
|
_s = _t = _r = 0;
|
|
|
|
_internalTextureFormat = 0;
|
|
|
|
_pixelFormat = (unsigned int)0;
|
|
|
|
_dataType = (unsigned int)0;
|
|
|
|
_packing = 4;
|
2001-01-11 00:32:10 +08:00
|
|
|
|
2003-02-25 19:56:18 +08:00
|
|
|
_allocationMode = USE_NEW_DELETE;
|
|
|
|
_data = (unsigned char *)0L;
|
2001-11-19 05:31:16 +08:00
|
|
|
|
|
|
|
_modifiedTag = 0;
|
2001-01-11 00:32:10 +08:00
|
|
|
}
|
|
|
|
|
2002-01-29 22:04:06 +08:00
|
|
|
Image::Image(const Image& image,const CopyOp& copyop):
|
|
|
|
Object(image,copyop),
|
Added support for shallow and deep copy of nodes, drawables and state, via a
copy constructor which takes an optional Cloner object, and the old
osg::Object::clone() has changed so that it now requires a Cloner as paramter.
This is passed on to the copy constructor to help control the shallow vs
deep copying. The old functionality of clone() which was clone of type has
been renamed to cloneType().
Updated all of the OSG to work with these new conventions, implemention all
the required copy constructors etc. A couple of areas will do shallow
copies by design, a couple of other still need to be updated to do either
shallow or deep.
Neither of the shallow or deep copy operations have been tested yet, only
the old functionality of the OSG has been checked so far, such running the
viewer on various demo datasets.
Also fixed a problem in osg::Optimize::RemoveRendundentNodesVisitor which
was not checking that Group didn't have have any attached StateSet's, Callbacks
or UserData. These checks have now been added, which fixes a bug which was
revealled by the new osgscribe demo, this related to removal of group acting
as state decorator.
method
2002-01-29 05:17:01 +08:00
|
|
|
_fileName(image._fileName),
|
|
|
|
_s(image._s), _t(image._t), _r(image._r),
|
2002-04-12 01:15:07 +08:00
|
|
|
_internalTextureFormat(image._internalTextureFormat),
|
Added support for shallow and deep copy of nodes, drawables and state, via a
copy constructor which takes an optional Cloner object, and the old
osg::Object::clone() has changed so that it now requires a Cloner as paramter.
This is passed on to the copy constructor to help control the shallow vs
deep copying. The old functionality of clone() which was clone of type has
been renamed to cloneType().
Updated all of the OSG to work with these new conventions, implemention all
the required copy constructors etc. A couple of areas will do shallow
copies by design, a couple of other still need to be updated to do either
shallow or deep.
Neither of the shallow or deep copy operations have been tested yet, only
the old functionality of the OSG has been checked so far, such running the
viewer on various demo datasets.
Also fixed a problem in osg::Optimize::RemoveRendundentNodesVisitor which
was not checking that Group didn't have have any attached StateSet's, Callbacks
or UserData. These checks have now been added, which fixes a bug which was
revealled by the new osgscribe demo, this related to removal of group acting
as state decorator.
method
2002-01-29 05:17:01 +08:00
|
|
|
_pixelFormat(image._pixelFormat),
|
|
|
|
_dataType(image._dataType),
|
|
|
|
_packing(image._packing),
|
|
|
|
_data(0L),
|
2002-04-23 22:58:33 +08:00
|
|
|
_modifiedTag(image._modifiedTag),
|
|
|
|
_mipmapData(image._mipmapData)
|
Added support for shallow and deep copy of nodes, drawables and state, via a
copy constructor which takes an optional Cloner object, and the old
osg::Object::clone() has changed so that it now requires a Cloner as paramter.
This is passed on to the copy constructor to help control the shallow vs
deep copying. The old functionality of clone() which was clone of type has
been renamed to cloneType().
Updated all of the OSG to work with these new conventions, implemention all
the required copy constructors etc. A couple of areas will do shallow
copies by design, a couple of other still need to be updated to do either
shallow or deep.
Neither of the shallow or deep copy operations have been tested yet, only
the old functionality of the OSG has been checked so far, such running the
viewer on various demo datasets.
Also fixed a problem in osg::Optimize::RemoveRendundentNodesVisitor which
was not checking that Group didn't have have any attached StateSet's, Callbacks
or UserData. These checks have now been added, which fixes a bug which was
revealled by the new osgscribe demo, this related to removal of group acting
as state decorator.
method
2002-01-29 05:17:01 +08:00
|
|
|
{
|
|
|
|
if (image._data)
|
|
|
|
{
|
2004-08-04 02:06:36 +08:00
|
|
|
int size = image.getTotalSizeInBytesIncludingMipmaps();
|
2003-02-25 19:56:18 +08:00
|
|
|
setData(new unsigned char [size],USE_NEW_DELETE);
|
Added support for shallow and deep copy of nodes, drawables and state, via a
copy constructor which takes an optional Cloner object, and the old
osg::Object::clone() has changed so that it now requires a Cloner as paramter.
This is passed on to the copy constructor to help control the shallow vs
deep copying. The old functionality of clone() which was clone of type has
been renamed to cloneType().
Updated all of the OSG to work with these new conventions, implemention all
the required copy constructors etc. A couple of areas will do shallow
copies by design, a couple of other still need to be updated to do either
shallow or deep.
Neither of the shallow or deep copy operations have been tested yet, only
the old functionality of the OSG has been checked so far, such running the
viewer on various demo datasets.
Also fixed a problem in osg::Optimize::RemoveRendundentNodesVisitor which
was not checking that Group didn't have have any attached StateSet's, Callbacks
or UserData. These checks have now been added, which fixes a bug which was
revealled by the new osgscribe demo, this related to removal of group acting
as state decorator.
method
2002-01-29 05:17:01 +08:00
|
|
|
memcpy(_data,image._data,size);
|
|
|
|
}
|
2002-04-23 05:13:33 +08:00
|
|
|
|
Added support for shallow and deep copy of nodes, drawables and state, via a
copy constructor which takes an optional Cloner object, and the old
osg::Object::clone() has changed so that it now requires a Cloner as paramter.
This is passed on to the copy constructor to help control the shallow vs
deep copying. The old functionality of clone() which was clone of type has
been renamed to cloneType().
Updated all of the OSG to work with these new conventions, implemention all
the required copy constructors etc. A couple of areas will do shallow
copies by design, a couple of other still need to be updated to do either
shallow or deep.
Neither of the shallow or deep copy operations have been tested yet, only
the old functionality of the OSG has been checked so far, such running the
viewer on various demo datasets.
Also fixed a problem in osg::Optimize::RemoveRendundentNodesVisitor which
was not checking that Group didn't have have any attached StateSet's, Callbacks
or UserData. These checks have now been added, which fixes a bug which was
revealled by the new osgscribe demo, this related to removal of group acting
as state decorator.
method
2002-01-29 05:17:01 +08:00
|
|
|
}
|
2001-01-11 00:32:10 +08:00
|
|
|
|
|
|
|
Image::~Image()
|
|
|
|
{
|
2003-02-25 19:56:18 +08:00
|
|
|
deallocateData();
|
2001-01-11 00:32:10 +08:00
|
|
|
}
|
|
|
|
|
2003-02-25 19:56:18 +08:00
|
|
|
void Image::deallocateData()
|
|
|
|
{
|
|
|
|
if (_data) {
|
|
|
|
if (_allocationMode==USE_NEW_DELETE) delete [] _data;
|
|
|
|
else if (_allocationMode==USE_MALLOC_FREE) ::free(_data);
|
|
|
|
_data = 0;
|
|
|
|
}
|
|
|
|
}
|
2001-01-11 00:32:10 +08:00
|
|
|
|
2003-03-11 21:30:03 +08:00
|
|
|
int Image::compare(const Image& rhs) const
|
|
|
|
{
|
|
|
|
if (getFileName()<rhs.getFileName()) return -1;
|
|
|
|
else if (getFileName()>rhs.getFileName()) return 1;
|
|
|
|
|
|
|
|
// need to test against image contents here...
|
|
|
|
COMPARE_StateAttribute_Parameter(_s)
|
|
|
|
COMPARE_StateAttribute_Parameter(_t)
|
|
|
|
COMPARE_StateAttribute_Parameter(_internalTextureFormat)
|
|
|
|
COMPARE_StateAttribute_Parameter(_pixelFormat)
|
|
|
|
COMPARE_StateAttribute_Parameter(_dataType)
|
|
|
|
COMPARE_StateAttribute_Parameter(_packing)
|
|
|
|
COMPARE_StateAttribute_Parameter(_modifiedTag)
|
|
|
|
|
2003-03-20 22:07:03 +08:00
|
|
|
if (_data<rhs._data) return -1;
|
|
|
|
if (_data>rhs._data) return 1;
|
|
|
|
|
|
|
|
if (_mipmapData<rhs._mipmapData) return -1;
|
|
|
|
if (_mipmapData>rhs._mipmapData) return 1;
|
|
|
|
|
2003-03-11 21:30:03 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2001-09-20 05:08:56 +08:00
|
|
|
void Image::setFileName(const std::string& fileName)
|
2001-01-11 00:32:10 +08:00
|
|
|
{
|
2001-09-20 05:08:56 +08:00
|
|
|
_fileName = fileName;
|
2001-01-11 00:32:10 +08:00
|
|
|
}
|
|
|
|
|
2003-02-25 19:56:18 +08:00
|
|
|
void Image::setData(unsigned char* data, AllocationMode mode)
|
|
|
|
{
|
|
|
|
deallocateData();
|
|
|
|
_data = data;
|
|
|
|
_allocationMode = mode;
|
|
|
|
}
|
|
|
|
|
2002-04-11 06:10:07 +08:00
|
|
|
|
2002-09-02 20:31:35 +08:00
|
|
|
bool Image::isPackedType(GLenum type)
|
2002-04-12 01:15:07 +08:00
|
|
|
{
|
2002-05-14 17:34:11 +08:00
|
|
|
switch(type)
|
2002-04-12 01:15:07 +08:00
|
|
|
{
|
|
|
|
case(GL_UNSIGNED_BYTE_3_3_2):
|
|
|
|
case(GL_UNSIGNED_BYTE_2_3_3_REV):
|
|
|
|
case(GL_UNSIGNED_SHORT_5_6_5):
|
|
|
|
case(GL_UNSIGNED_SHORT_5_6_5_REV):
|
|
|
|
case(GL_UNSIGNED_SHORT_4_4_4_4):
|
|
|
|
case(GL_UNSIGNED_SHORT_4_4_4_4_REV):
|
|
|
|
case(GL_UNSIGNED_SHORT_5_5_5_1):
|
|
|
|
case(GL_UNSIGNED_SHORT_1_5_5_5_REV):
|
|
|
|
case(GL_UNSIGNED_INT_8_8_8_8):
|
|
|
|
case(GL_UNSIGNED_INT_8_8_8_8_REV):
|
|
|
|
case(GL_UNSIGNED_INT_10_10_10_2):
|
|
|
|
case(GL_UNSIGNED_INT_2_10_10_10_REV): return true;
|
|
|
|
default: return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-09-22 17:13:22 +08:00
|
|
|
unsigned int Image::computeNumComponents(GLenum pixelFormat)
|
2002-04-12 01:15:07 +08:00
|
|
|
{
|
2003-09-22 17:13:22 +08:00
|
|
|
switch(pixelFormat)
|
2002-04-12 01:15:07 +08:00
|
|
|
{
|
2003-04-03 02:26:34 +08:00
|
|
|
case(GL_COMPRESSED_RGB_S3TC_DXT1_EXT): return 3;
|
2003-06-24 23:40:09 +08:00
|
|
|
case(GL_COMPRESSED_RGBA_S3TC_DXT1_EXT): return 4;
|
2003-04-03 02:26:34 +08:00
|
|
|
case(GL_COMPRESSED_RGBA_S3TC_DXT3_EXT): return 4;
|
|
|
|
case(GL_COMPRESSED_RGBA_S3TC_DXT5_EXT): return 4;
|
2002-04-12 01:15:07 +08:00
|
|
|
case(GL_COLOR_INDEX): return 1;
|
|
|
|
case(GL_STENCIL_INDEX): return 1;
|
|
|
|
case(GL_DEPTH_COMPONENT): return 1;
|
|
|
|
case(GL_RED): return 1;
|
|
|
|
case(GL_GREEN): return 1;
|
|
|
|
case(GL_BLUE): return 1;
|
|
|
|
case(GL_ALPHA): return 1;
|
|
|
|
case(GL_RGB): return 3;
|
|
|
|
case(GL_BGR): return 3;
|
|
|
|
case(GL_RGBA): return 4;
|
|
|
|
case(GL_BGRA): return 4;
|
|
|
|
case(GL_LUMINANCE): return 1;
|
2004-06-17 23:07:50 +08:00
|
|
|
case(GL_INTENSITY): return 1;
|
2002-04-16 22:09:46 +08:00
|
|
|
case(GL_LUMINANCE_ALPHA): return 2;
|
2004-03-14 23:08:12 +08:00
|
|
|
case(GL_HILO_NV): return 2;
|
|
|
|
case(GL_DSDT_NV): return 2;
|
|
|
|
case(GL_DSDT_MAG_NV): return 3;
|
|
|
|
case(GL_DSDT_MAG_VIB_NV): return 4;
|
2003-06-24 23:40:09 +08:00
|
|
|
default:
|
|
|
|
{
|
2003-09-22 17:13:22 +08:00
|
|
|
notify(WARN)<<"error pixelFormat = "<<std::hex<<pixelFormat<<std::endl;
|
2003-06-24 23:40:09 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2002-04-12 01:15:07 +08:00
|
|
|
}
|
2002-04-11 05:51:34 +08:00
|
|
|
}
|
|
|
|
|
2002-04-12 01:15:07 +08:00
|
|
|
|
2002-09-02 20:31:35 +08:00
|
|
|
unsigned int Image::computePixelSizeInBits(GLenum format,GLenum type)
|
2002-04-12 01:15:07 +08:00
|
|
|
{
|
2004-05-26 00:10:28 +08:00
|
|
|
|
|
|
|
if(format & 0x80F0) /* DXT* compressed formats */
|
2002-04-12 01:15:07 +08:00
|
|
|
{
|
2004-05-26 00:10:28 +08:00
|
|
|
switch(format)
|
|
|
|
{
|
|
|
|
case(GL_COMPRESSED_RGB_S3TC_DXT1_EXT): return 4;
|
2003-06-24 23:40:09 +08:00
|
|
|
|
2004-05-26 00:10:28 +08:00
|
|
|
case(GL_COMPRESSED_RGBA_S3TC_DXT1_EXT): return 4;
|
|
|
|
|
|
|
|
case(GL_COMPRESSED_RGBA_S3TC_DXT3_EXT): return 8;
|
|
|
|
|
|
|
|
case(GL_COMPRESSED_RGBA_S3TC_DXT5_EXT): return 8;
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
notify(WARN)<<"error format = "<<format<<std::endl;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(type)
|
|
|
|
{
|
|
|
|
|
2002-04-14 21:41:13 +08:00
|
|
|
case(GL_BITMAP): return computeNumComponents(format);
|
2002-04-12 01:15:07 +08:00
|
|
|
|
|
|
|
case(GL_BYTE):
|
2002-04-14 21:41:13 +08:00
|
|
|
case(GL_UNSIGNED_BYTE): return 8*computeNumComponents(format);
|
2002-04-12 01:15:07 +08:00
|
|
|
|
2004-09-01 16:49:48 +08:00
|
|
|
case(GL_HALF_FLOAT_NV):
|
2002-04-12 01:15:07 +08:00
|
|
|
case(GL_SHORT):
|
2002-04-14 21:41:13 +08:00
|
|
|
case(GL_UNSIGNED_SHORT): return 16*computeNumComponents(format);
|
2002-04-12 01:15:07 +08:00
|
|
|
|
|
|
|
case(GL_INT):
|
|
|
|
case(GL_UNSIGNED_INT):
|
2002-04-14 21:41:13 +08:00
|
|
|
case(GL_FLOAT): return 32*computeNumComponents(format);
|
2002-04-12 01:15:07 +08:00
|
|
|
|
|
|
|
|
|
|
|
case(GL_UNSIGNED_BYTE_3_3_2):
|
|
|
|
case(GL_UNSIGNED_BYTE_2_3_3_REV): return 8;
|
|
|
|
|
|
|
|
case(GL_UNSIGNED_SHORT_5_6_5):
|
|
|
|
case(GL_UNSIGNED_SHORT_5_6_5_REV):
|
|
|
|
case(GL_UNSIGNED_SHORT_4_4_4_4):
|
|
|
|
case(GL_UNSIGNED_SHORT_4_4_4_4_REV):
|
|
|
|
case(GL_UNSIGNED_SHORT_5_5_5_1):
|
|
|
|
case(GL_UNSIGNED_SHORT_1_5_5_5_REV): return 16;
|
|
|
|
|
|
|
|
case(GL_UNSIGNED_INT_8_8_8_8):
|
|
|
|
case(GL_UNSIGNED_INT_8_8_8_8_REV):
|
|
|
|
case(GL_UNSIGNED_INT_10_10_10_2):
|
|
|
|
case(GL_UNSIGNED_INT_2_10_10_10_REV): return 32;
|
2003-06-24 23:40:09 +08:00
|
|
|
default:
|
|
|
|
{
|
2003-06-25 05:57:13 +08:00
|
|
|
notify(WARN)<<"error type = "<<type<<std::endl;
|
2003-06-24 23:40:09 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2002-04-12 01:15:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2003-09-22 17:13:22 +08:00
|
|
|
unsigned int Image::computeRowWidthInBytes(int width,GLenum pixelFormat,GLenum type,int packing)
|
2002-04-12 01:15:07 +08:00
|
|
|
{
|
2003-09-22 17:13:22 +08:00
|
|
|
unsigned int pixelSize = computePixelSizeInBits(pixelFormat,type);
|
2002-04-12 01:15:07 +08:00
|
|
|
int widthInBits = width*pixelSize;
|
|
|
|
int packingInBits = packing*8;
|
2003-06-25 05:57:13 +08:00
|
|
|
//notify(INFO) << "width="<<width<<" pixelSize="<<pixelSize<<" width in bit="<<widthInBits<<" packingInBits="<<packingInBits<<" widthInBits%packingInBits="<<widthInBits%packingInBits<<std::endl;
|
2002-04-14 21:41:13 +08:00
|
|
|
return (widthInBits/packingInBits + ((widthInBits%packingInBits)?1:0))*packing;
|
2002-04-12 01:15:07 +08:00
|
|
|
}
|
|
|
|
|
2003-04-01 19:49:09 +08:00
|
|
|
int Image::computeNearestPowerOfTwo(int s,float bias)
|
2002-08-16 21:33:32 +08:00
|
|
|
{
|
|
|
|
if ((s & (s-1))!=0)
|
|
|
|
{
|
|
|
|
// it isn't so lets find the closest power of two.
|
|
|
|
// yes, logf and powf are slow, but this code should
|
|
|
|
// only be called during scene graph initilization,
|
|
|
|
// if at all, so not critical in the greater scheme.
|
|
|
|
float p2 = logf((float)s)/logf(2.0f);
|
|
|
|
float rounded_p2 = floorf(p2+bias);
|
|
|
|
s = (int)(powf(2.0f,rounded_p2));
|
|
|
|
}
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2003-06-24 23:40:09 +08:00
|
|
|
unsigned int Image::getTotalSizeInBytesIncludingMipmaps() const
|
|
|
|
{
|
|
|
|
if (_mipmapData.empty())
|
|
|
|
{
|
|
|
|
// no mips so just return size of main image
|
|
|
|
return getTotalSizeInBytes();
|
|
|
|
}
|
|
|
|
|
|
|
|
int s = _s;
|
|
|
|
int t = _t;
|
|
|
|
int r = _r;
|
|
|
|
|
|
|
|
unsigned int maxValue = 0;
|
|
|
|
for(unsigned int i=0;i<_mipmapData.size() && _mipmapData[i];++i)
|
|
|
|
{
|
|
|
|
s >>= 1;
|
|
|
|
t >>= 1;
|
|
|
|
r >>= 1;
|
|
|
|
maxValue = maximum(maxValue,_mipmapData[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (s==0) s=1;
|
|
|
|
if (t==0) t=1;
|
|
|
|
if (r==0) r=1;
|
|
|
|
|
2004-05-26 00:10:28 +08:00
|
|
|
unsigned int sizeOfLastMipMap;
|
|
|
|
if(_pixelFormat & 0x80F0)
|
|
|
|
{
|
|
|
|
unsigned int blockSize = (_pixelFormat == GL_COMPRESSED_RGBA_S3TC_DXT1_EXT) ? 8 : 16;
|
|
|
|
sizeOfLastMipMap = maximum(computeRowWidthInBytes(s,_pixelFormat,_dataType,_packing)* r*t, blockSize);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
sizeOfLastMipMap = computeRowWidthInBytes(s,_pixelFormat,_dataType,_packing)* r*t;
|
2003-06-24 23:40:09 +08:00
|
|
|
|
2003-06-25 05:57:13 +08:00
|
|
|
// notify(INFO)<<"sizeOfLastMipMap="<<sizeOfLastMipMap<<"\ts="<<s<<"\tt="<<t<<"\tr"<<r<<std::endl;
|
2004-05-26 00:10:28 +08:00
|
|
|
|
2003-06-24 23:40:09 +08:00
|
|
|
return maxValue+sizeOfLastMipMap;
|
2004-05-26 00:10:28 +08:00
|
|
|
|
2003-06-24 23:40:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-04-16 22:57:39 +08:00
|
|
|
void Image::setInternalTextureFormat(GLint internalFormat)
|
|
|
|
{
|
|
|
|
// won't do any sanity checking right now, leave it to
|
|
|
|
// OpenGL to make the call.
|
|
|
|
_internalTextureFormat = internalFormat;
|
|
|
|
}
|
|
|
|
|
2003-09-22 17:13:22 +08:00
|
|
|
void Image::setPixelFormat(GLenum pixelFormat)
|
2002-04-16 22:57:39 +08:00
|
|
|
{
|
2003-09-22 17:13:22 +08:00
|
|
|
if (_pixelFormat==pixelFormat) return; // do nothing if the same.
|
2002-04-16 22:57:39 +08:00
|
|
|
|
2003-09-22 17:13:22 +08:00
|
|
|
if (computeNumComponents(_pixelFormat)==computeNumComponents(pixelFormat))
|
2002-04-16 22:57:39 +08:00
|
|
|
{
|
|
|
|
// if the two formats have the same number of componets then
|
|
|
|
// we can do a straight swap.
|
2003-09-22 17:13:22 +08:00
|
|
|
_pixelFormat = pixelFormat;
|
2002-04-16 22:57:39 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
notify(WARN)<<"Image::setPixelFormat(..) - warning, attempt to reset the pixel format with a different number of components."<<std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-08-27 18:06:57 +08:00
|
|
|
void Image::allocateImage(int s,int t,int r,
|
2002-04-12 01:15:07 +08:00
|
|
|
GLenum format,GLenum type,
|
|
|
|
int packing)
|
|
|
|
{
|
2002-04-23 05:13:33 +08:00
|
|
|
_mipmapData.clear();
|
|
|
|
|
2003-07-23 16:18:19 +08:00
|
|
|
unsigned int previousTotalSize = 0;
|
|
|
|
|
|
|
|
if (_data) previousTotalSize = computeRowWidthInBytes(_s,_pixelFormat,_dataType,_packing)*_t*_r;
|
2002-04-12 01:15:07 +08:00
|
|
|
|
|
|
|
unsigned int newTotalSize = computeRowWidthInBytes(s,format,type,packing)*t*r;
|
|
|
|
|
|
|
|
if (newTotalSize!=previousTotalSize)
|
|
|
|
{
|
2002-04-14 21:41:13 +08:00
|
|
|
if (newTotalSize)
|
2003-02-25 19:56:18 +08:00
|
|
|
setData(new unsigned char [newTotalSize],USE_NEW_DELETE);
|
2002-04-14 21:41:13 +08:00
|
|
|
else
|
2003-02-25 19:56:18 +08:00
|
|
|
deallocateData(); // and sets it to NULL.
|
2002-04-12 01:15:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (_data)
|
|
|
|
{
|
|
|
|
_s = s;
|
|
|
|
_t = t;
|
|
|
|
_r = r;
|
|
|
|
_pixelFormat = format;
|
|
|
|
_dataType = type;
|
|
|
|
_packing = packing;
|
2004-03-18 04:03:56 +08:00
|
|
|
_internalTextureFormat = format;
|
2002-04-12 01:15:07 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2003-01-22 00:45:36 +08:00
|
|
|
|
2002-04-12 01:15:07 +08:00
|
|
|
// throw exception?? not for now, will simply set values to 0.
|
|
|
|
_s = 0;
|
|
|
|
_t = 0;
|
|
|
|
_r = 0;
|
|
|
|
_pixelFormat = 0;
|
|
|
|
_dataType = 0;
|
|
|
|
_packing = 0;
|
2004-03-18 04:03:56 +08:00
|
|
|
_internalTextureFormat = 0;
|
2002-04-12 01:15:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
++_modifiedTag;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Image::setImage(int s,int t,int r,
|
|
|
|
GLint internalTextureFormat,
|
|
|
|
GLenum format,GLenum type,
|
2001-01-11 00:32:10 +08:00
|
|
|
unsigned char *data,
|
2003-02-25 19:56:18 +08:00
|
|
|
AllocationMode mode,
|
2002-04-12 01:15:07 +08:00
|
|
|
int packing)
|
2001-01-11 00:32:10 +08:00
|
|
|
{
|
2002-04-23 05:13:33 +08:00
|
|
|
_mipmapData.clear();
|
2001-01-11 00:32:10 +08:00
|
|
|
|
|
|
|
_s = s;
|
|
|
|
_t = t;
|
|
|
|
_r = r;
|
|
|
|
|
2002-04-12 01:15:07 +08:00
|
|
|
_internalTextureFormat = internalTextureFormat;
|
|
|
|
_pixelFormat = format;
|
|
|
|
_dataType = type;
|
2001-01-11 00:32:10 +08:00
|
|
|
|
2003-02-25 19:56:18 +08:00
|
|
|
setData(data,mode);
|
|
|
|
|
2002-04-12 01:15:07 +08:00
|
|
|
_packing = packing;
|
2002-04-11 05:51:34 +08:00
|
|
|
|
2001-11-19 05:31:16 +08:00
|
|
|
++_modifiedTag;
|
2001-01-11 00:32:10 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2002-04-11 05:51:34 +08:00
|
|
|
void Image::readPixels(int x,int y,int width,int height,
|
2002-04-12 01:15:07 +08:00
|
|
|
GLenum format,GLenum type)
|
2002-04-11 05:51:34 +08:00
|
|
|
{
|
2002-08-27 18:06:57 +08:00
|
|
|
allocateImage(width,height,1,format,type);
|
2002-04-11 05:51:34 +08:00
|
|
|
|
|
|
|
glPixelStorei(GL_PACK_ALIGNMENT,_packing);
|
|
|
|
|
2002-04-12 01:15:07 +08:00
|
|
|
glReadPixels(x,y,width,height,format,type,_data);
|
2002-04-11 05:51:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-07-06 05:09:30 +08:00
|
|
|
void Image::readImageFromCurrentTexture(unsigned int contextID, bool copyMipMapsIfAvailable)
|
2003-04-03 02:26:34 +08:00
|
|
|
{
|
|
|
|
const osg::Texture::Extensions* extensions = osg::Texture::getExtensions(contextID,true);
|
2004-09-29 18:01:46 +08:00
|
|
|
const osg::Texture3D::Extensions* extensions3D = osg::Texture3D::getExtensions(contextID,true);
|
|
|
|
|
|
|
|
|
|
|
|
GLboolean binding1D, binding2D, binding3D;
|
|
|
|
glGetBooleanv(GL_TEXTURE_BINDING_1D, &binding1D);
|
|
|
|
glGetBooleanv(GL_TEXTURE_BINDING_2D, &binding2D);
|
|
|
|
glGetBooleanv(GL_TEXTURE_BINDING_3D, &binding3D);
|
|
|
|
|
|
|
|
GLenum textureMode = binding1D ? GL_TEXTURE_1D : binding2D ? GL_TEXTURE_2D : binding3D ? GL_TEXTURE_3D : 0;
|
|
|
|
|
|
|
|
if (textureMode==0) return;
|
2003-04-03 02:26:34 +08:00
|
|
|
|
|
|
|
GLint internalformat;
|
|
|
|
GLint width;
|
|
|
|
GLint height;
|
2004-07-06 05:09:30 +08:00
|
|
|
GLint depth;
|
2004-09-29 18:01:46 +08:00
|
|
|
|
2004-07-06 05:09:30 +08:00
|
|
|
GLint numMipMaps = 0;
|
|
|
|
if (copyMipMapsIfAvailable)
|
|
|
|
{
|
|
|
|
for(;numMipMaps<20;++numMipMaps)
|
|
|
|
{
|
2004-09-29 18:01:46 +08:00
|
|
|
glGetTexLevelParameteriv(textureMode, numMipMaps, GL_TEXTURE_WIDTH, &width);
|
|
|
|
glGetTexLevelParameteriv(textureMode, numMipMaps, GL_TEXTURE_HEIGHT, &height);
|
|
|
|
glGetTexLevelParameteriv(textureMode, numMipMaps, GL_TEXTURE_DEPTH, &depth);
|
2004-07-06 05:09:30 +08:00
|
|
|
if (width==0 || height==0 || depth==0) break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
numMipMaps = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
GLint compressed = 0;
|
2003-04-03 02:26:34 +08:00
|
|
|
|
2004-09-29 18:01:46 +08:00
|
|
|
if (textureMode==GL_TEXTURE_2D)
|
|
|
|
{
|
|
|
|
if (extensions->isCompressedTexImage2DSupported())
|
|
|
|
{
|
|
|
|
glGetTexLevelParameteriv(GL_TEXTURE_2D, 0, GL_TEXTURE_COMPRESSED_ARB,&compressed);
|
|
|
|
}
|
|
|
|
}
|
2004-09-30 17:10:29 +08:00
|
|
|
else if (textureMode==GL_TEXTURE_3D)
|
2003-04-03 02:26:34 +08:00
|
|
|
{
|
2004-09-29 18:01:46 +08:00
|
|
|
if (extensions3D->isCompressedTexImage3DSupported())
|
|
|
|
{
|
|
|
|
glGetTexLevelParameteriv(GL_TEXTURE_3D, 0, GL_TEXTURE_COMPRESSED_ARB,&compressed);
|
|
|
|
}
|
2004-07-06 05:09:30 +08:00
|
|
|
}
|
|
|
|
|
2004-09-29 18:01:46 +08:00
|
|
|
|
|
|
|
|
2004-07-06 05:09:30 +08:00
|
|
|
/* if the compression has been successful */
|
|
|
|
if (compressed == GL_TRUE)
|
|
|
|
{
|
2003-04-03 02:26:34 +08:00
|
|
|
|
2004-07-06 05:09:30 +08:00
|
|
|
MipmapDataType mipMapData;
|
|
|
|
|
|
|
|
unsigned int total_size = 0;
|
|
|
|
GLint i;
|
|
|
|
for(i=0;i<numMipMaps;++i)
|
2003-04-03 02:26:34 +08:00
|
|
|
{
|
2004-07-06 05:09:30 +08:00
|
|
|
if (i>0) mipMapData.push_back(total_size);
|
2003-04-03 02:26:34 +08:00
|
|
|
|
2004-07-06 05:09:30 +08:00
|
|
|
GLint compressed_size;
|
2004-09-29 18:01:46 +08:00
|
|
|
glGetTexLevelParameteriv(textureMode, i, GL_TEXTURE_COMPRESSED_IMAGE_SIZE_ARB, &compressed_size);
|
2003-04-03 02:26:34 +08:00
|
|
|
|
2004-07-06 05:09:30 +08:00
|
|
|
total_size += compressed_size;
|
2003-04-03 02:26:34 +08:00
|
|
|
}
|
|
|
|
|
2004-07-06 05:09:30 +08:00
|
|
|
|
|
|
|
unsigned char* data = new unsigned char[total_size];
|
|
|
|
if (!data)
|
|
|
|
{
|
|
|
|
osg::notify(osg::WARN)<<"Warning: Image::readImageFromCurrentTexture(..) out of memory, now image read."<<std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-07-06 05:46:02 +08:00
|
|
|
deallocateData(); // and sets it to NULL.
|
|
|
|
|
2004-09-29 18:01:46 +08:00
|
|
|
glGetTexLevelParameteriv(textureMode, 0, GL_TEXTURE_INTERNAL_FORMAT, &internalformat);
|
|
|
|
glGetTexLevelParameteriv(textureMode, 0, GL_TEXTURE_WIDTH, &width);
|
|
|
|
glGetTexLevelParameteriv(textureMode, 0, GL_TEXTURE_HEIGHT, &height);
|
|
|
|
glGetTexLevelParameteriv(textureMode, 0, GL_TEXTURE_DEPTH, &depth);
|
2004-07-06 05:09:30 +08:00
|
|
|
|
|
|
|
_data = data;
|
|
|
|
_s = width;
|
|
|
|
_t = height;
|
|
|
|
_r = depth;
|
|
|
|
|
|
|
|
_pixelFormat = internalformat;
|
|
|
|
_dataType = internalformat;
|
|
|
|
_internalTextureFormat = internalformat;
|
|
|
|
_mipmapData = mipMapData;
|
2004-07-06 05:46:02 +08:00
|
|
|
_allocationMode=USE_NEW_DELETE;
|
2004-07-06 05:09:30 +08:00
|
|
|
|
|
|
|
for(i=0;i<numMipMaps;++i)
|
|
|
|
{
|
2004-09-29 18:01:46 +08:00
|
|
|
extensions->glGetCompressedTexImage(textureMode, i, getMipmapData(i));
|
2004-07-06 05:09:30 +08:00
|
|
|
}
|
|
|
|
|
2004-07-06 05:46:02 +08:00
|
|
|
++_modifiedTag;
|
|
|
|
|
2003-04-03 02:26:34 +08:00
|
|
|
}
|
2004-07-06 05:09:30 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
MipmapDataType mipMapData;
|
2003-04-03 02:26:34 +08:00
|
|
|
|
2004-07-06 05:09:30 +08:00
|
|
|
unsigned int total_size = 0;
|
|
|
|
GLint i;
|
|
|
|
for(i=0;i<numMipMaps;++i)
|
|
|
|
{
|
|
|
|
if (i>0) mipMapData.push_back(total_size);
|
|
|
|
|
2004-09-29 18:01:46 +08:00
|
|
|
glGetTexLevelParameteriv(textureMode, i, GL_TEXTURE_WIDTH, &width);
|
|
|
|
glGetTexLevelParameteriv(textureMode, i, GL_TEXTURE_HEIGHT, &height);
|
|
|
|
glGetTexLevelParameteriv(textureMode, i, GL_TEXTURE_DEPTH, &depth);
|
2004-07-06 05:09:30 +08:00
|
|
|
|
|
|
|
unsigned int level_size = computeRowWidthInBytes(width,_pixelFormat,GL_UNSIGNED_BYTE,_packing)*height*depth;
|
2003-04-03 02:26:34 +08:00
|
|
|
|
2004-07-06 05:09:30 +08:00
|
|
|
total_size += level_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
unsigned char* data = new unsigned char[total_size];
|
|
|
|
if (!data)
|
|
|
|
{
|
|
|
|
osg::notify(osg::WARN)<<"Warning: Image::readImageFromCurrentTexture(..) out of memory, now image read."<<std::endl;
|
|
|
|
return;
|
|
|
|
}
|
2003-04-03 02:26:34 +08:00
|
|
|
|
2004-07-06 05:46:02 +08:00
|
|
|
deallocateData(); // and sets it to NULL.
|
|
|
|
|
2004-09-29 18:01:46 +08:00
|
|
|
glGetTexLevelParameteriv(textureMode, 0, GL_TEXTURE_INTERNAL_FORMAT, &internalformat);
|
|
|
|
glGetTexLevelParameteriv(textureMode, 0, GL_TEXTURE_WIDTH, &width);
|
|
|
|
glGetTexLevelParameteriv(textureMode, 0, GL_TEXTURE_HEIGHT, &height);
|
|
|
|
glGetTexLevelParameteriv(textureMode, 0, GL_TEXTURE_DEPTH, &depth);
|
2004-07-06 05:09:30 +08:00
|
|
|
|
|
|
|
_data = data;
|
|
|
|
_s = width;
|
|
|
|
_t = height;
|
|
|
|
_r = depth;
|
|
|
|
|
|
|
|
_pixelFormat = internalformat;
|
|
|
|
_dataType = GL_UNSIGNED_BYTE;
|
|
|
|
_internalTextureFormat = internalformat;
|
|
|
|
_mipmapData = mipMapData;
|
2004-07-06 05:46:02 +08:00
|
|
|
_allocationMode=USE_NEW_DELETE;
|
2004-07-06 05:09:30 +08:00
|
|
|
|
|
|
|
for(i=0;i<numMipMaps;++i)
|
|
|
|
{
|
2004-09-29 18:01:46 +08:00
|
|
|
glGetTexImage(textureMode,i,_pixelFormat,_dataType,getMipmapData(i));
|
2004-07-06 05:09:30 +08:00
|
|
|
}
|
|
|
|
|
2004-07-06 05:46:02 +08:00
|
|
|
++_modifiedTag;
|
2004-07-06 05:09:30 +08:00
|
|
|
}
|
2003-04-03 02:26:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-30 07:10:11 +08:00
|
|
|
void Image::scaleImage(int s,int t,int r, GLenum newDataType)
|
2001-01-11 00:32:10 +08:00
|
|
|
{
|
2002-08-27 18:06:57 +08:00
|
|
|
if (_s==s && _t==t && _r==r) return;
|
|
|
|
|
2002-04-12 01:15:07 +08:00
|
|
|
if (_data==NULL)
|
|
|
|
{
|
|
|
|
notify(WARN) << "Error Image::scaleImage() do not succeed : cannot scale NULL image."<<std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_r!=1 || r!=1)
|
|
|
|
{
|
|
|
|
notify(WARN) << "Error Image::scaleImage() do not succeed : scaling of volumes not implemented."<<std::endl;
|
|
|
|
return;
|
|
|
|
}
|
2002-08-27 18:06:57 +08:00
|
|
|
|
2002-04-12 01:15:07 +08:00
|
|
|
|
|
|
|
|
2003-10-30 07:10:11 +08:00
|
|
|
unsigned int newTotalSize = computeRowWidthInBytes(s,_pixelFormat,newDataType,_packing)*t;
|
2001-01-11 00:32:10 +08:00
|
|
|
|
2002-04-11 05:51:34 +08:00
|
|
|
// need to sort out what size to really use...
|
2003-02-25 19:56:18 +08:00
|
|
|
unsigned char* newData = new unsigned char [newTotalSize];
|
2002-04-14 21:41:13 +08:00
|
|
|
if (!newData)
|
2002-04-12 01:15:07 +08:00
|
|
|
{
|
|
|
|
// should we throw an exception??? Just return for time being.
|
2002-05-28 19:40:37 +08:00
|
|
|
notify(FATAL) << "Error Image::scaleImage() do not succeed : out of memory."<<newTotalSize<<std::endl;
|
2002-04-12 01:15:07 +08:00
|
|
|
return;
|
|
|
|
}
|
2001-01-11 00:32:10 +08:00
|
|
|
|
|
|
|
glPixelStorei(GL_PACK_ALIGNMENT,_packing);
|
|
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT,_packing);
|
|
|
|
|
2002-04-12 01:15:07 +08:00
|
|
|
GLint status = gluScaleImage(_pixelFormat,
|
2001-09-20 05:08:56 +08:00
|
|
|
_s,
|
|
|
|
_t,
|
2002-04-12 01:15:07 +08:00
|
|
|
_dataType,
|
2001-09-20 05:08:56 +08:00
|
|
|
_data,
|
|
|
|
s,
|
|
|
|
t,
|
2003-10-30 07:10:11 +08:00
|
|
|
newDataType,
|
2001-09-20 05:08:56 +08:00
|
|
|
newData);
|
|
|
|
|
|
|
|
if (status==0)
|
|
|
|
{
|
|
|
|
|
2001-01-11 00:32:10 +08:00
|
|
|
// free old image.
|
|
|
|
_s = s;
|
|
|
|
_t = t;
|
2003-10-30 07:10:11 +08:00
|
|
|
_dataType = newDataType;
|
2003-02-25 19:56:18 +08:00
|
|
|
setData(newData,USE_NEW_DELETE);
|
2001-01-11 00:32:10 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2003-02-25 19:56:18 +08:00
|
|
|
delete [] newData;
|
2001-01-11 00:32:10 +08:00
|
|
|
|
2003-12-24 08:14:45 +08:00
|
|
|
notify(WARN) << "Error Image::scaleImage() did not succeed : errorString = "<<gluErrorString((GLenum)status)<<std::endl;
|
2001-01-11 00:32:10 +08:00
|
|
|
}
|
2001-11-19 05:31:16 +08:00
|
|
|
|
|
|
|
++_modifiedTag;
|
2001-01-11 00:32:10 +08:00
|
|
|
}
|
|
|
|
|
2002-08-27 18:06:57 +08:00
|
|
|
void Image::copySubImage(int s_offset,int t_offset,int r_offset,osg::Image* source)
|
|
|
|
{
|
|
|
|
if (!source) return;
|
|
|
|
if (s_offset<0 || t_offset<0 || r_offset<0)
|
|
|
|
{
|
|
|
|
notify(WARN)<<"Warning: negative offsets passed to Image::copySubImage(..) not supported, operation ignored."<<std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!_data)
|
|
|
|
{
|
2003-06-25 05:57:13 +08:00
|
|
|
notify(INFO)<<"allocating image"<<endl;
|
2004-10-18 22:46:57 +08:00
|
|
|
allocateImage(s_offset+source->s(),t_offset+source->t(),r_offset+source->r(),
|
2002-08-27 18:06:57 +08:00
|
|
|
source->getPixelFormat(),source->getDataType(),
|
|
|
|
source->getPacking());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (s_offset>=_s || t_offset>=_t || r_offset>=_r)
|
|
|
|
{
|
|
|
|
notify(WARN)<<"Warning: offsets passed to Image::copySubImage(..) outside destination image, operation ignored."<<std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (_pixelFormat != source->getPixelFormat())
|
|
|
|
{
|
|
|
|
notify(WARN)<<"Warning: image with an incompatible pixel formats passed to Image::copySubImage(..), operation ignored."<<std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void* data_destination = data(s_offset,t_offset,r_offset);
|
|
|
|
|
|
|
|
glPixelStorei(GL_PACK_ALIGNMENT,source->getPacking());
|
|
|
|
glPixelStorei(GL_PACK_ROW_LENGTH,_s);
|
|
|
|
|
|
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT,_packing);
|
|
|
|
|
|
|
|
GLint status = gluScaleImage(_pixelFormat,
|
|
|
|
source->s(),
|
|
|
|
source->t(),
|
|
|
|
source->getDataType(),
|
|
|
|
source->data(),
|
|
|
|
source->s(),
|
|
|
|
source->t(),
|
|
|
|
_dataType,
|
|
|
|
data_destination);
|
|
|
|
|
|
|
|
glPixelStorei(GL_PACK_ROW_LENGTH,0);
|
|
|
|
|
|
|
|
if (status!=0)
|
|
|
|
{
|
|
|
|
notify(WARN) << "Error Image::scaleImage() do not succeed : errorString = "<<gluErrorString((GLenum)status)<<std::endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-08-17 04:57:24 +08:00
|
|
|
void Image::flipHorizontal()
|
2002-05-14 17:34:11 +08:00
|
|
|
{
|
|
|
|
if (_data==NULL)
|
|
|
|
{
|
2002-05-14 18:20:55 +08:00
|
|
|
notify(WARN) << "Error Image::flipHorizontal() do not succeed : cannot flip NULL image."<<std::endl;
|
2002-05-14 17:34:11 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int elemSize = getPixelSizeInBits()/8;
|
|
|
|
|
2004-08-17 04:57:24 +08:00
|
|
|
for(int r=0;r<_r;++r)
|
2002-05-14 17:34:11 +08:00
|
|
|
{
|
2004-08-17 04:57:24 +08:00
|
|
|
for (int t=0; t<_t; ++t)
|
2002-05-14 17:34:11 +08:00
|
|
|
{
|
2004-08-17 04:57:24 +08:00
|
|
|
unsigned char* rowData = _data+t*getRowSizeInBytes()+r*getImageSizeInBytes();
|
|
|
|
unsigned char* left = rowData ;
|
|
|
|
unsigned char* right = rowData + ((_s-1)*getPixelSizeInBits())/8;
|
|
|
|
|
|
|
|
while (left < right)
|
|
|
|
{
|
|
|
|
char tmp[32]; // max elem size is four floats
|
|
|
|
memcpy(tmp, left, elemSize);
|
|
|
|
memcpy(left, right, elemSize);
|
|
|
|
memcpy(right, tmp, elemSize);
|
|
|
|
left += elemSize;
|
|
|
|
right -= elemSize;
|
|
|
|
}
|
2002-05-14 17:34:11 +08:00
|
|
|
}
|
|
|
|
}
|
2004-08-17 04:57:24 +08:00
|
|
|
|
2002-05-14 17:34:11 +08:00
|
|
|
++_modifiedTag;
|
|
|
|
}
|
|
|
|
|
2004-08-17 04:57:24 +08:00
|
|
|
void flipImageVertical(unsigned char* top, unsigned char* bottom, unsigned int rowSize)
|
|
|
|
{
|
|
|
|
while(top<bottom)
|
|
|
|
{
|
|
|
|
for(unsigned int i=0;i<rowSize;++i, ++top,++bottom)
|
|
|
|
{
|
|
|
|
unsigned char temp=*top;
|
|
|
|
*top = *bottom;
|
|
|
|
*bottom = temp;
|
|
|
|
}
|
|
|
|
bottom -= 2*rowSize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-05-14 17:34:11 +08:00
|
|
|
|
2004-08-17 04:57:24 +08:00
|
|
|
void Image::flipVertical()
|
2002-05-14 17:34:11 +08:00
|
|
|
{
|
|
|
|
if (_data==NULL)
|
|
|
|
{
|
|
|
|
notify(WARN) << "Error Image::flipVertical() do not succeed : cannot flip NULL image."<<std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-08-17 04:57:24 +08:00
|
|
|
if (!_mipmapData.empty() && _r>1)
|
|
|
|
{
|
|
|
|
notify(WARN) << "Error Image::flipVertical() do not succeed : flipping of mipmap 3d textures not yet supported."<<std::endl;
|
|
|
|
return;
|
|
|
|
}
|
2002-05-14 17:34:11 +08:00
|
|
|
|
2004-08-17 04:57:24 +08:00
|
|
|
if (_mipmapData.empty())
|
2002-05-14 17:34:11 +08:00
|
|
|
{
|
2004-08-17 04:57:24 +08:00
|
|
|
// no mipmaps,
|
|
|
|
// so we can safely handle 3d textures
|
|
|
|
for(int r=0;r<_r;++r)
|
|
|
|
{
|
|
|
|
if (!dxtc_tool::VerticalFlip(_s,_t,_pixelFormat,data(0,0,r)))
|
|
|
|
{
|
|
|
|
// its not a compressed image, so implement flip oursleves.
|
|
|
|
|
|
|
|
unsigned int rowSize = computeRowWidthInBytes(_s,_pixelFormat,_dataType,_packing);
|
|
|
|
unsigned char* top = data(0,0,r);
|
|
|
|
unsigned char* bottom = top + (_t-1)*rowSize;
|
|
|
|
|
|
|
|
flipImageVertical(top, bottom, rowSize);
|
|
|
|
}
|
|
|
|
}
|
2002-05-14 17:34:11 +08:00
|
|
|
}
|
2004-08-17 04:57:24 +08:00
|
|
|
else if (_r==1)
|
|
|
|
{
|
|
|
|
if (!dxtc_tool::VerticalFlip(_s,_t,_pixelFormat,_data))
|
|
|
|
{
|
|
|
|
// its not a compressed image, so implement flip oursleves.
|
|
|
|
unsigned int rowSize = computeRowWidthInBytes(_s,_pixelFormat,_dataType,_packing);
|
|
|
|
unsigned char* top = data(0,0,0);
|
|
|
|
unsigned char* bottom = top + (_t-1)*rowSize;
|
2002-05-14 17:34:11 +08:00
|
|
|
|
2004-08-17 04:57:24 +08:00
|
|
|
flipImageVertical(top, bottom, rowSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
int s = _s;
|
|
|
|
int t = _t;
|
|
|
|
//int r = _r;
|
|
|
|
|
|
|
|
for(unsigned int i=0;i<_mipmapData.size() && _mipmapData[i];++i)
|
|
|
|
{
|
|
|
|
s >>= 1;
|
|
|
|
t >>= 1;
|
|
|
|
if (s==0) s=1;
|
|
|
|
if (t==0) t=1;
|
|
|
|
if (!dxtc_tool::VerticalFlip(s,t,_pixelFormat,_data+_mipmapData[i]))
|
|
|
|
{
|
|
|
|
// its not a compressed image, so implement flip oursleves.
|
|
|
|
unsigned int rowSize = computeRowWidthInBytes(s,_pixelFormat,_dataType,_packing);
|
|
|
|
unsigned char* top = _data+_mipmapData[i];
|
|
|
|
unsigned char* bottom = top + (t-1)*rowSize;
|
|
|
|
|
|
|
|
flipImageVertical(top, bottom, rowSize);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2002-05-14 17:34:11 +08:00
|
|
|
|
|
|
|
++_modifiedTag;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-09-20 05:08:56 +08:00
|
|
|
|
2002-09-17 04:58:05 +08:00
|
|
|
void Image::ensureValidSizeForTexturing(GLint maxTextureSize)
|
2001-01-11 00:32:10 +08:00
|
|
|
{
|
2002-08-16 21:33:32 +08:00
|
|
|
int new_s = computeNearestPowerOfTwo(_s);
|
|
|
|
int new_t = computeNearestPowerOfTwo(_t);
|
2001-10-05 18:38:16 +08:00
|
|
|
|
2002-09-17 04:58:05 +08:00
|
|
|
if (new_s>maxTextureSize) new_s = maxTextureSize;
|
|
|
|
if (new_t>maxTextureSize) new_t = maxTextureSize;
|
2001-10-05 18:38:16 +08:00
|
|
|
|
|
|
|
if (new_s!=_s || new_t!=_t)
|
2001-01-11 00:32:10 +08:00
|
|
|
{
|
2001-12-15 07:18:28 +08:00
|
|
|
if (!_fileName.empty()) notify(NOTICE) << "Scaling image '"<<_fileName<<"' from ("<<_s<<","<<_t<<") to ("<<new_s<<","<<new_t<<")"<<std::endl;
|
|
|
|
else notify(NOTICE) << "Scaling image from ("<<_s<<","<<_t<<") to ("<<new_s<<","<<new_t<<")"<<std::endl;
|
2001-09-20 05:08:56 +08:00
|
|
|
|
2001-01-11 00:32:10 +08:00
|
|
|
scaleImage(new_s,new_t,_r);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-11-26 00:39:05 +08:00
|
|
|
bool Image::isImageTranslucent() const
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
if (getPixelFormat()==GL_LUMINANCE_ALPHA)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
|
|
// unfinished code so commenting out right now...
|
|
|
|
// switch(type)
|
|
|
|
// {
|
|
|
|
// case(GL_BYTE):
|
|
|
|
// {
|
|
|
|
// int rowSizeInBytes = getRowSizeInBytes();
|
|
|
|
// int imageSizeInBytes = getImageSizeInBytes();
|
|
|
|
//
|
|
|
|
// for (int r=0;r<_r,++r)
|
|
|
|
// {
|
|
|
|
// for (int t=0;t<int _t,++t)
|
|
|
|
// {
|
|
|
|
// char* ptr = _data+r*getRowSizeInBytes()+image*getImageSizeInBytes();
|
|
|
|
// for (int s=0;s<_s,++s)
|
|
|
|
// {
|
|
|
|
//
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// break
|
|
|
|
// }
|
|
|
|
// case(GL_UNSIGNED_BYTE): return 8*computeNumComponents(format);
|
|
|
|
//
|
|
|
|
// case(GL_SHORT):
|
|
|
|
// case(GL_UNSIGNED_SHORT): return 16*computeNumComponents(format);
|
|
|
|
//
|
|
|
|
// case(GL_INT):
|
|
|
|
// case(GL_UNSIGNED_INT):
|
|
|
|
// case(GL_FLOAT): return 32*computeNumComponents(format);
|
|
|
|
//
|
|
|
|
// default: return 0;
|
|
|
|
// }
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (getPixelFormat()==GL_RGBA)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2002-08-16 21:33:32 +08:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2001-09-20 05:08:56 +08:00
|
|
|
|
2001-01-11 00:32:10 +08:00
|
|
|
Geode* osg::createGeodeForImage(osg::Image* image)
|
|
|
|
{
|
|
|
|
return createGeodeForImage(image,image->s(),image->t());
|
|
|
|
}
|
|
|
|
|
2001-09-20 05:08:56 +08:00
|
|
|
|
2002-09-02 20:31:35 +08:00
|
|
|
Geode* osg::createGeodeForImage(osg::Image* image,float s,float t)
|
2001-01-11 00:32:10 +08:00
|
|
|
{
|
|
|
|
if (image)
|
|
|
|
{
|
|
|
|
if (s>0 && t>0)
|
|
|
|
{
|
|
|
|
|
|
|
|
float y = 1.0;
|
|
|
|
float x = y*(s/t);
|
|
|
|
|
2001-09-20 05:08:56 +08:00
|
|
|
// set up the texture.
|
2002-12-16 21:40:58 +08:00
|
|
|
osg::Texture2D* texture = new osg::Texture2D;
|
2001-01-11 00:32:10 +08:00
|
|
|
texture->setImage(image);
|
|
|
|
|
2001-09-20 05:08:56 +08:00
|
|
|
// set up the drawstate.
|
2002-12-16 21:40:58 +08:00
|
|
|
osg::StateSet* dstate = new osg::StateSet;
|
2001-09-20 05:08:56 +08:00
|
|
|
dstate->setMode(GL_CULL_FACE,osg::StateAttribute::OFF);
|
|
|
|
dstate->setMode(GL_LIGHTING,osg::StateAttribute::OFF);
|
2002-07-10 19:22:24 +08:00
|
|
|
dstate->setTextureAttributeAndModes(0, texture,osg::StateAttribute::ON);
|
2001-01-11 00:32:10 +08:00
|
|
|
|
|
|
|
// set up the geoset.
|
2002-12-16 21:40:58 +08:00
|
|
|
Geometry* geom = new Geometry;
|
2002-06-26 04:27:51 +08:00
|
|
|
geom->setStateSet(dstate);
|
|
|
|
|
2002-12-16 21:40:58 +08:00
|
|
|
Vec3Array* coords = new Vec3Array(4);
|
2002-06-26 04:27:51 +08:00
|
|
|
(*coords)[0].set(-x,0.0f,y);
|
|
|
|
(*coords)[1].set(-x,0.0f,-y);
|
|
|
|
(*coords)[2].set(x,0.0f,-y);
|
|
|
|
(*coords)[3].set(x,0.0f,y);
|
|
|
|
geom->setVertexArray(coords);
|
|
|
|
|
2002-12-16 21:40:58 +08:00
|
|
|
Vec2Array* tcoords = new Vec2Array(4);
|
2002-06-26 04:27:51 +08:00
|
|
|
(*tcoords)[0].set(0.0f,1.0f);
|
|
|
|
(*tcoords)[1].set(0.0f,0.0f);
|
|
|
|
(*tcoords)[2].set(1.0f,0.0f);
|
|
|
|
(*tcoords)[3].set(1.0f,1.0f);
|
|
|
|
geom->setTexCoordArray(0,tcoords);
|
|
|
|
|
2002-12-16 21:40:58 +08:00
|
|
|
osg::Vec4Array* colours = new osg::Vec4Array(1);
|
2002-06-26 04:27:51 +08:00
|
|
|
(*colours)[0].set(1.0f,1.0f,1.0,1.0f);
|
|
|
|
geom->setColorArray(colours);
|
|
|
|
geom->setColorBinding(Geometry::BIND_OVERALL);
|
|
|
|
|
2002-12-16 21:40:58 +08:00
|
|
|
geom->addPrimitiveSet(new DrawArrays(PrimitiveSet::QUADS,0,4));
|
2001-01-11 00:32:10 +08:00
|
|
|
|
|
|
|
// set up the geode.
|
2002-12-16 21:40:58 +08:00
|
|
|
osg::Geode* geode = new osg::Geode;
|
2002-06-26 04:27:51 +08:00
|
|
|
geode->addDrawable(geom);
|
2001-01-11 00:32:10 +08:00
|
|
|
|
|
|
|
return geode;
|
|
|
|
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
2002-08-16 21:33:32 +08:00
|
|
|
|