| /* |
| * Win32 5.1 Theme drawing |
| * |
| * Copyright (C) 2003 Kevin Koltzau |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| */ |
| |
| #include "config.h" |
| |
| #include <stdlib.h> |
| #include <stdarg.h> |
| |
| #include "windef.h" |
| #include "winbase.h" |
| #include "winuser.h" |
| #include "wingdi.h" |
| #include "uxtheme.h" |
| #include "tmschema.h" |
| |
| #include "msstyles.h" |
| #include "uxthemedll.h" |
| |
| #include "wine/debug.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(uxtheme); |
| |
| /*********************************************************************** |
| * Defines and global variables |
| */ |
| |
| DWORD dwDialogTextureFlags; |
| |
| /***********************************************************************/ |
| |
| /*********************************************************************** |
| * EnableThemeDialogTexture (UXTHEME.@) |
| */ |
| HRESULT WINAPI EnableThemeDialogTexture(HWND hwnd, DWORD dwFlags) |
| { |
| TRACE("(%p,0x%08lx\n", hwnd, dwFlags); |
| dwDialogTextureFlags = dwFlags; |
| return S_OK; |
| } |
| |
| /*********************************************************************** |
| * IsThemeDialogTextureEnabled (UXTHEME.@) |
| */ |
| BOOL WINAPI IsThemeDialogTextureEnabled(void) |
| { |
| TRACE("\n"); |
| return (dwDialogTextureFlags & ETDT_ENABLE) && !(dwDialogTextureFlags & ETDT_DISABLE); |
| } |
| |
| /*********************************************************************** |
| * DrawThemeParentBackground (UXTHEME.@) |
| */ |
| HRESULT WINAPI DrawThemeParentBackground(HWND hwnd, HDC hdc, RECT *prc) |
| { |
| RECT rt; |
| POINT org; |
| HWND hParent; |
| TRACE("(%p,%p,%p)\n", hwnd, hdc, prc); |
| hParent = GetParent(hwnd); |
| if(!hParent) |
| hParent = hwnd; |
| if(prc) { |
| CopyRect(&rt, prc); |
| MapWindowPoints(hwnd, NULL, (LPPOINT)&rt, 2); |
| } |
| else { |
| GetClientRect(hParent, &rt); |
| MapWindowPoints(hParent, NULL, (LPPOINT)&rt, 2); |
| } |
| |
| SetViewportOrgEx(hdc, rt.left, rt.top, &org); |
| |
| SendMessageW(hParent, WM_ERASEBKGND, (WPARAM)hdc, 0); |
| SendMessageW(hParent, WM_PRINTCLIENT, (WPARAM)hdc, PRF_CLIENT); |
| |
| SetViewportOrgEx(hdc, org.x, org.y, NULL); |
| return S_OK; |
| } |
| |
| |
| /*********************************************************************** |
| * DrawThemeBackground (UXTHEME.@) |
| */ |
| HRESULT WINAPI DrawThemeBackground(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, const RECT *pRect, |
| const RECT *pClipRect) |
| { |
| DTBGOPTS opts; |
| opts.dwSize = sizeof(DTBGOPTS); |
| opts.dwFlags = 0; |
| if(pClipRect) { |
| opts.dwFlags |= DTBG_CLIPRECT; |
| CopyRect(&opts.rcClip, pClipRect); |
| } |
| return DrawThemeBackgroundEx(hTheme, hdc, iPartId, iStateId, pRect, &opts); |
| } |
| |
| /*********************************************************************** |
| * UXTHEME_SelectImage |
| * |
| * Select the image to use |
| */ |
| PTHEME_PROPERTY UXTHEME_SelectImage(HTHEME hTheme, HDC hdc, int iPartId, int iStateId, const RECT *pRect, BOOL glyph) |
| { |
| PTHEME_PROPERTY tp; |
| int imageselecttype = IST_NONE; |
| int i; |
| int image; |
| if(glyph) |
| image = TMT_GLYPHIMAGEFILE; |
| else |
| image = TMT_IMAGEFILE; |
| |
| if((tp=MSSTYLES_FindProperty(hTheme, iPartId, iStateId, TMT_FILENAME, image))) |
| return tp; |
| GetThemeEnumValue(hTheme, iPartId, iStateId, TMT_IMAGESELECTTYPE, &imageselecttype); |
| |
| if(imageselecttype == IST_DPI) { |
| int reqdpi = 0; |
| int screendpi = GetDeviceCaps(hdc, LOGPIXELSX); |
| for(i=4; i>=0; i--) { |
| reqdpi = 0; |
| if(SUCCEEDED(GetThemeInt(hTheme, iPartId, iStateId, i + TMT_MINDPI1, &reqdpi))) { |
| if(reqdpi != 0 && screendpi >= reqdpi) { |
| TRACE("Using %d DPI, image %d\n", reqdpi, i + TMT_IMAGEFILE1); |
| return MSSTYLES_FindProperty(hTheme, iPartId, iStateId, TMT_FILENAME, i + TMT_IMAGEFILE1); |
| } |
| } |
| } |
| /* If an image couldnt be selected, choose the first one */ |
| return MSSTYLES_FindProperty(hTheme, iPartId, iStateId, TMT_FILENAME, TMT_IMAGEFILE1); |
| } |
| else if(imageselecttype == IST_SIZE) { |
| POINT size = {pRect->right-pRect->left, pRect->bottom-pRect->top}; |
| POINT reqsize; |
| for(i=4; i>=0; i--) { |
| if(SUCCEEDED(GetThemePosition(hTheme, iPartId, iStateId, i + TMT_MINSIZE1, &reqsize))) { |
| if(reqsize.x >= size.x && reqsize.y >= size.y) { |
| TRACE("Using image size %ldx%ld, image %d\n", reqsize.x, reqsize.y, i + TMT_IMAGEFILE1); |
| return MSSTYLES_FindProperty(hTheme, iPartId, iStateId, TMT_FILENAME, i + TMT_IMAGEFILE1); |
| } |
| } |
| } |
| /* If an image couldnt be selected, choose the smallest one */ |
| return MSSTYLES_FindProperty(hTheme, iPartId, iStateId, TMT_FILENAME, TMT_IMAGEFILE1); |
| } |
| return NULL; |
| } |
| |
| /*********************************************************************** |
| * UXTHEME_LoadImage |
| * |
| * Load image for part/state |
| */ |
| HRESULT UXTHEME_LoadImage(HTHEME hTheme, HDC hdc, int iPartId, int iStateId, const RECT *pRect, BOOL glyph, |
| HBITMAP *hBmp, RECT *bmpRect) |
| { |
| int imagelayout = IL_VERTICAL; |
| int imagecount = 0; |
| BITMAP bmp; |
| WCHAR szPath[MAX_PATH]; |
| PTHEME_PROPERTY tp = UXTHEME_SelectImage(hTheme, hdc, iPartId, iStateId, pRect, glyph); |
| if(!tp) { |
| FIXME("Couldn't determine image for part/state %d/%d, invalid theme?\n", iPartId, iStateId); |
| return E_PROP_ID_UNSUPPORTED; |
| } |
| lstrcpynW(szPath, tp->lpValue, min(tp->dwValueLen+1, sizeof(szPath)/sizeof(szPath[0]))); |
| *hBmp = MSSTYLES_LoadBitmap(hdc, hTheme, szPath); |
| if(!*hBmp) { |
| TRACE("Failed to load bitmap %s\n", debugstr_w(szPath)); |
| return HRESULT_FROM_WIN32(GetLastError()); |
| } |
| |
| GetThemeEnumValue(hTheme, iPartId, iStateId, TMT_IMAGELAYOUT, &imagelayout); |
| GetThemeInt(hTheme, iPartId, iStateId, TMT_IMAGECOUNT, &imagecount); |
| |
| GetObjectW(*hBmp, sizeof(bmp), &bmp); |
| if(imagelayout == IL_VERTICAL) { |
| int height = bmp.bmHeight/imagecount; |
| bmpRect->left = 0; |
| bmpRect->right = bmp.bmWidth; |
| bmpRect->top = (min(imagecount, iStateId)-1) * height; |
| bmpRect->bottom = bmpRect->top + height; |
| } |
| else { |
| int width = bmp.bmWidth/imagecount; |
| bmpRect->left = (min(imagecount, iStateId)-1) * width; |
| bmpRect->right = bmpRect->left + width; |
| bmpRect->top = 0; |
| bmpRect->bottom = bmp.bmHeight; |
| } |
| return S_OK; |
| } |
| |
| /*********************************************************************** |
| * UXTHEME_StretchBlt |
| * |
| * Psudo TransparentBlt/StretchBlt |
| */ |
| static inline BOOL UXTHEME_StretchBlt(HDC hdcDst, int nXOriginDst, int nYOriginDst, int nWidthDst, int nHeightDst, |
| HDC hdcSrc, int nXOriginSrc, int nYOriginSrc, int nWidthSrc, int nHeightSrc, |
| BOOL transparent, COLORREF transcolor) |
| { |
| if(transparent) { |
| /* Ensure we don't pass any negative values to TransparentBlt */ |
| return TransparentBlt(hdcDst, nXOriginDst, nYOriginDst, abs(nWidthDst), abs(nHeightDst), |
| hdcSrc, nXOriginSrc, nYOriginSrc, abs(nWidthSrc), abs(nHeightSrc), |
| transcolor); |
| } |
| /* This should be using AlphaBlend */ |
| return StretchBlt(hdcDst, nXOriginDst, nYOriginDst, nWidthDst, nHeightDst, |
| hdcSrc, nXOriginSrc, nYOriginSrc, nWidthSrc, nHeightSrc, |
| SRCCOPY); |
| } |
| |
| /*********************************************************************** |
| * UXTHEME_Blt |
| * |
| * Simplify sending same width/height for both source and dest |
| */ |
| static inline BOOL UXTHEME_Blt(HDC hdcDest, int nXOriginDest, int nYOriginDest, int nWidthDest, int nHeightDest, |
| HDC hdcSrc, int nXOriginSrc, int nYOriginSrc, |
| BOOL transparent, COLORREF transcolor) |
| { |
| return UXTHEME_StretchBlt(hdcDest, nXOriginDest, nYOriginDest, nWidthDest, nHeightDest, |
| hdcSrc, nXOriginSrc, nYOriginSrc, nWidthDest, nHeightDest, |
| transparent, transcolor); |
| } |
| |
| |
| /*********************************************************************** |
| * UXTHEME_DrawImageGlyph |
| * |
| * Draw an imagefile glyph |
| */ |
| HRESULT UXTHEME_DrawImageGlyph(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, RECT *pRect, |
| const DTBGOPTS *pOptions) |
| { |
| HRESULT hr; |
| HBITMAP bmpSrc = NULL; |
| HDC hdcSrc = NULL; |
| HGDIOBJ oldSrc = NULL; |
| RECT rcSrc; |
| BOOL transparent = FALSE; |
| COLORREF transparentcolor = 0; |
| int valign = VA_CENTER; |
| int halign = HA_CENTER; |
| POINT dstSize; |
| POINT srcSize; |
| POINT topleft; |
| |
| hr = UXTHEME_LoadImage(hTheme, hdc, iPartId, iStateId, pRect, TRUE, &bmpSrc, &rcSrc); |
| if(FAILED(hr)) return hr; |
| hdcSrc = CreateCompatibleDC(hdc); |
| if(!hdcSrc) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| DeleteObject(bmpSrc); |
| return hr; |
| } |
| oldSrc = SelectObject(hdcSrc, bmpSrc); |
| |
| dstSize.x = pRect->right-pRect->left; |
| dstSize.y = pRect->bottom-pRect->top; |
| srcSize.x = rcSrc.right-rcSrc.left; |
| srcSize.y = rcSrc.bottom-rcSrc.top; |
| |
| GetThemeBool(hTheme, iPartId, iStateId, TMT_GLYPHTRANSPARENT, &transparent); |
| if(transparent) { |
| if(FAILED(GetThemeColor(hTheme, iPartId, iStateId, TMT_GLYPHTRANSPARENTCOLOR, &transparentcolor))) { |
| /* If image is transparent, but no color was specified, get the color of the upper left corner */ |
| transparentcolor = GetPixel(hdcSrc, 0, 0); |
| } |
| } |
| GetThemeEnumValue(hTheme, iPartId, iStateId, TMT_VALIGN, &valign); |
| GetThemeEnumValue(hTheme, iPartId, iStateId, TMT_HALIGN, &halign); |
| |
| topleft.x = pRect->left; |
| topleft.y = pRect->top; |
| if(halign == HA_CENTER) topleft.x += (dstSize.x/2)-(srcSize.x/2); |
| else if(halign == HA_RIGHT) topleft.x += dstSize.x-srcSize.x; |
| if(valign == VA_CENTER) topleft.y += (dstSize.y/2)-(srcSize.y/2); |
| else if(valign == VA_BOTTOM) topleft.y += dstSize.y-srcSize.y; |
| |
| if(!UXTHEME_Blt(hdc, topleft.x, topleft.y, srcSize.x, srcSize.y, |
| hdcSrc, rcSrc.left, rcSrc.top, |
| transparent, transparentcolor)) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| } |
| |
| SelectObject(hdcSrc, oldSrc); |
| DeleteDC(hdcSrc); |
| DeleteObject(bmpSrc); |
| return hr; |
| } |
| |
| /*********************************************************************** |
| * UXTHEME_DrawImageGlyph |
| * |
| * Draw glyph on top of background, if appropriate |
| */ |
| HRESULT UXTHEME_DrawGlyph(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, RECT *pRect, |
| const DTBGOPTS *pOptions) |
| { |
| int glyphtype = GT_NONE; |
| |
| GetThemeEnumValue(hTheme, iPartId, iStateId, TMT_GLYPHTYPE, &glyphtype); |
| |
| if(glyphtype == GT_IMAGEGLYPH) { |
| return UXTHEME_DrawImageGlyph(hTheme, hdc, iPartId, iStateId, pRect, pOptions); |
| } |
| else if(glyphtype == GT_FONTGLYPH) { |
| /* I don't know what a font glyph is, I've never seen it used in any themes */ |
| FIXME("Font glyph\n"); |
| } |
| return S_OK; |
| } |
| |
| /*********************************************************************** |
| * UXTHEME_DrawImageBackground |
| * |
| * Draw an imagefile background |
| */ |
| HRESULT UXTHEME_DrawImageBackground(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, RECT *pRect, |
| const DTBGOPTS *pOptions) |
| { |
| HRESULT hr = S_OK; |
| HBITMAP bmpSrc; |
| HGDIOBJ oldSrc; |
| HDC hdcSrc; |
| RECT rcSrc; |
| RECT rcDst; |
| POINT dstSize; |
| POINT srcSize; |
| int sizingtype = ST_TRUESIZE; |
| BOOL uniformsizing = FALSE; |
| BOOL transparent = FALSE; |
| COLORREF transparentcolor = 0; |
| |
| hr = UXTHEME_LoadImage(hTheme, hdc, iPartId, iStateId, pRect, FALSE, &bmpSrc, &rcSrc); |
| if(FAILED(hr)) return hr; |
| hdcSrc = CreateCompatibleDC(hdc); |
| if(!hdcSrc) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| DeleteObject(bmpSrc); |
| return hr; |
| } |
| oldSrc = SelectObject(hdcSrc, bmpSrc); |
| |
| CopyRect(&rcDst, pRect); |
| |
| GetThemeBool(hTheme, iPartId, iStateId, TMT_TRANSPARENT, &transparent); |
| if(transparent) { |
| if(FAILED(GetThemeColor(hTheme, iPartId, iStateId, TMT_TRANSPARENTCOLOR, &transparentcolor))) { |
| /* If image is transparent, but no color was specified, get the color of the upper left corner */ |
| transparentcolor = GetPixel(hdcSrc, 0, 0); |
| } |
| } |
| |
| dstSize.x = rcDst.right-rcDst.left; |
| dstSize.y = rcDst.bottom-rcDst.top; |
| srcSize.x = rcSrc.right-rcSrc.left; |
| srcSize.y = rcSrc.bottom-rcSrc.top; |
| |
| if(uniformsizing) { |
| /* Scale height and width equally */ |
| int widthDiff = abs(srcSize.x-dstSize.x); |
| int heightDiff = abs(srcSize.y-dstSize.x); |
| if(widthDiff > heightDiff) { |
| dstSize.y -= widthDiff-heightDiff; |
| rcDst.bottom = rcDst.top + dstSize.y; |
| } |
| else if(heightDiff > widthDiff) { |
| dstSize.x -= heightDiff-widthDiff; |
| rcDst.right = rcDst.left + dstSize.x; |
| } |
| } |
| |
| GetThemeEnumValue(hTheme, iPartId, iStateId, TMT_SIZINGTYPE, &sizingtype); |
| if(sizingtype == ST_TRUESIZE) { |
| int truesizestretchmark = 0; |
| |
| if(dstSize.x < 0 || dstSize.y < 0) { |
| BOOL mirrorimage = TRUE; |
| GetThemeBool(hTheme, iPartId, iStateId, TMT_MIRRORIMAGE, &mirrorimage); |
| if(mirrorimage) { |
| if(dstSize.x < 0) { |
| rcDst.left += dstSize.x; |
| rcDst.right += dstSize.x; |
| } |
| if(dstSize.y < 0) { |
| rcDst.top += dstSize.y; |
| rcDst.bottom += dstSize.y; |
| } |
| } |
| } |
| /* Only stretch when target exceeds source by truesizestretchmark percent */ |
| GetThemeInt(hTheme, iPartId, iStateId, TMT_TRUESIZESTRETCHMARK, &truesizestretchmark); |
| if(dstSize.x < 0 || dstSize.y < 0 || |
| MulDiv(srcSize.x, 100, dstSize.x) > truesizestretchmark || |
| MulDiv(srcSize.y, 100, dstSize.y) > truesizestretchmark) { |
| if(!UXTHEME_StretchBlt(hdc, rcDst.left, rcDst.top, dstSize.x, dstSize.y, |
| hdcSrc, rcSrc.left, rcSrc.top, srcSize.x, srcSize.y, |
| transparent, transparentcolor)) |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| } |
| else { |
| rcDst.left += (dstSize.x/2)-(srcSize.x/2); |
| rcDst.top += (dstSize.y/2)-(srcSize.y/2); |
| rcDst.right = rcDst.left + srcSize.x; |
| rcDst.bottom = rcDst.top + srcSize.y; |
| if(!UXTHEME_Blt(hdc, rcDst.left, rcDst.top, srcSize.x, srcSize.y, |
| hdcSrc, rcSrc.left, rcSrc.top, |
| transparent, transparentcolor)) |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| } |
| } |
| else { |
| HDC hdcDst = NULL; |
| HBITMAP bmpDst = NULL; |
| HGDIOBJ oldDst = NULL; |
| MARGINS sm; |
| |
| dstSize.x = abs(dstSize.x); |
| dstSize.y = abs(dstSize.y); |
| |
| GetThemeMargins(hTheme, hdc, iPartId, iStateId, TMT_SIZINGMARGINS, NULL, &sm); |
| |
| hdcDst = CreateCompatibleDC(hdc); |
| if(!hdcDst) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| goto draw_error; |
| } |
| bmpDst = CreateCompatibleBitmap(hdc, dstSize.x, dstSize.y); |
| if(!bmpDst) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| goto draw_error; |
| } |
| oldDst = SelectObject(hdcDst, bmpDst); |
| |
| /* Upper left corner */ |
| if(!BitBlt(hdcDst, 0, 0, sm.cxLeftWidth, sm.cyTopHeight, |
| hdcSrc, rcSrc.left, rcSrc.top, SRCCOPY)) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| goto draw_error; |
| } |
| /* Upper right corner */ |
| if(!BitBlt(hdcDst, dstSize.x-sm.cxRightWidth, 0, sm.cxRightWidth, sm.cyTopHeight, |
| hdcSrc, rcSrc.right-sm.cxRightWidth, rcSrc.top, SRCCOPY)) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| goto draw_error; |
| } |
| /* Lower left corner */ |
| if(!BitBlt(hdcDst, 0, dstSize.y-sm.cyBottomHeight, sm.cxLeftWidth, sm.cyBottomHeight, |
| hdcSrc, rcSrc.left, rcSrc.bottom-sm.cyBottomHeight, SRCCOPY)) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| goto draw_error; |
| } |
| /* Lower right corner */ |
| if(!BitBlt(hdcDst, dstSize.x-sm.cxRightWidth, dstSize.y-sm.cyBottomHeight, sm.cxRightWidth, sm.cyBottomHeight, |
| hdcSrc, rcSrc.right-sm.cxRightWidth, rcSrc.bottom-sm.cyBottomHeight, SRCCOPY)) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| goto draw_error; |
| } |
| |
| if(sizingtype == ST_TILE) { |
| FIXME("Tile\n"); |
| sizingtype = ST_STRETCH; /* Just use stretch for now */ |
| } |
| if(sizingtype == ST_STRETCH) { |
| int destCenterWidth = dstSize.x - (sm.cxLeftWidth + sm.cxRightWidth); |
| int srcCenterWidth = srcSize.x - (sm.cxLeftWidth + sm.cxRightWidth); |
| int destCenterHeight = dstSize.y - (sm.cyTopHeight + sm.cyBottomHeight); |
| int srcCenterHeight = srcSize.y - (sm.cyTopHeight + sm.cyBottomHeight); |
| |
| if(destCenterWidth > 0) { |
| /* Center top */ |
| if(!StretchBlt(hdcDst, sm.cxLeftWidth, 0, destCenterWidth, sm.cyTopHeight, |
| hdcSrc, rcSrc.left+sm.cxLeftWidth, rcSrc.top, srcCenterWidth, sm.cyTopHeight, SRCCOPY)) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| goto draw_error; |
| } |
| /* Center bottom */ |
| if(!StretchBlt(hdcDst, sm.cxLeftWidth, dstSize.y-sm.cyBottomHeight, destCenterWidth, sm.cyBottomHeight, |
| hdcSrc, rcSrc.left+sm.cxLeftWidth, rcSrc.bottom-sm.cyBottomHeight, srcCenterWidth, sm.cyTopHeight, SRCCOPY)) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| goto draw_error; |
| } |
| } |
| if(destCenterHeight > 0) { |
| /* Left center */ |
| if(!StretchBlt(hdcDst, 0, sm.cyTopHeight, sm.cxLeftWidth, destCenterHeight, |
| hdcSrc, rcSrc.left, rcSrc.top+sm.cyTopHeight, sm.cxLeftWidth, srcCenterHeight, SRCCOPY)) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| goto draw_error; |
| } |
| /* Right center */ |
| if(!StretchBlt(hdcDst, dstSize.x-sm.cxRightWidth, sm.cyTopHeight, sm.cxRightWidth, destCenterHeight, |
| hdcSrc, rcSrc.right-sm.cxRightWidth, rcSrc.top+sm.cyTopHeight, sm.cxRightWidth, srcCenterHeight, SRCCOPY)) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| goto draw_error; |
| } |
| } |
| if(destCenterHeight > 0 && destCenterWidth > 0) { |
| BOOL borderonly = FALSE; |
| GetThemeBool(hTheme, iPartId, iStateId, TMT_BORDERONLY, &borderonly); |
| if(!borderonly) { |
| /* Center */ |
| if(!StretchBlt(hdcDst, sm.cxLeftWidth, sm.cyTopHeight, destCenterWidth, destCenterHeight, |
| hdcSrc, rcSrc.left+sm.cxLeftWidth, rcSrc.top+sm.cyTopHeight, srcCenterWidth, srcCenterHeight, SRCCOPY)) { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| goto draw_error; |
| } |
| } |
| } |
| } |
| |
| if(!UXTHEME_Blt(hdc, rcDst.left, rcDst.top, dstSize.x, dstSize.y, |
| hdcDst, 0, 0, |
| transparent, transparentcolor)) |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| |
| draw_error: |
| if(hdcDst) { |
| SelectObject(hdcDst, oldDst); |
| DeleteDC(hdcDst); |
| } |
| if(bmpDst) DeleteObject(bmpDst); |
| } |
| SelectObject(hdcSrc, oldSrc); |
| DeleteObject(bmpSrc); |
| DeleteDC(hdcSrc); |
| CopyRect(pRect, &rcDst); |
| return hr; |
| } |
| |
| /*********************************************************************** |
| * UXTHEME_DrawBorderRectangle |
| * |
| * Draw the bounding rectangle for a borderfill background |
| */ |
| HRESULT UXTHEME_DrawBorderRectangle(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, RECT *pRect, |
| const DTBGOPTS *pOptions) |
| { |
| HRESULT hr = S_OK; |
| HPEN hPen; |
| HGDIOBJ oldPen; |
| COLORREF bordercolor = RGB(0,0,0); |
| int bordersize = 1; |
| |
| GetThemeInt(hTheme, iPartId, iStateId, TMT_BORDERSIZE, &bordersize); |
| if(bordersize > 0) { |
| POINT ptCorners[4]; |
| ptCorners[0].x = pRect->left; |
| ptCorners[0].y = pRect->top; |
| ptCorners[1].x = pRect->right; |
| ptCorners[1].y = pRect->top; |
| ptCorners[2].x = pRect->right; |
| ptCorners[2].y = pRect->bottom; |
| ptCorners[3].x = pRect->left; |
| ptCorners[3].y = pRect->bottom; |
| |
| InflateRect(pRect, -bordersize, -bordersize); |
| if(pOptions->dwFlags & DTBG_OMITBORDER) |
| return S_OK; |
| GetThemeColor(hTheme, iPartId, iStateId, TMT_BORDERCOLOR, &bordercolor); |
| hPen = CreatePen(PS_SOLID, bordersize, bordercolor); |
| if(!hPen) |
| return HRESULT_FROM_WIN32(GetLastError()); |
| oldPen = SelectObject(hdc, hPen); |
| |
| if(!Polyline(hdc, ptCorners, 4)) |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| |
| SelectObject(hdc, oldPen); |
| DeleteObject(hPen); |
| } |
| return hr; |
| } |
| |
| /*********************************************************************** |
| * UXTHEME_DrawBorderRectangle |
| * |
| * Fill a borderfill background rectangle |
| */ |
| HRESULT UXTHEME_DrawBackgroundFill(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, RECT *pRect, |
| const DTBGOPTS *pOptions) |
| { |
| HRESULT hr = S_OK; |
| int filltype = FT_SOLID; |
| |
| TRACE("(%d,%d,%ld)\n", iPartId, iStateId, pOptions->dwFlags); |
| |
| if(pOptions->dwFlags & DTBG_OMITCONTENT) |
| return S_OK; |
| |
| GetThemeEnumValue(hTheme, iPartId, iStateId, TMT_FILLTYPE, &filltype); |
| |
| if(filltype == FT_SOLID) { |
| HBRUSH hBrush; |
| COLORREF fillcolor = RGB(255,255,255); |
| |
| GetThemeColor(hTheme, iPartId, iStateId, TMT_FILLCOLOR, &fillcolor); |
| hBrush = CreateSolidBrush(fillcolor); |
| if(!FillRect(hdc, pRect, hBrush)) |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| DeleteObject(hBrush); |
| } |
| else if(filltype == FT_VERTGRADIENT || filltype == FT_HORZGRADIENT) { |
| /* FIXME: This only accounts for 2 gradient colors (out of 5) and ignores |
| the gradient ratios (no idea how those work) |
| Few themes use this, and the ones I've seen only use 2 colors with |
| a gradient ratio of 0 and 255 respectivly |
| */ |
| |
| COLORREF gradient1 = RGB(0,0,0); |
| COLORREF gradient2 = RGB(255,255,255); |
| TRIVERTEX vert[2]; |
| GRADIENT_RECT gRect; |
| |
| FIXME("Gradient implementation not complete\n"); |
| |
| GetThemeColor(hTheme, iPartId, iStateId, TMT_GRADIENTCOLOR1, &gradient1); |
| GetThemeColor(hTheme, iPartId, iStateId, TMT_GRADIENTCOLOR2, &gradient2); |
| |
| vert[0].x = pRect->left; |
| vert[0].y = pRect->top; |
| vert[0].Red = GetRValue(gradient1) << 8; |
| vert[0].Green = GetGValue(gradient1) << 8; |
| vert[0].Blue = GetBValue(gradient1) << 8; |
| vert[0].Alpha = 0x0000; |
| |
| vert[1].x = pRect->right; |
| vert[1].y = pRect->bottom; |
| vert[1].Red = GetRValue(gradient2) << 8; |
| vert[1].Green = GetGValue(gradient2) << 8; |
| vert[1].Blue = GetBValue(gradient2) << 8; |
| vert[1].Alpha = 0x0000; |
| |
| gRect.UpperLeft = 0; |
| gRect.LowerRight = 1; |
| GradientFill(hdc,vert,2,&gRect,1,filltype==FT_HORZGRADIENT?GRADIENT_FILL_RECT_H:GRADIENT_FILL_RECT_V); |
| } |
| else if(filltype == FT_RADIALGRADIENT) { |
| /* I've never seen this used in a theme */ |
| FIXME("Radial gradient\n"); |
| } |
| else if(filltype == FT_TILEIMAGE) { |
| /* I've never seen this used in a theme */ |
| FIXME("Tile image\n"); |
| } |
| return hr; |
| } |
| |
| /*********************************************************************** |
| * UXTHEME_DrawImageBackground |
| * |
| * Draw an imagefile background |
| */ |
| HRESULT UXTHEME_DrawBorderBackground(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, const RECT *pRect, |
| const DTBGOPTS *pOptions) |
| { |
| HRESULT hr; |
| RECT rt; |
| |
| CopyRect(&rt, pRect); |
| |
| hr = UXTHEME_DrawBorderRectangle(hTheme, hdc, iPartId, iStateId, &rt, pOptions); |
| if(FAILED(hr)) |
| return hr; |
| return UXTHEME_DrawBackgroundFill(hTheme, hdc, iPartId, iStateId, &rt, pOptions); |
| } |
| |
| /*********************************************************************** |
| * DrawThemeBackgroundEx (UXTHEME.@) |
| */ |
| HRESULT WINAPI DrawThemeBackgroundEx(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, const RECT *pRect, |
| const DTBGOPTS *pOptions) |
| { |
| HRESULT hr; |
| const DTBGOPTS defaultOpts = {sizeof(DTBGOPTS), 0, {0,0,0,0}}; |
| const DTBGOPTS *opts; |
| HRGN clip = NULL; |
| int hasClip = -1; |
| int bgtype = BT_BORDERFILL; |
| RECT rt; |
| |
| TRACE("(%p,%p,%d,%d,%ld,%ld)\n", hTheme, hdc, iPartId, iStateId,pRect->left,pRect->top); |
| if(!hTheme) |
| return E_HANDLE; |
| |
| /* Ensure we have a DTBGOPTS structure available, simplifies some of the code */ |
| opts = pOptions; |
| if(!opts) opts = &defaultOpts; |
| |
| if(opts->dwFlags & DTBG_CLIPRECT) { |
| clip = CreateRectRgn(0,0,1,1); |
| hasClip = GetClipRgn(hdc, clip); |
| if(hasClip == -1) |
| TRACE("Failed to get original clipping region\n"); |
| else |
| IntersectClipRect(hdc, opts->rcClip.left, opts->rcClip.top, opts->rcClip.right, opts->rcClip.bottom); |
| } |
| CopyRect(&rt, pRect); |
| |
| GetThemeEnumValue(hTheme, iPartId, iStateId, TMT_BGTYPE, &bgtype); |
| if(bgtype == BT_IMAGEFILE) |
| hr = UXTHEME_DrawImageBackground(hTheme, hdc, iPartId, iStateId, &rt, opts); |
| else if(bgtype == BT_BORDERFILL) |
| hr = UXTHEME_DrawBorderBackground(hTheme, hdc, iPartId, iStateId, pRect, opts); |
| else { |
| FIXME("Unknown background type\n"); |
| /* This should never happen, and hence I don't know what to return */ |
| hr = E_FAIL; |
| } |
| if(SUCCEEDED(hr)) |
| hr = UXTHEME_DrawGlyph(hTheme, hdc, iPartId, iStateId, &rt, opts); |
| if(opts->dwFlags & DTBG_CLIPRECT) { |
| if(hasClip == 0) |
| SelectClipRgn(hdc, NULL); |
| else if(hasClip == 1) |
| SelectClipRgn(hdc, clip); |
| DeleteObject(clip); |
| } |
| return hr; |
| } |
| |
| /*********************************************************************** |
| * DrawThemeEdge (UXTHEME.@) |
| */ |
| HRESULT WINAPI DrawThemeEdge(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, const RECT *pDestRect, UINT uEdge, |
| UINT uFlags, RECT *pContentRect) |
| { |
| FIXME("%d %d 0x%08x 0x%08x: stub\n", iPartId, iStateId, uEdge, uFlags); |
| if(!hTheme) |
| return E_HANDLE; |
| return ERROR_CALL_NOT_IMPLEMENTED; |
| } |
| |
| /*********************************************************************** |
| * DrawThemeIcon (UXTHEME.@) |
| */ |
| HRESULT WINAPI DrawThemeIcon(HTHEME hTheme, HDC hdc, int iPartId, int iStateId, |
| const RECT *pRect, HIMAGELIST himl, int iImageIndex) |
| { |
| FIXME("%d %d: stub\n", iPartId, iStateId); |
| if(!hTheme) |
| return E_HANDLE; |
| return ERROR_CALL_NOT_IMPLEMENTED; |
| } |
| |
| /*********************************************************************** |
| * DrawThemeText (UXTHEME.@) |
| */ |
| HRESULT WINAPI DrawThemeText(HTHEME hTheme, HDC hdc, int iPartId, int iStateId, |
| LPCWSTR pszText, int iCharCount, DWORD dwTextFlags, |
| DWORD dwTextFlags2, const RECT *pRect) |
| { |
| HRESULT hr; |
| HFONT hFont = NULL; |
| HGDIOBJ oldFont = NULL; |
| LOGFONTW logfont; |
| COLORREF textColor; |
| COLORREF oldTextColor; |
| int oldBkMode; |
| RECT rt; |
| |
| TRACE("%d %d: stub\n", iPartId, iStateId); |
| if(!hTheme) |
| return E_HANDLE; |
| |
| hr = GetThemeFont(hTheme, hdc, iPartId, iStateId, TMT_FONT, &logfont); |
| if(SUCCEEDED(hr)) { |
| hFont = CreateFontIndirectW(&logfont); |
| if(!hFont) |
| TRACE("Failed to create font\n"); |
| } |
| CopyRect(&rt, pRect); |
| if(hFont) |
| oldFont = SelectObject(hdc, hFont); |
| |
| if(dwTextFlags2 & DTT_GRAYED) |
| textColor = GetSysColor(COLOR_GRAYTEXT); |
| else { |
| if(FAILED(GetThemeColor(hTheme, iPartId, iStateId, TMT_TEXTCOLOR, &textColor))) |
| textColor = GetTextColor(hdc); |
| } |
| oldTextColor = SetTextColor(hdc, textColor); |
| oldBkMode = SetBkMode(hdc, TRANSPARENT); |
| DrawTextW(hdc, pszText, iCharCount, &rt, dwTextFlags); |
| SetBkMode(hdc, oldBkMode); |
| SetTextColor(hdc, oldTextColor); |
| |
| if(hFont) { |
| SelectObject(hdc, oldFont); |
| DeleteObject(hFont); |
| } |
| return S_OK; |
| } |
| |
| /*********************************************************************** |
| * GetThemeBackgroundContentRect (UXTHEME.@) |
| */ |
| HRESULT WINAPI GetThemeBackgroundContentRect(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, |
| const RECT *pBoundingRect, |
| RECT *pContentRect) |
| { |
| MARGINS margin; |
| HRESULT hr; |
| |
| TRACE("(%d,%d)\n", iPartId, iStateId); |
| if(!hTheme) |
| return E_HANDLE; |
| |
| hr = GetThemeMargins(hTheme, hdc, iPartId, iStateId, TMT_CONTENTMARGINS, NULL, &margin); |
| if(FAILED(hr)) { |
| TRACE("Margins not found\n"); |
| return hr; |
| } |
| pContentRect->left = pBoundingRect->left + margin.cxLeftWidth; |
| pContentRect->top = pBoundingRect->top + margin.cyTopHeight; |
| pContentRect->right = pBoundingRect->right - margin.cxRightWidth; |
| pContentRect->bottom = pBoundingRect->bottom - margin.cyBottomHeight; |
| |
| TRACE("left:%ld,top:%ld,right:%ld,bottom:%ld\n", pContentRect->left, pContentRect->top, pContentRect->right, pContentRect->bottom); |
| |
| return S_OK; |
| } |
| |
| /*********************************************************************** |
| * GetThemeBackgroundExtent (UXTHEME.@) |
| */ |
| HRESULT WINAPI GetThemeBackgroundExtent(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, const RECT *pContentRect, |
| RECT *pExtentRect) |
| { |
| FIXME("%d %d: stub\n", iPartId, iStateId); |
| if(!hTheme) |
| return E_HANDLE; |
| return ERROR_CALL_NOT_IMPLEMENTED; |
| } |
| |
| /*********************************************************************** |
| * GetThemeBackgroundRegion (UXTHEME.@) |
| */ |
| HRESULT WINAPI GetThemeBackgroundRegion(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, const RECT *pRect, |
| HRGN *pRegion) |
| { |
| FIXME("%d %d: stub\n", iPartId, iStateId); |
| if(!hTheme) |
| return E_HANDLE; |
| return ERROR_CALL_NOT_IMPLEMENTED; |
| } |
| |
| /*********************************************************************** |
| * GetThemePartSize (UXTHEME.@) |
| */ |
| HRESULT WINAPI GetThemePartSize(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, RECT *prc, THEMESIZE eSize, |
| SIZE *psz) |
| { |
| FIXME("%d %d %d: stub\n", iPartId, iStateId, eSize); |
| if(!hTheme) |
| return E_HANDLE; |
| return ERROR_CALL_NOT_IMPLEMENTED; |
| } |
| |
| |
| /*********************************************************************** |
| * GetThemeTextExtent (UXTHEME.@) |
| */ |
| HRESULT WINAPI GetThemeTextExtent(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, LPCWSTR pszText, int iCharCount, |
| DWORD dwTextFlags, const RECT *pBoundingRect, |
| RECT *pExtentRect) |
| { |
| HRESULT hr; |
| HFONT hFont = NULL; |
| HGDIOBJ oldFont = NULL; |
| LOGFONTW logfont; |
| RECT rt = {0,0,0xFFFF,0xFFFF}; |
| |
| TRACE("%d %d: stub\n", iPartId, iStateId); |
| if(!hTheme) |
| return E_HANDLE; |
| |
| if(pBoundingRect) |
| CopyRect(&rt, pBoundingRect); |
| |
| hr = GetThemeFont(hTheme, hdc, iPartId, iStateId, TMT_FONT, &logfont); |
| if(SUCCEEDED(hr)) { |
| hFont = CreateFontIndirectW(&logfont); |
| if(!hFont) |
| TRACE("Failed to create font\n"); |
| } |
| if(hFont) |
| oldFont = SelectObject(hdc, hFont); |
| |
| DrawTextW(hdc, pszText, iCharCount, &rt, dwTextFlags|DT_CALCRECT); |
| CopyRect(pExtentRect, &rt); |
| |
| if(hFont) { |
| SelectObject(hdc, oldFont); |
| DeleteObject(hFont); |
| } |
| return S_OK; |
| } |
| |
| /*********************************************************************** |
| * GetThemeTextMetrics (UXTHEME.@) |
| */ |
| HRESULT WINAPI GetThemeTextMetrics(HTHEME hTheme, HDC hdc, int iPartId, |
| int iStateId, TEXTMETRICW *ptm) |
| { |
| HRESULT hr; |
| HFONT hFont = NULL; |
| HGDIOBJ oldFont = NULL; |
| LOGFONTW logfont; |
| |
| TRACE("(%p, %p, %d, %d)\n", hTheme, hdc, iPartId, iStateId); |
| if(!hTheme) |
| return E_HANDLE; |
| |
| hr = GetThemeFont(hTheme, hdc, iPartId, iStateId, TMT_FONT, &logfont); |
| if(SUCCEEDED(hr)) { |
| hFont = CreateFontIndirectW(&logfont); |
| if(!hFont) |
| TRACE("Failed to create font\n"); |
| } |
| if(hFont) |
| oldFont = SelectObject(hdc, hFont); |
| |
| if(!GetTextMetricsW(hdc, ptm)) |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| |
| if(hFont) { |
| SelectObject(hdc, oldFont); |
| DeleteObject(hFont); |
| } |
| return hr; |
| } |
| |
| /*********************************************************************** |
| * IsThemeBackgroundPartiallyTransparent (UXTHEME.@) |
| */ |
| BOOL WINAPI IsThemeBackgroundPartiallyTransparent(HTHEME hTheme, int iPartId, |
| int iStateId) |
| { |
| BOOL transparent = FALSE; |
| TRACE("(%d,%d)\n", iPartId, iStateId); |
| GetThemeBool(hTheme, iPartId, iStateId, TMT_TRANSPARENT, &transparent); |
| return transparent; |
| } |