/* Miranda NG: the free IM client for Microsoft* Windows* Copyright (c) 2012-14 Miranda NG project (http://miranda-ng.org), Copyright (c) 2000-08 Miranda ICQ/IM project, all portions of this codebase are copyrighted to the people listed in contributors.txt. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program 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 GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ //Include #include "hdr/modern_commonheaders.h" #include #include "m_api/m_skin_eng.h" #include "hdr/modern_skinselector.h" #include "CLUIFrames/cluiframes.h" #define _EFFECTENUM_FULL_H #include "hdr/modern_effectenum.h" #undef _EFFECTENUM_FULL_H #include "hdr/modern_skinengine.h" #include "hdr/modern_commonprototypes.h" #include "hdr/modern_sync.h" //Implementation /* Global variables */ SKINOBJECTSLIST g_SkinObjectList = {0}; CURRWNDIMAGEDATA *g_pCachedWindow = NULL; BOOL g_flag_bPostWasCanceled = FALSE; BOOL g_flag_bFullRepaint = FALSE; BOOL g_mutex_bLockUpdating = FALSE; SortedList *gl_plGlyphTexts = NULL; SortedList *gl_plSkinFonts = NULL; /* Private module variables */ static HANDLE hSkinLoadedEvent; static GLYPHIMAGE *pLoadedImages = NULL; static DWORD dwLoadedImagesCount = 0; static DWORD dwLoadedImagesAlocated = 0; static BOOL flag_bUpdateQueued = FALSE; static BOOL flag_bJustDrawNonFramedObjects = FALSE; static BOOL mutex_bLockUpdate = FALSE; static LIST arEffectStack(10, HandleKeySortT); static SKINOBJECTSLIST *pCurrentSkin = NULL; static char **pszSettingName = NULL; static int nArrayLen = 0; static BYTE pbGammaWeight[256] = {0}; static BYTE pbGammaWeightAdv[256] = {0}; static BOOL bGammaWeightFilled = FALSE; static CRITICAL_SECTION cs_SkinChanging = {0}; static LISTMODERNMASK *MainModernMaskList = NULL; /* Private module procedures */ static BOOL ske_GetMaskBit(BYTE *line, int x); static INT_PTR ske_Service_AlphaTextOut(WPARAM wParam,LPARAM lParam); static INT_PTR ske_Service_DrawIconEx(WPARAM wParam,LPARAM lParam); static int ske_AlphaTextOut (HDC hDC, LPCTSTR lpString, int nCount, RECT *lpRect, UINT format, DWORD ARGBcolor); static void ske_AddParseTextGlyphObject(char * szGlyphTextID,char * szDefineString,SKINOBJECTSLIST *Skin); static void ske_AddParseSkinFont(char * szFontID,char * szDefineString,SKINOBJECTSLIST *Skin); static int ske_DeleteAllSettingInSection(char * SectionName); static int ske_GetSkinFromDB(char * szSection, SKINOBJECTSLIST * Skin); static LPSKINOBJECTDESCRIPTOR ske_FindObject(const char * szName, BYTE objType,SKINOBJECTSLIST* Skin); static int ske_LoadSkinFromResource(BOOL bOnlyObjects); static void ske_PreMultiplyChanells(HBITMAP hbmp,BYTE Mult); static int ske_ValidateSingleFrameImage(FRAMEWND * Frame, BOOL SkipBkgBlitting); static INT_PTR ske_Service_UpdateFrameImage(WPARAM wParam, LPARAM lParam); static INT_PTR ske_Service_InvalidateFrameImage(WPARAM wParam, LPARAM lParam); static INT_PTR ske_Service_DrawTextWithEffect(WPARAM wParam, LPARAM lParam); static MODERNEFFECT meCurrentEffect = {-1,{0}, 0, 0}; ////////////////////////////////////////////////////////////////////////// // Ini file parser ////////////////////////////////////////////////////////////////////////// IniParser::IniParser(TCHAR * tcsFileName, BYTE flags) : _Flags(flags) { _DoInit(); if (!tcsFileName) return; if (tcsFileName[0] == _T('%')) { //TODO: Add parser of resource filename here _LoadResourceIni(g_hInst, MAKEINTRESOURCEA(IDR_MSF_DEFAULT_SKIN), "MSF"); return; } _hFile = _tfopen(tcsFileName, _T("r")); if (_hFile != NULL) { _eType = IT_FILE; _isValid = true; } } IniParser::IniParser(HINSTANCE hInst, const char * resourceName, const char * resourceType, BYTE flags) : _Flags(flags) { _DoInit(); _LoadResourceIni(hInst, resourceName, resourceType); } IniParser::~IniParser() { mir_free(_szSection); if (_hFile) fclose(_hFile); if (_hGlobalRes) { UnlockResource(_hGlobalRes); FreeResource(_hGlobalRes); } _szSection = NULL; _hGlobalRes = NULL; _hFile = NULL; _isValid = false; _eType = IT_UNKNOWN; } HRESULT IniParser::Parse(ParserCallback_t pLineCallBackProc, LPARAM SecCheck) { if (_isValid && pLineCallBackProc) { _pLineCallBackProc = pLineCallBackProc; _SecCheck = SecCheck; switch (_eType) { case IT_FILE: return _DoParseFile(); case IT_RESOURCE: return _DoParseResource(); } } return E_FAIL; } HRESULT IniParser::WriteStrToDb( const char * szSection, const char * szName, const char * szValue, IniParser * This ) { if ( This->_SecCheck) { //TODO check security here if ( wildcmp( szSection,"Skin_Description_Section")) return S_OK; } if (( This->_Flags == IniParser::FLAG_ONLY_OBJECTS ) && !wildcmp( szSection, DEFAULTSKINSECTION)) return S_OK; // skip not objects switch (szValue[0]) { case 'b': db_set_b(NULL, szSection, szName, (BYTE)atoi(szValue + 1)); break; case 'w': db_set_w(NULL, szSection, szName, (WORD)atoi(szValue + 1)); break; case 'd': db_set_dw(NULL, szSection, szName, (DWORD)atoi(szValue + 1)); break; case 's': db_set_s(NULL, szSection, szName, szValue + 1); break; } return S_OK; } int IniParser::GetSkinFolder( IN const TCHAR * szFileName, OUT TCHAR * pszFolderName ) { TCHAR *szBuff = mir_tstrdup(szFileName); TCHAR *pszPos = szBuff + _tcslen(szBuff); while ( pszPos > szBuff && *pszPos != _T('.')) { pszPos--; } *pszPos = _T('\0'); _tcscpy( pszFolderName, szBuff ); TCHAR custom_folder[MAX_PATH]; TCHAR cus[MAX_PATH]; TCHAR *b3; _tcscpy( custom_folder, pszFolderName ); b3 = custom_folder + _tcslen( custom_folder ); while ( b3 > custom_folder && *b3 != _T('\\')) { b3--; } *b3 = _T('\0'); GetPrivateProfileString(_T("Skin_Description_Section"),_T("SkinFolder"),_T(""),cus,SIZEOF(custom_folder),szFileName); if (cus[0] != 0) mir_sntprintf(pszFolderName, MAX_PATH, _T("%s\\%s"), custom_folder, cus); mir_free(szBuff); PathToRelativeT(pszFolderName, pszFolderName); return 0; } void IniParser::_DoInit() { _isValid = false; _eType = IT_UNKNOWN; _szSection = NULL; _hFile = NULL; _hGlobalRes = NULL; _dwSizeOfRes = 0; _pPosition = NULL; _pLineCallBackProc = NULL; _SecCheck = 0; } void IniParser::_LoadResourceIni( HINSTANCE hInst, const char * resourceName, const char * resourceType ) { if (_eType != IT_UNKNOWN) return; HRSRC hRSrc = FindResourceA(hInst, resourceName, resourceType); if (!hRSrc) return; _hGlobalRes = LoadResource(hInst, hRSrc); if (!_hGlobalRes) return; _dwSizeOfRes = SizeofResource(hInst, hRSrc); _pPosition = (char*)LockResource(_hGlobalRes); _isValid = true; _eType = IT_RESOURCE; } HRESULT IniParser::_DoParseFile() { char szLine[MAX_LINE_LEN]; _nLine = 0; while (fgets(szLine, SIZEOF(szLine), _hFile) != NULL) { size_t len = 0; char * pLine = (char*)_RemoveTailings(szLine, len); if (len > 0) { pLine[len] = '\0'; if (!_DoParseLine(pLine)) return E_FAIL; } else _nLine++; }; return S_OK; } HRESULT IniParser::_DoParseResource() { _nLine = 0; char szLine[MAX_LINE_LEN]; char * pos = (char*)_pPosition; while (pos < _pPosition + _dwSizeOfRes) { int i = 0; while (pos < _pPosition + _dwSizeOfRes && *pos != '\n' && *pos != '\0' && i < MAX_LINE_LEN - 1) { if ((*pos) != '\r') szLine[i++] = *pos; pos++; } szLine[i] = '\0'; pos++; size_t len = 0; char * pLine = (char*)_RemoveTailings(szLine, len); if (len > 0) { pLine[len] = '\0'; if (!_DoParseLine(pLine)) return E_FAIL; } else _nLine++; } return S_OK; } const char * IniParser::_RemoveTailings( const char * szLine, size_t& len ) { const char * pStart = szLine; while (*pStart == ' ' || *pStart == '\t') pStart++; //skip spaces at begin const char * pEnd = pStart + strlen(pStart); while (pEnd > pStart && (*pEnd == ' ' || *pEnd == '\t' || *pEnd == '\n' || *pEnd == '\r')) pEnd--; len = pEnd - pStart; return pStart; } BOOL IniParser::_DoParseLine( char * szLine ) { _nLine++; DWORD len = strlen( szLine ); if (len == 0) return TRUE; switch( szLine[0] ) { case ';': return TRUE; // start of comment is found case '[': //New section start here mir_free(_szSection); _szSection = NULL; { char *tbuf = szLine + 1; // skip [ char *ebuf = tbuf; while (*ebuf != ']' && *ebuf != '\0') ebuf++; if (*ebuf == '\0') return FALSE; // no close bracket DWORD sectionLen = ebuf - tbuf; _szSection = (char*)mir_alloc(sectionLen + 1); strncpy(_szSection, tbuf, sectionLen); _szSection[sectionLen] = '\0'; } return TRUE; default: if (!_szSection ) return TRUE; //param found out of section char *keyName = szLine; char *keyValue = szLine; DWORD eqPlace = 0; DWORD len2 = strlen(keyName); while ( eqPlace < len2 && keyName[ eqPlace ] != '=' ) eqPlace++; //find '=' if (eqPlace == 0 || eqPlace == len2) return TRUE; // = not found or no key name //say false keyName[eqPlace] = '\0'; keyValue = keyName + eqPlace + 1; //remove tail spaces in Name { DWORD len3 = strlen(keyName); int j = len3-1; while (j>0 && (keyName[j] == ' ' || keyName[j] == '\t')) j--; if (j >= 0) keyName[j+1] = '\0'; } //remove start spaces in Value { DWORD len3 = strlen(keyValue); DWORD j = 0; while (j < len3 && (keyValue[j] == ' ' || keyValue[j] == '\t')) j++; if (j < len3) keyValue += j; } //remove tail spaces in Value { DWORD len3 = strlen(keyValue); int j = len3-1; while (j>0 && (keyValue[j] == ' ' || keyValue[j] == '\t' || keyValue[j] == '\n')) j--; if (j >= 0) keyValue[j+1] = '\0'; } _pLineCallBackProc( _szSection, keyName, keyValue, this ); } return TRUE; } ////////////////////////////////////////////////////////////////////////// // End of IniParser ////////////////////////////////////////////////////////////////////////// HRESULT SkinEngineLoadModule() { ModernSkinButtonLoadModule(); InitializeCriticalSection(&cs_SkinChanging); MainModernMaskList = (LISTMODERNMASK*)mir_calloc(sizeof(LISTMODERNMASK)); //init variables g_SkinObjectList.dwObjLPAlocated = 0; g_SkinObjectList.dwObjLPReserved = 0; g_SkinObjectList.pObjects = NULL; // Initialize GDI+ InitGdiPlus(); AniAva_InitModule(); //create services CreateServiceFunction(MS_SKIN_DRAWGLYPH,ske_Service_DrawGlyph); CreateServiceFunction(MS_SKINENG_UPTATEFRAMEIMAGE,ske_Service_UpdateFrameImage); CreateServiceFunction(MS_SKINENG_INVALIDATEFRAMEIMAGE,ske_Service_InvalidateFrameImage); CreateServiceFunction(MS_SKINENG_ALPHATEXTOUT,ske_Service_AlphaTextOut); CreateServiceFunction(MS_SKINENG_DRAWICONEXFIX,ske_Service_DrawIconEx); CreateServiceFunction(MS_DRAW_TEXT_WITH_EFFECT,ske_Service_DrawTextWithEffect); //create event handle hSkinLoadedEvent = HookEvent(ME_SKIN_SERVICESCREATED,CLUI_OnSkinLoad); NotifyEventHooks(g_CluiData.hEventSkinServicesCreated, 0, 0); return S_OK; } int SkinEngineUnloadModule() { //unload services ModernSkinButtonUnloadModule(0, 0); ske_UnloadSkin(&g_SkinObjectList); mir_free_and_nil(g_SkinObjectList.pObjects); mir_free_and_nil(g_SkinObjectList.pMaskList); mir_free_and_nil(MainModernMaskList); for (int i=0; i < arEffectStack.getCount(); i++) mir_free(arEffectStack[i]); arEffectStack.destroy(); if (g_pCachedWindow) { SelectObject(g_pCachedWindow->hBackDC,g_pCachedWindow->hBackOld); SelectObject(g_pCachedWindow->hImageDC,g_pCachedWindow->hImageOld); DeleteObject(g_pCachedWindow->hBackDIB); DeleteObject(g_pCachedWindow->hImageDIB); DeleteDC(g_pCachedWindow->hBackDC); DeleteDC(g_pCachedWindow->hImageDC); ReleaseDC(NULL,g_pCachedWindow->hScreenDC); mir_free_and_nil(g_pCachedWindow); } DeleteCriticalSection(&cs_SkinChanging); GdiFlush(); DestroyHookableEvent(g_CluiData.hEventSkinServicesCreated); AniAva_UnloadModule(); ShutdownGdiPlus(); //free variables return 1; } BOOL ske_AlphaBlend(HDC hdcDest,int nXOriginDest,int nYOriginDest,int nWidthDest,int nHeightDest,HDC hdcSrc,int nXOriginSrc,int nYOriginSrc,int nWidthSrc,int nHeightSrc,BLENDFUNCTION blendFunction) { if (g_CluiData.fDisableSkinEngine && !(blendFunction.BlendFlags&128)) { if (nWidthDest != nWidthSrc || nHeightDest != nHeightSrc) return StretchBlt(hdcDest,nXOriginDest,nYOriginDest,nWidthDest,nHeightDest,hdcSrc,nXOriginSrc,nYOriginSrc,nWidthSrc,nHeightSrc, SRCCOPY); else return BitBlt(hdcDest,nXOriginDest,nYOriginDest,nWidthDest,nHeightDest,hdcSrc,nXOriginSrc,nYOriginSrc, SRCCOPY); } if (blendFunction.BlendFlags&128) //Use gdi+ engine { return GDIPlus_AlphaBlend( hdcDest,nXOriginDest,nYOriginDest,nWidthDest,nHeightDest, hdcSrc,nXOriginSrc,nYOriginSrc,nWidthSrc,nHeightSrc, &blendFunction); } blendFunction.BlendFlags &= ~128; return AlphaBlend(hdcDest,nXOriginDest,nYOriginDest,nWidthDest,nHeightDest,hdcSrc,nXOriginSrc,nYOriginSrc,nWidthSrc,nHeightSrc,blendFunction); } static int ske_LockSkin() { EnterCriticalSection(&cs_SkinChanging); return 0; } static int ske_UnlockSkin() { LeaveCriticalSection(&cs_SkinChanging); return 0; } struct DCBUFFER { HDC hdcOwnedBy; int nUsageID; int width; int height; void* pImage; HDC hDC; HBITMAP oldBitmap; HBITMAP hBitmap; DWORD dwDestroyAfterTime; }; static int SortBufferList(const DCBUFFER *buf1, const DCBUFFER *buf2) { if (buf1->hdcOwnedBy != buf2->hdcOwnedBy) return (int)(buf1->hdcOwnedBy < buf2->hdcOwnedBy); else if (buf1->nUsageID != buf2->nUsageID) return (int)(buf1->nUsageID < buf2->nUsageID); else return (int)(buf1->hDC < buf2->hDC); } LIST BufferList(2, SortBufferList); CRITICAL_SECTION BufferListCS = {0}; enum { BUFFER_DRAWICON = 0, BUFFER_DRAWIMAGE }; HDC ske_RequestBufferDC(HDC hdcOwner, int dcID, int width, int height, BOOL fClear) { mir_cslock lck(BufferListCS); //Try to find DC in buffer list DCBUFFER buf; buf.hdcOwnedBy = hdcOwner; buf.nUsageID = dcID; buf.hDC = NULL; DCBUFFER *pBuf = BufferList.find(&buf); if (!pBuf) { //if not found - allocate it pBuf = (DCBUFFER *)mir_alloc(sizeof(DCBUFFER)); *pBuf = buf; pBuf->width = width; pBuf->height = height; pBuf->hBitmap = ske_CreateDIB32Point(width,height,&(pBuf->pImage)); pBuf->hDC = CreateCompatibleDC(hdcOwner); pBuf->oldBitmap = (HBITMAP)SelectObject(pBuf->hDC,pBuf->hBitmap); pBuf->dwDestroyAfterTime = 0; BufferList.insert(pBuf); } else { if (pBuf->width != width || pBuf->height != height) { //resize SelectObject(pBuf->hDC,pBuf->oldBitmap); DeleteObject(pBuf->hBitmap); pBuf->width = width; pBuf->height = height; pBuf->hBitmap = ske_CreateDIB32Point(width,height,&(pBuf->pImage)); pBuf->oldBitmap = (HBITMAP)SelectObject(pBuf->hDC,pBuf->hBitmap); } else if (fClear) memset(pBuf->pImage, 0, width*height*sizeof(DWORD)); } pBuf->dwDestroyAfterTime = 0; return pBuf->hDC; } int ske_ReleaseBufferDC(HDC hDC, int keepTime) { DWORD dwCurrentTime = GetTickCount(); //Try to find DC in buffer list - set flag to be release after time; mir_cslock lck(BufferListCS); for (int i=0; i < BufferList.getCount(); i++) { DCBUFFER *pBuf = BufferList[i]; if (pBuf) { if (hDC != NULL && pBuf->hDC == hDC) { pBuf->dwDestroyAfterTime = dwCurrentTime+keepTime; break; } if ((pBuf->dwDestroyAfterTime && pBuf->dwDestroyAfterTime < dwCurrentTime) || keepTime == -1) { SelectObject(pBuf->hDC,pBuf->oldBitmap); DeleteObject(pBuf->hBitmap); DeleteDC(pBuf->hDC); mir_free(pBuf); BufferList.remove(i); i--; } } } return 0; } BOOL ske_SetRgnOpaque(HDC memdc,HRGN hrgn, BOOL force) { RGNDATA * rdata; DWORD rgnsz; DWORD d; RECT *rect; if (g_CluiData.fDisableSkinEngine && !force) return TRUE; rgnsz = GetRegionData(hrgn, 0, NULL); rdata = (RGNDATA *) mir_alloc(rgnsz); GetRegionData(hrgn,rgnsz,rdata); rect = (RECT *)rdata->Buffer; for (d = 0; d < rdata->rdh.nCount; d++) { ske_SetRectOpaque(memdc,&rect[d], force); } mir_free(rdata); return TRUE; } BOOL ske_SetRectOpaque(HDC memdc,RECT *fr, BOOL force) { int f = 0; BYTE * bits; BITMAP bmp; HBITMAP hbmp; if ( g_CluiData.fDisableSkinEngine && !force ) return TRUE; hbmp = (HBITMAP)GetCurrentObject( memdc,OBJ_BITMAP ); GetObject( hbmp, sizeof(bmp), &bmp ); if ( bmp.bmPlanes != 1 ) return FALSE; if (!bmp.bmBits) { f = 1; bits = (BYTE*)malloc(bmp.bmWidthBytes*bmp.bmHeight); GetBitmapBits(hbmp,bmp.bmWidthBytes*bmp.bmHeight,bits); } else bits = (BYTE*)bmp.bmBits; int sx = ( fr->left > 0 ) ? fr->left : 0; int sy = ( fr->top > 0 ) ? fr->top : 0; int ex = ( fr->right < bmp.bmWidth ) ? fr->right : bmp.bmWidth; int ey = ( fr->bottom < bmp.bmHeight) ? fr->bottom : bmp.bmHeight; int width = ex-sx; BYTE* pLine = ((BYTE*)bits) + (bmp.bmHeight-sy-1)*bmp.bmWidthBytes + (sx << 2) + 3; for ( int y = 0; y < (ey - sy); y++ ) { BYTE * pColumn = pLine; for ( int x = 0; x < width; x++ ) { *pColumn = 255; pColumn += 4; } pLine -= bmp.bmWidthBytes; } if (f) { SetBitmapBits(hbmp,bmp.bmWidthBytes*bmp.bmHeight,bits); free(bits); } // DeleteObject(hbmp); return 1; } static BOOL ske_SkinFillRectByGlyph(HDC hDest, HDC hSource, RECT *rFill, RECT *rGlyph, RECT *rClip, BYTE mode, BYTE drawMode, int depth) { int destw = 0, desth = 0; int xstart = 0, xmax = 0; int ystart = 0, ymax = 0; BLENDFUNCTION bfa = {AC_SRC_OVER, 0, 255, AC_SRC_ALPHA }; BLENDFUNCTION bf = {AC_SRC_OVER, 0, 255, AC_SRC_ALPHA }; //initializations if (mode == FM_STRETCH) { HDC mem2dc; HBITMAP mem2bmp, oldbmp; RECT wr; IntersectRect(&wr,rClip,rFill); if ((wr.bottom-wr.top)*(wr.right-wr.left) == 0) return 0; if (drawMode != 2) { mem2dc = CreateCompatibleDC(hDest); mem2bmp = ske_CreateDIB32(wr.right-wr.left,wr.bottom-wr.top); oldbmp = (HBITMAP)SelectObject(mem2dc,mem2bmp); } if (drawMode == 0 || drawMode == 2) { if (drawMode == 0) { ske_AlphaBlend(mem2dc,rFill->left-wr.left,rFill->top-wr.top,rFill->right-rFill->left,rFill->bottom-rFill->top, hSource,rGlyph->left,rGlyph->top,rGlyph->right-rGlyph->left,rGlyph->bottom-rGlyph->top,bf); ske_AlphaBlend(hDest,wr.left,wr.top,wr.right-wr.left, wr.bottom -wr.top,mem2dc, 0, 0, wr.right-wr.left, wr.bottom -wr.top,bf); } else { ske_AlphaBlend(hDest,rFill->left,rFill->top,rFill->right-rFill->left,rFill->bottom-rFill->top, hSource,rGlyph->left,rGlyph->top,rGlyph->right-rGlyph->left,rGlyph->bottom-rGlyph->top,bf); } } else { // BLENDFUNCTION bf = {AC_SRC_OVER, 0, 255, 0 }; ske_AlphaBlend(mem2dc,rFill->left-wr.left,rFill->top-wr.top,rFill->right-rFill->left,rFill->bottom-rFill->top, hSource,rGlyph->left,rGlyph->top,rGlyph->right-rGlyph->left,rGlyph->bottom-rGlyph->top,bf); ske_AlphaBlend(hDest,wr.left,wr.top,wr.right-wr.left, wr.bottom -wr.top,mem2dc, 0, 0, wr.right-wr.left, wr.bottom -wr.top,bf); } if (drawMode != 2) { SelectObject(mem2dc,oldbmp); DeleteObject(mem2bmp); DeleteDC(mem2dc); } return 1; } else if (mode == FM_TILE_VERT && (rGlyph->bottom-rGlyph->top>0) && (rGlyph->right-rGlyph->left>0)) { HDC mem2dc; HBITMAP mem2bmp,oldbmp; RECT wr; IntersectRect(&wr,rClip,rFill); if ((wr.bottom-wr.top)*(wr.right-wr.left) == 0) return 0; mem2dc = CreateCompatibleDC(hDest); //SetStretchBltMode(mem2dc, HALFTONE); mem2bmp = ske_CreateDIB32(wr.right-wr.left, rGlyph->bottom-rGlyph->top); oldbmp = (HBITMAP)SelectObject(mem2dc,mem2bmp); if (!oldbmp) return 0; /// draw here { int y = 0, sy = 0, maxy = 0; int w = rFill->right-rFill->left; int h = rGlyph->bottom-rGlyph->top; if (h>0 && (wr.bottom-wr.top)*(wr.right-wr.left) != 0) { w = wr.right-wr.left; { // BLENDFUNCTION bf = {AC_SRC_OVER, 0, 255, 0 }; ske_AlphaBlend(mem2dc,-(wr.left-rFill->left), 0, rFill->right-rFill->left,h,hSource,rGlyph->left,rGlyph->top,rGlyph->right-rGlyph->left,h,bf); //StretchBlt(mem2dc,-(wr.left-rFill->left), 0, rFill->right-rFill->left,h,hSource,rGlyph->left,rGlyph->top,rGlyph->right-rGlyph->left,h,SRCCOPY); } if (drawMode == 0 || drawMode == 2) { if (drawMode == 0 ) { int dy; dy = (wr.top-rFill->top)%h; if (dy >= 0) { int ht; y = wr.top; ht = (y+h-dy <= wr.bottom)?(h-dy):(wr.bottom-wr.top); BitBlt(hDest,wr.left,y,w,ht,mem2dc, 0, dy,SRCCOPY); } y = wr.top+h-dy; while (y < wr.bottom-h){ BitBlt(hDest,wr.left,y,w,h,mem2dc, 0, 0, SRCCOPY); y += h; } if (y <= wr.bottom) BitBlt(hDest,wr.left,y,w,wr.bottom-y, mem2dc, 0, 0, SRCCOPY); } else { y = wr.top; while (y < wr.bottom-h) { // BLENDFUNCTION bf = {AC_SRC_OVER, 0, 255, 0 }; ske_AlphaBlend(hDest,wr.left,y,w,h, mem2dc, 0, 0, w,h,bf); y += h; } if (y <= wr.bottom) { // BLENDFUNCTION bf = {AC_SRC_OVER, 0, 255, 0 }; ske_AlphaBlend(hDest,wr.left,y,w,wr.bottom-y, mem2dc, 0, 0, w,wr.bottom-y,bf); } } } else { int dy; BLENDFUNCTION bf = {AC_SRC_OVER, 0, 255, AC_SRC_ALPHA }; dy = (wr.top-rFill->top)%h; if (dy >= 0) { int ht; y = wr.top; ht = (y+h-dy <= wr.bottom)?(h-dy):(wr.bottom-wr.top); ske_AlphaBlend(hDest,wr.left,y,w,ht,mem2dc, 0, dy,w,ht,bf); } y = wr.top+h-dy; while (y < wr.bottom-h) { ske_AlphaBlend(hDest,wr.left,y,w,h,mem2dc, 0, 0, w,h,bf); y += h; } if (y <= wr.bottom) ske_AlphaBlend(hDest,wr.left,y,w,wr.bottom-y, mem2dc, 0, 0, w,wr.bottom-y,bf); } } } SelectObject(mem2dc,oldbmp); DeleteObject(mem2bmp); DeleteDC(mem2dc); } else if (mode == FM_TILE_HORZ && (rGlyph->right-rGlyph->left>0) && (rGlyph->bottom-rGlyph->top>0) && (rFill->bottom-rFill->top)>0 && (rFill->right-rFill->left)>0) { HDC mem2dc; RECT wr; HBITMAP mem2bmp,oldbmp; int w = rGlyph->right-rGlyph->left; int h = rFill->bottom-rFill->top; IntersectRect(&wr,rClip,rFill); if ((wr.bottom-wr.top)*(wr.right-wr.left) == 0) return 0; h = wr.bottom-wr.top; mem2dc = CreateCompatibleDC(hDest); mem2bmp = ske_CreateDIB32(w,h); oldbmp = (HBITMAP)SelectObject(mem2dc,mem2bmp); if (!oldbmp) return 0; /// draw here { int x = 0, sy = 0, maxy = 0; { //SetStretchBltMode(mem2dc, HALFTONE); //StretchBlt(mem2dc, 0, 0, w,h,hSource,rGlyph->left+(wr.left-rFill->left),rGlyph->top,w,h,SRCCOPY); // BLENDFUNCTION bf = {AC_SRC_OVER, 0, 255, 0 }; ske_AlphaBlend(mem2dc, 0, -(wr.top-rFill->top),w,rFill->bottom-rFill->top,hSource,rGlyph->left,rGlyph->top,w,rGlyph->bottom-rGlyph->top,bf); if (drawMode == 0 || drawMode == 2) { if (drawMode == 0) { int dx; dx = (wr.left-rFill->left)%w; if (dx >= 0) { int wt; x = wr.left; wt = (x+w-dx <= wr.right)?(w-dx):(wr.right-wr.left); BitBlt(hDest,x,wr.top,wt,h,mem2dc,dx, 0, SRCCOPY); } x = wr.left+w-dx; while (x < wr.right-w){ BitBlt(hDest,x,wr.top,w,h,mem2dc, 0, 0, SRCCOPY); x += w; } if (x <= wr.right) BitBlt(hDest,x,wr.top,wr.right-x,h, mem2dc, 0, 0, SRCCOPY); } else { int dx; dx = (wr.left-rFill->left)%w; x = wr.left-dx; while (x < wr.right-w){ ske_AlphaBlend(hDest,x,wr.top,w,h,mem2dc, 0, 0, w,h,bf); x += w; } if (x <= wr.right) ske_AlphaBlend(hDest,x,wr.top,wr.right-x,h, mem2dc, 0, 0, wr.right-x,h,bf); } } else { BLENDFUNCTION bf = {AC_SRC_OVER, 0, 255, AC_SRC_ALPHA }; int dx; dx = (wr.left-rFill->left)%w; if (dx >= 0) { int wt; x = wr.left; wt = (x+w-dx <= wr.right)?(w-dx):(wr.right-wr.left); ske_AlphaBlend(hDest,x,wr.top,wt,h,mem2dc,dx, 0, wt,h,bf); } x = wr.left+w-dx; while (x < wr.right-w){ ske_AlphaBlend(hDest,x,wr.top,w,h,mem2dc, 0, 0, w,h,bf); x += w; } if (x <= wr.right) ske_AlphaBlend(hDest,x,wr.top,wr.right-x,h, mem2dc, 0, 0, wr.right-x,h,bf); } } } SelectObject(mem2dc,oldbmp); DeleteObject(mem2bmp); DeleteDC(mem2dc); } else if (mode == FM_TILE_BOTH && (rGlyph->right-rGlyph->left>0) && (rGlyph->bottom-rGlyph->top>0)) { HDC mem2dc; int w = rGlyph->right-rGlyph->left; int x = 0, sy = 0, maxy = 0; int h = rFill->bottom-rFill->top; HBITMAP mem2bmp,oldbmp; RECT wr; IntersectRect(&wr,rClip,rFill); if ((wr.bottom-wr.top)*(wr.right-wr.left) == 0) return 0; mem2dc = CreateCompatibleDC(hDest); mem2bmp = ske_CreateDIB32(w,wr.bottom-wr.top); h = wr.bottom-wr.top; oldbmp = (HBITMAP)SelectObject(mem2dc,mem2bmp); #ifdef _DEBUG if (!oldbmp) (NULL,"Tile bitmap not selected","ERROR", MB_OK); #endif /// draw here { //fill temp bitmap { int y; int dy; dy = (wr.top-rFill->top)%(rGlyph->bottom-rGlyph->top); y = -dy; while (y < wr.bottom-wr.top) { ske_AlphaBlend(mem2dc, 0, y,w,rGlyph->bottom-rGlyph->top, hSource,rGlyph->left,rGlyph->top,w,rGlyph->bottom-rGlyph->top,bf); y += rGlyph->bottom-rGlyph->top; } //-- //end temp bitmap if (drawMode == 0 || drawMode == 2) { if (drawMode == 0) { int dx; dx = (wr.left-rFill->left)%w; if (dx >= 0) { int wt; x = wr.left; wt = (x+w-dx <= wr.right)?(w-dx):(wr.right-wr.left); BitBlt(hDest,x,wr.top,wt,h,mem2dc,dx, 0, SRCCOPY); } x = wr.left+w-dx; while (x < wr.right-w){ BitBlt(hDest,x,wr.top,w,h,mem2dc, 0, 0, SRCCOPY); x += w; } if (x <= wr.right) BitBlt(hDest,x,wr.top,wr.right-x,h, mem2dc, 0, 0, SRCCOPY); } else { int dx; dx = (wr.left-rFill->left)%w; x = wr.left-dx; while (x < wr.right-w){ ske_AlphaBlend(hDest,x,wr.top,w,h,mem2dc, 0, 0, w,h,bf); x += w; } if (x <= wr.right) ske_AlphaBlend(hDest,x,wr.top,wr.right-x,h, mem2dc, 0, 0, wr.right-x,h,bf); } } else { BLENDFUNCTION bf = {AC_SRC_OVER, 0, 255, AC_SRC_ALPHA }; int dx; dx = (wr.left-rFill->left)%w; if (dx >= 0) { int wt; x = wr.left; wt = (x+w-dx <= wr.right)?(w-dx):(wr.right-wr.left); ske_AlphaBlend(hDest,x,wr.top,wt,h,mem2dc,dx, 0, wt,h,bf); } x = wr.left+w-dx; while (x < wr.right-w){ ske_AlphaBlend(hDest,x,wr.top,w,h,mem2dc, 0, 0, w,h,bf); x += w; } if (x <= wr.right) ske_AlphaBlend(hDest,x,wr.top,wr.right-x,h, mem2dc, 0, 0, wr.right-x,h,bf); } } } SelectObject(mem2dc,oldbmp); DeleteObject(mem2bmp); DeleteDC(mem2dc); } return 1; } HBITMAP ske_CreateDIB32(int cx, int cy) { return ske_CreateDIB32Point(cx,cy,NULL); } HBITMAP ske_CreateDIB32Point(int cx, int cy, void ** bits) { if (cx < 0 || cy < 0) return NULL; BITMAPINFO RGB32BitsBITMAPINFO = { 0 }; RGB32BitsBITMAPINFO.bmiHeader.biSize = sizeof(BITMAPINFOHEADER); RGB32BitsBITMAPINFO.bmiHeader.biWidth = cx;//bm.bmWidth; RGB32BitsBITMAPINFO.bmiHeader.biHeight = cy;//bm.bmHeight; RGB32BitsBITMAPINFO.bmiHeader.biPlanes = 1; RGB32BitsBITMAPINFO.bmiHeader.biBitCount = 32; // pointer used for direct Bitmap pixels access UINT *ptPixels; HBITMAP DirectBitmap = CreateDIBSection(NULL, (BITMAPINFO *)&RGB32BitsBITMAPINFO, DIB_RGB_COLORS, (void **)&ptPixels, NULL, 0); if ((DirectBitmap == NULL || ptPixels == NULL) && cx != 0 && cy != 0) { #ifdef _DEBUG MessageBoxA(NULL,"Object not allocated. Check GDI object count","ERROR",MB_OK|MB_ICONERROR); DebugBreak(); #endif ; } else memset(ptPixels, 0, cx*cy*4); if (bits != NULL) *bits = ptPixels; return DirectBitmap; } HRGN ske_CreateOpaqueRgn(BYTE Level, bool Opaque) { if (!g_pCachedWindow) return NULL; RGBQUAD * buf = (RGBQUAD *) g_pCachedWindow->hImageDIBByte; int x,y; unsigned int cRect = 64; PRGNDATA pRgnData = (PRGNDATA)malloc(sizeof(RGNDATAHEADER) + (cRect)*sizeof(RECT)); memset(pRgnData, 0, sizeof(RGNDATAHEADER)); pRgnData->rdh.dwSize = sizeof(RGNDATAHEADER); pRgnData->rdh.iType = RDH_RECTANGLES; for (y = 0; y < g_pCachedWindow->Height; ++y) { bool inside = false; bool lastin = false; unsigned int entry = 0; for (x = 0; x < g_pCachedWindow->Width; ++x) { inside = Opaque ? (buf->rgbReserved > Level) : (buf->rgbReserved < Level); ++buf; if (inside != lastin) { if (inside) { lastin = true; entry = x; } else { if (pRgnData->rdh.nCount == cRect) { cRect = cRect + 64; pRgnData = (PRGNDATA)realloc(pRgnData, sizeof(RGNDATAHEADER) + (cRect)*sizeof(RECT)); } SetRect(((LPRECT)pRgnData->Buffer) + pRgnData->rdh.nCount, entry, g_pCachedWindow->Height - y, x, g_pCachedWindow->Height - y + 1); pRgnData->rdh.nCount++; lastin = false; } } } if (lastin) { if (pRgnData->rdh.nCount == cRect) { cRect = cRect + 64; pRgnData = (PRGNDATA)realloc(pRgnData, sizeof(RGNDATAHEADER) + (cRect)*sizeof(RECT)); } SetRect(((LPRECT)pRgnData->Buffer) + pRgnData->rdh.nCount, entry, g_pCachedWindow->Height - y, x, g_pCachedWindow->Height - y + 1); pRgnData->rdh.nCount++; } } HRGN hRgn = ExtCreateRegion(NULL, sizeof(RGNDATAHEADER) + pRgnData->rdh.nCount*sizeof(RECT), (LPRGNDATA)pRgnData); free(pRgnData); return hRgn; } static int ske_DrawSkinObject(SKINDRAWREQUEST * preq, GLYPHOBJECT * pobj) { HDC memdc = NULL, glyphdc = NULL; int k = 0; //BITMAP bmp = {0}; HBITMAP membmp = 0, oldbmp = 0, oldglyph = 0; BYTE Is32Bit = 0; RECT PRect; POINT mode2offset = {0}; int depth = 0; int mode = 0; //0-FastDraw, 1-DirectAlphaDraw, 2-BufferedAlphaDraw if (!(preq && pobj)) return -1; if ((!pobj->hGlyph || pobj->hGlyph == (HBITMAP)-1) && ((pobj->Style&7) == ST_IMAGE || (pobj->Style&7) == ST_FRAGMENT || (pobj->Style&7) == ST_SOLARIZE)) return 0; // Determine painting mode depth = GetDeviceCaps(preq->hDC,BITSPIXEL); depth = depth < 16?16:depth; Is32Bit = pobj->bmBitsPixel == 32; if ((!Is32Bit && pobj->dwAlpha == 255) && pobj->Style != ST_BRUSH) mode = 0; else if (pobj->dwAlpha == 255 && pobj->Style != ST_BRUSH) mode = 1; else mode = 2; // End painting mode //force mode if (preq->rcClipRect.bottom-preq->rcClipRect.top*preq->rcClipRect.right-preq->rcClipRect.left == 0) preq->rcClipRect = preq->rcDestRect; IntersectRect(&PRect,&preq->rcDestRect,&preq->rcClipRect); if (IsRectEmpty(&PRect)) { return 0; } if (mode == 2) { memdc = CreateCompatibleDC(preq->hDC); membmp = ske_CreateDIB32(PRect.right-PRect.left,PRect.bottom-PRect.top); oldbmp = (HBITMAP)SelectObject(memdc,membmp); if (oldbmp == NULL) { if (mode == 2) { SelectObject(memdc,oldbmp); DeleteDC(memdc); DeleteObject(membmp); } return 0; } } if (mode != 2) memdc = preq->hDC; { if (pobj->hGlyph && pobj->hGlyph != (HBITMAP)-1) { glyphdc = CreateCompatibleDC(preq->hDC); if (!oldglyph) oldglyph = (HBITMAP)SelectObject(glyphdc,pobj->hGlyph); else SelectObject(glyphdc,pobj->hGlyph); } // Drawing { RECT rFill, rGlyph, rClip; if ((pobj->Style&7) == ST_BRUSH) { HBRUSH br = CreateSolidBrush(pobj->dwColor); RECT fr; if (mode == 2) { SetRect(&fr, 0, 0, PRect.right-PRect.left,PRect.bottom-PRect.top); FillRect(memdc,&fr,br); ske_SetRectOpaque(memdc,&fr); // FillRectAlpha(memdc,&fr,pobj->dwColor|0xFF000000); } else { fr = PRect; // SetRect(&fr, 0, 0, PRect.right-PRect.left,PRect.bottom-PRect.top); FillRect(preq->hDC,&fr,br); } DeleteObject(br); k = -1; } else { if (mode == 2) { mode2offset.x = PRect.left; mode2offset.y = PRect.top; OffsetRect(&PRect,-mode2offset.x,-mode2offset.y); } rClip = (preq->rcClipRect); { int lft = 0; int top = 0; int rgh = pobj->bmWidth; int btm = pobj->bmHeight; if ((pobj->Style&7) == ST_FRAGMENT) { lft = pobj->clipArea.x; top = pobj->clipArea.y; rgh = min(rgh,lft+pobj->szclipArea.cx); btm = min(btm,top+pobj->szclipArea.cy); } // Draw center... if (1) { rFill.top = preq->rcDestRect.top+pobj->dwTop; rFill.bottom = preq->rcDestRect.bottom-pobj->dwBottom; rFill.left = preq->rcDestRect.left+pobj->dwLeft; rFill.right = preq->rcDestRect.right-pobj->dwRight; if (mode == 2) OffsetRect(&rFill,-mode2offset.x,-mode2offset.y); rGlyph.top = top+pobj->dwTop; rGlyph.left = lft+pobj->dwLeft; rGlyph.right = rgh-pobj->dwRight; rGlyph.bottom = btm-pobj->dwBottom; k += ske_SkinFillRectByGlyph(memdc,glyphdc,&rFill,&rGlyph,&PRect,pobj->FitMode,mode,depth); } // Draw top side... if (1) { rFill.top = preq->rcDestRect.top; rFill.bottom = preq->rcDestRect.top+pobj->dwTop; rFill.left = preq->rcDestRect.left+pobj->dwLeft; rFill.right = preq->rcDestRect.right-pobj->dwRight; if (mode == 2) OffsetRect(&rFill,-mode2offset.x,-mode2offset.y); rGlyph.top = top+0; rGlyph.left = lft+pobj->dwLeft; rGlyph.right = rgh-pobj->dwRight; rGlyph.bottom = top+pobj->dwTop; k += ske_SkinFillRectByGlyph(memdc,glyphdc,&rFill,&rGlyph,&PRect,pobj->FitMode&FM_TILE_HORZ,mode,depth); } // Draw bottom side... if (1) { rFill.top = preq->rcDestRect.bottom-pobj->dwBottom; rFill.bottom = preq->rcDestRect.bottom; rFill.left = preq->rcDestRect.left+pobj->dwLeft; rFill.right = preq->rcDestRect.right-pobj->dwRight; if (mode == 2) OffsetRect(&rFill,-mode2offset.x,-mode2offset.y); rGlyph.top = btm-pobj->dwBottom; rGlyph.left = lft+pobj->dwLeft; rGlyph.right = rgh-pobj->dwRight; rGlyph.bottom = btm; k += ske_SkinFillRectByGlyph(memdc,glyphdc,&rFill,&rGlyph,&PRect,pobj->FitMode&FM_TILE_HORZ,mode,depth); } // Draw left side... if (1) { rFill.top = preq->rcDestRect.top+pobj->dwTop; rFill.bottom = preq->rcDestRect.bottom-pobj->dwBottom; rFill.left = preq->rcDestRect.left; rFill.right = preq->rcDestRect.left+pobj->dwLeft; if (mode == 2) OffsetRect(&rFill,-mode2offset.x,-mode2offset.y); rGlyph.top = top+pobj->dwTop; rGlyph.left = lft; rGlyph.right = lft+pobj->dwLeft; rGlyph.bottom = btm-pobj->dwBottom; k += ske_SkinFillRectByGlyph(memdc,glyphdc,&rFill,&rGlyph,&PRect,pobj->FitMode&FM_TILE_VERT,mode,depth); } // Draw right side... if (1) { rFill.top = preq->rcDestRect.top+pobj->dwTop; rFill.bottom = preq->rcDestRect.bottom-pobj->dwBottom; rFill.left = preq->rcDestRect.right-pobj->dwRight; rFill.right = preq->rcDestRect.right; if (mode == 2) OffsetRect(&rFill,-mode2offset.x,-mode2offset.y); rGlyph.top = top+pobj->dwTop; rGlyph.left = rgh-pobj->dwRight; rGlyph.right = rgh; rGlyph.bottom = btm-pobj->dwBottom; k += ske_SkinFillRectByGlyph(memdc,glyphdc,&rFill,&rGlyph,&PRect,pobj->FitMode&FM_TILE_VERT,mode,depth); } // Draw Top-Left corner... if (1) { rFill.top = preq->rcDestRect.top; rFill.bottom = preq->rcDestRect.top+pobj->dwTop; rFill.left = preq->rcDestRect.left; rFill.right = preq->rcDestRect.left+pobj->dwLeft; if (mode == 2) OffsetRect(&rFill,-mode2offset.x,-mode2offset.y); rGlyph.top = top; rGlyph.left = lft; rGlyph.right = lft+pobj->dwLeft; rGlyph.bottom = top+pobj->dwTop; k += ske_SkinFillRectByGlyph(memdc,glyphdc,&rFill,&rGlyph,&PRect, 0, mode,depth); } // Draw Top-Right corner... if (1) { rFill.top = preq->rcDestRect.top; rFill.bottom = preq->rcDestRect.top+pobj->dwTop; rFill.left = preq->rcDestRect.right-pobj->dwRight; rFill.right = preq->rcDestRect.right; if (mode == 2) OffsetRect(&rFill,-mode2offset.x,-mode2offset.y); rGlyph.top = top; rGlyph.left = rgh-pobj->dwRight; rGlyph.right = rgh; rGlyph.bottom = top+pobj->dwTop; k += ske_SkinFillRectByGlyph(memdc,glyphdc,&rFill,&rGlyph,&PRect, 0, mode,depth); } // Draw Bottom-Left corner... if (1) { rFill.top = preq->rcDestRect.bottom-pobj->dwBottom; rFill.bottom = preq->rcDestRect.bottom; rFill.left = preq->rcDestRect.left; rFill.right = preq->rcDestRect.left+pobj->dwLeft; if (mode == 2) OffsetRect(&rFill,-mode2offset.x,-mode2offset.y); rGlyph.left = lft; rGlyph.right = lft+pobj->dwLeft; rGlyph.top = btm-pobj->dwBottom; rGlyph.bottom = btm; k += ske_SkinFillRectByGlyph(memdc,glyphdc,&rFill,&rGlyph,&PRect, 0, mode,depth); } // Draw Bottom-Right corner... if (1) { rFill.top = preq->rcDestRect.bottom-pobj->dwBottom; rFill.bottom = preq->rcDestRect.bottom; rFill.left = preq->rcDestRect.right-pobj->dwRight; rFill.right = preq->rcDestRect.right; if (mode == 2) OffsetRect(&rFill,-mode2offset.x,-mode2offset.y); rGlyph.left = rgh-pobj->dwRight; rGlyph.right = rgh; rGlyph.top = btm-pobj->dwBottom; rGlyph.bottom = btm; k += ske_SkinFillRectByGlyph(memdc,glyphdc,&rFill,&rGlyph,&PRect, 0, mode,depth); } } } if ((k>0 || k == -1) && mode == 2) { { BLENDFUNCTION bf = {AC_SRC_OVER, 0, /*(bm.bmBitsPixel == 32)?255:*/pobj->dwAlpha, (pobj->bmBitsPixel == 32 && pobj->Style != ST_BRUSH)?AC_SRC_ALPHA:0}; if (mode == 2) OffsetRect(&PRect,mode2offset.x,mode2offset.y); ske_AlphaBlend( preq->hDC,PRect.left,PRect.top,PRect.right-PRect.left,PRect.bottom-PRect.top, memdc, 0, 0, PRect.right-PRect.left,PRect.bottom-PRect.top,bf); } } } //free GDI resources //--++-- //free GDI resources { if (oldglyph) SelectObject(glyphdc,oldglyph); if (glyphdc) DeleteDC(glyphdc); } if (mode == 2) { SelectObject(memdc,oldbmp); DeleteDC(memdc); DeleteObject(membmp); } } if (pobj->plTextList && pobj->plTextList->realCount>0) { int i; HFONT hOldFont; for (i=0; i < pobj->plTextList->realCount; i++) { GLYPHTEXT * gt = (GLYPHTEXT *)pobj->plTextList->items[i]; if (!gt->hFont) { if (gl_plSkinFonts && gl_plSkinFonts->realCount>0) { int j = 0; for (j = 0; j < gl_plSkinFonts->realCount; j++) { SKINFONT * sf; sf = (SKINFONT*)gl_plSkinFonts->items[j]; if (sf->szFontID && !strcmp(sf->szFontID,gt->szFontID)) { gt->hFont = sf->hFont; break; } } } if (!gt->hFont) gt->hFont = (HFONT)-1; } if (gt->hFont != (HFONT)-1) { RECT rc = {0}; hOldFont = (HFONT)SelectObject(preq->hDC,gt->hFont); if (gt->RelativeFlags&2) rc.left = preq->rcDestRect.right+gt->iLeft; else if (gt->RelativeFlags&1) rc.left = ((preq->rcDestRect.right-preq->rcDestRect.left)>>1)+gt->iLeft; else rc.left = preq->rcDestRect.left+gt->iLeft; if (gt->RelativeFlags&8) rc.top = preq->rcDestRect.bottom+gt->iTop; else if (gt->RelativeFlags&4) rc.top = ((preq->rcDestRect.bottom-preq->rcDestRect.top)>>1)+gt->iTop; else rc.top = preq->rcDestRect.top+gt->iTop; if (gt->RelativeFlags&32) rc.right = preq->rcDestRect.right+gt->iRight; else if (gt->RelativeFlags&16) rc.right = ((preq->rcDestRect.right-preq->rcDestRect.left)>>1)+gt->iRight; else rc.right = preq->rcDestRect.left+gt->iRight; if (gt->RelativeFlags&128) rc.bottom = preq->rcDestRect.bottom+gt->iBottom; else if (gt->RelativeFlags&64) rc.bottom = ((preq->rcDestRect.bottom-preq->rcDestRect.top)>>1)+gt->iBottom; else rc.bottom = preq->rcDestRect.top+gt->iBottom; ske_AlphaTextOut(preq->hDC, gt->stText, -1, &rc,gt->dwFlags, gt->dwColor); SelectObject(preq->hDC,hOldFont); } } } return 0; } int ske_AddDescriptorToSkinObjectList (LPSKINOBJECTDESCRIPTOR lpDescr, SKINOBJECTSLIST* Skin) { SKINOBJECTSLIST *sk; if (Skin) sk = Skin; else sk = &g_SkinObjectList; if (!sk) return 0; if (mir_bool_strcmpi(lpDescr->szObjectID,"_HEADER_")) return 0; {//check if new object allready presents. DWORD i=0; for (i=0; i < sk->dwObjLPAlocated; i++) if (!mir_strcmp(sk->pObjects[i].szObjectID,lpDescr->szObjectID)) return 0; } if (sk->dwObjLPAlocated+1>sk->dwObjLPReserved) { // Realocated list to add space for new object sk->pObjects = (SKINOBJECTDESCRIPTOR*)mir_realloc(sk->pObjects,sizeof(SKINOBJECTDESCRIPTOR)*(sk->dwObjLPReserved+1)/*alloc step*/); sk->dwObjLPReserved++; } { //filling new objects field sk->pObjects[sk->dwObjLPAlocated].bType = lpDescr->bType; sk->pObjects[sk->dwObjLPAlocated].Data = NULL; sk->pObjects[sk->dwObjLPAlocated].szObjectID = mir_strdup(lpDescr->szObjectID); // sk->Objects[sk->dwObjLPAlocated].szObjectName = mir_strdup(lpDescr->szObjectName); if (lpDescr->Data != NULL) { //Copy defaults values switch (lpDescr->bType) { case OT_GLYPHOBJECT: { GLYPHOBJECT * obdat; GLYPHOBJECT * gl = (GLYPHOBJECT*)lpDescr->Data; sk->pObjects[sk->dwObjLPAlocated].Data = mir_alloc(sizeof(GLYPHOBJECT)); obdat = (GLYPHOBJECT*)sk->pObjects[sk->dwObjLPAlocated].Data; memcpy(obdat,gl,sizeof(GLYPHOBJECT)); if (gl->szFileName != NULL) { obdat->szFileName = mir_strdup(gl->szFileName); mir_free_and_nil(gl->szFileName); } else obdat->szFileName = NULL; obdat->hGlyph = NULL; break; } } } } sk->dwObjLPAlocated++; return 1; } static LPSKINOBJECTDESCRIPTOR ske_FindObject(const char * szName, BYTE objType, SKINOBJECTSLIST* Skin) { // DWORD i; SKINOBJECTSLIST* sk; sk = (Skin == NULL)?(&g_SkinObjectList):Skin; return skin_FindObjectByRequest((char *)szName,sk->pMaskList); } static LPSKINOBJECTDESCRIPTOR ske_FindObjectByMask(MODERNMASK *pModernMask, BYTE objType, SKINOBJECTSLIST* Skin) { // DWORD i; SKINOBJECTSLIST* sk; sk = (Skin == NULL)?(&g_SkinObjectList):Skin; if (!sk->pMaskList) return NULL; return skin_FindObjectByMask(pModernMask,sk->pMaskList); } LPSKINOBJECTDESCRIPTOR ske_FindObjectByName(const char * szName, BYTE objType, SKINOBJECTSLIST* Skin) { DWORD i; SKINOBJECTSLIST* sk; sk = (Skin == NULL)?(&g_SkinObjectList):Skin; for (i=0; i < sk->dwObjLPAlocated; i++) { if (sk->pObjects[i].bType == objType || objType == OT_ANY) { if (!mir_strcmp(sk->pObjects[i].szObjectID,szName)) return &(sk->pObjects[i]); } } return NULL; } ////////////////////////////////////////////////////////////////////////// // Paint glyph // wParam - LPSKINDRAWREQUEST // lParam - possible direct pointer to modern mask ////////////////////////////////////////////////////////////////////////// INT_PTR ske_Service_DrawGlyph(WPARAM wParam,LPARAM lParam) { LPSKINDRAWREQUEST preq; LPSKINOBJECTDESCRIPTOR pgl; LPGLYPHOBJECT gl; if (!wParam) return -1; ske_LockSkin(); __try { preq = (LPSKINDRAWREQUEST)wParam; if (lParam) pgl = ske_FindObjectByMask((MODERNMASK*)lParam, OT_GLYPHOBJECT,NULL); else pgl = ske_FindObject(preq->szObjectID, OT_GLYPHOBJECT,NULL); if (pgl == NULL) return -1; if (pgl->Data == NULL) return -1; gl = (LPGLYPHOBJECT)pgl->Data; int iStyle = gl->Style & 7; if (iStyle == ST_SKIP) return ST_SKIP; if (gl->hGlyph == NULL && gl->hGlyph != (HBITMAP)-1 && (iStyle == ST_IMAGE || iStyle == ST_FRAGMENT || iStyle == ST_SOLARIZE)) if (gl->szFileName) { gl->hGlyph = ske_LoadGlyphImage( _A2T(gl->szFileName)); if (gl->hGlyph) { BITMAP bmp = {0}; GetObject(gl->hGlyph,sizeof(BITMAP),&bmp); gl->bmBitsPixel = (BYTE)bmp.bmBitsPixel; gl->bmHeight = bmp.bmHeight; gl->bmWidth = bmp.bmWidth; } else gl->hGlyph = (HBITMAP)-1; //invalid } return ske_DrawSkinObject(preq,gl); } __finally { ske_UnlockSkin(); } return -1; } void ske_PreMultiplyChanells(HBITMAP hbmp,BYTE Mult) { BITMAP bmp; BOOL flag = FALSE; BYTE * pBitmapBits; DWORD Len; int bh,bw,y,x; GetObject(hbmp, sizeof(BITMAP), (LPSTR)&bmp); bh = bmp.bmHeight; bw = bmp.bmWidth; Len = bh*bw*4; flag = (bmp.bmBits == NULL); if (flag) { pBitmapBits = (LPBYTE)malloc(Len); GetBitmapBits(hbmp,Len,pBitmapBits); } else pBitmapBits = (BYTE*)bmp.bmBits; for (y = 0; y < bh; ++y) { BYTE *pPixel = pBitmapBits + bw * 4 * y; for (x = 0; x < bw ; ++x) { if (Mult) { pPixel[0] = pPixel[0]*pPixel[3]/255; pPixel[1] = pPixel[1]*pPixel[3]/255; pPixel[2] = pPixel[2]*pPixel[3]/255; } else { pPixel[3] = 255; } pPixel += 4; } } if (flag) { Len = SetBitmapBits(hbmp,Len,pBitmapBits); free (pBitmapBits); } return; } int ske_GetFullFilename(TCHAR *buf, const TCHAR *file, TCHAR *skinfolder, BOOL madeAbsolute) { TCHAR *SkinPlace = db_get_tsa(NULL,SKIN,"SkinFolder"); if (SkinPlace == NULL) SkinPlace = mir_tstrdup( _T("\\Skin\\default")); TCHAR b2[MAX_PATH]; if (file[0] != '\\' && file[1] != ':') mir_sntprintf(b2, MAX_PATH, _T("%s\\%s"), (skinfolder == NULL) ? SkinPlace : ((INT_PTR)skinfolder != -1) ? skinfolder : _T(""), file); else _tcsncpy(b2, file, SIZEOF(b2)); if (madeAbsolute) { if (b2[0] == '\\' && b2[1] != '\\') PathToAbsoluteT(b2+1, buf); else PathToAbsoluteT(b2, buf); } else _tcsncpy(buf, b2, MAX_PATH); mir_free(SkinPlace); return 0; } /* This function is required to load TGA to dib buffer myself Major part of routines is from http://tfcduke.developpez.com/tutoriel/format/tga/fichiers/tga.c */ static BOOL ske_ReadTGAImageData(void * From, DWORD fromSize, BYTE * destBuf, DWORD bufSize, BOOL RLE) { BYTE * pos = destBuf; BYTE * from = fromSize?(BYTE*)From:NULL; FILE * fp = !fromSize?(FILE*)From:NULL; DWORD destCount = 0; DWORD fromCount = 0; if (!RLE) { while (((from && fromCount < fromSize) || (fp && fromCount < bufSize)) && (destCount < bufSize)) { BYTE r = from?from[fromCount++]:(BYTE)fgetc(fp); BYTE g = from?from[fromCount++]:(BYTE)fgetc(fp); BYTE b = from?from[fromCount++]:(BYTE)fgetc(fp); BYTE a = from?from[fromCount++]:(BYTE)fgetc(fp); pos[destCount++] = r; pos[destCount++] = g; pos[destCount++] = b; pos[destCount++] = a; if (destCount>bufSize) break; if (from) if (fromCount < fromSize) break; } } else { BYTE rgba[4]; BYTE packet_header; BYTE *ptr = pos; BYTE size; int i; while (ptr < pos + bufSize) { /* read first byte */ packet_header = from?from[fromCount]:(BYTE)fgetc(fp); if (from) from++; size = 1 + (packet_header & 0x7f); if (packet_header & 0x80) { /* run-length packet */ if (from) { *((DWORD*)rgba) = *((DWORD*)(from+fromCount)); fromCount += 4; } else fread (rgba, sizeof (BYTE), 4, fp); for (i=0; i < size; ++i, ptr += 4) { ptr[2] = rgba[2]; ptr[1] = rgba[1]; ptr[0] = rgba[0]; ptr[3] = rgba[3]; } } else { /* not run-length packet */ for (i=0; i < size; ++i, ptr += 4) { ptr[0] = from? from[fromCount++]:(BYTE)fgetc (fp); ptr[1] = from? from[fromCount++]:(BYTE)fgetc (fp); ptr[2] = from? from[fromCount++]:(BYTE)fgetc (fp); ptr[3] = from? from[fromCount++]:(BYTE)fgetc (fp); } } } } return TRUE; } static HBITMAP ske_LoadGlyphImage_TGA(const TCHAR *szFilename) { BYTE *colormap = NULL; int cx = 0, cy = 0; BOOL err = FALSE; tga_header_t header; if (!szFilename) return NULL; if (!wildcmpit(szFilename, _T("*\\*%.tga"))) { //Loading TGA image from file FILE *fp = _tfopen (szFilename, _T("rb")); if (!fp) { TRACEVAR("error: couldn't open \"%s\"!\n", szFilename); return NULL; } /* read header */ fread (&header, sizeof (tga_header_t), 1, fp); if ((header.pixel_depth != 32) || ((header.image_type != 10) && (header.image_type != 2))) { fclose(fp); return NULL; } /*memory allocation */ colormap = (BYTE*)malloc(header.width*header.height*4); cx = header.width; cy = header.height; fseek (fp, header.id_lenght, SEEK_CUR); fseek (fp, header.cm_length, SEEK_CUR); err = !ske_ReadTGAImageData((void*)fp, 0, colormap, header.width*header.height*4,header.image_type == 10); fclose(fp); } else { /* reading from resources IDR_TGA_DEFAULT_SKIN */ DWORD size = 0; BYTE * mem; HGLOBAL hRes; HRSRC hRSrc = FindResourceA(g_hInst,MAKEINTRESOURCEA(IDR_TGA_DEFAULT_SKIN),"TGA"); if (!hRSrc) return NULL; hRes = LoadResource(g_hInst,hRSrc); if (!hRes) return NULL; size = SizeofResource(g_hInst,hRSrc); mem = (BYTE*) LockResource(hRes); if (size>sizeof(header)) { tga_header_t * header = (tga_header_t *)mem; if (header->pixel_depth == 32 && (header->image_type == 2 || header->image_type == 10)) { colormap = (BYTE*)malloc(header->width*header->height*4); cx = header->width; cy = header->height; ske_ReadTGAImageData((void*)(mem+sizeof(tga_header_t)+header->id_lenght+header->cm_length), size-(sizeof(tga_header_t)+header->id_lenght+header->cm_length), colormap, cx*cy*4,header->image_type == 10); } } FreeResource(hRes); } if (colormap) { //create dib section BYTE * pt; HBITMAP hbmp = ske_CreateDIB32Point(cx,cy,(void**)&pt); if (hbmp) memcpy(pt,colormap,cx*cy*4); free(colormap); return hbmp; } return NULL; } //this function is required to load PNG to dib buffer myself static HBITMAP ske_LoadGlyphImage_Png2Dib(const TCHAR *tszFilename) { HANDLE hFile, hMap = NULL; BYTE* ppMap = NULL; long cbFileSize = 0; BITMAPINFOHEADER* pDib; BYTE* pDibBits; if (!ServiceExists( MS_PNG2DIB )) { MessageBox( NULL, TranslateT("You need an image services plugin to process PNG images."), TranslateT("Error"), MB_OK ); return (HBITMAP)NULL; } if ((hFile = CreateFile(tszFilename, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL)) != INVALID_HANDLE_VALUE) if ((hMap = CreateFileMapping(hFile, NULL, PAGE_READONLY, 0, 0, NULL )) != NULL) if ((ppMap = ( BYTE* )MapViewOfFile(hMap, FILE_MAP_READ, 0, 0, 0 )) != NULL) cbFileSize = GetFileSize(hFile, NULL); if (cbFileSize != 0) { PNG2DIB param; param.pSource = ppMap; param.cbSourceSize = cbFileSize; param.pResult = &pDib; if ( CallService(MS_PNG2DIB, 0, (LPARAM)¶m )) pDibBits = ( BYTE* )( pDib+1 ); else cbFileSize = 0; } if ( ppMap != NULL ) UnmapViewOfFile( ppMap ); if ( hMap != NULL ) CloseHandle( hMap ); if ( hFile != NULL ) CloseHandle( hFile ); if ( cbFileSize == 0 ) return (HBITMAP)NULL; HBITMAP hBitmap; BITMAPINFO* bi = ( BITMAPINFO* )pDib; BYTE *pt = (BYTE*)bi; pt += bi->bmiHeader.biSize; if (bi->bmiHeader.biBitCount != 32) { HDC sDC = GetDC( NULL ); hBitmap = CreateDIBitmap( sDC, pDib, CBM_INIT, pDibBits, bi, DIB_PAL_COLORS ); SelectObject( sDC, hBitmap ); DeleteDC( sDC ); } else { BYTE *ptPixels = pt; hBitmap = CreateDIBSection(NULL,bi, DIB_RGB_COLORS, (void **)&ptPixels, NULL, 0); memcpy(ptPixels,pt,bi->bmiHeader.biSizeImage); } GlobalFree( pDib ); return hBitmap; } static HBITMAP ske_LoadGlyphImageByDecoders(const TCHAR *tszFileName) { // Loading image from file by imgdecoder... HBITMAP hBitmap = NULL; TCHAR ext[5]; BYTE f = 0; LPVOID pImg = NULL; BITMAP bmpInfo; { int l = lstrlen(tszFileName); lstrcpyn(ext, tszFileName+(l-4),5); } if (!_tcschr(tszFileName,'%') && !PathFileExists(tszFileName)) return NULL; if (mir_bool_tstrcmpi(ext, _T(".tga"))) { hBitmap = ske_LoadGlyphImage_TGA(tszFileName); f = 1; } else if ( ServiceExists("Image/Png2Dib") && mir_bool_tstrcmpi(ext, _T(".png"))) { hBitmap = ske_LoadGlyphImage_Png2Dib(tszFileName); GetObject(hBitmap, sizeof(BITMAP), &bmpInfo); f = (bmpInfo.bmBits != NULL); } else if (!mir_bool_tstrcmpi(ext, _T(".png"))) { hBitmap = (HBITMAP)CallService(MS_UTILS_LOADBITMAPT, 0, (LPARAM)tszFileName); } if (hBitmap) { GetObject(hBitmap, sizeof(BITMAP), &bmpInfo); if (bmpInfo.bmBitsPixel == 32) ske_PreMultiplyChanells(hBitmap,f); else { HDC dc24,dc32; HBITMAP hBitmap32,obmp24,obmp32; dc32 = CreateCompatibleDC(NULL); dc24 = CreateCompatibleDC(NULL); hBitmap32 = ske_CreateDIB32(bmpInfo.bmWidth,bmpInfo.bmHeight); obmp24 = (HBITMAP)SelectObject(dc24,hBitmap); obmp32 = (HBITMAP)SelectObject(dc32,hBitmap32); BitBlt(dc32, 0, 0, bmpInfo.bmWidth,bmpInfo.bmHeight,dc24, 0, 0, SRCCOPY); SelectObject(dc24,obmp24); SelectObject(dc32,obmp32); DeleteDC(dc24); DeleteDC(dc32); DeleteObject(hBitmap); hBitmap = hBitmap32; ske_PreMultiplyChanells(hBitmap,0); } } return hBitmap; } static HBITMAP ske_skinLoadGlyphImage(const TCHAR *tszFileName) { if (!wildcmpit(tszFileName, _T("*.tga"))) return GDIPlus_LoadGlyphImage(tszFileName); return ske_LoadGlyphImageByDecoders(tszFileName); } HBITMAP ske_LoadGlyphImage(const TCHAR *tszFileName) { // try to find image in loaded DWORD i; HBITMAP hbmp; TCHAR szFile [MAX_PATH] = {0}; ske_GetFullFilename(szFile, tszFileName, g_SkinObjectList.szSkinPlace, TRUE); ske_LockSkin(); if (pLoadedImages) { for (i=0; i < dwLoadedImagesCount; i++) { if (mir_bool_tstrcmpi(pLoadedImages[i].szFileName, szFile)) { pLoadedImages[i].dwLoadedTimes++; ske_UnlockSkin(); return pLoadedImages[i].hGlyph; } } } // load new image hbmp = ske_skinLoadGlyphImage(szFile); if (hbmp == NULL) { ske_UnlockSkin(); return NULL; } // add to loaded list if (dwLoadedImagesCount+1>dwLoadedImagesAlocated) { pLoadedImages = (GLYPHIMAGE*)mir_realloc(pLoadedImages,sizeof(GLYPHIMAGE)*(dwLoadedImagesCount+1)); if (pLoadedImages) dwLoadedImagesAlocated++; else { ske_UnlockSkin(); return NULL; } } pLoadedImages[dwLoadedImagesCount].dwLoadedTimes = 1; pLoadedImages[dwLoadedImagesCount].hGlyph = hbmp; pLoadedImages[dwLoadedImagesCount].szFileName = mir_tstrdup(szFile); dwLoadedImagesCount++; ske_UnlockSkin(); return hbmp; } int ske_UnloadGlyphImage(HBITMAP hbmp) { DWORD i; for (i=0; i < dwLoadedImagesCount; i++) { if (hbmp == pLoadedImages[i].hGlyph) { pLoadedImages[i].dwLoadedTimes--; if (pLoadedImages[i].dwLoadedTimes == 0) { LPGLYPHIMAGE gl = &(pLoadedImages[i]); mir_free_and_nil(gl->szFileName); memmove(&(pLoadedImages[i]),&(pLoadedImages[i+1]),sizeof(GLYPHIMAGE)*(dwLoadedImagesCount-i-1)); dwLoadedImagesCount--; DeleteObject(hbmp); if (pLoadedImages && dwLoadedImagesCount == 0) { dwLoadedImagesAlocated = 0; mir_free_and_nil(pLoadedImages); } } return 0; } } DeleteObject(hbmp); return 0; } int ske_UnloadSkin(SKINOBJECTSLIST * Skin) { DWORD i; ske_LockSkin(); ClearMaskList(Skin->pMaskList); //clear font list if (gl_plSkinFonts && gl_plSkinFonts->realCount > 0) { for (int i=0; i < gl_plSkinFonts->realCount; i++) { SKINFONT * sf = (SKINFONT *)gl_plSkinFonts->items[i]; if (sf) { mir_free(sf->szFontID); DeleteObject(sf->hFont); mir_free(sf); } } List_Destroy(gl_plSkinFonts); mir_free_and_nil(gl_plSkinFonts); } mir_free_and_nil(Skin->szSkinPlace); if (Skin->pTextList) List_Destroy(Skin->pTextList); mir_free_and_nil(Skin->pTextList); ModernSkinButtonDeleteAll(); if (Skin->dwObjLPAlocated == 0) { ske_UnlockSkin(); return 0;} for (i=0; i < Skin->dwObjLPAlocated; i++) { switch(Skin->pObjects[i].bType) { case OT_GLYPHOBJECT: { GLYPHOBJECT * dt; dt = (GLYPHOBJECT*)Skin->pObjects[i].Data; if (dt->hGlyph && dt->hGlyph != (HBITMAP)-1) ske_UnloadGlyphImage(dt->hGlyph); dt->hGlyph = NULL; mir_free_and_nil(dt->szFileName); {// delete texts int i; if (dt->plTextList && dt->plTextList->realCount>0) { for (i=0; i < dt->plTextList->realCount; i++) { GLYPHTEXT * gt = (GLYPHTEXT *)dt->plTextList->items[i]; if (gt) { mir_free(gt->stText); mir_free(gt->stValueText); mir_free(gt->szFontID); mir_free(gt->szGlyphTextID); mir_free(gt); } } List_Destroy(dt->plTextList); mir_free(dt->plTextList); } } mir_free(dt); } break; } mir_free_and_nil(Skin->pObjects[i].szObjectID); } mir_free_and_nil(Skin->pObjects); Skin->pTextList = NULL; Skin->dwObjLPAlocated = 0; Skin->dwObjLPReserved = 0; ske_UnlockSkin(); return 0; } static void RegisterMaskByParce(const char * szSetting, char * szValue, SKINOBJECTSLIST * pSkin) { int i; DWORD ID=atoi(szSetting+1); for (i=0; i < mir_strlen(szValue); i++) if (szValue[i] == ':') break; if (i < mir_strlen(szValue)) { char * Obj, *Mask; int res; Mask = szValue+i+1; Obj = (char*)mir_alloc(i+1); strncpy(Obj,szValue,i); Obj[i] = '\0'; res = AddStrModernMaskToList(ID,Mask,Obj,pSkin->pMaskList,pSkin); mir_free(Obj); } } static int ske_ProcessLoadindString(const char * szSetting, char *szValue) { if (!pCurrentSkin) return 0; if (szSetting[0] == '$') RegisterObjectByParce((char *)szSetting, szValue); else if (szSetting[0] == '#') RegisterButtonByParce((char *)szSetting,szValue); else if (szSetting[0] == '@') RegisterMaskByParce((char *)szSetting, szValue, pCurrentSkin); /// else if (szSetting[0] == 't') ske_AddParseTextGlyphObject((char*)szSetting,szValue,pCurrentSkin); else if (szSetting[0] == 'f') ske_AddParseSkinFont((char*)szSetting,szValue,pCurrentSkin); else return 0; return 1; } static int ske_enumdb_SkinObjectsProc (const char *szSetting,LPARAM lParam) { char *value; value = db_get_sa(NULL,SKIN,szSetting); ske_ProcessLoadindString(szSetting,value); mir_free_and_nil(value); return 0; } static int ske_SortTextGlyphObjectFunc(void * first, void * second) { return strcmp(((GLYPHTEXT*)(((int*)first)[0]))->szGlyphTextID,((GLYPHTEXT*)(((int*)second)[0]))->szGlyphTextID); } static void ske_LinkSkinObjects(SKINOBJECTSLIST * pObjectList) { DWORD i; // LINK Mask with objects for (i=0; i < pObjectList->pMaskList->dwMaskCnt; i++) { MODERNMASK *mm = &(pObjectList->pMaskList->pl_Masks[i]); void * pObject = (void*) ske_FindObjectByName(mm->szObjectName, OT_ANY, (SKINOBJECTSLIST*) pObjectList); mir_free_and_nil(mm->szObjectName); mm->bObjectFound = TRUE; mm->pObject = pObject; } if (pObjectList->pTextList) { int i; // LINK Text with objects for (i=0; i < pObjectList->pTextList->realCount; i++) { GLYPHTEXT * glText; GLYPHOBJECT *globj = NULL; SKINOBJECTDESCRIPTOR * lpobj; glText = (GLYPHTEXT *)pObjectList->pTextList->items[i]; lpobj = ske_FindObjectByName(glText->szObjectName,OT_GLYPHOBJECT, pObjectList); mir_free_and_nil(glText->szObjectName); if (lpobj) globj = (GLYPHOBJECT*)lpobj->Data; if (globj) { if (!globj->plTextList) { globj->plTextList = List_Create(0, 1); globj->plTextList->sortFunc = ske_SortTextGlyphObjectFunc; } List_Insert(globj->plTextList,(void*)glText,globj->plTextList->realCount); qsort(globj->plTextList->items,globj->plTextList->realCount,sizeof(void*),(int(*)(const void*, const void*))globj->plTextList->sortFunc); pObjectList->pTextList->items[i] = NULL; } else { GLYPHTEXT *gt = glText; if (gt) { mir_free(gt->stText); mir_free(gt->stValueText); mir_free(gt->szFontID); mir_free(gt->szGlyphTextID); mir_free(gt); } } } List_Destroy(pObjectList->pTextList); mir_free_and_nil(pObjectList->pTextList); } } // Getting skin objects and masks from DB static int ske_GetSkinFromDB(char * szSection, SKINOBJECTSLIST * Skin) { if (Skin == NULL) return 0; ske_UnloadSkin(Skin); g_CluiData.fDisableSkinEngine = db_get_b(NULL,"ModernData","DisableEngine", SETTING_DISABLESKIN_DEFAULT); //window borders if (g_CluiData.fDisableSkinEngine) { g_CluiData.LeftClientMargin = 0; g_CluiData.RightClientMargin = 0; g_CluiData.TopClientMargin = 0; g_CluiData.BottomClientMargin = 0; } else { //window borders g_CluiData.LeftClientMargin = (int)db_get_b(NULL,"CLUI","LeftClientMargin",SETTING_LEFTCLIENTMARIGN_DEFAULT); g_CluiData.RightClientMargin = (int)db_get_b(NULL,"CLUI","RightClientMargin",SETTING_RIGHTCLIENTMARIGN_DEFAULT); g_CluiData.TopClientMargin = (int)db_get_b(NULL,"CLUI","TopClientMargin",SETTING_TOPCLIENTMARIGN_DEFAULT); g_CluiData.BottomClientMargin = (int)db_get_b(NULL,"CLUI","BottomClientMargin",SETTING_BOTTOMCLIENTMARIGN_DEFAULT); } if (g_CluiData.fDisableSkinEngine) return 0; Skin->pMaskList = (LISTMODERNMASK*)mir_alloc(sizeof(LISTMODERNMASK)); memset(Skin->pMaskList, 0, sizeof(LISTMODERNMASK)); Skin->szSkinPlace = db_get_tsa(NULL, SKIN, "SkinFolder"); if (!Skin->szSkinPlace || (_tcschr(Skin->szSkinPlace, '%') && !db_get_b(NULL,SKIN,"Modified",0))) { BOOL bOnlyObjects = FALSE; if (Skin->szSkinPlace && _tcschr(Skin->szSkinPlace, '%')) bOnlyObjects = TRUE; mir_free(Skin->szSkinPlace); Skin->szSkinPlace = mir_tstrdup( _T("%Default%")); ske_LoadSkinFromResource( bOnlyObjects ); } //Load objects { DBCONTACTENUMSETTINGS dbces; pCurrentSkin = Skin; dbces.pfnEnumProc = ske_enumdb_SkinObjectsProc; dbces.szModule = SKIN; dbces.ofsSettings = 0; CallService(MS_DB_CONTACT_ENUMSETTINGS, 0, (LPARAM)&dbces); SortMaskList(pCurrentSkin->pMaskList); ske_LinkSkinObjects(pCurrentSkin); } //Load Masks return 0; } //surrogate to be called from outside void ske_LoadSkinFromDB(void) { ske_GetSkinFromDB(SKIN,&g_SkinObjectList); g_CluiData.dwKeyColor = db_get_dw(NULL,"ModernSettings","KeyColor",(DWORD)SETTING_KEYCOLOR_DEFAULT); } // static int ske_LoadSkinFromResource(BOOL bOnlyObjects) { IniParser parser(g_hInst, MAKEINTRESOURCEA(IDR_MSF_DEFAULT_SKIN), "MSF", bOnlyObjects ? IniParser::FLAG_ONLY_OBJECTS : IniParser::FLAG_WITH_SETTINGS); if (parser.CheckOK()) { ske_DeleteAllSettingInSection("ModernSkin"); db_set_s(NULL, SKIN, "SkinFolder", "%Default%"); db_set_s(NULL, SKIN, "SkinFile", "%Default%"); parser.Parse(IniParser::WriteStrToDb, 0); } return 0; } //Load data from ini file int ske_LoadSkinFromIniFile(TCHAR *szFileName, BOOL bOnlyObjects) { if (_tcschr(szFileName,_T('%'))) return ske_LoadSkinFromResource( bOnlyObjects ); IniParser parser(szFileName, bOnlyObjects ? IniParser::FLAG_ONLY_OBJECTS : IniParser::FLAG_WITH_SETTINGS); if (!parser.CheckOK()) return 0; ske_DeleteAllSettingInSection("ModernSkin"); TCHAR skinFolder[MAX_PATH], skinFile[MAX_PATH]; IniParser::GetSkinFolder(szFileName, skinFolder); PathToRelativeT(szFileName, skinFile); db_set_ts(NULL,SKIN,"SkinFolder", skinFolder); db_set_ts(NULL,SKIN,"SkinFile", skinFile); parser.Parse(IniParser::WriteStrToDb, 1); return 0; } static int ske_enumdb_SkinSectionDeletionProc (const char *szSetting,LPARAM lParam) { if (szSetting == NULL) return 0; nArrayLen++; pszSettingName = (char **)realloc(pszSettingName, nArrayLen*sizeof(char *)); pszSettingName[nArrayLen-1] = _strdup(szSetting); return 0; } static int ske_DeleteAllSettingInSection(char * SectionName) { DBCONTACTENUMSETTINGS dbces; nArrayLen = 0; pszSettingName = NULL; dbces.pfnEnumProc = ske_enumdb_SkinSectionDeletionProc; dbces.szModule = SectionName; dbces.ofsSettings = 0; CallService(MS_DB_CONTACT_ENUMSETTINGS, 0, (LPARAM)&dbces); //delete all settings if (nArrayLen == 0) return 0; for (int i=0; i < nArrayLen; i++) { db_unset(0, SectionName,pszSettingName[i]); free(pszSettingName[i]); } free(pszSettingName); pszSettingName = NULL; nArrayLen = 0; return 0; } BOOL ske_TextOut(HDC hdc, int x, int y, LPCTSTR lpString, int nCount) { SIZE sz; GetTextExtentPoint32(hdc, lpString, nCount, &sz); int ta = GetTextAlign(hdc); RECT rc = { 0 }; SetRect(&rc, x, y, x + sz.cx, y + sz.cy); ske_DrawText(hdc, lpString, nCount, &rc, DT_NOCLIP | DT_SINGLELINE | DT_LEFT); return 1; } static INT_PTR ske_Service_AlphaTextOut(WPARAM wParam,LPARAM lParam) { if (!wParam) return 0; AlphaTextOutParams ap = *(AlphaTextOutParams*)wParam; return ske_AlphaTextOut(ap.hDC,ap.lpString,ap.nCount,ap.lpRect,ap.format,ap.ARGBcolor); } static __inline void ske_SetMatrix( sbyte * matrix, sbyte a, sbyte b, sbyte c, sbyte d, sbyte e, sbyte f, sbyte g, sbyte h, sbyte i) { matrix[0] = a; matrix[1] = b; matrix[2] = c; matrix[3] = d; matrix[4] = e; matrix[5] = f; matrix[6] = g; matrix[7] = h; matrix[8] = i; } static void ske_SetTextEffect(BYTE EffectID, DWORD FirstColor, DWORD SecondColor) { if (EffectID > MAXPREDEFINEDEFFECTS) return; if (EffectID == -1) meCurrentEffect.EffectID = -1; else { meCurrentEffect.EffectID = EffectID; meCurrentEffect.EffectMatrix = ModernEffectsEnum[EffectID]; meCurrentEffect.EffectColor1 = FirstColor; meCurrentEffect.EffectColor2 = SecondColor; } } bool ske_ResetTextEffect(HDC hdc) { int idx = arEffectStack.getIndex((EFFECTSSTACKITEM*)&hdc); if (idx == -1) return false; mir_free(arEffectStack[idx]); arEffectStack.remove(idx); return true; } bool ske_SelectTextEffect(HDC hdc, BYTE EffectID, DWORD FirstColor, DWORD SecondColor) { if (EffectID > MAXPREDEFINEDEFFECTS) return false; if (EffectID == -1) return ske_ResetTextEffect(hdc); EFFECTSSTACKITEM *effect = arEffectStack.find((EFFECTSSTACKITEM*)&hdc); if (effect == NULL) { effect = (EFFECTSSTACKITEM *)mir_alloc(sizeof(EFFECTSSTACKITEM)); effect->hdc = hdc; arEffectStack.insert(effect); } effect->EffectID = EffectID; effect->FirstColor = FirstColor; effect->SecondColor = SecondColor; return true; } static bool ske_GetTextEffect(HDC hdc, MODERNEFFECT *modernEffect) { if (!modernEffect) return false; EFFECTSSTACKITEM *effect = arEffectStack.find((EFFECTSSTACKITEM*)&hdc); if (effect == NULL) return false; modernEffect->EffectID = effect->EffectID; modernEffect->EffectColor1 = effect->FirstColor; modernEffect->EffectColor2 = effect->SecondColor; modernEffect->EffectMatrix = ModernEffectsEnum[effect->EffectID]; return true; } static bool ske_DrawTextEffect(BYTE* destPt, BYTE* maskPt, DWORD width, DWORD height, MODERNEFFECT *effect) { sbyte *buf; sbyte *outbuf; sbyte *bufline, *buflineTop, *buflineMid; int sign = 0; BYTE *maskline,*destline; BYTE al,rl,gl,bl,ad,rd,gd,bd; int k = 0; DWORD x,y; sbyte *matrix; BYTE mcTopStart; BYTE mcBottomEnd; BYTE mcLeftStart; BYTE mcRightEnd; BYTE effectCount; int minX = width; int maxX = 0; int minY = height; int maxY = 0; if (effect->EffectID == 0xFF) return false; if (!width || !height) return false; if (!destPt) return false; buf = (sbyte*)malloc(width*height*sizeof(BYTE)); { matrix = effect->EffectMatrix.matrix; mcTopStart = 2-effect->EffectMatrix.topEffect; mcBottomEnd = 3+effect->EffectMatrix.bottomEffect; mcLeftStart = 2-effect->EffectMatrix.leftEffect; mcRightEnd = 3+effect->EffectMatrix.rightEffect; effectCount = effect->EffectMatrix.cycleCount; } al = 255-((BYTE)(effect->EffectColor1>>24)); rl = GetRValue(effect->EffectColor1); gl = GetGValue(effect->EffectColor1); bl = GetBValue(effect->EffectColor1); ad = 255-((BYTE)(effect->EffectColor2>>24)); rd = GetRValue(effect->EffectColor2); gd = GetGValue(effect->EffectColor2); bd = GetBValue(effect->EffectColor2); //Fill buffer by mid values of image for (y = 0; y < height; y++) { bufline = buf+y*width; maskline = maskPt+((y*width) << 2); for (x = 0; x < width; x++) { BYTE a = (sbyte)(DWORD)((maskline[0]+maskline[2]+maskline[1]+maskline[1])>>4); *bufline = a; if (a != 0) { minX = min((int)x,minX); minY = min((int)y,minY); maxX = max((int)x,maxX); maxY = max((int)y,maxY); } bufline++; maskline += 4; } } //Here perform effect on buffer and place results to outbuf for (k = 0; k < (effectCount&0x7F); k++) { minX = max( 0, minX + mcLeftStart - 2 ); minY = max( 0, minY + mcTopStart - 2 ); maxX = min((int)width, maxX + mcRightEnd - 1 ); maxY = min((int)height, maxY + mcBottomEnd - 1 ); outbuf = (sbyte*)malloc(width*height*sizeof(sbyte)); memset(outbuf, 0, width*height*sizeof(sbyte)); for (y = (DWORD)minY; y < (DWORD)maxY; y++) { int val; bufline = outbuf+y*width+minX; buflineMid = buf+y*width+minX; for (x = (DWORD)minX; x < (DWORD)maxX; x++) { int matrixHor,matrixVer; val = 0; for (matrixVer = mcTopStart; matrixVer < mcBottomEnd; matrixVer++) { int buflineStep = width*(matrixVer-2); int as = y+matrixVer-2; sbyte * buflineTopS = NULL; if (as >= 0 && (DWORD)as < height) buflineTopS = buflineMid+buflineStep; for (matrixHor = mcLeftStart; matrixHor < mcRightEnd;matrixHor++) { buflineTop = buflineTopS; int a = x+matrixHor-2; if (buflineTop && a >= 0 && (DWORD)a < width) buflineTop += matrixHor-2; else buflineTop = NULL; if (buflineTop) val += ((*buflineTop)*matrix[matrixVer*5+matrixHor]); } } val = (val+1)>>5; *bufline = (sbyte)((val>127)?127:(val < -125)?-125:val); bufline++; buflineMid++; } } free(buf); buf = outbuf; } { BYTE r1,b1,g1,a1; b1 = bl; r1 = rl; g1 = gl; a1 = al; sign = 1; //perform out to dest for (y = 0; y < height; y++) { bufline = buf+y*width; destline = destPt+((y*width) << 2); for (x = 0; x < width; x++) { sbyte val = *bufline; BYTE absVal = ((val < 0)?-val:val); if (val != 0) { if (val>0 && sign < 0) { b1 = bl; r1 = rl; g1 = gl; a1 = al; sign = 1;} else if (val < 0 && sign>0) { b1 = bd; r1 = rd; g1 = gd; a1 = ad; sign = -1;} absVal = absVal*a1/255; destline[0] = ((destline[0]*(128-absVal))+absVal*b1)>>7; destline[1] = ((destline[1]*(128-absVal))+absVal*g1)>>7; destline[2] = ((destline[2]*(128-absVal))+absVal*r1)>>7; destline[3] += ((255-destline[3])*(a1*absVal))/32640; } bufline++; destline += 4; } } free(buf); } return false; } static int ske_AlphaTextOut (HDC hDC, LPCTSTR lpString, int nCount, RECT *lpRect, UINT format, DWORD ARGBcolor) { if (!(lpString && lpRect)) return 0; // Step first fill fast calc correction tables: static bool _tables_empty = true; static BYTE gammaTbl[256]; // Gamma correction table static WORD blueMulTbl[256]; // blue coefficient multiplication table static WORD greenMulTbl[256]; // green coefficient multiplication table static WORD redMulTbl[256]; // red coefficient multiplication table if ( _tables_empty ) { // fill tables double gammaCfPw = 1000 / (double)DBGetContactSettingRangedWord(NULL,"ModernData","AlphaTextOutGamma", 700, 1, 5000 ); BYTE blueCf = db_get_b(NULL,"ModernData","AlphaTextOutBlueCorrection", 28 ); BYTE redCf = db_get_b(NULL,"ModernData","AlphaTextOutRed Correction", 77 ); BYTE greenCf = db_get_b(NULL,"ModernData","AlphaTextOutGreen Correction", 151 ); for ( int i=0; i < 256; i++ ) { gammaTbl[i] = (BYTE)( 255 * pow((double)i / 255, gammaCfPw )); blueMulTbl[i] = i * blueCf; redMulTbl[i] = i * redCf; greenMulTbl[i] = i * greenCf; } } // Calc len of input string if (nCount == -1) nCount = lstrlen(lpString); // retrieve destination bitmap bits HBITMAP hDestBitmap = (HBITMAP)GetCurrentObject(hDC, OBJ_BITMAP); BITMAP bmpDest; GetObject(hDestBitmap, sizeof(BITMAP), &bmpDest); BOOL destHasNotDIB = FALSE; BYTE *pDestBits = NULL; if (bmpDest.bmBits == NULL) { destHasNotDIB = TRUE; pDestBits = (BYTE*) malloc ( bmpDest.bmHeight * bmpDest.bmWidthBytes ); GetBitmapBits( hDestBitmap, bmpDest.bmHeight*bmpDest.bmWidthBytes, pDestBits ); } else pDestBits = (BYTE*)bmpDest.bmBits; BOOL isDest16bit = (bmpDest.bmBitsPixel) != 32; // Creating offscreen buffer HDC hOffscreenDC = CreateCompatibleDC(hDC); // Font to be used to draw text HFONT hFont = (HFONT)GetCurrentObject(hDC, OBJ_FONT); HFONT hOldOffscreenFont = (HFONT)SelectObject(hOffscreenDC, hFont); // Calculating text geometric size RECT workRect = *lpRect; int workRectWidth = workRect.right - workRect.left; int workRectHeight = workRect.bottom - workRect.top; if (workRectWidth <= 0 || workRectHeight <= 0) return 0; SIZE textSize; GetTextExtentPoint32( hOffscreenDC, lpString, nCount, &textSize ); LPCTSTR lpWorkString = lpString; BOOL bNeedFreeWorkString = FALSE; // if we need to cut the text with ellipsis if ((format & DT_END_ELLIPSIS) && textSize.cx > workRectWidth) { // Calc geometric width of ellipsis SIZE szEllipsis; GetTextExtentPoint32A(hOffscreenDC,"...",3,&szEllipsis); szEllipsis.cx++; // CORRECTION: some width correction // Calc count of visible chars int visibleCharCount = nCount; if (workRectWidth > szEllipsis.cx) GetTextExtentExPoint(hOffscreenDC, lpString, nCount, workRectWidth - szEllipsis.cx, &visibleCharCount, NULL, &textSize); else GetTextExtentExPoint(hOffscreenDC, lpString, nCount, 0, &visibleCharCount, NULL, &textSize); // replace end of string by elipsis bNeedFreeWorkString = TRUE; lpWorkString = (TCHAR*) malloc(( visibleCharCount + 4) * sizeof(TCHAR)); memcpy((void*) lpWorkString, lpString, visibleCharCount * sizeof(TCHAR)); memcpy((void*) ( lpWorkString + visibleCharCount ), _T("..."), 4 * sizeof(TCHAR)); // 3 + 1 nCount = visibleCharCount + 3; } // Calc sizes and offsets textSize.cx += 2; // CORRECTION: for italic int drx = 0; // x-axis offset of draw point if (workRectWidth > textSize.cx) { if ( format & ( DT_RIGHT | DT_RTLREADING )) drx = workRectWidth - textSize.cx; else if ( format & DT_CENTER ) drx = ( workRectWidth - textSize.cx ) >> 1; } else textSize.cx = workRectWidth; int dry = 0; // y-axis offset of draw point if (workRectHeight > textSize.cy) { if (format & DT_BOTTOM) dry = workRectHeight - textSize.cy; else if (format & DT_VCENTER) dry = (workRectHeight - textSize.cy) >> 1; } else textSize.cy = workRectHeight; textSize.cx += 4; // CORRECTION: for effects ??? textSize.cy += 4; // CORRECTION: for effects ??? if (textSize.cx > 0 && textSize.cy > 0) { // Ok we need to paint // probably here are mess ofscreen and temp buff dc //Create bitmap image for offscreen BYTE *bits = NULL; HBITMAP hbmp = ske_CreateDIB32Point(textSize.cx, textSize.cy, (void**)&bits); if (bits != NULL) { HBITMAP holdbmp = (HBITMAP)SelectObject( hOffscreenDC, hbmp ); //Create buffer bitmap image for temp text BYTE *bufbits = NULL; HBITMAP bufbmp = ske_CreateDIB32Point(textSize.cx, textSize.cy, (void**)&bufbits); if ( bufbits != NULL ) { HDC bufDC = CreateCompatibleDC(hDC); HBITMAP bufoldbmp = (HBITMAP)SelectObject(bufDC, bufbmp); HFONT hOldBufFont = (HFONT)SelectObject(bufDC, hFont); SetBkColor(bufDC, RGB(0, 0, 0)); SetTextColor(bufDC, RGB(255, 255, 255)); // Copy from destination to temp buffer BitBlt(hOffscreenDC, 0, 0, textSize.cx, textSize.cy, hDC, workRect.left + drx - 2, workRect.top + dry - 2, SRCCOPY); //Draw text on offscreen bitmap TextOut(bufDC, 2, 2, lpWorkString, nCount); MODERNEFFECT effect; if (ske_GetTextEffect(hDC, &effect)) ske_DrawTextEffect(bits, bufbits, textSize.cx, textSize.cy, &effect); // RenderText RECT drawRect; drawRect.left = 0; drawRect.top = 0; drawRect.right = textSize.cx; drawRect.bottom = textSize.cy; DWORD width = textSize.cx; DWORD heigh = textSize.cy; BYTE *pDestScanLine; BYTE *pBufScanLine; BYTE *pix; BYTE *bufpix; BYTE al = 255 - ((BYTE)( ARGBcolor >> 24 )); BYTE r = GetRValue( ARGBcolor ); BYTE g = GetGValue( ARGBcolor ); BYTE b = GetBValue( ARGBcolor ); for (DWORD y = 2; y < heigh - 2; y++) { int lineBytes = y * (width << 2); pDestScanLine = bits + lineBytes; pBufScanLine = bufbits + lineBytes; for (DWORD x = 2; x < width - 2; x++) { pix = pDestScanLine + ( x << 2 ); bufpix = pBufScanLine + ( x << 2 ); // Monochromatic BYTE bx = gammaTbl[ bufpix[0] ]; BYTE gx = gammaTbl[ bufpix[1] ]; BYTE rx = gammaTbl[ bufpix[2] ]; if ( al != 255 ) { bx *= al/255; gx *= al/255; rx *= al/255; } BYTE ax = (BYTE)(( (DWORD)rx*77 + (DWORD)gx * 151 + (DWORD)bx *28 + 128 ) / 256 ); if (ax) { //Normalize components to gray BYTE axx = 255 - (( r + g + b ) >> 2 ) ; // Coefficient of grayance, more white font - more gray edges WORD atx = ax * (255 - axx); bx = ( atx + bx * axx )/255; gx = ( atx + gx * axx )/255; rx = ( atx + rx * axx )/255; short brx = (short)((b - pix[0])*bx / 255); short grx = (short)((g - pix[1])*gx / 255); short rrx = (short)((r - pix[2])*rx / 255); pix[0] += brx; pix[1] += grx; pix[2] += rrx; pix[3] = (BYTE)(ax+(BYTE)(255-ax)*pix[3]/255); } } } // Blit to destination BitBlt(hDC, workRect.left + drx - 2, workRect.top + dry - 2, textSize.cx, textSize.cy, hOffscreenDC, 0, 0, SRCCOPY); //free resources SelectObject(bufDC, bufoldbmp); DeleteObject(bufbmp); SelectObject(bufDC, hOldBufFont); DeleteDC(bufDC); } SelectObject(hOffscreenDC, holdbmp); DeleteObject(hbmp); } } // Final cleanup SelectObject(hOffscreenDC, hOldOffscreenFont); DeleteDC(hOffscreenDC); if (destHasNotDIB) free(pDestBits); if (bNeedFreeWorkString) free((void*)lpWorkString); return 0; } static int ske_DrawTextWithEffectWorker( HDC hdc, LPCTSTR lpString, int nCount, RECT *lpRect, UINT format, MODERNFONTEFFECT * effect ) { if (format & DT_CALCRECT) return DrawText(hdc, lpString, nCount, lpRect, format); if (format & DT_RTLREADING) SetTextAlign(hdc, TA_RTLREADING); DWORD color = GetTextColor(hdc); RECT r = *lpRect; OffsetRect(&r, 1, 1); DWORD form = format; if (effect && effect->effectIndex) ske_SelectTextEffect(hdc, effect->effectIndex - 1, effect->baseColour, effect->secondaryColour); int res = ske_AlphaTextOut(hdc, lpString, nCount, lpRect, form, color); if (effect && effect->effectIndex) ske_ResetTextEffect(hdc); return res; } INT_PTR ske_Service_DrawTextWithEffect(WPARAM wParam, LPARAM lParam) { DrawTextWithEffectParam *p = (DrawTextWithEffectParam *)wParam; if (p->cbSize != sizeof(DrawTextWithEffectParam)) return FALSE; return ske_DrawTextWithEffectWorker(p->hdc, p->lpchText, p->cchText, p->lprc, p->dwDTFormat, p->pEffect); } BOOL ske_DrawText(HDC hdc, LPCTSTR lpString, int nCount, RECT *lpRect, UINT format) { DWORD form = 0, color = 0; RECT r = *lpRect; OffsetRect(&r, 1, 1); if (format & DT_RTLREADING) SetTextAlign(hdc, TA_RTLREADING); if (format & DT_CALCRECT) return DrawText(hdc, lpString, nCount, lpRect, format); if (format & DT_FORCENATIVERENDER || g_CluiData.fDisableSkinEngine) return DrawText(hdc, lpString, nCount, lpRect, format&~DT_FORCENATIVERENDER); form = format; color = GetTextColor(hdc); return ske_AlphaTextOut(hdc, lpString, nCount, lpRect, form, color); } HICON ske_ImageList_GetIcon(HIMAGELIST himl, int i, UINT fStyle) { IMAGEINFO imi = {0}; BITMAP bm = {0}; if (i != -1) { ImageList_GetImageInfo(himl,i,&imi); GetObject(imi.hbmImage,sizeof(bm),&bm); if (bm.bmBitsPixel == 32) //stupid bug of Microsoft // Icons bitmaps are not premultiplied // So Imagelist_AddIcon - premultiply alpha // But incorrect - it is possible that alpha will // be less than color and // ImageList_GetIcon will return overflowed colors // TODO: Direct draw Icon from imagelist without // extracting of icon { BYTE *bits = (BYTE*)bm.bmBits; if (!bits) { bits = (BYTE*)malloc(bm.bmWidthBytes*bm.bmHeight); GetBitmapBits(imi.hbmImage,bm.bmWidthBytes*bm.bmHeight,bits); } int wb = ((imi.rcImage.right - imi.rcImage.left)*bm.bmBitsPixel >> 3); BYTE *bcbits = bits + (bm.bmHeight - imi.rcImage.bottom)*bm.bmWidthBytes + (imi.rcImage.left*bm.bmBitsPixel >> 3); for (int iy = 0; iy < imi.rcImage.bottom - imi.rcImage.top; iy++) { int x; // Dummy microsoft fix - alpha can be less than r,g or b // Looks like color channels in icons should be non-premultiplied with alpha // But AddIcon store it premultiplied (incorrectly cause can be Alpha == 7F, but R,G or B == 80 // So i check that alpha is 0x7F and set it to 0x80 DWORD *c = ((DWORD*)bcbits); for (x = 0; x < imi.rcImage.right - imi.rcImage.left; x++) { DWORD val = *c; BYTE a = (BYTE)((val) >> 24); if (a != 0) { BYTE r = (BYTE)((val & 0xFF0000) >> 16); BYTE g = (BYTE)((val & 0xFF00) >> 8); BYTE b = (BYTE)(val & 0xFF); if (a < r || a < g || a < b) { a = max(max(r, g), b); val = a << 24 | r << 16 | g << 8 | b; *c = val; } } c++; } bcbits += bm.bmWidthBytes; } if (!bm.bmBits) { SetBitmapBits(imi.hbmImage, bm.bmWidthBytes*bm.bmHeight, bits); free(bits); } } } return ImageList_GetIcon(himl,i,ILD_NORMAL); } //////////////////////////////////////////////////////////////////////////// // This creates new dib image from Imagelist icon ready to alphablend it HBITMAP ske_ExtractDIBFromImagelistIcon( HIMAGELIST himl,int index, int * outWidth, int * outHeight) { return NULL; } BOOL ske_ImageList_DrawEx( HIMAGELIST himl,int i,HDC hdcDst,int x,int y,int dx,int dy,COLORREF rgbBk,COLORREF rgbFg,UINT fStyle) { //the routine to directly draw icon from image list without creating icon from there - should be some faster if (i < 0) return FALSE; if (g_CluiData.fDisableSkinEngine) return ImageList_DrawEx(himl, i, hdcDst, x, y, dx, dy, rgbBk, rgbFg, fStyle); BYTE alpha; if (fStyle&ILD_BLEND25) alpha = 64; else if (fStyle&ILD_BLEND50) alpha = 128; else alpha = 255; HICON hIcon = ske_ImageList_GetIcon(himl, i, ILD_NORMAL); if (hIcon == NULL) return FALSE; ske_DrawIconEx(hdcDst, x, y, hIcon, dx ? dx : GetSystemMetrics(SM_CXSMICON), dy ? dy : GetSystemMetrics(SM_CYSMICON), 0, NULL, DI_NORMAL | (alpha << 24)); DestroyIcon(hIcon); return TRUE; } static INT_PTR ske_Service_DrawIconEx(WPARAM wParam,LPARAM lParam) { DrawIconFixParam *p = (DrawIconFixParam*)wParam; if (!p) return 0; return ske_DrawIconEx(p->hdc,p->xLeft,p->yTop,p->hIcon,p->cxWidth,p->cyWidth,p->istepIfAniCur,p->hbrFlickerFreeDraw,p->diFlags); } BOOL ske_DrawIconEx(HDC hdcDst,int xLeft,int yTop,HICON hIcon,int cxWidth,int cyWidth, UINT istepIfAniCur, HBRUSH hbrFlickerFreeDraw, UINT diFlags) { ICONINFO ici; BYTE alpha = (BYTE)((diFlags&0xFF000000)>>24); HDC imDC; HBITMAP oldBmp, imBmp,tBmp; BITMAP imbt,immaskbt; BYTE * imbits; BYTE * imimagbits; BYTE * immaskbits; DWORD cx,cy,icy; BYTE *t1, *t2, *t3; BOOL NoDIBImage = FALSE; //lockimagelist BYTE hasmask = FALSE; BYTE no32bit = FALSE; BYTE noMirrorMask = FALSE; BYTE hasalpha = FALSE; alpha = alpha?alpha:255; if ( g_CluiData.fDisableSkinEngine && !(diFlags&0x80)) return DrawIconEx(hdcDst,xLeft,yTop,hIcon,cxWidth,cyWidth,istepIfAniCur,hbrFlickerFreeDraw,diFlags&0xFFFF7F); if (!GetIconInfo(hIcon, &ici)) return 0; GetObject(ici.hbmColor,sizeof(BITMAP),&imbt); if (imbt.bmWidth*imbt.bmHeight == 0) { DeleteObject(ici.hbmColor); DeleteObject(ici.hbmMask); return 0; } GetObject(ici.hbmMask,sizeof(BITMAP),&immaskbt); cy = imbt.bmHeight; if (imbt.bmBitsPixel != 32) { HBITMAP otBmp; no32bit = TRUE; HDC tempDC1 = CreateCompatibleDC(hdcDst); tBmp = ske_CreateDIB32(imbt.bmWidth, imbt.bmHeight); if (tBmp) { GetObject(tBmp, sizeof(BITMAP), &imbt); otBmp = (HBITMAP)SelectObject(tempDC1, tBmp); DrawIconEx(tempDC1, 0, 0, hIcon, imbt.bmWidth, imbt.bmHeight, istepIfAniCur, hbrFlickerFreeDraw, DI_IMAGE); noMirrorMask = TRUE; } SelectObject(tempDC1, otBmp); DeleteDC(tempDC1); } if (imbt.bmBits == NULL) { NoDIBImage = TRUE; imimagbits = (BYTE*)malloc(cy*imbt.bmWidthBytes); GetBitmapBits(ici.hbmColor, cy*imbt.bmWidthBytes, (void*)imimagbits); } else imimagbits = (BYTE*)imbt.bmBits; if (immaskbt.bmBits == NULL) { immaskbits = (BYTE*)malloc(cy*immaskbt.bmWidthBytes); GetBitmapBits(ici.hbmMask, cy*immaskbt.bmWidthBytes, (void*)immaskbits); } else immaskbits = (BYTE*)immaskbt.bmBits; icy = imbt.bmHeight; cx = imbt.bmWidth; imDC = CreateCompatibleDC(hdcDst); imBmp = ske_CreateDIB32Point(cx, icy, (void**)&imbits); oldBmp = (HBITMAP)SelectObject(imDC, imBmp); if (imbits != NULL && imimagbits != NULL && immaskbits != NULL) { int x; int y; int bottom, right, top, h; int mwb, mwb2; mwb = immaskbt.bmWidthBytes; mwb2 = imbt.bmWidthBytes; bottom = icy; right = cx; top = 0; h = icy; for (y = top; (y < bottom) && !hasmask; y++) { t1 = immaskbits + y*mwb; for (x = 0; (x < mwb) && !hasmask; x++) hasmask |= (*(t1 + x) != 0); } for (y = top; (y < bottom) && !hasalpha; y++) { t1 = imimagbits + (cy - y - 1)*mwb2; for (x = 0; (x < right) && !hasalpha; x++) hasalpha |= (*(t1+(x << 2)+3) != 0); } for (y = 0; y < (int)icy; y++) { t1 = imimagbits+(h-y-1-top)*mwb2; t2 = imbits+(!no32bit?y:(icy-y-1))*mwb2; t3 = immaskbits+(noMirrorMask?y:(h-y-1-top))*mwb; for (x = 0; x < right; x++) { BYTE mask = 0; BYTE a = 0; DWORD *src = (DWORD*)(t1 + (x << 2)); DWORD *dest = (DWORD*)(t2 + (x << 2)); if (hasalpha && !hasmask) a = ((BYTE*)src)[3]; else { mask = ((1 << (7 - x % 8))&(*(t3 + (x >> 3)))) != 0; if (mask) { if (!hasalpha) { *dest = 0; continue; } if (((BYTE*)src)[3]>0 ) a = ((BYTE*)src)[3]; else a = 0; } else if (hasalpha || hasmask) a = (((BYTE*)src)[3]>0?((BYTE*)src)[3]:255); else if (!hasalpha && !hasmask) a = 255; else { *dest = 0; continue; } } if (a > 0) { ((BYTE*)dest)[3] = a; ((BYTE*)dest)[0] = ((BYTE*)src)[0] * a / 255; ((BYTE*)dest)[1] = ((BYTE*)src)[1] * a / 255; ((BYTE*)dest)[2] = ((BYTE*)src)[2] * a / 255; } else *dest = 0; } } } BLENDFUNCTION bf = { AC_SRC_OVER, diFlags & 128, alpha, AC_SRC_ALPHA }; ske_AlphaBlend(hdcDst, xLeft, yTop, cxWidth, cyWidth, imDC, 0, 0, cx, icy, bf); if (immaskbt.bmBits == NULL) free(immaskbits); if (imbt.bmBits == NULL) free(imimagbits); SelectObject(imDC, oldBmp); DeleteObject(imBmp); if (no32bit)DeleteObject(tBmp); DeleteObject(ici.hbmColor); DeleteObject(ici.hbmMask); SelectObject(imDC, GetStockObject(DEFAULT_GUI_FONT)); DeleteDC(imDC); return 1; } int ske_PrepareImageButDontUpdateIt(RECT *r) { if (!g_CluiData.fLayered) return ske_ReCreateBackImage(FALSE, r); mutex_bLockUpdate = 1; ske_DrawNonFramedObjects(TRUE, r); ske_ValidateFrameImageProc(r); mutex_bLockUpdate = 0; return 0; } int ske_RedrawCompleteWindow() { if (g_CluiData.fLayered) { ske_DrawNonFramedObjects(TRUE,0); CallService(MS_SKINENG_INVALIDATEFRAMEIMAGE, 0, 0); } else RedrawWindow(pcli->hwndContactList,NULL,NULL,RDW_ALLCHILDREN|RDW_ERASE|RDW_INVALIDATE|RDW_FRAME); return 0; } // Request to repaint frame or change/drop callback data // wParam = hWnd of called frame // lParam = pointer to sPaintRequest (or NULL to redraw all) // return 2 - already queued, data updated, 1-have been queued, 0 - failure static INT_PTR ske_Service_UpdateFrameImage(WPARAM wParam, LPARAM lParam) // Immideately recall paint routines for frame and refresh image { if ( MirandaLoading()) return 0; RECT wnd; FRAMEWND *frm; BOOL NoCancelPost = 0; BOOL IsAnyQueued = 0; if (!g_CluiData.mutexOnEdgeSizing) GetWindowRect(pcli->hwndContactList,&wnd); else wnd = g_rcEdgeSizingRect; if (!g_CluiData.fLayered) { RedrawWindow((HWND)wParam,NULL,NULL,RDW_UPDATENOW|RDW_ERASE|RDW_INVALIDATE|RDW_FRAME); return 0; } if (g_pCachedWindow == NULL) ske_ValidateFrameImageProc(&wnd); else if (g_pCachedWindow->Width != wnd.right-wnd.left || g_pCachedWindow->Height != wnd.bottom-wnd.top) ske_ValidateFrameImageProc(&wnd); else if (wParam == 0) ske_ValidateFrameImageProc(&wnd); else { // all Ok Update Single Frame // TO BE LOCKED OR PROXIED frm = FindFrameByItsHWND((HWND)wParam); if (!frm) ske_ValidateFrameImageProc(&wnd); // Validate frame, update window image and remove it from queue else { if (frm->UpdateRgn) { DeleteObject(frm->UpdateRgn); frm->UpdateRgn = 0; } ske_ValidateSingleFrameImage(frm, 0); ske_UpdateWindowImage(); NoCancelPost = 1; //-- Remove frame from queue if (flag_bUpdateQueued) { frm->bQueued = 0; for (int i = 0; i < g_nFramesCount; i++) if (IsAnyQueued |= g_pfwFrames[i].bQueued) break; } } } if ((!NoCancelPost || !IsAnyQueued) && flag_bUpdateQueued) { // no any queued updating cancel post or need to cancel post flag_bUpdateQueued = 0; g_flag_bPostWasCanceled = 1; } return 1; } static INT_PTR ske_Service_InvalidateFrameImage(WPARAM wParam, LPARAM lParam) // Post request for updating { if (MirandaLoading()) return 0; if (wParam) { FRAMEWND *frm = FindFrameByItsHWND((HWND)wParam); sPaintRequest *pr = (sPaintRequest*)lParam; if (!g_CluiData.fLayered || (frm && frm->floating)) return InvalidateRect((HWND)wParam, pr ? (RECT*)&(pr->rcUpdate) : NULL, FALSE); if (frm) { if (frm->PaintCallbackProc != NULL) { frm->PaintData = (sPaintRequest *)pr; frm->bQueued = 1; if (pr) { HRGN r2; if (!IsRectEmpty(&pr->rcUpdate)) { RECT rcClient; RECT rcUpdate; GetClientRect(frm->hWnd, &rcClient); IntersectRect(&rcUpdate, &rcClient, &pr->rcUpdate); if (IsRectEmpty(&rcUpdate)) return 0; r2 = CreateRectRgn(rcUpdate.left, rcUpdate.top, rcUpdate.right, rcUpdate.bottom); } else { RECT r; GetClientRect(frm->hWnd, &r); r2 = CreateRectRgn(r.left, r.top, r.right, r.bottom); } if (!frm->UpdateRgn) { frm->UpdateRgn = CreateRectRgn(0, 0, 1, 1); CombineRgn(frm->UpdateRgn, r2, 0, RGN_COPY); } else CombineRgn(frm->UpdateRgn, frm->UpdateRgn, r2, RGN_OR); DeleteObject(r2); } } } else Sync(QueueAllFramesUpdating, 1); } else Sync(QueueAllFramesUpdating, 1); if (!flag_bUpdateQueued || g_flag_bPostWasCanceled) if (PostMessage(pcli->hwndContactList, UM_UPDATE, 0, 0)) { flag_bUpdateQueued = 1; g_flag_bPostWasCanceled = 0; } return 1; } static int ske_ValidateSingleFrameImage(FRAMEWND * Frame, BOOL SkipBkgBlitting) // Calling frame paint proc { if (!g_pCachedWindow) { TRACE("ske_ValidateSingleFrameImage calling without cached\n"); return 0;} if (Frame->hWnd == (HWND)-1 && !Frame->PaintCallbackProc) { TRACE("ske_ValidateSingleFrameImage calling without FrameProc\n"); return 0;} // if ok update image HDC hdc; HBITMAP o,n; RECT rcPaint,wnd; RECT ru = {0}; int w,h,x,y; int w1,h1,x1,y1; CLUI_SizingGetWindowRect(pcli->hwndContactList,&wnd); rcPaint = Frame->wndSize; { int dx,dy,bx,by; if (g_CluiData.mutexOnEdgeSizing) { dx = rcPaint.left-wnd.left; dy = rcPaint.top-wnd.top; bx = rcPaint.right-wnd.right; by = rcPaint.bottom-wnd.bottom; wnd = g_rcEdgeSizingRect; rcPaint.left = wnd.left+dx; rcPaint.top = wnd.top+dy; rcPaint.right = wnd.right+bx; rcPaint.bottom = wnd.bottom+by; } } //OffsetRect(&rcPaint,-wnd.left,-wnd.top); w = rcPaint.right-rcPaint.left; h = rcPaint.bottom-rcPaint.top; if (w <= 0 || h <= 0) { TRACE("Frame size smaller than 0\n"); return 0; } x = rcPaint.left; y = rcPaint.top; hdc = CreateCompatibleDC(g_pCachedWindow->hImageDC); n = ske_CreateDIB32(w,h); o = (HBITMAP)SelectObject(hdc,n); { if (Frame->UpdateRgn && !SkipBkgBlitting) { GetRgnBox(Frame->UpdateRgn,&ru); { RECT rc; GetClientRect(Frame->hWnd,&rc); if (ru.top < 0) ru.top = 0; if (ru.left < 0) ru.left = 0; if (ru.right>rc.right) ru.right = rc.right; if (ru.bottom>rc.bottom) ru.bottom = rc.bottom; } if (!IsRectEmpty(&ru)) { x1 = ru.left; y1 = ru.top; w1 = ru.right-ru.left; h1 = ru.bottom-ru.top; } else {x1 = 0; y1 = 0; w1 = w; h1 = h;} // copy image at hdc if (SkipBkgBlitting) //image already at foreground { BitBlt(hdc,x1,y1,w1,h1,g_pCachedWindow->hImageDC,x+x1,y+y1,SRCCOPY); } else { BitBlt(hdc,x1,y1,w1,h1,g_pCachedWindow->hBackDC,x+x1,y+y1,SRCCOPY); } Frame->PaintCallbackProc(Frame->hWnd,hdc,&ru,Frame->UpdateRgn, Frame->dwFlags,Frame->PaintData); } else { HRGN rgnUpdate; RECT r; GetClientRect(Frame->hWnd,&r); rgnUpdate = CreateRectRgn(r.left,r.top,r.right,r.bottom); ru = r; if (!IsRectEmpty(&ru)) { x1 = ru.left; y1 = ru.top; w1 = ru.right-ru.left; h1 = ru.bottom-ru.top; } else {x1 = 0; y1 = 0; w1 = w; h1 = h;} // copy image at hdc if (SkipBkgBlitting) //image already at foreground { BitBlt(hdc,x1,y1,w1,h1,g_pCachedWindow->hImageDC,x+x1,y+y1,SRCCOPY); } else { BitBlt(hdc,x1,y1,w1,h1,g_pCachedWindow->hBackDC,x+x1,y+y1,SRCCOPY); } Frame->PaintCallbackProc(Frame->hWnd,hdc,&r,rgnUpdate, Frame->dwFlags,Frame->PaintData); ru = r; DeleteObject(rgnUpdate); } DeleteObject(Frame->UpdateRgn); Frame->UpdateRgn = 0; } if (!IsRectEmpty(&ru)) { x1 = ru.left; y1 = ru.top; w1 = ru.right-ru.left; h1 = ru.bottom-ru.top; } else { x1 = 0; y1 = 0; w1 = w; h1 = h; } BitBlt(g_pCachedWindow->hImageDC,x+x1,y+y1,w1,h1,hdc,x1,y1,SRCCOPY); if (GetWindowLongPtr(Frame->hWnd, GWL_STYLE) & WS_VSCROLL) { //Draw vertical scroll bar // SCROLLBARINFO si = {0}; si.cbSize = sizeof(SCROLLBARINFO); GetScrollBarInfo(Frame->hWnd, OBJID_VSCROLL, &si); RECT rLine = (si.rcScrollBar); RECT rUpBtn = rLine; RECT rDnBtn = rLine; RECT rThumb = rLine; rUpBtn.bottom = rUpBtn.top+si.dxyLineButton; rDnBtn.top = rDnBtn.bottom-si.dxyLineButton; rThumb.top = rLine.top+si.xyThumbTop; rThumb.bottom = rLine.top+si.xyThumbBottom; int dx = Frame->wndSize.right-rLine.right; int dy = -rLine.top + Frame->wndSize.top; OffsetRect(&rLine, dx, dy); OffsetRect(&rUpBtn, dx, dy); OffsetRect(&rDnBtn, dx, dy); OffsetRect(&rThumb, dx, dy); BitBlt(g_pCachedWindow->hImageDC, rLine.left, rLine.top, rLine.right - rLine.left, rLine.bottom - rLine.top, g_pCachedWindow->hBackDC, rLine.left, rLine.top, SRCCOPY); char req[255]; mir_snprintf(req, sizeof(req), "Main,ID=ScrollBar,Frame=%s,Part=Back", Frame->szName); SkinDrawGlyph(g_pCachedWindow->hImageDC,&rLine,&rLine,req); mir_snprintf(req, sizeof(req), "Main,ID=ScrollBar,Frame=%s,Part=Thumb", Frame->szName); SkinDrawGlyph(g_pCachedWindow->hImageDC,&rThumb,&rThumb,req); mir_snprintf(req, sizeof(req), "Main,ID=ScrollBar, Frame=%s,Part=UpLineButton", Frame->szName); SkinDrawGlyph(g_pCachedWindow->hImageDC,&rUpBtn,&rUpBtn,req); mir_snprintf(req, sizeof(req), "Main,ID=ScrollBar,Frame=%s,Part=DownLineButton", Frame->szName); SkinDrawGlyph(g_pCachedWindow->hImageDC,&rDnBtn,&rDnBtn,req); } SelectObject(hdc, o); DeleteObject(n); DeleteDC(hdc); return 1; } int ske_BltBackImage (HWND destHWND, HDC destDC, RECT *BltClientRect) { POINT ptMainWnd = {0}; POINT ptChildWnd = {0}; RECT from = {0}; RECT w = {0}; if (g_CluiData.fDisableSkinEngine) { FillRect(destDC,BltClientRect,GetSysColorBrush(COLOR_3DFACE)); return 0; } ske_ReCreateBackImage(FALSE,NULL); if (BltClientRect) w = *BltClientRect; else GetClientRect(destHWND,&w); ptChildWnd.x = w.left; ptChildWnd.y = w.top; ClientToScreen(destHWND,&ptChildWnd); ClientToScreen(pcli->hwndContactList,&ptMainWnd); //TODO if main not relative to client area return BitBlt(destDC,w.left,w.top,(w.right-w.left),(w.bottom-w.top),g_pCachedWindow->hBackDC,(ptChildWnd.x-ptMainWnd.x),(ptChildWnd.y-ptMainWnd.y),SRCCOPY); } int ske_ReCreateBackImage(BOOL Erase,RECT *w) { HBITMAP hb2; RECT wnd = {0}; BOOL IsNewCache = 0; if (g_CluiData.fDisableSkinEngine) return 0; GetClientRect(pcli->hwndContactList,&wnd); if (w) wnd = *w; //-- Check cached. if (g_pCachedWindow == NULL) { //-- Create New Cache g_pCachedWindow = (CURRWNDIMAGEDATA*)mir_calloc(sizeof(CURRWNDIMAGEDATA)); g_pCachedWindow->hScreenDC = GetDC(NULL); g_pCachedWindow->hBackDC = CreateCompatibleDC(g_pCachedWindow->hScreenDC); g_pCachedWindow->hImageDC = CreateCompatibleDC(g_pCachedWindow->hScreenDC); g_pCachedWindow->Width = wnd.right-wnd.left; g_pCachedWindow->Height = wnd.bottom-wnd.top; if (g_pCachedWindow->Width != 0 && g_pCachedWindow->Height != 0) { g_pCachedWindow->hImageDIB = ske_CreateDIB32Point(g_pCachedWindow->Width,g_pCachedWindow->Height,(void**)&(g_pCachedWindow->hImageDIBByte)); g_pCachedWindow->hBackDIB = ske_CreateDIB32Point(g_pCachedWindow->Width,g_pCachedWindow->Height,(void**)&(g_pCachedWindow->hBackDIBByte)); g_pCachedWindow->hImageOld = (HBITMAP)SelectObject(g_pCachedWindow->hImageDC,g_pCachedWindow->hImageDIB); g_pCachedWindow->hBackOld = (HBITMAP)SelectObject(g_pCachedWindow->hBackDC,g_pCachedWindow->hBackDIB); } IsNewCache = 1; } if (g_pCachedWindow->Width != wnd.right-wnd.left || g_pCachedWindow->Height != wnd.bottom-wnd.top) { HBITMAP hb1 = NULL,hb2 = NULL; g_pCachedWindow->Width = wnd.right-wnd.left; g_pCachedWindow->Height = wnd.bottom-wnd.top; if (g_pCachedWindow->Width != 0 && g_pCachedWindow->Height != 0) { hb1 = ske_CreateDIB32Point(g_pCachedWindow->Width,g_pCachedWindow->Height,(void**)&(g_pCachedWindow->hImageDIBByte)); hb2 = ske_CreateDIB32Point(g_pCachedWindow->Width,g_pCachedWindow->Height,(void**)&(g_pCachedWindow->hBackDIBByte)); SelectObject(g_pCachedWindow->hImageDC,hb1); SelectObject(g_pCachedWindow->hBackDC,hb2); } else { SelectObject(g_pCachedWindow->hImageDC,g_pCachedWindow->hImageOld); SelectObject(g_pCachedWindow->hBackDC,g_pCachedWindow->hBackOld); } if (g_pCachedWindow->hImageDIB) DeleteObject(g_pCachedWindow->hImageDIB); if (g_pCachedWindow->hBackDIB) DeleteObject(g_pCachedWindow->hBackDIB); g_pCachedWindow->hImageDIB = hb1; g_pCachedWindow->hBackDIB = hb2; IsNewCache = 1; } if ((Erase || IsNewCache ) && (g_pCachedWindow->Width != 0 && g_pCachedWindow->Height != 0)) { hb2 = ske_CreateDIB32(g_pCachedWindow->Width,g_pCachedWindow->Height); SelectObject(g_pCachedWindow->hBackDC,hb2); DeleteObject(g_pCachedWindow->hBackDIB); g_pCachedWindow->hBackDIB = hb2; FillRect(g_pCachedWindow->hBackDC,&wnd,GetSysColorBrush(COLOR_BTNFACE)); SkinDrawGlyph(g_pCachedWindow->hBackDC,&wnd,&wnd,"Main,ID=Background,Opt=Non-Layered"); ske_SetRectOpaque(g_pCachedWindow->hBackDC,&wnd); } return 1; } int ske_DrawNonFramedObjects(BOOL Erase,RECT *r) { RECT w, wnd; if (r) w = *r; else CLUI_SizingGetWindowRect(pcli->hwndContactList, &w); if (!g_CluiData.fLayered) return ske_ReCreateBackImage(FALSE, 0); if (g_pCachedWindow == NULL) return ske_ValidateFrameImageProc(&w); wnd = w; OffsetRect(&w, -w.left, -w.top); if (Erase) { HBITMAP hb2; hb2 = ske_CreateDIB32(g_pCachedWindow->Width, g_pCachedWindow->Height); SelectObject(g_pCachedWindow->hBackDC, hb2); DeleteObject(g_pCachedWindow->hBackDIB); g_pCachedWindow->hBackDIB = hb2; } SkinDrawGlyph(g_pCachedWindow->hBackDC, &w, &w, "Main,ID=Background"); //--Draw frames captions for (int i = 0; i < g_nFramesCount; i++) { if (g_pfwFrames[i].TitleBar.ShowTitleBar && g_pfwFrames[i].visible && !g_pfwFrames[i].floating) { RECT rc; SetRect(&rc, g_pfwFrames[i].wndSize.left, g_pfwFrames[i].wndSize.top - g_nTitleBarHeight - g_CluiData.nGapBetweenTitlebar, g_pfwFrames[i].wndSize.right, g_pfwFrames[i].wndSize.top - g_CluiData.nGapBetweenTitlebar); Sync(DrawTitleBar, g_pCachedWindow->hBackDC, &rc, g_pfwFrames[i].id); } } g_mutex_bLockUpdating = 1; flag_bJustDrawNonFramedObjects = 1; return 0; } int ske_ValidateFrameImageProc(RECT *r) // Calling queued frame paint procs and refresh image { RECT wnd = {0}; BOOL IsNewCache = 0; BOOL IsForceAllPainting = 0; if (r) wnd = *r; else GetWindowRect(pcli->hwndContactList,&wnd); if (wnd.right-wnd.left == 0 || wnd.bottom-wnd.top == 0) return 0; g_mutex_bLockUpdating = 1; //-- Check cached. if (g_pCachedWindow == NULL) { //-- Create New Cache g_pCachedWindow = (CURRWNDIMAGEDATA*)mir_calloc(sizeof(CURRWNDIMAGEDATA)); g_pCachedWindow->hScreenDC = GetDC(NULL); g_pCachedWindow->hBackDC = CreateCompatibleDC(g_pCachedWindow->hScreenDC); g_pCachedWindow->hImageDC = CreateCompatibleDC(g_pCachedWindow->hScreenDC); g_pCachedWindow->Width = wnd.right-wnd.left; g_pCachedWindow->Height = wnd.bottom-wnd.top; g_pCachedWindow->hImageDIB = ske_CreateDIB32Point(g_pCachedWindow->Width,g_pCachedWindow->Height,(void**)&(g_pCachedWindow->hImageDIBByte)); g_pCachedWindow->hBackDIB = ske_CreateDIB32Point(g_pCachedWindow->Width,g_pCachedWindow->Height,(void**)&(g_pCachedWindow->hBackDIBByte)); g_pCachedWindow->hImageOld = (HBITMAP)SelectObject(g_pCachedWindow->hImageDC,g_pCachedWindow->hImageDIB); g_pCachedWindow->hBackOld = (HBITMAP)SelectObject(g_pCachedWindow->hBackDC,g_pCachedWindow->hBackDIB); IsNewCache = 1; } if (g_pCachedWindow->Width != wnd.right-wnd.left || g_pCachedWindow->Height != wnd.bottom-wnd.top) { HBITMAP hb1,hb2; g_pCachedWindow->Width = wnd.right-wnd.left; g_pCachedWindow->Height = wnd.bottom-wnd.top; hb1 = ske_CreateDIB32Point(g_pCachedWindow->Width,g_pCachedWindow->Height,(void**)&(g_pCachedWindow->hImageDIBByte)); hb2 = ske_CreateDIB32Point(g_pCachedWindow->Width,g_pCachedWindow->Height,(void**)&(g_pCachedWindow->hBackDIBByte)); SelectObject(g_pCachedWindow->hImageDC,hb1); SelectObject(g_pCachedWindow->hBackDC,hb2); DeleteObject(g_pCachedWindow->hImageDIB); DeleteObject(g_pCachedWindow->hBackDIB); g_pCachedWindow->hImageDIB = hb1; g_pCachedWindow->hBackDIB = hb2; IsNewCache = 1; } if (IsNewCache) { ske_DrawNonFramedObjects(0, &wnd); IsForceAllPainting = 1; } if (flag_bJustDrawNonFramedObjects) { IsForceAllPainting = 1; flag_bJustDrawNonFramedObjects = 0; } if (IsForceAllPainting) { BitBlt(g_pCachedWindow->hImageDC, 0, 0, g_pCachedWindow->Width,g_pCachedWindow->Height,g_pCachedWindow->hBackDC, 0, 0, SRCCOPY); Sync( QueueAllFramesUpdating, (BYTE)1 ); } //-- Validating frames for (int i=0; i < g_nFramesCount; i++) if (g_pfwFrames[i].PaintCallbackProc && g_pfwFrames[i].visible && !g_pfwFrames[i].floating) if (g_pfwFrames[i].bQueued || IsForceAllPainting) ske_ValidateSingleFrameImage(&g_pfwFrames[i], IsForceAllPainting); g_mutex_bLockUpdating = 1; ModernSkinButtonRedrawAll(0); g_mutex_bLockUpdating = 0; if (!mutex_bLockUpdate) ske_UpdateWindowImageRect(&wnd); //-- Clear queue Sync(QueueAllFramesUpdating, 0); flag_bUpdateQueued = 0; g_flag_bPostWasCanceled = 0; return 1; } int ske_UpdateWindowImage() { if (MirandaExiting()) return 0; if (g_CluiData.fLayered) { RECT r; GetWindowRect(pcli->hwndContactList,&r); return ske_UpdateWindowImageRect(&r); } else ske_ReCreateBackImage(FALSE,0); ske_ApplyTranslucency(); return 0; } int ske_UpdateWindowImageRect(RECT *r) // Update window with current image and { //if not validity -> ValidateImageProc //else Update using current alpha RECT wnd = *r; if (!g_CluiData.fLayered) return ske_ReCreateBackImage(FALSE, 0); if (g_pCachedWindow == NULL) return ske_ValidateFrameImageProc(&wnd); if (g_pCachedWindow->Width != wnd.right-wnd.left || g_pCachedWindow->Height != wnd.bottom-wnd.top) return ske_ValidateFrameImageProc(&wnd); if (g_flag_bFullRepaint) { g_flag_bFullRepaint = 0; return ske_ValidateFrameImageProc(&wnd); } ske_JustUpdateWindowImageRect(&wnd); return 0; } void ske_ApplyTranslucency() { int IsTransparancy; HWND hwnd = pcli->hwndContactList; BOOL layered = (GetWindowLongPtr(hwnd, GWL_EXSTYLE) & WS_EX_LAYERED)?TRUE:FALSE; IsTransparancy = g_CluiData.fSmoothAnimation || g_bTransparentFlag; if (!g_bTransparentFlag && !g_CluiData.fSmoothAnimation && g_CluiData.bCurrentAlpha != 0) g_CluiData.bCurrentAlpha = 255; if (!g_CluiData.fLayered && IsTransparancy) { if (!layered) SetWindowLongPtr(hwnd, GWL_EXSTYLE, GetWindowLongPtr(hwnd, GWL_EXSTYLE) | WS_EX_LAYERED); SetLayeredWindowAttributes(hwnd, RGB(0, 0, 0), (BYTE)g_CluiData.bCurrentAlpha, LWA_ALPHA); } AniAva_RedrawAllAvatars(FALSE); return; } int ske_JustUpdateWindowImage() { RECT r; if (!g_CluiData.fLayered) { ske_ApplyTranslucency(); return 0; } GetWindowRect(pcli->hwndContactList,&r); return ske_JustUpdateWindowImageRect(&r); } // Update window image int ske_JustUpdateWindowImageRect(RECT *rty) { if (!g_CluiData.fLayered) { ske_ApplyTranslucency(); return 0; } if (!pcli->hwndContactList) return 0; RECT wnd = *rty; RECT rect = wnd; POINT dest = { 0 }, src = { 0 }; dest.x = rect.left; dest.y = rect.top; SIZE sz = { rect.right - rect.left, rect.bottom - rect.top }; if (g_CluiData.fLayered) { if (!(GetWindowLongPtr(pcli->hwndContactList, GWL_EXSTYLE)&WS_EX_LAYERED)) SetWindowLongPtr(pcli->hwndContactList, GWL_EXSTYLE, GetWindowLongPtr(pcli->hwndContactList, GWL_EXSTYLE) | WS_EX_LAYERED); Sync(SetAlpha, g_CluiData.bCurrentAlpha); BLENDFUNCTION bf = { AC_SRC_OVER, 0, g_CluiData.bCurrentAlpha, AC_SRC_ALPHA }; UpdateLayeredWindow(pcli->hwndContactList, g_pCachedWindow->hScreenDC, &dest, &sz, g_pCachedWindow->hImageDC, &src, RGB(1, 1, 1), &bf, ULW_ALPHA); g_CluiData.fAeroGlass = false; CLUI_UpdateAeroGlass(); } else InvalidateRect(pcli->hwndContactList,NULL,TRUE); return 0; } int ske_DrawImageAt(HDC hdc, RECT *rc) { BLENDFUNCTION bf = {AC_SRC_OVER, 0, 255, AC_SRC_ALPHA }; BitBlt(g_pCachedWindow->hImageDC,rc->left,rc->top,rc->right-rc->left,rc->bottom-rc->top,g_pCachedWindow->hBackDC,rc->left,rc->top,SRCCOPY); ske_AlphaBlend(g_pCachedWindow->hImageDC,rc->left,rc->top,rc->right-rc->left,rc->bottom-rc->top,hdc, 0, 0, rc->right-rc->left,rc->bottom-rc->top,bf); if (!g_mutex_bLockUpdating) ske_UpdateWindowImage(); return 0; } HBITMAP ske_GetCurrentWindowImage() { return g_pCachedWindow->hImageDIB; } /* * Glyph text routine */ static DWORD ske_HexToARGB(char * Hex) { char buf[10] = {0}; char buf2[11] = {0}; mir_snprintf(buf, SIZEOF(buf), "%s\n", Hex); if (buf[1] == 'x' || buf[1] == 'X') mir_snprintf(buf2, SIZEOF(buf2), "0x%s\n", buf+2); else mir_snprintf(buf2, SIZEOF(buf2), "0x%s\n", buf); buf2[10] = '\0'; char *st; DWORD AARRGGBB = strtoul(buf2, &st, 16); BYTE alpha = (BYTE)((AARRGGBB & 0xFF000000) >> 24); alpha = 255-((alpha == 0)?255:alpha); AARRGGBB = (alpha << 24)+((AARRGGBB&0x00FF0000)>>16)+((AARRGGBB&0x000000FF) << 16)+(AARRGGBB&0x0000FF00); return AARRGGBB; } static TCHAR *ske_ReAppend(TCHAR *lfirst, TCHAR * lsecond, int len) { int l1 = lfirst?lstrlen(lfirst):0; int l2 = (len?len:(lstrlen(lsecond)+1)); TCHAR *buf = (TCHAR *)mir_alloc((l1+l2+1)*sizeof(TCHAR)); if (lfirst) memmove(buf,lfirst,l1*sizeof(TCHAR)); memmove(buf+l1,lsecond,l2*sizeof(TCHAR)); mir_free(lfirst); if (len) buf[l1+l2] = _T('\0'); return buf; } TCHAR* ske_ReplaceVar(TCHAR *var) { if (!var) return mir_tstrdup(_T("")); if (!lstrcmpi(var,_T("Profile"))) { char buf[MAX_PATH] = {0}; CallService(MS_DB_GETPROFILENAME,(WPARAM)MAX_PATH,(LPARAM)buf); int i = strlen(buf); while (buf[i] != '.' && i>0) i--; buf[i] = '\0'; mir_free(var); return mir_a2u(buf); } mir_free(var); return mir_tstrdup(_T("")); } TCHAR *ske_ParseText(TCHAR *stzText) { int len = lstrlen(stzText); TCHAR *result = NULL; int stpos = 0; int curpos = 0; while (curpos < len) { //1 find first % while (curpos < len && stzText[curpos] != (TCHAR)'%') curpos++; if (curpos < len) { //% found if (curpos-stpos>0) result = ske_ReAppend(result,stzText+stpos,curpos-stpos); stpos = curpos+1; curpos++; //3 find second % while(curpos < len && stzText[curpos] != (TCHAR)'%') curpos++; if (curpos >= len) break; if (curpos - stpos > 0) { TCHAR *var = (TCHAR *)mir_alloc((curpos - stpos + 1)*sizeof(TCHAR)); memmove(var, stzText + stpos, (curpos - stpos)*sizeof(TCHAR)); var[curpos - stpos] = (TCHAR)'\0'; var = ske_ReplaceVar(var); result = ske_ReAppend(result, var, 0); mir_free(var); } else result = ske_ReAppend(result, _T("%"), 0); curpos++; stpos = curpos; } else { if (curpos - stpos > 0) result = ske_ReAppend(result, stzText + stpos, curpos - stpos); break; } } return result; } /* * Parse text object string, find glyph object and add text to it. * szGlyphTextID and Define string is: * t[szGlyphTextID] = s[HostObjectID],[Left],[Top],[Right],[Bottom],[LTRBHV],[FontID],[Color1],[reservedforColor2],[Text] */ static void ske_AddParseTextGlyphObject(char * szGlyphTextID,char * szDefineString,SKINOBJECTSLIST *Skin) { char buf[255] = {0}; GetParamN(szDefineString,buf,sizeof(buf), 0, ',',TRUE); if (buf[0] == 0) return; GLYPHTEXT *glText = (GLYPHTEXT*)mir_calloc(sizeof(GLYPHTEXT)); glText->szGlyphTextID = mir_strdup(szGlyphTextID); glText->szObjectName = mir_strdup(buf); glText->iLeft = atoi(GetParamN(szDefineString,buf,sizeof(buf),1,',',TRUE)); glText->iTop = atoi(GetParamN(szDefineString,buf,sizeof(buf),2,',',TRUE)); glText->iRight = atoi(GetParamN(szDefineString,buf,sizeof(buf),3,',',TRUE)); glText->iBottom = atoi(GetParamN(szDefineString,buf,sizeof(buf),4,',',TRUE)); { memset(buf, 0, 6); GetParamN(szDefineString,buf,sizeof(buf),5,',',TRUE); buf[0] &= 95; buf[1] &= 95; buf[2] &= 95; buf[3] &= 95; buf[4] &= 95; buf[5] &= 95; //to uppercase: &01011111 (0-95) glText->RelativeFlags = (buf[0] == 'C'?1:((buf[0] == 'R')?2:0)) //[BC][RC][BC][RC] --- Left relative |(buf[1] == 'C'?4:((buf[1] == 'B')?8:0)) // | | |--------- Top relative |(buf[2] == 'C'?16:((buf[2] == 'R')?32:0)) // | |--------------Right relative |(buf[3] == 'C'?64:((buf[3] == 'B')?128:0)); // |------------------Bottom relative glText->dwFlags = (buf[4] == 'C'?DT_CENTER:((buf[4] == 'R')?DT_RIGHT:DT_LEFT)) |(buf[5] == 'C'?DT_VCENTER:((buf[5] == 'B')?DT_BOTTOM:DT_TOP)); } glText->szFontID = mir_strdup(GetParamN(szDefineString,buf,sizeof(buf),6,',',TRUE)); glText->dwColor = ske_HexToARGB(GetParamN(szDefineString,buf,sizeof(buf),7,',',TRUE)); glText->dwShadow = ske_HexToARGB(GetParamN(szDefineString,buf,sizeof(buf),8,',',TRUE)); glText->stValueText = mir_a2u(GetParamN(szDefineString,buf,sizeof(buf),9,',',TRUE)); glText->stText = ske_ParseText(glText->stValueText); if (!Skin->pTextList) Skin->pTextList = List_Create(0, 1); List_InsertPtr(Skin->pTextList,glText); } /* * Parse font definition string. * szGlyphTextID and Define string is: * f[szFontID] = s[FontTypefaceName],[size],[BIU] */ static void ske_AddParseSkinFont(char * szFontID,char * szDefineString,SKINOBJECTSLIST *Skin) { SKINFONT *sf = (SKINFONT*)mir_calloc(sizeof(SKINFONT)); if (!sf) return; char buf[255]; int fntSize = 0; BOOL fntBold = FALSE, fntItalic = FALSE, fntUnderline = FALSE; LOGFONTA logfont = {0}; logfont.lfCharSet = DEFAULT_CHARSET; logfont.lfOutPrecision = OUT_DEFAULT_PRECIS; logfont.lfClipPrecision = CLIP_DEFAULT_PRECIS; logfont.lfQuality = DEFAULT_QUALITY; logfont.lfPitchAndFamily = DEFAULT_PITCH|FF_DONTCARE; strncpy(logfont.lfFaceName,GetParamN(szDefineString,buf,sizeof(buf), 0, ',',TRUE),32); logfont.lfHeight = atoi(GetParamN(szDefineString,buf,sizeof(buf),1,',',TRUE)); if (logfont.lfHeight < 0) { HDC hdc = CreateCompatibleDC(NULL); logfont.lfHeight = (long)-MulDiv(logfont.lfHeight, GetDeviceCaps(hdc, LOGPIXELSY), 72); DeleteDC(hdc); } logfont.lfHeight = -logfont.lfHeight; GetParamN(szDefineString,buf,sizeof(buf),2,',',TRUE); buf[0] &= 95; buf[1] &= 95; buf[2] &= 95; logfont.lfWeight = (buf[0] == 'B')?FW_BOLD:FW_NORMAL; logfont.lfItalic = (buf[1] == 'I')?1:0; logfont.lfUnderline = (buf[2] == 'U')?1:0; sf->hFont = CreateFontIndirectA(&logfont); if (sf->hFont) { sf->szFontID = mir_strdup(szFontID); if (!gl_plSkinFonts) gl_plSkinFonts = List_Create(0, 1); if (gl_plSkinFonts) List_Insert(gl_plSkinFonts,(void*)sf,gl_plSkinFonts->realCount); } } /* * ske_CheckHasAlfaChannel - checks if image has at least one BYTE in alpha chennel * that is not a 0. (is image real 32 bit or just 24 bit) */ static BOOL ske_CheckHasAlfaChannel(BYTE * from, int widthByte, int height) { DWORD *pt = (DWORD*)from; for (int j = 0; j < height; j++) { BYTE *add = (BYTE*)pt+widthByte; while (pt < (DWORD*)add) { if ((*pt & 0xFF000000) != 0) return TRUE; pt++; } pt = (DWORD*)(from+widthByte*j); } return FALSE; } /* * ske_CheckIconHasMask - checks if mask image has at least one that is not a 0. * Not sure is ir required or not */ static BOOL ske_CheckIconHasMask(BYTE * from) { for (int i=0; i < 16*16/8; i++) if (from[i] != 0) return TRUE; return FALSE; } /* * ske_GetMaskBit - return value of apropriate mask bit in line at x position */ static BOOL ske_GetMaskBit(BYTE *line, int x) { return ((*(line+(x>>3)))&(0x01 << (7-(x&0x07)))) != 0; } /* * ske_Blend - alpha ske_Blend ARGB values of 2 pixels. X1 - underlaying, * X2 - overlaying points. */ static DWORD ske_Blend(DWORD X1,DWORD X2, BYTE alpha) { BYTE a1 = (BYTE)(X1>>24); BYTE a2 = (BYTE)(((X2>>24)*alpha)>>8); BYTE r1 = (BYTE)(X1>>16); BYTE r2 = (BYTE)(X2>>16); BYTE g1 = (BYTE)(X1>>8); BYTE g2 = (BYTE)(X2>>8); BYTE b1 = (BYTE)(X1); BYTE b2 = (BYTE)(X2); BYTE a_1 = ~a1; BYTE a_2 = ~a2; WORD am = (WORD)a1*a_2; /* it is possible to use >>8 instead of /255 but it is require additional * checking of alphavalues */ WORD ar = a1+(((WORD)a_1*a2)/255); // if a2 more than 0 than result should be more // or equal (if a1 == 0) to a2, else in combination // with mask we can get here black points ar = (a2>ar)?a2:ar; if (ar == 0) return 0; WORD arm = ar*255; WORD rr = (((WORD)r1*am+(WORD)r2*a2*255))/arm; WORD gr = (((WORD)g1*am+(WORD)g2*a2*255))/arm; WORD br = (((WORD)b1*am+(WORD)b2*a2*255))/arm; return (ar << 24)|(rr << 16)|(gr << 8)|br; } /* * CreateJoinedIcon - creates new icon by drawing hTop over hBottom. */ HICON ske_CreateJoinedIcon(HICON hBottom, HICON hTop, BYTE alpha) { HBITMAP nMask; BITMAP bmp = { 0 }; BYTE *ptPixels; ICONINFO iNew = { 0 }; ICONINFO iciBottom = { 0 }; ICONINFO iciTop = { 0 }; BITMAP bmp_top = { 0 }; BITMAP bmp_top_mask = { 0 }; BITMAP bmp_bottom = { 0 }; BITMAP bmp_bottom_mask = { 0 }; HDC tempDC = CreateCompatibleDC(NULL); HBITMAP nImage = ske_CreateDIB32Point(16, 16, (void**)&ptPixels); HBITMAP oImage = (HBITMAP)SelectObject(tempDC, nImage); GetIconInfo(hBottom, &iciBottom); GetObject(iciBottom.hbmColor, sizeof(BITMAP), &bmp_bottom); GetObject(iciBottom.hbmMask, sizeof(BITMAP), &bmp_bottom_mask); GetIconInfo(hTop, &iciTop); GetObject(iciTop.hbmColor, sizeof(BITMAP), &bmp_top); GetObject(iciTop.hbmMask, sizeof(BITMAP), &bmp_top_mask); if (bmp_bottom.bmBitsPixel == 32 && bmp_top.bmBitsPixel == 32) { BYTE *BottomBuffer, *TopBuffer, *BottomMaskBuffer, *TopMaskBuffer; BYTE *bb, *tb, *bmb, *tmb; BYTE *db = ptPixels; int vstep_d = 16 * 4; int vstep_b = bmp_bottom.bmWidthBytes; int vstep_t = bmp_top.bmWidthBytes; int vstep_bm = bmp_bottom_mask.bmWidthBytes; int vstep_tm = bmp_top_mask.bmWidthBytes; alpha = alpha ? alpha : 255; if (bmp_bottom.bmBits) bb = BottomBuffer = (BYTE*)bmp_bottom.bmBits; else { BottomBuffer = (BYTE*)malloc(bmp_bottom.bmHeight*bmp_bottom.bmWidthBytes); GetBitmapBits(iciBottom.hbmColor, bmp_bottom.bmHeight*bmp_bottom.bmWidthBytes, BottomBuffer); bb = BottomBuffer + vstep_b*(bmp_bottom.bmHeight - 1); vstep_b = -vstep_b; } if (bmp_top.bmBits) tb = TopBuffer = (BYTE*)bmp_top.bmBits; else { TopBuffer = (BYTE*)malloc(bmp_top.bmHeight*bmp_top.bmWidthBytes); GetBitmapBits(iciTop.hbmColor,bmp_top.bmHeight*bmp_top.bmWidthBytes,TopBuffer); tb = TopBuffer+vstep_t*(bmp_top.bmHeight-1); vstep_t = -vstep_t; } if (bmp_bottom_mask.bmBits) { BottomMaskBuffer = (BYTE*)bmp_bottom_mask.bmBits; bmb = BottomMaskBuffer; } else { BottomMaskBuffer = (BYTE*)malloc(bmp_bottom_mask.bmHeight*bmp_bottom_mask.bmWidthBytes); GetBitmapBits(iciBottom.hbmMask,bmp_bottom_mask.bmHeight*bmp_bottom_mask.bmWidthBytes,BottomMaskBuffer); bmb = BottomMaskBuffer+vstep_bm*(bmp_bottom_mask.bmHeight-1); vstep_bm = -vstep_bm; } if (bmp_top_mask.bmBits) { TopMaskBuffer = (BYTE*)bmp_top_mask.bmBits; tmb = TopMaskBuffer; } else { TopMaskBuffer = (BYTE*)malloc(bmp_top_mask.bmHeight*bmp_top_mask.bmWidthBytes); GetBitmapBits(iciTop.hbmMask,bmp_top_mask.bmHeight*bmp_top_mask.bmWidthBytes,TopMaskBuffer); tmb = TopMaskBuffer+vstep_tm*(bmp_top_mask.bmHeight-1); vstep_tm = -vstep_tm; } BOOL topHasAlpha = ske_CheckHasAlfaChannel(TopBuffer, bmp_top.bmWidthBytes, bmp_top.bmHeight); BOOL bottomHasAlpha = ske_CheckHasAlfaChannel(BottomBuffer, bmp_bottom.bmWidthBytes, bmp_bottom.bmHeight); BOOL topHasMask = ske_CheckIconHasMask(TopMaskBuffer); BOOL bottomHasMask = ske_CheckIconHasMask(BottomMaskBuffer); for (int y = 0; y < 16; y++) { for (int x = 0; x < 16; x++) { BOOL mask_b = ske_GetMaskBit(bmb, x); BOOL mask_t = ske_GetMaskBit(tmb, x); DWORD bottom_d = ((DWORD*)bb)[x]; DWORD top_d = ((DWORD*)tb)[x]; if (topHasMask) { if (mask_t == 1 && !topHasAlpha) top_d &= 0xFFFFFF; else if (!topHasAlpha) top_d |= 0xFF000000; } if (bottomHasMask) { if (mask_b == 1 && !bottomHasAlpha) bottom_d &= 0xFFFFFF; else if (!bottomHasAlpha) bottom_d |= 0xFF000000; } ((DWORD*)db)[x] = ske_Blend(bottom_d, top_d, alpha); } bb += vstep_b; tb += vstep_t; bmb += vstep_bm; tmb += vstep_tm; db += vstep_d; } if (!bmp_bottom.bmBits) free(BottomBuffer); if (!bmp_top.bmBits) free(TopBuffer); if (!bmp_bottom_mask.bmBits) free(BottomMaskBuffer); if (!bmp_top_mask.bmBits) free(TopMaskBuffer); } else { ske_DrawIconEx(tempDC, 0, 0, hBottom,16,16, 0, NULL,DI_NORMAL); ske_DrawIconEx(tempDC, 0, 0, hTop,16,16, 0, NULL,DI_NORMAL|(alpha << 24)); } DeleteObject(iciBottom.hbmColor); DeleteObject(iciTop.hbmColor); DeleteObject(iciBottom.hbmMask); DeleteObject(iciTop.hbmMask); SelectObject(tempDC,oImage); DeleteDC(tempDC); BYTE p[32] = {0}; nMask = CreateBitmap(16,16,1,1,(void*)&p); { HDC tempDC2 = CreateCompatibleDC(NULL); HDC tempDC3 = CreateCompatibleDC(NULL); HBITMAP hbm = CreateCompatibleBitmap(tempDC3,16,16); HBITMAP obmp = (HBITMAP)SelectObject(tempDC2,nMask); HBITMAP obmp2 = (HBITMAP)SelectObject(tempDC3,hbm); DrawIconEx(tempDC2, 0, 0, hBottom,16,16, 0, NULL,DI_MASK); DrawIconEx(tempDC3, 0, 0, hTop,16,16, 0, NULL,DI_MASK); BitBlt(tempDC2, 0, 0, 16,16,tempDC3, 0, 0, SRCAND); SelectObject(tempDC2,obmp); SelectObject(tempDC3,obmp2); DeleteObject(hbm); DeleteDC(tempDC2); DeleteDC(tempDC3); } iNew.fIcon = TRUE; iNew.hbmColor = nImage; iNew.hbmMask = nMask; HICON res = CreateIconIndirect(&iNew); DeleteObject(nImage); DeleteObject(nMask); return res; } #define NEWJOINEDSTR( destination, first, separator, last) \ destination = (char*)alloca(strlen(first)+strlen(separator)+strlen(last)+1); \ if (destination) { \ *destination = '\0'; \ strcat(destination,first); \ strcat(destination,separator); \ strcat(destination,last); \ } #define SKINSETSECTION "SkinnedSettings" BOOL SkinDBGetContactSetting(HANDLE hContact, const char* szSection, const char*szKey, DBVARIANT * retdbv, BOOL * bSkinned ) { if (!hContact) { //only for not contact settings char *szSkinKey; NEWJOINEDSTR(szSkinKey, szSection, "@", szKey); if (!db_get(hContact, SKINSETSECTION, szSkinKey, retdbv)) { if (bSkinned) *bSkinned = TRUE; return FALSE; } } // not skinned if (bSkinned) bSkinned = FALSE; return db_get(hContact, szSection, szKey, retdbv); } BYTE SkinDBGetContactSettingByte(HANDLE hContact, const char* szSection, const char*szKey, BYTE bDefault) { DBVARIANT dbv = { 0 }; BOOL bSkinned = FALSE; if (!SkinDBGetContactSetting(hContact, szSection, szKey, &dbv, &bSkinned)) { if (dbv.type == DBVT_BYTE) { BYTE retVal = dbv.bVal; db_free(&dbv); return retVal; } else { db_free(&dbv); if (!bSkinned) return db_get_b(hContact, szSection, szKey, bDefault); } } return bDefault; } WORD SkinDBGetContactSettingWord(HANDLE hContact, const char* szSection, const char*szKey, WORD wDefault) { BOOL bSkinned = FALSE; DBVARIANT dbv = { 0 }; if (!SkinDBGetContactSetting(hContact, szSection, szKey, &dbv, &bSkinned)) { if (dbv.type == DBVT_WORD) { WORD retVal = dbv.wVal; db_free(&dbv); return retVal; } db_free(&dbv); if (!bSkinned) return db_get_w(hContact, szSection, szKey, wDefault); } return wDefault; } DWORD SkinDBGetContactSettingDword(HANDLE hContact, const char* szSection, const char*szKey, DWORD dwDefault) { DBVARIANT dbv = { 0 }; BOOL bSkinned = FALSE; if (!SkinDBGetContactSetting(hContact, szSection, szKey, &dbv, &bSkinned)) { if (dbv.type == DBVT_DWORD) { DWORD retVal = dbv.dVal; db_free(&dbv); return retVal; } db_free(&dbv); if (!bSkinned) return db_get_dw(hContact, szSection, szKey, dwDefault); } return dwDefault; }