2065 lines
62 KiB
C++
2065 lines
62 KiB
C++
|
//---------------------------------------------------------------------------
|
||
|
// ImageFile.cpp - implements the drawing API for bgtype = ImageFile
|
||
|
//---------------------------------------------------------------------------
|
||
|
#include "stdafx.h"
|
||
|
#include "Render.h"
|
||
|
#include "Utils.h"
|
||
|
#include "tmutils.h"
|
||
|
#include "rgn.h"
|
||
|
#include "ImageFile.h"
|
||
|
#include "CacheList.h"
|
||
|
#include "DrawHelp.h"
|
||
|
#include "ninegrid2.h"
|
||
|
#include "TmReg.h"
|
||
|
#include "globals.h"
|
||
|
#include "bmpcache.h"
|
||
|
//---------------------------------------------------------------------------
|
||
|
void AdjustSizeMin(SIZE *psz, int ixMin, int iyMin)
|
||
|
{
|
||
|
if (psz->cx < ixMin)
|
||
|
{
|
||
|
psz->cx = ixMin;
|
||
|
}
|
||
|
|
||
|
if (psz->cy < iyMin)
|
||
|
{
|
||
|
psz->cy = iyMin;
|
||
|
}
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CMaxImageFile::PackMaxProperties(CRenderObj *pRender, int iPartId, int iStateId,
|
||
|
OUT int *piMultiDibCount)
|
||
|
{
|
||
|
HRESULT hr = PackProperties(pRender, iPartId, iStateId);
|
||
|
|
||
|
*piMultiDibCount = _iMultiImageCount;
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::PackProperties(CRenderObj *pRender, int iPartId, int iStateId)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
memset(this, 0, sizeof(CImageFile)); // allowed because we have no vtable
|
||
|
_eBgType = BT_IMAGEFILE;
|
||
|
|
||
|
//---- save off partid, stateid for debugging ----
|
||
|
_iSourcePartId = iPartId;
|
||
|
_iSourceStateId = iStateId;
|
||
|
|
||
|
DIBINFO *pdi = &_ImageInfo;
|
||
|
|
||
|
pdi->iMinDpi = 96; // only way this gets set for now
|
||
|
|
||
|
pdi->iDibOffset = pRender->GetValueIndex(iPartId, iStateId, TMT_DIBDATA);
|
||
|
if (pdi->iDibOffset == -1) // not found
|
||
|
pdi->iDibOffset = 0;
|
||
|
|
||
|
//---- image-related fields ----
|
||
|
if (FAILED(pRender->GetInt(iPartId, iStateId, TMT_IMAGECOUNT, &_iImageCount)))
|
||
|
_iImageCount = 1; // default value
|
||
|
|
||
|
if (_iImageCount < 1) // avoid divide by zero problems
|
||
|
_iImageCount = 1;
|
||
|
|
||
|
if (FAILED(pRender->GetEnumValue(iPartId, iStateId, TMT_IMAGELAYOUT, (int *)&_eImageLayout)))
|
||
|
_eImageLayout = IL_HORIZONTAL; // default value until we are converted
|
||
|
|
||
|
if (pdi->iDibOffset)
|
||
|
{
|
||
|
//---- compute some fields from bitmap ----
|
||
|
hr = SetImageInfo(pdi, pRender, iPartId, iStateId);
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
//---- get MinSize ----
|
||
|
if (FAILED(pRender->GetPosition(iPartId, iStateId, TMT_MINSIZE, (POINT *)&pdi->szMinSize)))
|
||
|
{
|
||
|
pdi->szMinSize.cx = pdi->iSingleWidth;
|
||
|
pdi->szMinSize.cy = pdi->iSingleHeight;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
AdjustSizeMin(&pdi->szMinSize, 1, 1);
|
||
|
}
|
||
|
|
||
|
//---- get TrueSizeScalingType ----
|
||
|
if (FAILED(pRender->GetEnumValue(iPartId, iStateId, TMT_TRUESIZESCALINGTYPE, (int *)&_eTrueSizeScalingType)))
|
||
|
_eTrueSizeScalingType = TSST_NONE; // default
|
||
|
|
||
|
//---- sizing ----
|
||
|
if (FAILED(pRender->GetEnumValue(iPartId, iStateId, TMT_SIZINGTYPE, (int *)&pdi->eSizingType)))
|
||
|
pdi->eSizingType = ST_STRETCH; // default
|
||
|
|
||
|
if (FAILED(pRender->GetBool(iPartId, iStateId, TMT_BORDERONLY, &pdi->fBorderOnly)))
|
||
|
pdi->fBorderOnly = FALSE;
|
||
|
|
||
|
if (FAILED(pRender->GetInt(iPartId, iStateId, TMT_TRUESIZESTRETCHMARK, &_iTrueSizeStretchMark)))
|
||
|
_iTrueSizeStretchMark = 0; // default
|
||
|
|
||
|
if (FAILED(pRender->GetBool(iPartId, iStateId, TMT_UNIFORMSIZING, &_fUniformSizing)))
|
||
|
_fUniformSizing = FALSE; // default
|
||
|
|
||
|
if (FAILED(pRender->GetBool(iPartId, iStateId, TMT_INTEGRALSIZING, &_fIntegralSizing)))
|
||
|
_fIntegralSizing = FALSE; // default
|
||
|
|
||
|
//---- transparency ----
|
||
|
if (FAILED(pRender->GetBool(iPartId, iStateId, TMT_TRANSPARENT, &pdi->fTransparent)))
|
||
|
pdi->fTransparent = FALSE;
|
||
|
|
||
|
if (pdi->fTransparent)
|
||
|
{
|
||
|
if (FAILED(pRender->GetColor(iPartId, iStateId, TMT_TRANSPARENTCOLOR, &pdi->crTransparent)))
|
||
|
pdi->crTransparent = DEFAULT_TRANSPARENT_COLOR;
|
||
|
}
|
||
|
|
||
|
//---- MirrorImage ----
|
||
|
if (FAILED(pRender->GetBool(iPartId, iStateId, TMT_MIRRORIMAGE, &_fMirrorImage)))
|
||
|
_fMirrorImage = TRUE; // default setting
|
||
|
|
||
|
//---- alignment ----
|
||
|
if (FAILED(pRender->GetEnumValue(iPartId, iStateId, TMT_HALIGN, (int *)&_eHAlign)))
|
||
|
_eHAlign = HA_CENTER; // default value
|
||
|
|
||
|
if (FAILED(pRender->GetEnumValue(iPartId, iStateId, TMT_VALIGN, (int *)&_eVAlign)))
|
||
|
_eVAlign = VA_CENTER; // default value
|
||
|
|
||
|
//---- for regular or glyph truesize images ----
|
||
|
if (SUCCEEDED(pRender->GetBool(iPartId, iStateId, TMT_BGFILL, &_fBgFill)))
|
||
|
{
|
||
|
//---- get fill color ----
|
||
|
if (FAILED(pRender->GetColor(iPartId, iStateId, TMT_FILLCOLOR, &_crFill)))
|
||
|
_crFill = RGB(255, 255, 255);
|
||
|
}
|
||
|
|
||
|
//---- SizingMargins ----
|
||
|
if (FAILED(pRender->GetMargins(NULL, iPartId, iStateId, TMT_SIZINGMARGINS,
|
||
|
NULL, &_SizingMargins)))
|
||
|
{
|
||
|
_SizingMargins.cxLeftWidth = 0;
|
||
|
_SizingMargins.cxRightWidth = 0;
|
||
|
_SizingMargins.cyTopHeight = 0;
|
||
|
_SizingMargins.cyBottomHeight = 0;
|
||
|
}
|
||
|
|
||
|
//---- ContentMargins ----
|
||
|
if (FAILED(pRender->GetMargins(NULL, iPartId, iStateId, TMT_CONTENTMARGINS,
|
||
|
NULL, &_ContentMargins)))
|
||
|
{
|
||
|
_ContentMargins = _SizingMargins;
|
||
|
}
|
||
|
|
||
|
//---- SourceGrow ----
|
||
|
if (FAILED(pRender->GetBool(iPartId, iStateId, TMT_SOURCEGROW, &_fSourceGrow)))
|
||
|
_fSourceGrow = FALSE; // default
|
||
|
|
||
|
//---- SourceShrink ----
|
||
|
if (FAILED(pRender->GetBool(iPartId, iStateId, TMT_SOURCESHRINK, &_fSourceShrink)))
|
||
|
_fSourceShrink = FALSE; // default
|
||
|
|
||
|
//---- NormalSize ----
|
||
|
if (FAILED(pRender->GetPosition(iPartId, iStateId, TMT_NORMALSIZE, (POINT *)&_szNormalSize)))
|
||
|
{
|
||
|
_szNormalSize.cx = 60;
|
||
|
_szNormalSize.cy = 30;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
AdjustSizeMin(&_szNormalSize, 1, 1);
|
||
|
}
|
||
|
|
||
|
//---- glphytype ----
|
||
|
if (FAILED(pRender->GetEnumValue(iPartId, iStateId, TMT_GLYPHTYPE, (int *)&_eGlyphType)))
|
||
|
_eGlyphType = GT_NONE; // default value
|
||
|
|
||
|
if (_eGlyphType == GT_FONTGLYPH)
|
||
|
{
|
||
|
//---- font-based glyphs ----
|
||
|
if (FAILED(pRender->GetFont(NULL, iPartId, iStateId, TMT_GLYPHFONT, FALSE, &_lfGlyphFont)))
|
||
|
goto exit; // required
|
||
|
|
||
|
if (FAILED(pRender->GetColor(iPartId, iStateId, TMT_GLYPHTEXTCOLOR, &_crGlyphTextColor)))
|
||
|
_crGlyphTextColor = RGB(0, 0, 0); // default color
|
||
|
|
||
|
if (FAILED(pRender->GetInt(iPartId, iStateId, TMT_GLYPHINDEX, &_iGlyphIndex)))
|
||
|
_iGlyphIndex = 1; // default index
|
||
|
}
|
||
|
else if (_eGlyphType == GT_IMAGEGLYPH)
|
||
|
{
|
||
|
//---- image-based glyphs ----
|
||
|
pdi = &_GlyphInfo;
|
||
|
|
||
|
pdi->iMinDpi = 96; // only way this gets set for now
|
||
|
|
||
|
pdi->iDibOffset = pRender->GetValueIndex(iPartId, iStateId, TMT_GLYPHDIBDATA);
|
||
|
if (pdi->iDibOffset == -1)
|
||
|
pdi->iDibOffset = 0;
|
||
|
|
||
|
if (pdi->iDibOffset > 0) // found
|
||
|
{
|
||
|
hr = SetImageInfo(pdi, pRender, iPartId, iStateId);
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(pRender->GetBool(iPartId, iStateId, TMT_GLYPHTRANSPARENT, &pdi->fTransparent)))
|
||
|
{
|
||
|
if (FAILED(pRender->GetColor(iPartId, iStateId, TMT_GLYPHTRANSPARENTCOLOR, &pdi->crTransparent)))
|
||
|
pdi->crTransparent = DEFAULT_TRANSPARENT_COLOR;
|
||
|
}
|
||
|
|
||
|
pdi->eSizingType = ST_TRUESIZE; // glyphs are always true size
|
||
|
pdi->fBorderOnly = FALSE; // glyphs are never borderonly (for now)
|
||
|
}
|
||
|
|
||
|
if (_eGlyphType != GT_NONE)
|
||
|
{
|
||
|
if (FAILED(pRender->GetBool(iPartId, iStateId, TMT_GLYPHONLY, &_fGlyphOnly)))
|
||
|
_fGlyphOnly = FALSE;
|
||
|
}
|
||
|
|
||
|
//---- multi files specified? ----
|
||
|
if (FAILED(pRender->GetEnumValue(iPartId, iStateId, TMT_IMAGESELECTTYPE, (int *)&_eImageSelectType)))
|
||
|
_eImageSelectType = IST_NONE;
|
||
|
|
||
|
//---- fill in multi DIBINFO's ----
|
||
|
if (_eImageSelectType != IST_NONE)
|
||
|
{
|
||
|
DIBINFO *pParent;
|
||
|
|
||
|
if (_eGlyphType == GT_IMAGEGLYPH)
|
||
|
{
|
||
|
pParent = &_GlyphInfo;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pParent = &_ImageInfo;
|
||
|
}
|
||
|
|
||
|
for (int i=0; i < MAX_IMAGEFILE_SIZES; i++)
|
||
|
{
|
||
|
//---- get ImageFileN ----
|
||
|
int iDibOffset = pRender->GetValueIndex(iPartId, iStateId, TMT_DIBDATA1 + i);
|
||
|
if (iDibOffset == -1)
|
||
|
break;
|
||
|
|
||
|
_iMultiImageCount++;
|
||
|
|
||
|
DIBINFO *pdi = MultiDibPtr(i);
|
||
|
|
||
|
*pdi = *pParent; // inherit some props from parent
|
||
|
pdi->iDibOffset = iDibOffset;
|
||
|
|
||
|
hr = SetImageInfo(pdi, pRender, iPartId, iStateId);
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
|
||
|
//---- get MinDpiN ----
|
||
|
if (FAILED(pRender->GetInt(iPartId, iStateId, TMT_MINDPI1 + i, &pdi->iMinDpi)))
|
||
|
{
|
||
|
pdi->iMinDpi = 96; // default
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//---- ensure value >= 1 ----
|
||
|
if (pdi->iMinDpi < 1)
|
||
|
{
|
||
|
pdi->iMinDpi = 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//---- get MinSizeN ----
|
||
|
if (FAILED(pRender->GetPosition(iPartId, iStateId, TMT_MINSIZE1 + i,
|
||
|
(POINT *)&pdi->szMinSize)))
|
||
|
{
|
||
|
pdi->szMinSize.cx = pdi->iSingleWidth;
|
||
|
pdi->szMinSize.cy = pdi->iSingleHeight;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
AdjustSizeMin(&pdi->szMinSize, 1, 1);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
if (_iMultiImageCount > 0)
|
||
|
{
|
||
|
*pParent = *MultiDibPtr(0); // use first multi entry as primary object
|
||
|
}
|
||
|
}
|
||
|
|
||
|
exit:
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
BOOL CImageFile::KeyProperty(int iPropId)
|
||
|
{
|
||
|
BOOL fKey = FALSE;
|
||
|
|
||
|
switch (iPropId)
|
||
|
{
|
||
|
case TMT_BGTYPE:
|
||
|
case TMT_TRANSPARENT:
|
||
|
case TMT_AUTOSIZE:
|
||
|
case TMT_BORDERONLY:
|
||
|
case TMT_IMAGECOUNT:
|
||
|
case TMT_ALPHALEVEL:
|
||
|
case TMT_ALPHATHRESHOLD:
|
||
|
case TMT_IMAGEFILE:
|
||
|
case TMT_IMAGEFILE1:
|
||
|
case TMT_IMAGEFILE2:
|
||
|
case TMT_IMAGEFILE3:
|
||
|
case TMT_IMAGEFILE4:
|
||
|
case TMT_IMAGEFILE5:
|
||
|
case TMT_SIZINGMARGINS:
|
||
|
case TMT_CONTENTMARGINS:
|
||
|
case TMT_TRANSPARENTCOLOR:
|
||
|
case TMT_SIZINGTYPE:
|
||
|
case TMT_HALIGN:
|
||
|
case TMT_VALIGN:
|
||
|
case TMT_IMAGELAYOUT:
|
||
|
case TMT_BGFILL:
|
||
|
case TMT_MIRRORIMAGE:
|
||
|
case TMT_TRUESIZESTRETCHMARK:
|
||
|
case TMT_TRUESIZESCALINGTYPE:
|
||
|
case TMT_IMAGESELECTTYPE:
|
||
|
case TMT_UNIFORMSIZING:
|
||
|
case TMT_INTEGRALSIZING:
|
||
|
case TMT_SOURCEGROW:
|
||
|
case TMT_SOURCESHRINK:
|
||
|
case TMT_NORMALSIZE:
|
||
|
case TMT_MINSIZE:
|
||
|
case TMT_MINSIZE1:
|
||
|
case TMT_MINSIZE2:
|
||
|
case TMT_MINSIZE3:
|
||
|
case TMT_MINSIZE4:
|
||
|
case TMT_MINSIZE5:
|
||
|
case TMT_MINDPI1:
|
||
|
case TMT_MINDPI2:
|
||
|
case TMT_MINDPI3:
|
||
|
case TMT_MINDPI4:
|
||
|
case TMT_MINDPI5:
|
||
|
|
||
|
//---- glyph properties ----
|
||
|
case TMT_GLYPHTYPE:
|
||
|
case TMT_GLYPHIMAGEFILE:
|
||
|
case TMT_GLYPHTRANSPARENT:
|
||
|
case TMT_GLYPHTRANSPARENTCOLOR:
|
||
|
case TMT_GLYPHFONT:
|
||
|
case TMT_GLYPHINDEX:
|
||
|
case TMT_GLYPHTEXTCOLOR:
|
||
|
case TMT_GLYPHONLY:
|
||
|
|
||
|
// case TMT_FILLCOLOR: - this prop belongs to BorderFill (we borrow it)
|
||
|
|
||
|
fKey = TRUE;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return fKey;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
DIBINFO *CImageFile::EnumImageFiles(int iIndex)
|
||
|
{
|
||
|
DIBINFO *pdi = NULL;
|
||
|
BOOL fHasGlyph = (_eGlyphType == GT_IMAGEGLYPH);
|
||
|
|
||
|
//---- enum in this order: primary, glyph, multi images ----
|
||
|
|
||
|
if (iIndex == 0)
|
||
|
{
|
||
|
pdi = &_ImageInfo;
|
||
|
}
|
||
|
else if (iIndex == 1)
|
||
|
{
|
||
|
if (fHasGlyph)
|
||
|
pdi = &_GlyphInfo;
|
||
|
}
|
||
|
|
||
|
if (! pdi) // not yet set
|
||
|
{
|
||
|
if (fHasGlyph)
|
||
|
iIndex -= 2;
|
||
|
else
|
||
|
iIndex -= 1;
|
||
|
|
||
|
if (iIndex < _iMultiImageCount)
|
||
|
{
|
||
|
pdi = MultiDibPtr(iIndex);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return pdi;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
void CImageFile::DumpProperties(CSimpleFile *pFile, BYTE *pbThemeData, BOOL fFullInfo)
|
||
|
{
|
||
|
if (fFullInfo)
|
||
|
pFile->OutLine(L"Dump of CImageFile at offset=0x%x", (BYTE *)this - pbThemeData);
|
||
|
else
|
||
|
pFile->OutLine(L"Dump of CImageFile");
|
||
|
|
||
|
pFile->OutLine(L" _eBgType=%d", _eBgType);
|
||
|
|
||
|
DIBINFO *pdi = &_ImageInfo;
|
||
|
|
||
|
if (fFullInfo)
|
||
|
{
|
||
|
pFile->OutLine(L" iDibOffset=%d, _iImageCount=%d, _eImageLayout=%d",
|
||
|
pdi->iDibOffset, _iImageCount, _eImageLayout);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pFile->OutLine(L" _iImageCount=%d, _eImageLayout=%d, MinSize=(%d, %d)",
|
||
|
_iImageCount, _eImageLayout, pdi->szMinSize.cx, pdi->szMinSize.cy);
|
||
|
}
|
||
|
|
||
|
pFile->OutLine(L" _iSingleWidth=%d, _iSingleHeight=%d, _fMirrorImage=%d",
|
||
|
pdi->iSingleWidth, pdi->iSingleHeight, _fMirrorImage);
|
||
|
|
||
|
//---- dump multiple image info ----
|
||
|
for (int i=0; i < _iMultiImageCount; i++)
|
||
|
{
|
||
|
DIBINFO *pdi = MultiDibPtr(i);
|
||
|
|
||
|
pFile->OutLine(L" Multi[%d]: sw=%d, sh=%d, diboff=%d, rgnoff=%d",
|
||
|
i, pdi->iSingleWidth, pdi->iSingleHeight,
|
||
|
(pdi->iDibOffset > 0), (pdi->iRgnListOffset > 0));
|
||
|
|
||
|
pFile->OutLine(L" MinDpi=%d, MinSize=(%d, %d)",
|
||
|
pdi->iMinDpi, pdi->szMinSize.cx, pdi->szMinSize.cy);
|
||
|
|
||
|
pFile->OutLine(L" sizetype=%d, bordonly=%d, fTrans=%d, crTrans=0x%x, fAlpha=%d, iAlphaThres=%d",
|
||
|
pdi->eSizingType, pdi->fBorderOnly, pdi->fTransparent, pdi->crTransparent,
|
||
|
pdi->fAlphaChannel, pdi->iAlphaThreshold);
|
||
|
}
|
||
|
|
||
|
pFile->OutLine(L" _eSizingType=%d, _fBorderOnly=%d, _eTrueSizeScalingType=%d",
|
||
|
pdi->eSizingType, pdi->fBorderOnly, _eTrueSizeScalingType);
|
||
|
|
||
|
pFile->OutLine(L" _fTransparent=%d, _crTransparent=0x%08x",
|
||
|
pdi->fTransparent, pdi->crTransparent);
|
||
|
|
||
|
pFile->OutLine(L" _fAlphaChannel=%d, _iAlphaThreshold=%d",
|
||
|
pdi->fAlphaChannel, pdi->iAlphaThreshold);
|
||
|
|
||
|
pFile->OutLine(L" _eHAlign=%d, _eVAlign=%d, _iTrueSizeStretchMark=%d",
|
||
|
_eHAlign, _eVAlign, _iTrueSizeStretchMark);
|
||
|
|
||
|
pFile->OutLine(L" _fUniformSizing=%d, _fIntegralSizing=%d",
|
||
|
_fUniformSizing, _fIntegralSizing);
|
||
|
|
||
|
pFile->OutLine(L" _fBgFill=%d, _crFill=0x%08x",
|
||
|
_fBgFill, _crFill);
|
||
|
|
||
|
pFile->OutLine(L" _fSourceGrow=%d, _fSourceShrink=%d, _szNormalSize=(%d, %d)",
|
||
|
_fSourceGrow, _fSourceShrink, _szNormalSize.cx, _szNormalSize.cy);
|
||
|
|
||
|
pFile->OutLine(L" _SizingMargins=%d, %d, %d, %d",
|
||
|
_SizingMargins.cxLeftWidth, _SizingMargins.cxRightWidth,
|
||
|
_SizingMargins.cyTopHeight, _SizingMargins.cyBottomHeight);
|
||
|
|
||
|
pFile->OutLine(L" _ContentMargins=%d, %d, %d, %d",
|
||
|
_ContentMargins.cxLeftWidth, _ContentMargins.cxRightWidth,
|
||
|
_ContentMargins.cyTopHeight, _ContentMargins.cyBottomHeight);
|
||
|
|
||
|
pFile->OutLine(L" _fFontGlyph=%d, _iGlyphIndex=%d, _crGlyphTextColor=0x%x",
|
||
|
(_eGlyphType==GT_FONTGLYPH), _iGlyphIndex, _crGlyphTextColor);
|
||
|
|
||
|
pFile->OutLine(L" _lfGlyphFont=%s, _fGlyphOnly=%d, _fImageGlyph=%d",
|
||
|
_lfGlyphFont.lfFaceName, _fGlyphOnly, (_eGlyphType==GT_IMAGEGLYPH));
|
||
|
|
||
|
//---- dump glyph properties ----
|
||
|
pdi = &_GlyphInfo;
|
||
|
|
||
|
if (fFullInfo)
|
||
|
{
|
||
|
pFile->OutLine(L" Glyph: iDibOffset=%d, iSingleWidth=%d, iSingleHeight=%d",
|
||
|
pdi->iDibOffset, pdi->iSingleWidth, pdi->iSingleHeight);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pFile->OutLine(L" _iGlyphSingleWidth=%d, _iGlyphSingleHeight=%d",
|
||
|
pdi->iSingleWidth, pdi->iSingleHeight);
|
||
|
}
|
||
|
|
||
|
pFile->OutLine(L" _fGlyphTransparent=%d, _crGlyphTransparent=0x%x, _fGlyphAlpha=%d",
|
||
|
pdi->fTransparent, pdi->crTransparent, pdi->fAlphaChannel);
|
||
|
|
||
|
//pFile->OutLine(L" Glyph: iAlphaThreshold=%d", pdi->iAlphaThreshold);
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::SetImageInfo(DIBINFO *pdi, CRenderObj *pRender, int iPartId, int iStateId)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
if (! pRender->_pbThemeData)
|
||
|
{
|
||
|
hr = E_FAIL;
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
TMBITMAPHEADER *pThemeBitmapHeader = NULL;
|
||
|
|
||
|
pThemeBitmapHeader = reinterpret_cast<TMBITMAPHEADER*>(pRender->_pbThemeData + pdi->iDibOffset);
|
||
|
ASSERT(pThemeBitmapHeader->dwSize == TMBITMAPSIZE);
|
||
|
|
||
|
pdi->fAlphaChannel = pThemeBitmapHeader->fTrueAlpha;
|
||
|
if (pdi->fAlphaChannel)
|
||
|
{
|
||
|
if (FAILED(pRender->GetBool(iPartId, iStateId, TMT_ALPHATHRESHOLD, &pdi->iAlphaThreshold)))
|
||
|
pdi->iAlphaThreshold = 255;
|
||
|
}
|
||
|
|
||
|
int iWidth = - 1;
|
||
|
int iHeight = -1;
|
||
|
|
||
|
if (pThemeBitmapHeader->hBitmap)
|
||
|
{
|
||
|
BITMAP bmInfo;
|
||
|
if (GetObject(pThemeBitmapHeader->hBitmap, sizeof(bmInfo), &bmInfo))
|
||
|
{
|
||
|
iWidth = bmInfo.bmWidth;
|
||
|
iHeight = bmInfo.bmHeight;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hr = E_FAIL;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
BITMAPINFOHEADER* pbmInfo = BITMAPDATA(pThemeBitmapHeader);
|
||
|
if (pbmInfo)
|
||
|
{
|
||
|
iWidth = pbmInfo->biWidth;
|
||
|
iHeight = pbmInfo->biHeight;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hr = E_FAIL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//---- get SingleWidth/SingleHeight of bitmap ----
|
||
|
if ((iWidth != -1) && (iHeight != -1))
|
||
|
{
|
||
|
if (_eImageLayout == IL_HORIZONTAL)
|
||
|
{
|
||
|
pdi->iSingleWidth = iWidth / _iImageCount;
|
||
|
pdi->iSingleHeight = iHeight;
|
||
|
}
|
||
|
else // vertical
|
||
|
{
|
||
|
pdi->iSingleWidth = iWidth;
|
||
|
pdi->iSingleHeight = iHeight / _iImageCount;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
exit:
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
BOOL CImageFile::HasRegionImageFile(DIBINFO *pdi, int *piMaxState)
|
||
|
{
|
||
|
BOOL fGot = FALSE;
|
||
|
|
||
|
if ((pdi->fTransparent) || (pdi->fAlphaChannel))
|
||
|
{
|
||
|
if (pdi->iDibOffset > 0)
|
||
|
{
|
||
|
fGot = TRUE;
|
||
|
*piMaxState = _iImageCount;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return fGot;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
void CImageFile::SetRgnListOffset(DIBINFO *pdi, int iOffset)
|
||
|
{
|
||
|
//---- get offset to the actual jump table ----
|
||
|
pdi->iRgnListOffset = iOffset + ENTRYHDR_SIZE;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::BuildRgnData(DIBINFO *pdi, CRenderObj *pRender, int iStateId, RGNDATA **ppRgnData,
|
||
|
int *piDataLen)
|
||
|
{
|
||
|
RESOURCE HRGN hrgn = NULL;
|
||
|
RESOURCE RGNDATA *pRgnData = NULL;
|
||
|
int iTotalBytes = 0;
|
||
|
int iRectCount;
|
||
|
DWORD len, len2;
|
||
|
HBITMAP hBitmap = NULL;
|
||
|
HRESULT hr = S_OK;
|
||
|
BOOL fStock = FALSE;
|
||
|
|
||
|
if ((! pdi->fAlphaChannel) && (! pdi->fTransparent)) // empty region
|
||
|
goto gotit;
|
||
|
|
||
|
if (pRender->_pbThemeData && pdi->iDibOffset > 0)
|
||
|
{
|
||
|
fStock = ((reinterpret_cast<TMBITMAPHEADER*>(pRender->_pbThemeData + pdi->iDibOffset))->hBitmap != NULL);
|
||
|
}
|
||
|
|
||
|
hr = pRender->GetBitmap(NULL, pdi->iDibOffset, &hBitmap);
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
|
||
|
int iXOffset, iYOffset;
|
||
|
GetOffsets(iStateId, pdi, &iXOffset, &iYOffset);
|
||
|
|
||
|
//---- create a region ----
|
||
|
hr = CreateBitmapRgn(hBitmap, iXOffset, iYOffset, pdi->iSingleWidth, pdi->iSingleHeight,
|
||
|
pdi->fAlphaChannel, pdi->iAlphaThreshold, pdi->crTransparent, 0, &hrgn);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
//---- soft error - author said it was transparent but it wasn't ----
|
||
|
hr = S_OK;
|
||
|
goto gotit;
|
||
|
}
|
||
|
|
||
|
//---- extract region data ----
|
||
|
len = GetRegionData(hrgn, 0, NULL); // get required length
|
||
|
if (! len)
|
||
|
{
|
||
|
hr = MakeErrorLast();
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
iRectCount = len/sizeof(RECT); // # of rects
|
||
|
len += ((sizeof(BYTE)+sizeof(BYTE))*iRectCount); // room for grid id's for each point
|
||
|
|
||
|
iTotalBytes = len + sizeof(RGNDATAHEADER);
|
||
|
pRgnData = (RGNDATA *) new BYTE[iTotalBytes];
|
||
|
len2 = GetRegionData(hrgn, len, pRgnData);
|
||
|
if (! len2)
|
||
|
{
|
||
|
hr = MakeErrorLast();
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
//---- grid-ize the point values within each rect ----
|
||
|
RECT rcImage;
|
||
|
SetRect( &rcImage, 0, 0, pdi->iSingleWidth, pdi->iSingleHeight );
|
||
|
|
||
|
hr = pRender->PrepareRegionDataForScaling(pRgnData, &rcImage, &_SizingMargins);
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
|
||
|
gotit:
|
||
|
*ppRgnData = pRgnData;
|
||
|
*piDataLen = iTotalBytes;
|
||
|
|
||
|
exit:
|
||
|
|
||
|
if (hBitmap && !fStock)
|
||
|
{
|
||
|
pRender->ReturnBitmap(hBitmap);
|
||
|
}
|
||
|
|
||
|
if (hrgn)
|
||
|
DeleteObject(hrgn);
|
||
|
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
if (pRgnData)
|
||
|
delete [] pRgnData;
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
// Helper function for DrawBackgroundDS
|
||
|
void StreamSetSource(BYTE** pvStream, HBITMAP hbmSrc)
|
||
|
{
|
||
|
DS_SETSOURCE* pdsSetSource = (DS_SETSOURCE*)*pvStream;
|
||
|
pdsSetSource->ulCmdID = DS_SETSOURCEID;
|
||
|
pdsSetSource->hbm = HandleToULong(hbmSrc);
|
||
|
*pvStream += sizeof(DS_SETSOURCE);
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
void StreamInit(BYTE** pvStream, HDC hdcDest, HBITMAP hbmSrc, RECTL* prcl)
|
||
|
{
|
||
|
DS_HEADER* pdsHeader = (DS_HEADER*)*pvStream;
|
||
|
pdsHeader->magic = DS_MAGIC;
|
||
|
*pvStream += sizeof(DS_HEADER);
|
||
|
|
||
|
DS_SETTARGET* pdsSetTarget = (DS_SETTARGET*)*pvStream;
|
||
|
pdsSetTarget->ulCmdID = DS_SETTARGETID;
|
||
|
pdsSetTarget->hdc = HandleToULong(hdcDest);
|
||
|
pdsSetTarget->rclDstClip = *prcl;
|
||
|
*pvStream += sizeof(DS_SETTARGET);
|
||
|
|
||
|
StreamSetSource(pvStream, hbmSrc);
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HBITMAP CreateScaledTempBitmap(HDC hdc, HBITMAP hSrcBitmap, int ixSrcOffset, int iySrcOffset,
|
||
|
int iSrcWidth, int iSrcHeight, int iDestWidth, int iDestHeight)
|
||
|
{
|
||
|
HBITMAP hTempBitmap = NULL;
|
||
|
|
||
|
if (hSrcBitmap) // create a DIB from caller's bitmap (Clipper test program)
|
||
|
{
|
||
|
//---- reuse our bitmap ----
|
||
|
hTempBitmap = g_pBitmapCacheScaled->AcquireBitmap(hdc, iDestWidth, iDestHeight);
|
||
|
if (hTempBitmap)
|
||
|
{
|
||
|
HDC hdcDest = CreateCompatibleDC(hdc);
|
||
|
if (hdcDest)
|
||
|
{
|
||
|
HBITMAP hOldDestBitmap = (HBITMAP)SelectObject(hdcDest, hTempBitmap);
|
||
|
|
||
|
HDC hdcSrc = CreateCompatibleDC(hdc);
|
||
|
if (hdcSrc)
|
||
|
{
|
||
|
SetLayout(hdcSrc, 0);
|
||
|
SetLayout(hdcDest, 0);
|
||
|
|
||
|
HBITMAP hOldSrcBitmap = (HBITMAP) SelectObject(hdcSrc, hSrcBitmap);
|
||
|
|
||
|
int iOldSM = SetStretchBltMode(hdcDest, COLORONCOLOR);
|
||
|
|
||
|
//---- stretch src to dest ----
|
||
|
StretchBlt(hdcDest, 0, 0, iDestWidth, iDestHeight,
|
||
|
hdcSrc, ixSrcOffset, iySrcOffset, iSrcWidth, iSrcHeight,
|
||
|
SRCCOPY);
|
||
|
|
||
|
SetStretchBltMode(hdcDest, iOldSM);
|
||
|
|
||
|
SelectObject(hdcSrc, hOldSrcBitmap);
|
||
|
DeleteDC(hdcSrc);
|
||
|
}
|
||
|
|
||
|
SelectObject(hdcDest, hOldDestBitmap);
|
||
|
DeleteDC(hdcDest);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return hTempBitmap;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HBITMAP CreateUnscaledTempBitmap(HDC hdc, HBITMAP hSrcBitmap, int ixSrcOffset, int iySrcOffset,
|
||
|
int iDestWidth, int iDestHeight)
|
||
|
{
|
||
|
HBITMAP hTempBitmap = NULL;
|
||
|
|
||
|
if (hSrcBitmap) // create a DIB from caller's bitmap (Clipper test program)
|
||
|
{
|
||
|
//---- reuse our bitmap ----
|
||
|
hTempBitmap = g_pBitmapCacheUnscaled->AcquireBitmap(hdc, iDestWidth, iDestHeight);
|
||
|
if (hTempBitmap)
|
||
|
{
|
||
|
HDC hdcDest = CreateCompatibleDC(hdc);
|
||
|
if (hdcDest)
|
||
|
{
|
||
|
HBITMAP hOldDestBitmap = (HBITMAP) SelectObject(hdcDest, hTempBitmap);
|
||
|
|
||
|
HDC hdcSrc = CreateCompatibleDC(hdc);
|
||
|
if (hdcSrc)
|
||
|
{
|
||
|
SetLayout(hdcSrc, 0);
|
||
|
SetLayout(hdcDest, 0);
|
||
|
|
||
|
HBITMAP hOldSrcBitmap = (HBITMAP) SelectObject(hdcSrc, hSrcBitmap);
|
||
|
|
||
|
//---- copy src to dest ----
|
||
|
BitBlt(hdcDest, 0, 0, iDestWidth, iDestHeight, hdcSrc, ixSrcOffset, iySrcOffset,
|
||
|
SRCCOPY);
|
||
|
|
||
|
SelectObject(hdcSrc, hOldSrcBitmap);
|
||
|
DeleteDC(hdcSrc);
|
||
|
}
|
||
|
|
||
|
SelectObject(hdcDest, hOldDestBitmap);
|
||
|
DeleteDC(hdcDest);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return hTempBitmap;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::DrawBackgroundDS(DIBINFO *pdi, TMBITMAPHEADER *pThemeBitmapHeader, BOOL fStock,
|
||
|
CRenderObj *pRender, HDC hdc, int iStateId, const RECT *pRect, BOOL fForceStretch,
|
||
|
MARGINS *pmarDest, float xMarginFactor, float yMarginFactor, OPTIONAL const DTBGOPTS *pOptions)
|
||
|
{
|
||
|
//---- bitmaps we may create ----
|
||
|
HBITMAP hBitmapStock = NULL;
|
||
|
HBITMAP hBitmapTempScaled = NULL;
|
||
|
HBITMAP hBitmapTempUnscaled = NULL;
|
||
|
|
||
|
//---- copy of bitmap handle to use ----
|
||
|
HBITMAP hDsBitmap = NULL;
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
int iTempSrcWidth = pdi->iSingleWidth;
|
||
|
int iTempSrcHeight = pdi->iSingleHeight;
|
||
|
|
||
|
int iXOffset, iYOffset;
|
||
|
GetOffsets(iStateId, pdi, &iXOffset, &iYOffset);
|
||
|
|
||
|
if (pThemeBitmapHeader) // get stock bitmap (32 bit format)
|
||
|
{
|
||
|
hr = pRender->GetBitmap(hdc, pdi->iDibOffset, &hBitmapStock);
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
|
||
|
hDsBitmap = hBitmapStock;
|
||
|
}
|
||
|
else // caller passed in bitmap (unknown format)
|
||
|
{
|
||
|
hBitmapTempUnscaled = CreateUnscaledTempBitmap(hdc, pdi->hProcessBitmap, iXOffset, iYOffset,
|
||
|
pdi->iSingleWidth, pdi->iSingleHeight);
|
||
|
if (! hBitmapTempUnscaled)
|
||
|
{
|
||
|
hr = E_FAIL;
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
hDsBitmap = hBitmapTempUnscaled;
|
||
|
|
||
|
//---- src is now just a single image ----
|
||
|
iXOffset = iYOffset = 0;
|
||
|
}
|
||
|
|
||
|
//---- handle scaled margins ----
|
||
|
if ((xMarginFactor != 1) || (yMarginFactor != 1))
|
||
|
{
|
||
|
iTempSrcWidth = int(pdi->iSingleWidth * xMarginFactor);
|
||
|
iTempSrcHeight = int(pdi->iSingleHeight * yMarginFactor);
|
||
|
|
||
|
hBitmapTempScaled = CreateScaledTempBitmap(hdc, hDsBitmap, iXOffset, iYOffset,
|
||
|
pdi->iSingleWidth, pdi->iSingleHeight, iTempSrcWidth, iTempSrcHeight);
|
||
|
if (! hBitmapTempScaled)
|
||
|
{
|
||
|
hr = E_FAIL;
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
hDsBitmap = hBitmapTempScaled;
|
||
|
|
||
|
//---- src is now just a single image ----
|
||
|
iXOffset = iYOffset = 0;
|
||
|
}
|
||
|
|
||
|
if (hDsBitmap)
|
||
|
{
|
||
|
|
||
|
RECTL rclSrc = { iXOffset, iYOffset, iXOffset + iTempSrcWidth, iYOffset + iTempSrcHeight };
|
||
|
RECTL rclDest = { pRect->left, pRect->top, pRect->right, pRect->bottom };
|
||
|
|
||
|
// Flip Dest Rect if someone passed us inverted co-ordinates
|
||
|
if (rclDest.left > rclDest.right)
|
||
|
{
|
||
|
int xTemp = rclDest.left;
|
||
|
rclDest.left = rclDest.right;
|
||
|
rclDest.right = xTemp;
|
||
|
}
|
||
|
if (rclDest.top > rclDest.bottom)
|
||
|
{
|
||
|
int yTemp = rclDest.bottom;
|
||
|
rclDest.bottom = rclDest.top;
|
||
|
rclDest.top = yTemp;
|
||
|
}
|
||
|
|
||
|
DWORD dwOptionFlags = 0;
|
||
|
if (pOptions)
|
||
|
{
|
||
|
dwOptionFlags = pOptions->dwFlags;
|
||
|
}
|
||
|
|
||
|
// Initialize Drawing Stream
|
||
|
BYTE stream[500];
|
||
|
BYTE* pvStreamStart = stream;
|
||
|
BYTE* pvStream = stream;
|
||
|
|
||
|
RECTL rclClip = rclDest;
|
||
|
|
||
|
if (dwOptionFlags & DTBG_CLIPRECT)
|
||
|
{
|
||
|
IntersectRect((LPRECT)&rclClip, (LPRECT)&rclDest, &pOptions->rcClip);
|
||
|
}
|
||
|
|
||
|
StreamInit(&pvStream, hdc, hDsBitmap, &rclClip);
|
||
|
|
||
|
DS_NINEGRID* pvNineGrid = (DS_NINEGRID*)pvStream;
|
||
|
pvNineGrid->ulCmdID = DS_NINEGRIDID;
|
||
|
|
||
|
if ((fForceStretch) || (pdi->eSizingType == ST_STRETCH))
|
||
|
{
|
||
|
pvNineGrid->ngi.flFlags = DSDNG_STRETCH;
|
||
|
}
|
||
|
else if (pdi->eSizingType == ST_TRUESIZE)
|
||
|
{
|
||
|
pvNineGrid->ngi.flFlags = DSDNG_TRUESIZE;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pvNineGrid->ngi.flFlags = DSDNG_TILE;
|
||
|
}
|
||
|
|
||
|
if (pdi->fAlphaChannel)
|
||
|
{
|
||
|
pvNineGrid->ngi.flFlags |= DSDNG_PERPIXELALPHA;
|
||
|
}
|
||
|
else if (pdi->fTransparent)
|
||
|
{
|
||
|
pvNineGrid->ngi.flFlags |= DSDNG_TRANSPARENT;
|
||
|
}
|
||
|
|
||
|
if ((dwOptionFlags & DTBG_MIRRORDC) || (IsMirrored(hdc)))
|
||
|
{
|
||
|
if (_fMirrorImage)
|
||
|
{
|
||
|
pvNineGrid->ngi.flFlags |= DSDNG_MUSTFLIP;
|
||
|
|
||
|
//---- workaround: needed by GdiDrawStream if we don't have a mirrored DC ----
|
||
|
//---- gdi should only look at the DSDNG_MUSTFLIP flag ----
|
||
|
if (! IsMirrored(hdc))
|
||
|
{
|
||
|
int xTemp = rclDest.left;
|
||
|
rclDest.left = rclDest.right;
|
||
|
rclDest.right = xTemp;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pvNineGrid->rclDst = rclDest;
|
||
|
pvNineGrid->rclSrc = rclSrc;
|
||
|
|
||
|
if (pdi->eSizingType == ST_TRUESIZE)
|
||
|
{
|
||
|
pvNineGrid->ngi.ulLeftWidth = 0;
|
||
|
pvNineGrid->ngi.ulRightWidth = 0;
|
||
|
pvNineGrid->ngi.ulTopHeight = 0;
|
||
|
pvNineGrid->ngi.ulBottomHeight = 0;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//---- copy scaled Src margins ----
|
||
|
pvNineGrid->ngi.ulLeftWidth = pmarDest->cxLeftWidth;
|
||
|
pvNineGrid->ngi.ulRightWidth = pmarDest->cxRightWidth;
|
||
|
pvNineGrid->ngi.ulTopHeight = pmarDest->cyTopHeight;
|
||
|
pvNineGrid->ngi.ulBottomHeight = pmarDest->cyBottomHeight;
|
||
|
}
|
||
|
|
||
|
pvNineGrid->ngi.crTransparent = pdi->crTransparent;
|
||
|
|
||
|
pvStream += sizeof(DS_NINEGRID);
|
||
|
|
||
|
GdiDrawStream(hdc, (int)(pvStream - pvStreamStart), (char*) pvStreamStart);
|
||
|
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hr = E_FAIL;
|
||
|
}
|
||
|
|
||
|
exit:
|
||
|
//---- clean up temp bitmaps ----
|
||
|
if (hBitmapTempScaled)
|
||
|
{
|
||
|
g_pBitmapCacheScaled->ReturnBitmap();
|
||
|
}
|
||
|
|
||
|
if (hBitmapTempUnscaled)
|
||
|
{
|
||
|
g_pBitmapCacheUnscaled->ReturnBitmap();
|
||
|
}
|
||
|
|
||
|
if ((hBitmapStock) && (! fStock)) // not really stock (was "create on demand")
|
||
|
{
|
||
|
pRender->ReturnBitmap(hBitmapStock);
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::DrawBackgroundDNG(DIBINFO *pdi, TMBITMAPHEADER *pThemeBitmapHeader, BOOL fStock, CRenderObj *pRender,
|
||
|
HDC hdc, int iStateId, const RECT *pRect, BOOL fForceStretch,
|
||
|
MARGINS *pmarDest, OPTIONAL const DTBGOPTS *pOptions)
|
||
|
{
|
||
|
HRESULT hr = E_FAIL;
|
||
|
|
||
|
//---- options ----
|
||
|
DWORD dwOptionFlags = 0;
|
||
|
const RECT *pClipRect = NULL;
|
||
|
|
||
|
if (pOptions)
|
||
|
{
|
||
|
dwOptionFlags = pOptions->dwFlags;
|
||
|
|
||
|
if (dwOptionFlags & DTBG_CLIPRECT)
|
||
|
pClipRect = &pOptions->rcClip;
|
||
|
}
|
||
|
|
||
|
int iXOffset, iYOffset;
|
||
|
GetOffsets(iStateId, pdi, &iXOffset, &iYOffset);
|
||
|
|
||
|
DWORD dwFlags = 0;
|
||
|
|
||
|
if (! (dwOptionFlags & DTBG_DRAWSOLID))
|
||
|
{
|
||
|
if (pdi->fTransparent)
|
||
|
{
|
||
|
dwFlags = NGI_TRANS;
|
||
|
}
|
||
|
if (pdi->fAlphaChannel)
|
||
|
{
|
||
|
dwFlags = NGI_ALPHA;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ULONG* pvSrcBits = NULL;
|
||
|
int iWidth = 0;
|
||
|
int iHeight = 0;
|
||
|
BOOL fFreeBits = FALSE;
|
||
|
|
||
|
if (pThemeBitmapHeader)
|
||
|
{
|
||
|
BITMAPINFOHEADER* pHeader = BITMAPDATA(pThemeBitmapHeader);
|
||
|
if (pHeader && pHeader->biBitCount == 32)
|
||
|
{
|
||
|
pvSrcBits = (ULONG*)DIBDATA(pHeader);
|
||
|
}
|
||
|
iWidth = pHeader->biWidth;
|
||
|
iHeight = pHeader->biHeight;
|
||
|
}
|
||
|
else if (pdi->hProcessBitmap)
|
||
|
{
|
||
|
BITMAP bm;
|
||
|
if (GetObject(pdi->hProcessBitmap, sizeof(bm), &bm))
|
||
|
{
|
||
|
BITMAPINFO bmInfo = {{sizeof(BITMAPINFOHEADER), bm.bmWidth, bm.bmHeight, 1, 32, BI_RGB, 0, 0, 0, 0, 0}, NULL};
|
||
|
pvSrcBits = new DWORD[bm.bmWidth * bm.bmHeight];
|
||
|
if (pvSrcBits)
|
||
|
{
|
||
|
if (GetDIBits(hdc, pdi->hProcessBitmap, 0, bm.bmHeight, pvSrcBits, &bmInfo, DIB_RGB_COLORS))
|
||
|
{
|
||
|
fFreeBits = TRUE;
|
||
|
iWidth = bm.bmWidth;
|
||
|
iHeight = bm.bmHeight;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
delete[] pvSrcBits;
|
||
|
pvSrcBits = NULL;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (pvSrcBits)
|
||
|
{
|
||
|
NGIMAGE ngi;
|
||
|
ngi.hbm = NULL;
|
||
|
ngi.iWidth = pdi->iSingleWidth;
|
||
|
ngi.iHeight = pdi->iSingleHeight;
|
||
|
ngi.margin = _SizingMargins;
|
||
|
//ngi.marDest = *pmarDest;
|
||
|
ngi.dwFlags = dwFlags;
|
||
|
ngi.crTrans = pdi->crTransparent;
|
||
|
|
||
|
if (fForceStretch)
|
||
|
ngi.eSize = ST_STRETCH;
|
||
|
else
|
||
|
ngi.eSize = pdi->eSizingType;
|
||
|
|
||
|
DWORD dwDNGFlags = 0;
|
||
|
|
||
|
if ((dwOptionFlags & DTBG_MIRRORDC) || (IsMirrored(hdc)))
|
||
|
{
|
||
|
if (_fMirrorImage)
|
||
|
{
|
||
|
dwDNGFlags |= DNG_MUSTFLIP;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ngi.iBufWidth = iWidth;
|
||
|
int iDibOffset = (iHeight - iYOffset) - pdi->iSingleHeight;
|
||
|
ngi.pvBits = pvSrcBits + (iDibOffset * ngi.iBufWidth) + iXOffset;
|
||
|
|
||
|
RECT rcDest = *pRect;
|
||
|
|
||
|
if (pdi->fBorderOnly &&
|
||
|
(RECTHEIGHT(&rcDest) > ngi.margin.cyTopHeight + ngi.margin.cyBottomHeight) &&
|
||
|
(RECTWIDTH(&rcDest) > ngi.margin.cxLeftWidth + ngi.margin.cxRightWidth))
|
||
|
{
|
||
|
RECT rcTop = rcDest;
|
||
|
RECT rcLeft = rcDest;
|
||
|
RECT rcBottom = rcDest;
|
||
|
RECT rcRight = rcDest;
|
||
|
|
||
|
rcLeft.top = rcRight.top = rcTop.bottom = rcTop.top + ngi.margin.cyTopHeight;
|
||
|
rcLeft.bottom = rcRight.bottom = rcBottom.top = rcBottom.bottom - ngi.margin.cyBottomHeight;
|
||
|
|
||
|
rcLeft.right = rcLeft.left + ngi.margin.cxLeftWidth;
|
||
|
rcRight.left = rcRight.right - ngi.margin.cxRightWidth;
|
||
|
|
||
|
if (pClipRect)
|
||
|
{
|
||
|
IntersectRect(&rcLeft, pClipRect, &rcLeft);
|
||
|
IntersectRect(&rcTop, pClipRect, &rcTop);
|
||
|
IntersectRect(&rcRight, pClipRect, &rcRight);
|
||
|
IntersectRect(&rcBottom, pClipRect, &rcBottom);
|
||
|
}
|
||
|
|
||
|
hr = DrawNineGrid2(hdc, &ngi, &rcDest, &rcTop, dwDNGFlags);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = DrawNineGrid2(hdc, &ngi, &rcDest, &rcLeft, dwDNGFlags);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = DrawNineGrid2(hdc, &ngi, &rcDest, &rcRight, dwDNGFlags);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = DrawNineGrid2(hdc, &ngi, &rcDest, &rcBottom, dwDNGFlags);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hr = DrawNineGrid2(hdc, &ngi, &rcDest, pClipRect, dwDNGFlags);
|
||
|
}
|
||
|
|
||
|
if (fFreeBits)
|
||
|
{
|
||
|
delete[] pvSrcBits;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
Log(LOG_ALWAYS, L"DrawBackground FAILED: class=%s, hr=0x%x",
|
||
|
SHARECLASS(pRender), hr);
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
DIBINFO *CImageFile::SelectCorrectImageFile(CRenderObj *pRender, HDC hdc, OPTIONAL const RECT *prc,
|
||
|
BOOL fForGlyph, OPTIONAL TRUESTRETCHINFO *ptsInfo)
|
||
|
{
|
||
|
DIBINFO *pdiDefault = (fForGlyph) ? &_GlyphInfo : &_ImageInfo;
|
||
|
DIBINFO *pdi = NULL;
|
||
|
BOOL fForceRectSizing = FALSE;
|
||
|
int iWidth = 1;
|
||
|
int iHeight = 1;
|
||
|
|
||
|
//---- do we need a screen dc? ----
|
||
|
BOOL fReleaseDC = FALSE;
|
||
|
if (! hdc)
|
||
|
{
|
||
|
hdc = GetWindowDC(NULL);
|
||
|
if (hdc)
|
||
|
fReleaseDC = TRUE;
|
||
|
}
|
||
|
|
||
|
if (prc)
|
||
|
{
|
||
|
iWidth = WIDTH(*prc);
|
||
|
iHeight = HEIGHT(*prc);
|
||
|
}
|
||
|
|
||
|
//---- see if our clients wants to force a TRUESIZE to stretch ----
|
||
|
if ((fForGlyph) || (_ImageInfo.eSizingType == ST_TRUESIZE))
|
||
|
{
|
||
|
if ((pRender) && (pRender->_dwOtdFlags & OTD_FORCE_RECT_SIZING))
|
||
|
{
|
||
|
fForceRectSizing = TRUE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//---- find correct file by DPI or Size ----
|
||
|
if ((fForGlyph) || (_eGlyphType != GT_IMAGEGLYPH)) // match multifiles to reg or glyph
|
||
|
{
|
||
|
BOOL fSizing = FALSE;
|
||
|
BOOL fDpi = FALSE;
|
||
|
|
||
|
if ((fForceRectSizing) || (_eImageSelectType == IST_SIZE) || (_fSourceGrow))
|
||
|
{
|
||
|
if (prc)
|
||
|
fSizing = TRUE;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
fDpi = (_eImageSelectType == IST_DPI);
|
||
|
}
|
||
|
|
||
|
if (fDpi) // DPI-based image selection
|
||
|
{
|
||
|
int iMinDestDpi = __min(GetDeviceCaps(hdc, LOGPIXELSX), GetDeviceCaps(hdc, LOGPIXELSY));
|
||
|
|
||
|
//---- search from largest to smallest ----
|
||
|
for (int i=_iMultiImageCount-1; i >= 0; i--)
|
||
|
{
|
||
|
if (MultiDibPtr(i)->iMinDpi <= iMinDestDpi) // got him
|
||
|
{
|
||
|
pdi = MultiDibPtr(i);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else if (fSizing) // Sizing-base image selection
|
||
|
{
|
||
|
if (_iMultiImageCount)
|
||
|
{
|
||
|
//---- search from largest to smallest ----
|
||
|
for (int i=_iMultiImageCount-1; i >= 0; i--)
|
||
|
{
|
||
|
DIBINFO *pdii = MultiDibPtr(i);
|
||
|
if ((pdii->szMinSize.cx <= iWidth) && (pdii->szMinSize.cy <= iHeight))
|
||
|
{
|
||
|
pdi = pdii;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (! pdi) // no match found
|
||
|
{
|
||
|
pdi = pdiDefault;
|
||
|
}
|
||
|
|
||
|
//---- determine drawing size of selected file (MultiImage or regular) ----
|
||
|
if (ptsInfo)
|
||
|
{
|
||
|
ptsInfo->fForceStretch = FALSE;
|
||
|
ptsInfo->fFullStretch = FALSE;
|
||
|
|
||
|
ptsInfo->szDrawSize.cx = 0;
|
||
|
ptsInfo->szDrawSize.cy = 0;
|
||
|
|
||
|
//---- this sizing only applies to TRUESIZE images ----
|
||
|
if ((pdi->eSizingType == ST_TRUESIZE) && (_eTrueSizeScalingType != TSST_NONE))
|
||
|
{
|
||
|
if (prc)
|
||
|
{
|
||
|
//---- force an exact stretch match? ----
|
||
|
if ((fForceRectSizing) || (pdi->iSingleWidth > iWidth) || (pdi->iSingleHeight > iHeight))
|
||
|
{
|
||
|
//---- either Forced to stretch by caller or image is too big for dest RECT ----
|
||
|
ptsInfo->fForceStretch = TRUE;
|
||
|
ptsInfo->fFullStretch = TRUE;
|
||
|
|
||
|
ptsInfo->szDrawSize.cx = iWidth;
|
||
|
ptsInfo->szDrawSize.cy = iHeight;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (! ptsInfo->fForceStretch) // keep trying..
|
||
|
{
|
||
|
//---- see if image is too small for dest RECT ---
|
||
|
SIZE szTargetSize = {0, 0};
|
||
|
|
||
|
if (_eTrueSizeScalingType == TSST_DPI)
|
||
|
{
|
||
|
int ixDpiDc = GetDeviceCaps(hdc, LOGPIXELSX);
|
||
|
int iyDpiDc = GetDeviceCaps(hdc, LOGPIXELSY);
|
||
|
|
||
|
szTargetSize.cx = MulDiv(pdi->iSingleWidth, ixDpiDc, pdi->iMinDpi);
|
||
|
szTargetSize.cy = MulDiv(pdi->iSingleHeight, iyDpiDc, pdi->iMinDpi);
|
||
|
}
|
||
|
else if ((_eTrueSizeScalingType == TSST_SIZE) && (prc))
|
||
|
{
|
||
|
szTargetSize.cx = MulDiv(pdi->iSingleWidth, iWidth, pdi->szMinSize.cx);
|
||
|
szTargetSize.cy = MulDiv(pdi->iSingleHeight, iHeight, pdi->szMinSize.cy);
|
||
|
}
|
||
|
|
||
|
if (szTargetSize.cx) // was set
|
||
|
{
|
||
|
//---- clip targetsize against dest rect ----
|
||
|
if (prc)
|
||
|
{
|
||
|
szTargetSize.cx = __min(szTargetSize.cx, iWidth);
|
||
|
szTargetSize.cy = __min(szTargetSize.cy, iHeight);
|
||
|
}
|
||
|
|
||
|
int ixPercentExceed = 100*(szTargetSize.cx - pdi->iSingleWidth)/pdi->iSingleWidth;
|
||
|
int iyPercentExceed = 100*(szTargetSize.cy - pdi->iSingleHeight)/pdi->iSingleHeight;
|
||
|
|
||
|
if ((ixPercentExceed >= _iTrueSizeStretchMark) && (iyPercentExceed >= _iTrueSizeStretchMark))
|
||
|
{
|
||
|
ptsInfo->fForceStretch = TRUE;
|
||
|
ptsInfo->szDrawSize = szTargetSize;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (! pdi)
|
||
|
{
|
||
|
pdi = pdiDefault;
|
||
|
}
|
||
|
|
||
|
if (fReleaseDC)
|
||
|
{
|
||
|
ReleaseDC(NULL, hdc);
|
||
|
}
|
||
|
|
||
|
return pdi;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
void CImageFile::GetDrawnImageSize(DIBINFO *pdi, const RECT *pRect, TRUESTRETCHINFO *ptsInfo,
|
||
|
SIZE *pszDraw)
|
||
|
{
|
||
|
//---- szDraw is the size image will be drawn to ----
|
||
|
if (pdi->eSizingType == ST_TRUESIZE)
|
||
|
{
|
||
|
if (ptsInfo->fForceStretch)
|
||
|
{
|
||
|
*pszDraw = ptsInfo->szDrawSize;
|
||
|
|
||
|
//---- integral sizing (stretched truesize only) ----
|
||
|
if ((_fIntegralSizing) && (! ptsInfo->fFullStretch))
|
||
|
{
|
||
|
float flFactX = float(ptsInfo->szDrawSize.cx)/pdi->iSingleWidth;
|
||
|
float flFactY = float(ptsInfo->szDrawSize.cy)/pdi->iSingleHeight;
|
||
|
|
||
|
//---- cast float's to int to get lowest int (vs. rounded) ----
|
||
|
pszDraw->cx = pdi->iSingleWidth * int(flFactX);
|
||
|
pszDraw->cy = pdi->iSingleHeight * int(flFactY);
|
||
|
}
|
||
|
}
|
||
|
else // use original image size
|
||
|
{
|
||
|
pszDraw->cx = pdi->iSingleWidth;
|
||
|
pszDraw->cy = pdi->iSingleHeight;
|
||
|
}
|
||
|
|
||
|
//---- Uniform Sizing ----
|
||
|
if (_fUniformSizing)
|
||
|
{
|
||
|
int iSingleWidth = pdi->iSingleWidth;
|
||
|
int iSingleHeight = pdi->iSingleHeight;
|
||
|
|
||
|
double fact1 = double(pszDraw->cx)/iSingleWidth;
|
||
|
double fact2 = double(pszDraw->cy)/iSingleHeight;
|
||
|
|
||
|
//---- select the smallest factor to use for both dims ----
|
||
|
if (fact1 < fact2)
|
||
|
{
|
||
|
pszDraw->cy = int(iSingleHeight*fact1);
|
||
|
}
|
||
|
else if (fact1 > fact2)
|
||
|
{
|
||
|
pszDraw->cx = int(iSingleWidth*fact2);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else // ST_TILE or ST_STRETCH: pRect determines size
|
||
|
{
|
||
|
if (pRect)
|
||
|
{
|
||
|
pszDraw->cx = WIDTH(*pRect);
|
||
|
pszDraw->cy = HEIGHT(*pRect);
|
||
|
}
|
||
|
else // void function so just return 0
|
||
|
{
|
||
|
pszDraw->cx = 0;
|
||
|
pszDraw->cy = 0;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::DrawImageInfo(DIBINFO *pdi, CRenderObj *pRender, HDC hdc, int iStateId,
|
||
|
const RECT *pRect, const DTBGOPTS *pOptions, TRUESTRETCHINFO *ptsInfo)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
TMBITMAPHEADER *pThemeBitmapHeader = NULL;
|
||
|
BOOL fStock = FALSE;
|
||
|
RECT rcLocal;
|
||
|
DWORD dwFlags;
|
||
|
SIZE szDraw;
|
||
|
BOOL fRectFilled;
|
||
|
MARGINS marDest;
|
||
|
float xFactor;
|
||
|
float yFactor;
|
||
|
|
||
|
if (pOptions)
|
||
|
dwFlags = pOptions->dwFlags;
|
||
|
else
|
||
|
dwFlags = 0;
|
||
|
|
||
|
//---- validate bitmap header ----
|
||
|
if (! pdi->hProcessBitmap) // regular, section based DIB
|
||
|
{
|
||
|
if (pRender->_pbThemeData && pdi->iDibOffset > 0)
|
||
|
{
|
||
|
pThemeBitmapHeader = reinterpret_cast<TMBITMAPHEADER*>(pRender->_pbThemeData + pdi->iDibOffset);
|
||
|
ASSERT(pThemeBitmapHeader->dwSize == TMBITMAPSIZE);
|
||
|
fStock = (pThemeBitmapHeader->hBitmap != NULL);
|
||
|
}
|
||
|
|
||
|
if (!pRender->IsReady())
|
||
|
{
|
||
|
// Stock bitmaps in section are cleaning, don't try to paint with an old HBITMAP
|
||
|
hr = E_FAIL;
|
||
|
//Log(LOG_TMBITMAP, L"Obsolete theme section: class=%s", SHARECLASS(pRender));
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
if (pThemeBitmapHeader == NULL)
|
||
|
{
|
||
|
hr = E_FAIL;
|
||
|
Log(LOG_ALWAYS, L"No TMBITMAPHEADER: class=%s, hr=0x%x", SHARECLASS(pRender), hr);
|
||
|
goto exit;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//----- set szDraw to size image will be drawn at ----
|
||
|
GetDrawnImageSize(pdi, pRect, ptsInfo, &szDraw);
|
||
|
|
||
|
rcLocal = *pRect;
|
||
|
fRectFilled = TRUE;
|
||
|
|
||
|
//---- horizontal alignment ----
|
||
|
if (WIDTH(rcLocal) > szDraw.cx)
|
||
|
{
|
||
|
fRectFilled = FALSE;
|
||
|
|
||
|
if (_eHAlign == HA_LEFT)
|
||
|
{
|
||
|
rcLocal.right = rcLocal.left + szDraw.cx;
|
||
|
}
|
||
|
else if (_eHAlign == HA_CENTER)
|
||
|
{
|
||
|
rcLocal.left += (WIDTH(rcLocal) - szDraw.cx) / 2;
|
||
|
rcLocal.right = rcLocal.left + szDraw.cx;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
rcLocal.left = rcLocal.right - szDraw.cx;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//---- vertical alignment ----
|
||
|
if (HEIGHT(rcLocal) > szDraw.cy)
|
||
|
{
|
||
|
fRectFilled = FALSE;
|
||
|
|
||
|
if (_eVAlign == VA_TOP)
|
||
|
{
|
||
|
rcLocal.bottom = rcLocal.top + szDraw.cy;
|
||
|
}
|
||
|
else if (_eVAlign == VA_CENTER)
|
||
|
{
|
||
|
rcLocal.top += (HEIGHT(rcLocal) - szDraw.cy) / 2;
|
||
|
rcLocal.bottom = rcLocal.top + szDraw.cy;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
rcLocal.top = rcLocal.bottom - szDraw.cy;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//---- BgFill ----
|
||
|
if ((! fRectFilled) && (! pdi->fBorderOnly) && (_fBgFill))
|
||
|
{
|
||
|
if (! (dwFlags & DTBG_OMITCONTENT))
|
||
|
{
|
||
|
//---- paint bg ----
|
||
|
HBRUSH hbr = CreateSolidBrush(_crFill);
|
||
|
if (! hbr)
|
||
|
{
|
||
|
hr = GetLastError();
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
FillRect(hdc, pRect, hbr);
|
||
|
DeleteObject(hbr);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//---- calculate source/margin scaling factors ----
|
||
|
marDest = _SizingMargins;
|
||
|
|
||
|
if (pdi->eSizingType == ST_TRUESIZE) // sizing margins ignored - no scaling needed
|
||
|
{
|
||
|
xFactor = 1;
|
||
|
yFactor = 1;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//---- scale destination sizing margins ----
|
||
|
ScaleMargins(&marDest, hdc, pRender, pdi, &szDraw, &xFactor, &yFactor);
|
||
|
}
|
||
|
|
||
|
#if 1 // keep this in sync with #if in parser.cpp
|
||
|
|
||
|
//---- new GDI drawing ----
|
||
|
hr = DrawBackgroundDS(pdi, pThemeBitmapHeader, fStock, pRender, hdc, iStateId, &rcLocal,
|
||
|
ptsInfo->fForceStretch, &marDest, xFactor, yFactor, pOptions);
|
||
|
|
||
|
#else
|
||
|
|
||
|
//---- old drawing (keep around until DS is burned in) ----
|
||
|
hr = DrawBackgroundDNG(pdi, pThemeBitmapHeader, fStock, pRender, hdc, iStateId, &rcLocal,
|
||
|
ptsInfo->fForceStretch, &marDest, pOptions);
|
||
|
|
||
|
#endif
|
||
|
|
||
|
exit:
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::DrawBackground(CRenderObj *pRender, HDC hdc, int iStateId,
|
||
|
const RECT *pRect, OPTIONAL const DTBGOPTS *pOptions)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
TRUESTRETCHINFO tsInfo;
|
||
|
|
||
|
if (! _fGlyphOnly)
|
||
|
{
|
||
|
DIBINFO *pdi = SelectCorrectImageFile(pRender, hdc, pRect, FALSE, &tsInfo);
|
||
|
|
||
|
//---- draw normal image ----
|
||
|
hr = DrawImageInfo(pdi, pRender, hdc, iStateId, pRect, pOptions, &tsInfo);
|
||
|
}
|
||
|
|
||
|
//---- draw glyph, if needed ----
|
||
|
if (SUCCEEDED(hr) && (_eGlyphType != GT_NONE))
|
||
|
{
|
||
|
RECT rc;
|
||
|
hr = GetBackgroundContentRect(pRender, hdc, pRect, &rc);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (_eGlyphType == GT_FONTGLYPH)
|
||
|
{
|
||
|
hr = DrawFontGlyph(pRender, hdc, &rc, pOptions);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
DIBINFO *pdi = SelectCorrectImageFile(pRender, hdc, &rc, TRUE, &tsInfo);
|
||
|
|
||
|
//---- draw glyph image ----
|
||
|
hr = DrawImageInfo(pdi, pRender, hdc, iStateId, &rc, pOptions, &tsInfo);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::DrawFontGlyph(CRenderObj *pRender, HDC hdc, RECT *prc,
|
||
|
OPTIONAL const DTBGOPTS *pOptions)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
DWORD dwFlags = DT_SINGLELINE;
|
||
|
HFONT hFont = NULL;
|
||
|
HFONT hOldFont = NULL;
|
||
|
COLORREF crOld = 0;
|
||
|
CSaveClipRegion scrOrig;
|
||
|
int iOldMode = 0;
|
||
|
WCHAR szText[2] = { (WCHAR)_iGlyphIndex, 0 };
|
||
|
|
||
|
//---- options ----
|
||
|
DWORD dwOptionFlags = 0;
|
||
|
const RECT *pClipRect = NULL;
|
||
|
|
||
|
if (pOptions)
|
||
|
{
|
||
|
dwOptionFlags = pOptions->dwFlags;
|
||
|
|
||
|
if (dwOptionFlags & DTBG_CLIPRECT)
|
||
|
pClipRect = &pOptions->rcClip;
|
||
|
}
|
||
|
|
||
|
//---- create the font ----
|
||
|
hr = pRender->GetScaledFontHandle(hdc, &_lfGlyphFont, &hFont);
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
|
||
|
//---- make it active ----
|
||
|
hOldFont = (HFONT)SelectObject(hdc, hFont);
|
||
|
if (! hOldFont)
|
||
|
{
|
||
|
hr = MakeErrorLast();
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
//---- set the text color ----
|
||
|
crOld = SetTextColor(hdc, _crGlyphTextColor);
|
||
|
|
||
|
//---- draw text with transparent background ----
|
||
|
iOldMode = SetBkMode(hdc, TRANSPARENT);
|
||
|
|
||
|
//---- set the HORZ alignment flags ----
|
||
|
if (_eHAlign == HA_LEFT)
|
||
|
dwFlags |= DT_LEFT;
|
||
|
else if (_eHAlign == HA_CENTER)
|
||
|
dwFlags |= DT_CENTER;
|
||
|
else
|
||
|
dwFlags |= DT_RIGHT;
|
||
|
|
||
|
//---- set the VERT alignment flags ----
|
||
|
if (_eVAlign == VA_TOP)
|
||
|
dwFlags |= DT_TOP;
|
||
|
else if (_eVAlign == VA_CENTER)
|
||
|
dwFlags |= DT_VCENTER;
|
||
|
else
|
||
|
dwFlags |= DT_BOTTOM;
|
||
|
|
||
|
//---- add clipping ----
|
||
|
if (pClipRect)
|
||
|
{
|
||
|
//---- get previous clipping region (for restoring at end) ----
|
||
|
hr = scrOrig.Save(hdc);
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
|
||
|
//---- add "pClipRect" to the GDI clipping region ----
|
||
|
int iRetVal = IntersectClipRect(hdc, pClipRect->left, pClipRect->top,
|
||
|
pClipRect->right, pClipRect->bottom);
|
||
|
if (iRetVal == ERROR)
|
||
|
{
|
||
|
hr = MakeErrorLast();
|
||
|
goto exit;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//---- draw the char ----
|
||
|
if (! DrawTextEx(hdc, szText, 1, prc, dwFlags, NULL))
|
||
|
{
|
||
|
hr = MakeErrorLast();
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
exit:
|
||
|
|
||
|
if (pClipRect)
|
||
|
scrOrig.Restore(hdc);
|
||
|
|
||
|
//---- reset the background mode ----
|
||
|
if (iOldMode != TRANSPARENT)
|
||
|
SetBkMode(hdc, iOldMode);
|
||
|
|
||
|
//---- restore text color ----
|
||
|
if (crOld != _crGlyphTextColor)
|
||
|
SetTextColor(hdc, crOld);
|
||
|
|
||
|
//---- restore font ----
|
||
|
if (hOldFont)
|
||
|
SelectObject(hdc, hOldFont);
|
||
|
|
||
|
if (hFont)
|
||
|
pRender->ReturnFontHandle(hFont);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
BOOL CImageFile::IsBackgroundPartiallyTransparent(int iStateId)
|
||
|
{
|
||
|
DIBINFO *pdi = &_ImageInfo; // primary image determines transparency
|
||
|
|
||
|
return ((pdi->fAlphaChannel) || (pdi->fTransparent));
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::HitTestBackground(CRenderObj *pRender, OPTIONAL HDC hdc, int iStateId,
|
||
|
DWORD dwHTFlags, const RECT *pRect, HRGN hrgn, POINT ptTest, OUT WORD *pwHitCode)
|
||
|
{
|
||
|
*pwHitCode = HTNOWHERE;
|
||
|
|
||
|
if (! PtInRect(pRect, ptTest))
|
||
|
return S_OK; // nowhere
|
||
|
|
||
|
//---- background might have transparent parts - get its region ----
|
||
|
HRESULT hr = S_OK;
|
||
|
HRGN hrgnBk = NULL;
|
||
|
|
||
|
if( !hrgn && IsBackgroundPartiallyTransparent(iStateId) )
|
||
|
{
|
||
|
hr = GetBackgroundRegion(pRender, hdc, iStateId, pRect, &hrgnBk);
|
||
|
if( SUCCEEDED(hr) )
|
||
|
hrgn = hrgnBk;
|
||
|
}
|
||
|
|
||
|
MARGINS margins;
|
||
|
if( TESTFLAG(dwHTFlags, HTTB_SYSTEMSIZINGMARGINS) &&
|
||
|
TESTFLAG(dwHTFlags, HTTB_RESIZINGBORDER) &&
|
||
|
!TESTFLAG(dwHTFlags, HTTB_SIZINGTEMPLATE) )
|
||
|
{
|
||
|
ZeroMemory( &margins, sizeof(margins) );
|
||
|
|
||
|
int cxBorder = ClassicGetSystemMetrics( SM_CXSIZEFRAME );
|
||
|
int cyBorder = ClassicGetSystemMetrics( SM_CXSIZEFRAME );
|
||
|
|
||
|
if( TESTFLAG(dwHTFlags, HTTB_RESIZINGBORDER_LEFT) )
|
||
|
margins.cxLeftWidth = cxBorder;
|
||
|
|
||
|
if( TESTFLAG(dwHTFlags, HTTB_RESIZINGBORDER_RIGHT) )
|
||
|
margins.cxRightWidth = cxBorder;
|
||
|
|
||
|
if( TESTFLAG(dwHTFlags, HTTB_RESIZINGBORDER_TOP) )
|
||
|
margins.cyTopHeight = cyBorder;
|
||
|
|
||
|
if( TESTFLAG(dwHTFlags, HTTB_RESIZINGBORDER_BOTTOM) )
|
||
|
margins.cyBottomHeight = cyBorder;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hr = GetScaledContentMargins(pRender, hdc, pRect, &margins);
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
if( hrgn )
|
||
|
{
|
||
|
// 122013 - we originally delegated to a sophisticated but broken
|
||
|
// resizing area region hit test algorithm for regioned windows,
|
||
|
// but for whistler we'll just do the bounding
|
||
|
// rectangle thang instead.
|
||
|
//*pwHitCode = HitTestRgn( dwHTFlags, pRect, hrgn, margins, ptTest );
|
||
|
|
||
|
RECT rcRgn;
|
||
|
if( GetRgnBox( hrgn, &rcRgn ) )
|
||
|
{
|
||
|
if( TESTFLAG(dwHTFlags, HTTB_SIZINGTEMPLATE) )
|
||
|
{
|
||
|
*pwHitCode = HitTestTemplate( dwHTFlags, &rcRgn, hrgn, margins, ptTest );
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
*pwHitCode = HitTestRect( dwHTFlags, &rcRgn, margins, ptTest );
|
||
|
}
|
||
|
}
|
||
|
SAFE_DELETE_GDIOBJ(hrgnBk);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
*pwHitCode = HitTestRect( dwHTFlags, pRect, margins, ptTest );
|
||
|
}
|
||
|
|
||
|
exit:
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::GetBackgroundRegion(CRenderObj *pRender, OPTIONAL HDC hdc, int iStateId,
|
||
|
const RECT *pRect, HRGN *pRegion)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
RGNDATA *pRgnData;
|
||
|
CMemoryDC hdcMemory;
|
||
|
HRGN hrgn;
|
||
|
int iRgnDataOffset = 0;
|
||
|
MIXEDPTRS u;
|
||
|
|
||
|
DIBINFO *pdi = SelectCorrectImageFile(pRender, hdc, pRect, FALSE);
|
||
|
|
||
|
//---- get rgndata offset ----
|
||
|
if ((pdi->iRgnListOffset) && (pRender->_pbThemeData))
|
||
|
{
|
||
|
u.pb = pRender->_pbThemeData + pdi->iRgnListOffset;
|
||
|
int iMaxState = (*u.pb++) - 1;
|
||
|
if (iStateId > iMaxState)
|
||
|
iStateId = 0;
|
||
|
iRgnDataOffset = u.pi[iStateId];
|
||
|
}
|
||
|
|
||
|
//---- see if it even has a transparent part ----
|
||
|
if (iRgnDataOffset)
|
||
|
{
|
||
|
//---- stretch those puppies & create a new region ----
|
||
|
pRgnData = (RGNDATA *)(pRender->_pbThemeData + iRgnDataOffset
|
||
|
+ sizeof(RGNDATAHDR) + ENTRYHDR_SIZE);
|
||
|
|
||
|
SIZE szSrcImage = {pdi->iSingleWidth, pdi->iSingleHeight};
|
||
|
|
||
|
hr = _ScaleRectsAndCreateRegion(pRgnData, pRect, &_SizingMargins, &szSrcImage, &hrgn);
|
||
|
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//---- return the bounding rect as the region ----
|
||
|
hrgn = CreateRectRgn(pRect->left, pRect->top,
|
||
|
pRect->right, pRect->bottom);
|
||
|
|
||
|
if (! hrgn)
|
||
|
{
|
||
|
hr = MakeErrorLast();
|
||
|
goto exit;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
*pRegion = hrgn;
|
||
|
|
||
|
exit:
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::GetBackgroundContentRect(CRenderObj *pRender, OPTIONAL HDC hdc,
|
||
|
const RECT *pBoundingRect, RECT *pContentRect)
|
||
|
{
|
||
|
MARGINS margins;
|
||
|
HRESULT hr = GetScaledContentMargins(pRender, hdc, pBoundingRect, &margins);
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
|
||
|
pContentRect->left = pBoundingRect->left + margins.cxLeftWidth;
|
||
|
pContentRect->top = pBoundingRect->top + margins.cyTopHeight;
|
||
|
|
||
|
pContentRect->right = pBoundingRect->right - margins.cxRightWidth;
|
||
|
pContentRect->bottom = pBoundingRect->bottom - margins.cyBottomHeight;
|
||
|
|
||
|
exit:
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::GetBackgroundExtent(CRenderObj *pRender, OPTIONAL HDC hdc,
|
||
|
const RECT *pContentRect, RECT *pExtentRect)
|
||
|
{
|
||
|
MARGINS margins;
|
||
|
HRESULT hr = GetScaledContentMargins(pRender, hdc, pContentRect, &margins);
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
|
||
|
pExtentRect->left = pContentRect->left - margins.cxLeftWidth;
|
||
|
pExtentRect->top = pContentRect->top-+ margins.cyTopHeight;
|
||
|
|
||
|
pExtentRect->right = pContentRect->right + margins.cxRightWidth;
|
||
|
pExtentRect->bottom = pContentRect->bottom + margins.cyBottomHeight;
|
||
|
|
||
|
exit:
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::GetScaledContentMargins(CRenderObj *pRender, OPTIONAL HDC hdc,
|
||
|
OPTIONAL const RECT *prcDest, MARGINS *pMargins)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
*pMargins = _ContentMargins;
|
||
|
|
||
|
//---- now scale the margins ----
|
||
|
SIZE szDraw;
|
||
|
TRUESTRETCHINFO tsInfo;
|
||
|
|
||
|
DIBINFO *pdi = SelectCorrectImageFile(pRender, hdc, prcDest, FALSE, NULL);
|
||
|
|
||
|
GetDrawnImageSize(pdi, prcDest, &tsInfo, &szDraw);
|
||
|
|
||
|
hr = ScaleMargins(pMargins, hdc, pRender, pdi, &szDraw);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::GetPartSize(CRenderObj *pRender, HDC hdc, OPTIONAL const RECT *prc,
|
||
|
THEMESIZE eSize, SIZE *psz)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
TRUESTRETCHINFO tsInfo;
|
||
|
|
||
|
DIBINFO *pdi = SelectCorrectImageFile(pRender, hdc, prc, FALSE, &tsInfo);
|
||
|
|
||
|
if (eSize == TS_MIN)
|
||
|
{
|
||
|
MARGINS margins;
|
||
|
hr = GetScaledContentMargins(pRender, hdc, prc, &margins);
|
||
|
if (FAILED(hr))
|
||
|
goto exit;
|
||
|
|
||
|
psz->cx = max(1, margins.cxLeftWidth + margins.cxRightWidth);
|
||
|
psz->cy = max(1, margins.cyTopHeight + margins.cyBottomHeight);
|
||
|
}
|
||
|
else if (eSize == TS_TRUE)
|
||
|
{
|
||
|
psz->cx = pdi->iSingleWidth;
|
||
|
psz->cy = pdi->iSingleHeight;
|
||
|
}
|
||
|
else if (eSize == TS_DRAW)
|
||
|
{
|
||
|
GetDrawnImageSize(pdi, prc, &tsInfo, psz);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hr = MakeError32(E_INVALIDARG);
|
||
|
goto exit;
|
||
|
}
|
||
|
|
||
|
exit:
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::GetBitmap(CRenderObj *pRender, HDC hdc, const RECT *prc, HBITMAP *phBitmap)
|
||
|
{
|
||
|
int iStockDibOffset = pRender->GetValueIndex(_iSourcePartId, _iSourceStateId, TMT_STOCKDIBDATA);
|
||
|
if (iStockDibOffset > 0)
|
||
|
{
|
||
|
return pRender->GetBitmap(NULL, iStockDibOffset, phBitmap);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return E_INVALIDARG;
|
||
|
}
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
void CImageFile::GetOffsets(int iStateId, DIBINFO *pdi, int *piXOffset, int *piYOffset)
|
||
|
{
|
||
|
if (_eImageLayout == IL_HORIZONTAL)
|
||
|
{
|
||
|
//---- iStateId in the image index ----
|
||
|
if ((iStateId <= 0) || (iStateId > _iImageCount))
|
||
|
*piXOffset = 0;
|
||
|
else
|
||
|
*piXOffset = (iStateId-1) * (pdi->iSingleWidth);
|
||
|
|
||
|
*piYOffset = 0;
|
||
|
}
|
||
|
else // vertical
|
||
|
{
|
||
|
//---- iStateId in the image index ----
|
||
|
if ((iStateId <= 0) || (iStateId > _iImageCount))
|
||
|
*piYOffset = 0;
|
||
|
else
|
||
|
*piYOffset = (iStateId-1) * (pdi->iSingleHeight);
|
||
|
|
||
|
*piXOffset = 0;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|
||
|
HRESULT CImageFile::ScaleMargins(IN OUT MARGINS *pMargins, HDC hdcOrig, CRenderObj *pRender,
|
||
|
DIBINFO *pdi, const SIZE *pszDraw, OPTIONAL float *pfx, OPTIONAL float *pfy)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
COptionalDC hdc(hdcOrig);
|
||
|
BOOL fForceRectSizing = FALSE;
|
||
|
|
||
|
if ((pRender) && (pRender->_dwOtdFlags & OTD_FORCE_RECT_SIZING))
|
||
|
{
|
||
|
fForceRectSizing = TRUE;
|
||
|
}
|
||
|
|
||
|
float xFactor = 1;
|
||
|
float yFactor = 1;
|
||
|
|
||
|
//---- any margins to size? ----
|
||
|
if ((pMargins->cxLeftWidth) || (pMargins->cxRightWidth) || (pMargins->cyBottomHeight)
|
||
|
|| (pMargins->cyTopHeight))
|
||
|
{
|
||
|
if ((pszDraw->cx > 0) && (pszDraw->cy > 0))
|
||
|
{
|
||
|
BOOL fxNeedScale = FALSE;
|
||
|
BOOL fyNeedScale = FALSE;
|
||
|
|
||
|
//---- scale if dest rect is too small in one dimension ----
|
||
|
if ((_fSourceShrink) || (fForceRectSizing))
|
||
|
{
|
||
|
if (pszDraw->cx < pdi->szMinSize.cx)
|
||
|
{
|
||
|
fxNeedScale = TRUE;
|
||
|
}
|
||
|
|
||
|
if (pszDraw->cy < pdi->szMinSize.cy)
|
||
|
{
|
||
|
fyNeedScale = TRUE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ((_fSourceGrow) || (fForceRectSizing))
|
||
|
{
|
||
|
if ((! fxNeedScale) && (! fyNeedScale))
|
||
|
{
|
||
|
//---- calculate our Dest DPI ----
|
||
|
int iDestDpi;
|
||
|
|
||
|
if (fForceRectSizing)
|
||
|
{
|
||
|
iDestDpi = (pRender) ? (pRender->GetDpiOverride()) : 0;
|
||
|
|
||
|
if (! iDestDpi)
|
||
|
{
|
||
|
//---- make up a DPI based on sizes (IE will pass us the actual DPI soon) ----
|
||
|
int ixFakeDpi = MulDiv(pdi->iMinDpi, pszDraw->cx, _szNormalSize.cx);
|
||
|
int iyFakeDpi = MulDiv(pdi->iMinDpi, pszDraw->cy, _szNormalSize.cy);
|
||
|
|
||
|
iDestDpi = (ixFakeDpi + iyFakeDpi)/2;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
iDestDpi = GetDeviceCaps(hdc, LOGPIXELSX);
|
||
|
}
|
||
|
|
||
|
//---- scale source/margins by Dest DPI ----
|
||
|
if (iDestDpi >= 2*pdi->iMinDpi)
|
||
|
{
|
||
|
xFactor *= iDestDpi/pdi->iMinDpi;
|
||
|
yFactor *= iDestDpi/pdi->iMinDpi;
|
||
|
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//---- scale by ratio of our image to draw size ----
|
||
|
if (fxNeedScale)
|
||
|
{
|
||
|
xFactor *= float(pszDraw->cx)/float(_szNormalSize.cx);
|
||
|
}
|
||
|
|
||
|
if (fyNeedScale)
|
||
|
{
|
||
|
yFactor *= float(pszDraw->cy)/float(_szNormalSize.cy);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//---- use smallest factor for both ----
|
||
|
if (xFactor < yFactor)
|
||
|
{
|
||
|
yFactor = xFactor;
|
||
|
}
|
||
|
else if (yFactor < xFactor)
|
||
|
{
|
||
|
xFactor = yFactor;
|
||
|
}
|
||
|
|
||
|
//---- integer truncation ----
|
||
|
if (xFactor > 1.0)
|
||
|
{
|
||
|
xFactor = float(int(xFactor));
|
||
|
}
|
||
|
|
||
|
if (yFactor > 1.0)
|
||
|
{
|
||
|
yFactor = float(int(yFactor));
|
||
|
}
|
||
|
|
||
|
//---- scale the margin values ----
|
||
|
if (xFactor != 1)
|
||
|
{
|
||
|
pMargins->cxLeftWidth = ROUND(xFactor*pMargins->cxLeftWidth);
|
||
|
pMargins->cxRightWidth = ROUND(xFactor*pMargins->cxRightWidth);
|
||
|
}
|
||
|
|
||
|
if (yFactor != 1)
|
||
|
{
|
||
|
pMargins->cyTopHeight = ROUND(yFactor*pMargins->cyTopHeight);
|
||
|
pMargins->cyBottomHeight = ROUND(yFactor*pMargins->cyBottomHeight);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//---- return factors to interested callers ----
|
||
|
if (pfx)
|
||
|
{
|
||
|
*pfx = xFactor;
|
||
|
}
|
||
|
|
||
|
if (pfy)
|
||
|
{
|
||
|
*pfy = yFactor;
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
//---------------------------------------------------------------------------
|