Files
UnrealEngine/Engine/Source/Runtime/ImageWrapper/Private/Formats/IcoImageWrapper.cpp
2025-05-18 13:04:45 +08:00

229 lines
6.0 KiB
C++

// Copyright Epic Games, Inc. All Rights Reserved.
#include "Formats/IcoImageWrapper.h"
#include "BmpImageSupport.h"
#include "Formats/PngImageWrapper.h"
#include "Formats/BmpImageWrapper.h"
#pragma pack(push,1)
struct FIconDirEntry
{
uint8 bWidth; // Width, in pixels, of the image
uint8 bHeight; // Height, in pixels, of the image
uint8 bColorCount; // Number of colors in image (0 if >=8bpp)
uint8 bReserved; // Reserved ( must be 0)
uint16 wPlanes; // Color Planes
uint16 wBitCount; // Bits per pixel
uint32 dwBytesInRes; // How many bytes in this resource?
uint32 dwImageOffset; // Where in the file is this image?
};
#pragma pack(pop)
#pragma pack(push,1)
struct FIconDir
{
uint16 idReserved; // Reserved (must be 0)
uint16 idType; // Resource Type (1 for icons)
uint16 idCount; // How many images?
FIconDirEntry idEntries[1]; // An entry for each image (idCount of 'em)
};
#pragma pack(pop)
#pragma pack(push,1)
struct FRGBQuad
{
uint8 rgbBlue; // Blue channel
uint8 rgbGreen; // Green channel
uint8 rgbRed; // Red channel
uint8 rgbReserved; // Reserved (alpha)
};
#pragma pack(pop)
#pragma pack(push,1)
struct FIconImage
{
FBitmapInfoHeader icHeader; // DIB header
FRGBQuad icColors[1]; // Color table
uint8 icXOR[1]; // DIB bits for XOR mask
uint8 icAND[1]; // DIB bits for AND mask
};
#pragma pack(pop)
/* FJpegImageWrapper structors
*****************************************************************************/
FIcoImageWrapper::FIcoImageWrapper()
: FImageWrapperBase()
{ }
/* FImageWrapper interface
*****************************************************************************/
// CanSetRawFormat returns true if SetRaw will accept this format
bool FIcoImageWrapper::CanSetRawFormat(const ERGBFormat InFormat, const int32 InBitDepth) const
{
//checkf(false, TEXT("ICO compression not supported"));
return false;
}
// returns InFormat if supported, else maps to something supported
ERawImageFormat::Type FIcoImageWrapper::GetSupportedRawFormat(const ERawImageFormat::Type InFormat) const
{
//checkf(false, TEXT("ICO compression not supported"));
return ERawImageFormat::BGRA8;
}
void FIcoImageWrapper::Compress( int32 Quality )
{
checkf(false, TEXT("ICO compression not supported"));
}
void FIcoImageWrapper::Uncompress( const ERGBFormat InFormat, const int32 InBitDepth )
{
const uint8* Buffer = CompressedData.GetData();
if (ImageOffset != 0 && ImageSize != 0 && SubImageWrapper != nullptr)
{
SubImageWrapper->Uncompress(InFormat, InBitDepth);
// Uncompress has no return value
// we can tell it failed if it set an error, or has no rawdata
LastError = SubImageWrapper->GetLastError();
}
}
bool FIcoImageWrapper::SetCompressed( const void* InCompressedData, int64 InCompressedSize )
{
bool bResult = FImageWrapperBase::SetCompressed( InCompressedData, InCompressedSize );
return bResult && LoadICOHeader(); // Fetch the variables from the header info
}
bool FIcoImageWrapper::GetRaw( const ERGBFormat InFormat, int32 InBitDepth, TArray64<uint8>& OutRawData )
{
LastError.Empty();
Uncompress(InFormat, InBitDepth);
if (LastError.IsEmpty() && SubImageWrapper != nullptr)
{
SubImageWrapper->MoveRawData(OutRawData);
if ( OutRawData.IsEmpty() )
{
return false;
}
}
return LastError.IsEmpty();
}
/* FImageWrapper implementation
*****************************************************************************/
bool FIcoImageWrapper::LoadICOHeader()
{
const uint8* Buffer = CompressedData.GetData();
uint64 BufferSize = CompressedData.Num();
#if WITH_UNREALPNG
TSharedPtr<FPngImageWrapper> PngWrapper = MakeShareable(new FPngImageWrapper);
#endif
TSharedPtr<FBmpImageWrapper> BmpWrapper = MakeShareable(new FBmpImageWrapper(false, true));
bool bFoundImage = false;
SubImageWrapper = nullptr;
if ( BufferSize < sizeof(FIconDir) )
{
SetError(TEXT("LoadICOHeader corrupt; insufficient data"));
return false;
}
const FIconDir* IconHeader = (FIconDir*)(Buffer);
if (IconHeader->idReserved == 0 && IconHeader->idType == 1)
{
// use the largest-width 32-bit dir entry we find
uint32 LargestWidth = 0;
const FIconDirEntry* IconDirEntry = IconHeader->idEntries;
if ( IconHeader->idCount == 0 )
{
SetError(TEXT("LoadICOHeader corrupt; no entries"));
return false;
}
// FIconDir has 1 entry in it
if ( BufferSize < sizeof(FIconDir) + (IconHeader->idCount-1)*sizeof(FIconDirEntry) )
{
SetError(TEXT("LoadICOHeader corrupt; insufficient data"));
return false;
}
for (int32 Entry = 0; Entry < (int32)IconHeader->idCount; Entry++, IconDirEntry++)
{
const uint32 RealWidth = IconDirEntry->bWidth == 0 ? 256 : IconDirEntry->bWidth;
if ( IconDirEntry->wBitCount == 32 && RealWidth > LargestWidth )
{
if ( (uint64)IconDirEntry->dwImageOffset + IconDirEntry->dwBytesInRes > BufferSize )
{
SetError(TEXT("LoadICOHeader corrupt; insufficient data"));
return false;
}
#if WITH_UNREALPNG
if (PngWrapper->SetCompressed(Buffer + IconDirEntry->dwImageOffset, (int64)IconDirEntry->dwBytesInRes))
{
Width = PngWrapper->GetWidth();
Height = PngWrapper->GetHeight();
Format = PngWrapper->GetFormat();
BitDepth = PngWrapper->GetBitDepth();
LargestWidth = RealWidth;
bFoundImage = true;
bIsPng = true;
ImageOffset = IconDirEntry->dwImageOffset;
ImageSize = IconDirEntry->dwBytesInRes;
}
else
#endif
if (BmpWrapper->SetCompressed(Buffer + IconDirEntry->dwImageOffset, (int64)IconDirEntry->dwBytesInRes))
{
// otherwise this should be a BMP icon
Width = BmpWrapper->GetWidth();
Height = BmpWrapper->GetHeight();
Format = BmpWrapper->GetFormat();
BitDepth = BmpWrapper->GetBitDepth();
LargestWidth = RealWidth;
bFoundImage = true;
bIsPng = false;
ImageOffset = IconDirEntry->dwImageOffset;
ImageSize = IconDirEntry->dwBytesInRes;
}
}
}
}
if (bFoundImage)
{
#if WITH_UNREALPNG
if (bIsPng)
{
SubImageWrapper = PngWrapper;
}
else
#endif //WITH_UNREALPNG
{
SubImageWrapper = BmpWrapper;
}
}
return bFoundImage;
}