|  | /* | 
|  | * Progress control | 
|  | * | 
|  | * Copyright 1997, 2002 Dimitrie O. Paun | 
|  | * Copyright 1998, 1999 Eric Kohl | 
|  | * | 
|  | * 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA | 
|  | * | 
|  | * NOTE | 
|  | * | 
|  | * This code was audited for completeness against the documented features | 
|  | * of Comctl32.dll version 6.0 on Sep. 9, 2002, by Dimitrie O. Paun. | 
|  | * | 
|  | * Unless otherwise noted, we believe this code to be complete, as per | 
|  | * the specification mentioned above. | 
|  | * If you discover missing features, or bugs, please note them below. | 
|  | * | 
|  | * TODO: | 
|  | * | 
|  | * Messages: | 
|  | *    -- PBM_GETSTEP | 
|  | *    -- PBM_SETSTATE | 
|  | *    -- PBM_GETSTATE | 
|  | * | 
|  | * Styles: | 
|  | *    -- PBS_SMOOTHREVERSE | 
|  | * | 
|  | */ | 
|  |  | 
|  | #include <stdarg.h> | 
|  | #include <string.h> | 
|  | #include "windef.h" | 
|  | #include "winbase.h" | 
|  | #include "wingdi.h" | 
|  | #include "winuser.h" | 
|  | #include "winnls.h" | 
|  | #include "commctrl.h" | 
|  | #include "comctl32.h" | 
|  | #include "uxtheme.h" | 
|  | #include "tmschema.h" | 
|  | #include "wine/debug.h" | 
|  |  | 
|  | WINE_DEFAULT_DEBUG_CHANNEL(progress); | 
|  |  | 
|  | typedef struct | 
|  | { | 
|  | HWND      Self;         /* The window handle for this control */ | 
|  | INT       CurVal;       /* Current progress value */ | 
|  | INT       MinVal;       /* Minimum progress value */ | 
|  | INT       MaxVal;       /* Maximum progress value */ | 
|  | INT       Step;         /* Step to use on PMB_STEPIT */ | 
|  | INT       MarqueePos;   /* Marquee animation position */ | 
|  | BOOL      Marquee;      /* Whether the marquee animation is enabled */ | 
|  | COLORREF  ColorBar;     /* Bar color */ | 
|  | COLORREF  ColorBk;      /* Background color */ | 
|  | HFONT     Font;         /* Handle to font (not unused) */ | 
|  | } PROGRESS_INFO; | 
|  |  | 
|  | /* Control configuration constants */ | 
|  |  | 
|  | #define LED_GAP           2 | 
|  | #define MARQUEE_LEDS      5 | 
|  | #define ID_MARQUEE_TIMER  1 | 
|  |  | 
|  | /* Helper to obtain size of a progress bar chunk ("led"). */ | 
|  | static inline int get_led_size ( const PROGRESS_INFO *infoPtr, LONG style, | 
|  | const RECT* rect ) | 
|  | { | 
|  | HTHEME theme = GetWindowTheme (infoPtr->Self); | 
|  | if (theme) | 
|  | { | 
|  | int chunkSize; | 
|  | if (SUCCEEDED( GetThemeInt( theme, 0, 0, TMT_PROGRESSCHUNKSIZE, &chunkSize ))) | 
|  | return chunkSize; | 
|  | } | 
|  |  | 
|  | if (style & PBS_VERTICAL) | 
|  | return MulDiv (rect->right - rect->left, 2, 3); | 
|  | else | 
|  | return MulDiv (rect->bottom - rect->top, 2, 3); | 
|  | } | 
|  |  | 
|  | /* Helper to obtain gap between progress bar chunks */ | 
|  | static inline int get_led_gap ( const PROGRESS_INFO *infoPtr ) | 
|  | { | 
|  | HTHEME theme = GetWindowTheme (infoPtr->Self); | 
|  | if (theme) | 
|  | { | 
|  | int spaceSize; | 
|  | if (SUCCEEDED( GetThemeInt( theme, 0, 0, TMT_PROGRESSSPACESIZE, &spaceSize ))) | 
|  | return spaceSize; | 
|  | } | 
|  |  | 
|  | return LED_GAP; | 
|  | } | 
|  |  | 
|  | /* Get client rect. Takes into account that theming needs no adjustment. */ | 
|  | static inline void get_client_rect (HWND hwnd, RECT* rect) | 
|  | { | 
|  | HTHEME theme = GetWindowTheme (hwnd); | 
|  | GetClientRect (hwnd, rect); | 
|  | if (!theme) | 
|  | InflateRect(rect, -1, -1); | 
|  | else | 
|  | { | 
|  | DWORD dwStyle = GetWindowLongW (hwnd, GWL_STYLE); | 
|  | int part = (dwStyle & PBS_VERTICAL) ? PP_BARVERT : PP_BAR; | 
|  | GetThemeBackgroundContentRect (theme, 0, part, 0, rect, rect); | 
|  | } | 
|  | } | 
|  |  | 
|  | /* Compute the extend of the bar */ | 
|  | static inline int get_bar_size( LONG style, const RECT* rect ) | 
|  | { | 
|  | if (style & PBS_VERTICAL) | 
|  | return rect->bottom - rect->top; | 
|  | else | 
|  | return rect->right - rect->left; | 
|  | } | 
|  |  | 
|  | /* Compute the pixel position of a progress value */ | 
|  | static inline int get_bar_position( const PROGRESS_INFO *infoPtr, LONG style, | 
|  | const RECT* rect, INT value ) | 
|  | { | 
|  | return MulDiv (value - infoPtr->MinVal, get_bar_size (style, rect), | 
|  | infoPtr->MaxVal - infoPtr->MinVal); | 
|  | } | 
|  |  | 
|  | /*********************************************************************** | 
|  | * PROGRESS_Invalidate | 
|  | * | 
|  | * Don't be too clever about invalidating the progress bar. | 
|  | * InstallShield depends on this simple behaviour. | 
|  | */ | 
|  | static void PROGRESS_Invalidate( const PROGRESS_INFO *infoPtr, INT old, INT new ) | 
|  | { | 
|  | InvalidateRect( infoPtr->Self, NULL, old > new ); | 
|  | } | 
|  |  | 
|  | /* Information for a progress bar drawing helper */ | 
|  | typedef struct tagProgressDrawInfo | 
|  | { | 
|  | HDC hdc; | 
|  | RECT rect; | 
|  | HBRUSH hbrBar; | 
|  | HBRUSH hbrBk; | 
|  | int ledW, ledGap; | 
|  | HTHEME theme; | 
|  | RECT bgRect; | 
|  | } ProgressDrawInfo; | 
|  |  | 
|  | typedef void (*ProgressDrawProc)(const ProgressDrawInfo* di, int start, int end); | 
|  |  | 
|  | /* draw solid horizontal bar from 'start' to 'end' */ | 
|  | static void draw_solid_bar_H (const ProgressDrawInfo* di, int start, int end) | 
|  | { | 
|  | RECT r; | 
|  | r.left = di->rect.left + start; | 
|  | r.top = di->rect.top; | 
|  | r.right = di->rect.left + end; | 
|  | r.bottom = di->rect.bottom; | 
|  | FillRect (di->hdc, &r, di->hbrBar); | 
|  | } | 
|  |  | 
|  | /* draw solid horizontal background from 'start' to 'end' */ | 
|  | static void draw_solid_bkg_H (const ProgressDrawInfo* di, int start, int end) | 
|  | { | 
|  | RECT r; | 
|  | r.left = di->rect.left + start; | 
|  | r.top = di->rect.top; | 
|  | r.right = di->rect.left + end; | 
|  | r.bottom = di->rect.bottom; | 
|  | FillRect (di->hdc, &r, di->hbrBk); | 
|  | } | 
|  |  | 
|  | /* draw solid vertical bar from 'start' to 'end' */ | 
|  | static void draw_solid_bar_V (const ProgressDrawInfo* di, int start, int end) | 
|  | { | 
|  | RECT r; | 
|  | r.left = di->rect.left; | 
|  | r.top = di->rect.bottom - end; | 
|  | r.right = di->rect.right; | 
|  | r.bottom = di->rect.bottom - start; | 
|  | FillRect (di->hdc, &r, di->hbrBar); | 
|  | } | 
|  |  | 
|  | /* draw solid vertical background from 'start' to 'end' */ | 
|  | static void draw_solid_bkg_V (const ProgressDrawInfo* di, int start, int end) | 
|  | { | 
|  | RECT r; | 
|  | r.left = di->rect.left; | 
|  | r.top = di->rect.bottom - end; | 
|  | r.right = di->rect.right; | 
|  | r.bottom = di->rect.bottom - start; | 
|  | FillRect (di->hdc, &r, di->hbrBk); | 
|  | } | 
|  |  | 
|  | /* draw chunky horizontal bar from 'start' to 'end' */ | 
|  | static void draw_chunk_bar_H (const ProgressDrawInfo* di, int start, int end) | 
|  | { | 
|  | RECT r; | 
|  | int right = di->rect.left + end; | 
|  | r.left = di->rect.left + start; | 
|  | r.top = di->rect.top; | 
|  | r.bottom = di->rect.bottom; | 
|  | while (r.left < right) | 
|  | { | 
|  | r.right = min (r.left + di->ledW, right); | 
|  | FillRect (di->hdc, &r, di->hbrBar); | 
|  | r.left = r.right; | 
|  | r.right = min (r.left + di->ledGap, right); | 
|  | FillRect (di->hdc, &r, di->hbrBk); | 
|  | r.left = r.right; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* draw chunky vertical bar from 'start' to 'end' */ | 
|  | static void draw_chunk_bar_V (const ProgressDrawInfo* di, int start, int end) | 
|  | { | 
|  | RECT r; | 
|  | int top = di->rect.bottom - end; | 
|  | r.left = di->rect.left; | 
|  | r.right = di->rect.right; | 
|  | r.bottom = di->rect.bottom - start; | 
|  | while (r.bottom > top) | 
|  | { | 
|  | r.top = max (r.bottom - di->ledW, top); | 
|  | FillRect (di->hdc, &r, di->hbrBar); | 
|  | r.bottom = r.top; | 
|  | r.top = max (r.bottom - di->ledGap, top); | 
|  | FillRect (di->hdc, &r, di->hbrBk); | 
|  | r.bottom = r.top; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* drawing functions for "classic" style */ | 
|  | static const ProgressDrawProc drawProcClassic[8] = { | 
|  | /* Smooth */ | 
|  | /* Horizontal */ | 
|  | draw_solid_bar_H, draw_solid_bkg_H, | 
|  | /* Vertical */ | 
|  | draw_solid_bar_V, draw_solid_bkg_V, | 
|  | /* Chunky */ | 
|  | /* Horizontal */ | 
|  | draw_chunk_bar_H, draw_solid_bkg_H, | 
|  | /* Vertical */ | 
|  | draw_chunk_bar_V, draw_solid_bkg_V, | 
|  | }; | 
|  |  | 
|  | /* draw themed horizontal bar from 'start' to 'end' */ | 
|  | static void draw_theme_bar_H (const ProgressDrawInfo* di, int start, int end) | 
|  | { | 
|  | RECT r; | 
|  | int right = di->rect.left + end; | 
|  | r.left = di->rect.left + start; | 
|  | r.top = di->rect.top; | 
|  | r.bottom = di->rect.bottom; | 
|  | while (r.left < right) | 
|  | { | 
|  | r.right = min (r.left + di->ledW, right); | 
|  | DrawThemeBackground (di->theme, di->hdc, PP_CHUNK, 0, &r, NULL); | 
|  | r.left = r.right; | 
|  | r.right = min (r.left + di->ledGap, right); | 
|  | DrawThemeBackground (di->theme, di->hdc, PP_BAR, 0, &di->bgRect, &r); | 
|  | r.left = r.right; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* draw themed horizontal bar from 'start' to 'end' */ | 
|  | static void draw_theme_bar_V (const ProgressDrawInfo* di, int start, int end) | 
|  | { | 
|  | RECT r; | 
|  | int top = di->rect.bottom - end; | 
|  | r.left = di->rect.left; | 
|  | r.right = di->rect.right; | 
|  | r.bottom = di->rect.bottom - start; | 
|  | while (r.bottom > top) | 
|  | { | 
|  | r.top = max (r.bottom - di->ledW, top); | 
|  | DrawThemeBackground (di->theme, di->hdc, PP_CHUNKVERT, 0, &r, NULL); | 
|  | r.bottom = r.top; | 
|  | r.top = max (r.bottom - di->ledGap, top); | 
|  | DrawThemeBackground (di->theme, di->hdc, PP_BARVERT, 0, &di->bgRect, &r); | 
|  | r.bottom = r.top; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* draw themed horizontal background from 'start' to 'end' */ | 
|  | static void draw_theme_bkg_H (const ProgressDrawInfo* di, int start, int end) | 
|  | { | 
|  | RECT r; | 
|  | r.left = di->rect.left + start; | 
|  | r.top = di->rect.top; | 
|  | r.right = di->rect.left + end; | 
|  | r.bottom = di->rect.bottom; | 
|  | DrawThemeBackground (di->theme, di->hdc, PP_BAR, 0, &di->bgRect, &r); | 
|  | } | 
|  |  | 
|  | /* draw themed vertical background from 'start' to 'end' */ | 
|  | static void draw_theme_bkg_V (const ProgressDrawInfo* di, int start, int end) | 
|  | { | 
|  | RECT r; | 
|  | r.left = di->rect.left; | 
|  | r.top = di->rect.bottom - end; | 
|  | r.right = di->rect.right; | 
|  | r.bottom = di->rect.bottom - start; | 
|  | DrawThemeBackground (di->theme, di->hdc, PP_BARVERT, 0, &di->bgRect, &r); | 
|  | } | 
|  |  | 
|  | /* drawing functions for themed style */ | 
|  | static const ProgressDrawProc drawProcThemed[8] = { | 
|  | /* Smooth */ | 
|  | /* Horizontal */ | 
|  | draw_theme_bar_H, draw_theme_bkg_H, | 
|  | /* Vertical */ | 
|  | draw_theme_bar_V, draw_theme_bkg_V, | 
|  | /* Chunky */ | 
|  | /* Horizontal */ | 
|  | draw_theme_bar_H, draw_theme_bkg_H, | 
|  | /* Vertical */ | 
|  | draw_theme_bar_V, draw_theme_bkg_V, | 
|  | }; | 
|  |  | 
|  | /*********************************************************************** | 
|  | * PROGRESS_Draw | 
|  | * Draws the progress bar. | 
|  | */ | 
|  | static LRESULT PROGRESS_Draw (PROGRESS_INFO *infoPtr, HDC hdc) | 
|  | { | 
|  | int barSize; | 
|  | DWORD dwStyle; | 
|  | BOOL barSmooth; | 
|  | const ProgressDrawProc* drawProcs; | 
|  | ProgressDrawInfo pdi; | 
|  |  | 
|  | TRACE("(infoPtr=%p, hdc=%p)\n", infoPtr, hdc); | 
|  |  | 
|  | pdi.hdc = hdc; | 
|  | pdi.theme = GetWindowTheme (infoPtr->Self); | 
|  |  | 
|  | /* get the required bar brush */ | 
|  | if (infoPtr->ColorBar == CLR_DEFAULT) | 
|  | pdi.hbrBar = GetSysColorBrush(COLOR_HIGHLIGHT); | 
|  | else | 
|  | pdi.hbrBar = CreateSolidBrush (infoPtr->ColorBar); | 
|  |  | 
|  | if (infoPtr->ColorBk == CLR_DEFAULT) | 
|  | pdi.hbrBk = GetSysColorBrush(COLOR_3DFACE); | 
|  | else | 
|  | pdi.hbrBk = CreateSolidBrush(infoPtr->ColorBk); | 
|  |  | 
|  | /* get the window style */ | 
|  | dwStyle = GetWindowLongW (infoPtr->Self, GWL_STYLE); | 
|  |  | 
|  | /* get client rectangle */ | 
|  | GetClientRect (infoPtr->Self, &pdi.rect); | 
|  | if (!pdi.theme) { | 
|  | FrameRect( hdc, &pdi.rect, pdi.hbrBk ); | 
|  | InflateRect(&pdi.rect, -1, -1); | 
|  | } | 
|  | else | 
|  | { | 
|  | RECT cntRect; | 
|  | int part = (dwStyle & PBS_VERTICAL) ? PP_BARVERT : PP_BAR; | 
|  |  | 
|  | GetThemeBackgroundContentRect (pdi.theme, hdc, part, 0, &pdi.rect, | 
|  | &cntRect); | 
|  |  | 
|  | /* Exclude content rect - content background will be drawn later */ | 
|  | ExcludeClipRect (hdc, cntRect.left, cntRect.top, | 
|  | cntRect.right, cntRect.bottom); | 
|  | if (IsThemeBackgroundPartiallyTransparent (pdi.theme, part, 0)) | 
|  | DrawThemeParentBackground (infoPtr->Self, hdc, NULL); | 
|  | DrawThemeBackground (pdi.theme, hdc, part, 0, &pdi.rect, NULL); | 
|  | SelectClipRgn (hdc, NULL); | 
|  | CopyRect (&pdi.rect, &cntRect); | 
|  | } | 
|  |  | 
|  | /* compute some drawing parameters */ | 
|  | barSmooth = (dwStyle & PBS_SMOOTH) && !pdi.theme; | 
|  | drawProcs = &((pdi.theme ? drawProcThemed : drawProcClassic)[(barSmooth ? 0 : 4) | 
|  | + ((dwStyle & PBS_VERTICAL) ? 2 : 0)]); | 
|  | barSize = get_bar_size( dwStyle, &pdi.rect ); | 
|  | if (pdi.theme) | 
|  | { | 
|  | GetWindowRect( infoPtr->Self, &pdi.bgRect ); | 
|  | ScreenToClient( infoPtr->Self, (POINT*)&pdi.bgRect ); | 
|  | ScreenToClient( infoPtr->Self, (POINT*)&pdi.bgRect.right ); | 
|  | } | 
|  |  | 
|  | if (!barSmooth) | 
|  | pdi.ledW = get_led_size( infoPtr, dwStyle, &pdi.rect); | 
|  | pdi.ledGap = get_led_gap( infoPtr ); | 
|  |  | 
|  | if (dwStyle & PBS_MARQUEE) | 
|  | { | 
|  | const int ledW = !barSmooth ? (pdi.ledW + pdi.ledGap) : 1; | 
|  | const int leds = (barSize + ledW - 1) / ledW; | 
|  | const int ledMEnd = infoPtr->MarqueePos + MARQUEE_LEDS; | 
|  |  | 
|  | if (ledMEnd > leds) | 
|  | { | 
|  | /* case 1: the marquee bar extends over the end and wraps around to | 
|  | * the start */ | 
|  | const int gapStart = max((ledMEnd - leds) * ledW, 0); | 
|  | const int gapEnd = min(infoPtr->MarqueePos * ledW, barSize); | 
|  |  | 
|  | drawProcs[0]( &pdi, 0, gapStart); | 
|  | drawProcs[1]( &pdi, gapStart, gapEnd); | 
|  | drawProcs[0]( &pdi, gapEnd, barSize); | 
|  | } | 
|  | else | 
|  | { | 
|  | /* case 2: the marquee bar is between start and end */ | 
|  | const int barStart = infoPtr->MarqueePos * ledW; | 
|  | const int barEnd = min (ledMEnd * ledW, barSize); | 
|  |  | 
|  | drawProcs[1]( &pdi, 0, barStart); | 
|  | drawProcs[0]( &pdi, barStart, barEnd); | 
|  | drawProcs[1]( &pdi, barEnd, barSize); | 
|  | } | 
|  | } | 
|  | else | 
|  | { | 
|  | int barEnd = get_bar_position( infoPtr, dwStyle, &pdi.rect, | 
|  | infoPtr->CurVal); | 
|  | if (!barSmooth) | 
|  | { | 
|  | const int ledW = pdi.ledW + pdi.ledGap; | 
|  | barEnd = min (((barEnd + ledW - 1) / ledW) * ledW, barSize); | 
|  | } | 
|  | drawProcs[0]( &pdi, 0, barEnd); | 
|  | drawProcs[1]( &pdi, barEnd, barSize); | 
|  | } | 
|  |  | 
|  | /* delete bar brush */ | 
|  | if (infoPtr->ColorBar != CLR_DEFAULT) DeleteObject (pdi.hbrBar); | 
|  | if (infoPtr->ColorBk != CLR_DEFAULT) DeleteObject (pdi.hbrBk); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /*********************************************************************** | 
|  | * PROGRESS_Paint | 
|  | * Draw the progress bar. The background need not be erased. | 
|  | * If dc!=0, it draws on it | 
|  | */ | 
|  | static LRESULT PROGRESS_Paint (PROGRESS_INFO *infoPtr, HDC hdc) | 
|  | { | 
|  | PAINTSTRUCT ps; | 
|  | if (hdc) return PROGRESS_Draw (infoPtr, hdc); | 
|  | hdc = BeginPaint (infoPtr->Self, &ps); | 
|  | PROGRESS_Draw (infoPtr, hdc); | 
|  | EndPaint (infoPtr->Self, &ps); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | * PROGRESS_Timer | 
|  | * Handle the marquee timer messages | 
|  | */ | 
|  | static LRESULT PROGRESS_Timer (PROGRESS_INFO *infoPtr, INT idTimer) | 
|  | { | 
|  | if(idTimer == ID_MARQUEE_TIMER) | 
|  | { | 
|  | LONG style = GetWindowLongW (infoPtr->Self, GWL_STYLE); | 
|  | RECT rect; | 
|  | int ledWidth, leds; | 
|  | HTHEME theme = GetWindowTheme (infoPtr->Self); | 
|  | BOOL barSmooth = (style & PBS_SMOOTH) && !theme; | 
|  |  | 
|  | get_client_rect (infoPtr->Self, &rect); | 
|  |  | 
|  | if(!barSmooth) | 
|  | ledWidth = get_led_size( infoPtr, style, &rect ) + | 
|  | get_led_gap( infoPtr ); | 
|  | else | 
|  | ledWidth = 1; | 
|  |  | 
|  | leds = (get_bar_size( style, &rect ) + ledWidth - 1) / | 
|  | ledWidth; | 
|  |  | 
|  | /* increment the marquee progress */ | 
|  | if(++infoPtr->MarqueePos >= leds) | 
|  | { | 
|  | infoPtr->MarqueePos = 0; | 
|  | } | 
|  |  | 
|  | InvalidateRect(infoPtr->Self, &rect, FALSE); | 
|  | UpdateWindow(infoPtr->Self); | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           PROGRESS_CoercePos | 
|  | * Makes sure the current position (CurVal) is within bounds. | 
|  | */ | 
|  | static void PROGRESS_CoercePos(PROGRESS_INFO *infoPtr) | 
|  | { | 
|  | if(infoPtr->CurVal < infoPtr->MinVal) | 
|  | infoPtr->CurVal = infoPtr->MinVal; | 
|  | if(infoPtr->CurVal > infoPtr->MaxVal) | 
|  | infoPtr->CurVal = infoPtr->MaxVal; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           PROGRESS_SetFont | 
|  | * Set new Font for progress bar | 
|  | */ | 
|  | static HFONT PROGRESS_SetFont (PROGRESS_INFO *infoPtr, HFONT hFont, BOOL bRedraw) | 
|  | { | 
|  | HFONT hOldFont = infoPtr->Font; | 
|  | infoPtr->Font = hFont; | 
|  | /* Since infoPtr->Font is not used, there is no need for repaint */ | 
|  | return hOldFont; | 
|  | } | 
|  |  | 
|  | static DWORD PROGRESS_SetRange (PROGRESS_INFO *infoPtr, int low, int high) | 
|  | { | 
|  | DWORD res = MAKELONG(LOWORD(infoPtr->MinVal), LOWORD(infoPtr->MaxVal)); | 
|  |  | 
|  | /* if nothing changes, simply return */ | 
|  | if(infoPtr->MinVal == low && infoPtr->MaxVal == high) return res; | 
|  |  | 
|  | infoPtr->MinVal = low; | 
|  | infoPtr->MaxVal = high; | 
|  | PROGRESS_CoercePos(infoPtr); | 
|  | InvalidateRect(infoPtr->Self, NULL, TRUE); | 
|  | return res; | 
|  | } | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           ProgressWindowProc | 
|  | */ | 
|  | static LRESULT WINAPI ProgressWindowProc(HWND hwnd, UINT message, | 
|  | WPARAM wParam, LPARAM lParam) | 
|  | { | 
|  | PROGRESS_INFO *infoPtr; | 
|  | static const WCHAR themeClass[] = {'P','r','o','g','r','e','s','s',0}; | 
|  | HTHEME theme; | 
|  |  | 
|  | TRACE("hwnd=%p msg=%04x wparam=%lx lParam=%lx\n", hwnd, message, wParam, lParam); | 
|  |  | 
|  | infoPtr = (PROGRESS_INFO *)GetWindowLongPtrW(hwnd, 0); | 
|  |  | 
|  | if (!infoPtr && message != WM_CREATE) | 
|  | return DefWindowProcW( hwnd, message, wParam, lParam ); | 
|  |  | 
|  | switch(message) { | 
|  | case WM_CREATE: | 
|  | { | 
|  | DWORD dwExStyle = GetWindowLongW (hwnd, GWL_EXSTYLE); | 
|  |  | 
|  | theme = OpenThemeData (hwnd, themeClass); | 
|  |  | 
|  | dwExStyle &= ~(WS_EX_CLIENTEDGE | WS_EX_WINDOWEDGE); | 
|  | if (!theme) dwExStyle |= WS_EX_STATICEDGE; | 
|  | SetWindowLongW (hwnd, GWL_EXSTYLE, dwExStyle); | 
|  | /* Force recalculation of a non-client area */ | 
|  | SetWindowPos(hwnd, 0, 0, 0, 0, 0, | 
|  | SWP_FRAMECHANGED | SWP_NOSIZE | SWP_NOMOVE | SWP_NOZORDER | SWP_NOACTIVATE); | 
|  |  | 
|  | /* allocate memory for info struct */ | 
|  | infoPtr = Alloc (sizeof(PROGRESS_INFO)); | 
|  | if (!infoPtr) return -1; | 
|  | SetWindowLongPtrW (hwnd, 0, (DWORD_PTR)infoPtr); | 
|  |  | 
|  | /* initialize the info struct */ | 
|  | infoPtr->Self = hwnd; | 
|  | infoPtr->MinVal = 0; | 
|  | infoPtr->MaxVal = 100; | 
|  | infoPtr->CurVal = 0; | 
|  | infoPtr->Step = 10; | 
|  | infoPtr->MarqueePos = 0; | 
|  | infoPtr->Marquee = FALSE; | 
|  | infoPtr->ColorBar = CLR_DEFAULT; | 
|  | infoPtr->ColorBk = CLR_DEFAULT; | 
|  | infoPtr->Font = 0; | 
|  |  | 
|  | TRACE("Progress Ctrl creation, hwnd=%p\n", hwnd); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | case WM_DESTROY: | 
|  | TRACE("Progress Ctrl destruction, hwnd=%p\n", hwnd); | 
|  | Free (infoPtr); | 
|  | SetWindowLongPtrW(hwnd, 0, 0); | 
|  | theme = GetWindowTheme (hwnd); | 
|  | CloseThemeData (theme); | 
|  | return 0; | 
|  |  | 
|  | case WM_ERASEBKGND: | 
|  | return 1; | 
|  |  | 
|  | case WM_GETFONT: | 
|  | return (LRESULT)infoPtr->Font; | 
|  |  | 
|  | case WM_SETFONT: | 
|  | return (LRESULT)PROGRESS_SetFont(infoPtr, (HFONT)wParam, (BOOL)lParam); | 
|  |  | 
|  | case WM_PRINTCLIENT: | 
|  | case WM_PAINT: | 
|  | return PROGRESS_Paint (infoPtr, (HDC)wParam); | 
|  |  | 
|  | case WM_TIMER: | 
|  | return PROGRESS_Timer (infoPtr, (INT)wParam); | 
|  |  | 
|  | case WM_THEMECHANGED: | 
|  | { | 
|  | DWORD dwExStyle = GetWindowLongW (hwnd, GWL_EXSTYLE); | 
|  |  | 
|  | theme = GetWindowTheme (hwnd); | 
|  | CloseThemeData (theme); | 
|  | theme = OpenThemeData (hwnd, themeClass); | 
|  |  | 
|  | /* WS_EX_STATICEDGE disappears when the control is themed */ | 
|  | if (theme) | 
|  | dwExStyle &= ~WS_EX_STATICEDGE; | 
|  | else | 
|  | dwExStyle |= WS_EX_STATICEDGE; | 
|  | SetWindowLongW (hwnd, GWL_EXSTYLE, dwExStyle); | 
|  |  | 
|  | InvalidateRect (hwnd, NULL, FALSE); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | case PBM_DELTAPOS: | 
|  | { | 
|  | INT oldVal; | 
|  | oldVal = infoPtr->CurVal; | 
|  | if(wParam != 0) { | 
|  | infoPtr->CurVal += (INT)wParam; | 
|  | PROGRESS_CoercePos (infoPtr); | 
|  | TRACE("PBM_DELTAPOS: current pos changed from %d to %d\n", oldVal, infoPtr->CurVal); | 
|  | PROGRESS_Invalidate( infoPtr, oldVal, infoPtr->CurVal ); | 
|  | UpdateWindow( infoPtr->Self ); | 
|  | } | 
|  | return oldVal; | 
|  | } | 
|  |  | 
|  | case PBM_SETPOS: | 
|  | { | 
|  | UINT oldVal; | 
|  | oldVal = infoPtr->CurVal; | 
|  | if(oldVal != wParam) { | 
|  | infoPtr->CurVal = (INT)wParam; | 
|  | PROGRESS_CoercePos(infoPtr); | 
|  | TRACE("PBM_SETPOS: current pos changed from %d to %d\n", oldVal, infoPtr->CurVal); | 
|  | PROGRESS_Invalidate( infoPtr, oldVal, infoPtr->CurVal ); | 
|  | UpdateWindow( infoPtr->Self ); | 
|  | } | 
|  | return oldVal; | 
|  | } | 
|  |  | 
|  | case PBM_SETRANGE: | 
|  | return PROGRESS_SetRange (infoPtr, (int)LOWORD(lParam), (int)HIWORD(lParam)); | 
|  |  | 
|  | case PBM_SETSTEP: | 
|  | { | 
|  | INT oldStep; | 
|  | oldStep = infoPtr->Step; | 
|  | infoPtr->Step = (INT)wParam; | 
|  | return oldStep; | 
|  | } | 
|  |  | 
|  | case PBM_STEPIT: | 
|  | { | 
|  | INT oldVal; | 
|  | oldVal = infoPtr->CurVal; | 
|  | infoPtr->CurVal += infoPtr->Step; | 
|  | if(infoPtr->CurVal > infoPtr->MaxVal) | 
|  | infoPtr->CurVal = infoPtr->MinVal; | 
|  | if(oldVal != infoPtr->CurVal) | 
|  | { | 
|  | TRACE("PBM_STEPIT: current pos changed from %d to %d\n", oldVal, infoPtr->CurVal); | 
|  | PROGRESS_Invalidate( infoPtr, oldVal, infoPtr->CurVal ); | 
|  | UpdateWindow( infoPtr->Self ); | 
|  | } | 
|  | return oldVal; | 
|  | } | 
|  |  | 
|  | case PBM_SETRANGE32: | 
|  | return PROGRESS_SetRange (infoPtr, (int)wParam, (int)lParam); | 
|  |  | 
|  | case PBM_GETRANGE: | 
|  | if (lParam) { | 
|  | ((PPBRANGE)lParam)->iLow = infoPtr->MinVal; | 
|  | ((PPBRANGE)lParam)->iHigh = infoPtr->MaxVal; | 
|  | } | 
|  | return wParam ? infoPtr->MinVal : infoPtr->MaxVal; | 
|  |  | 
|  | case PBM_GETPOS: | 
|  | return infoPtr->CurVal; | 
|  |  | 
|  | case PBM_SETBARCOLOR: | 
|  | infoPtr->ColorBar = (COLORREF)lParam; | 
|  | InvalidateRect(hwnd, NULL, TRUE); | 
|  | return 0; | 
|  |  | 
|  | case PBM_GETBARCOLOR: | 
|  | return infoPtr->ColorBar; | 
|  |  | 
|  | case PBM_SETBKCOLOR: | 
|  | infoPtr->ColorBk = (COLORREF)lParam; | 
|  | InvalidateRect(hwnd, NULL, TRUE); | 
|  | return 0; | 
|  |  | 
|  | case PBM_GETBKCOLOR: | 
|  | return infoPtr->ColorBk; | 
|  |  | 
|  | case PBM_SETMARQUEE: | 
|  | if(wParam != 0) | 
|  | { | 
|  | infoPtr->Marquee = TRUE; | 
|  | SetTimer(infoPtr->Self, ID_MARQUEE_TIMER, (UINT)lParam, NULL); | 
|  | } | 
|  | else | 
|  | { | 
|  | infoPtr->Marquee = FALSE; | 
|  | KillTimer(infoPtr->Self, ID_MARQUEE_TIMER); | 
|  | } | 
|  | return infoPtr->Marquee; | 
|  |  | 
|  | default: | 
|  | if ((message >= WM_USER) && (message < WM_APP) && !COMCTL32_IsReflectedMessage(message)) | 
|  | ERR("unknown msg %04x wp=%04lx lp=%08lx\n", message, wParam, lParam ); | 
|  | return DefWindowProcW( hwnd, message, wParam, lParam ); | 
|  | } | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | * PROGRESS_Register [Internal] | 
|  | * | 
|  | * Registers the progress bar window class. | 
|  | */ | 
|  | void PROGRESS_Register (void) | 
|  | { | 
|  | WNDCLASSW wndClass; | 
|  |  | 
|  | ZeroMemory (&wndClass, sizeof(wndClass)); | 
|  | wndClass.style         = CS_GLOBALCLASS | CS_VREDRAW | CS_HREDRAW; | 
|  | wndClass.lpfnWndProc   = (WNDPROC)ProgressWindowProc; | 
|  | wndClass.cbClsExtra    = 0; | 
|  | wndClass.cbWndExtra    = sizeof (PROGRESS_INFO *); | 
|  | wndClass.hCursor       = LoadCursorW (0, (LPWSTR)IDC_ARROW); | 
|  | wndClass.lpszClassName = PROGRESS_CLASSW; | 
|  |  | 
|  | RegisterClassW (&wndClass); | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | * PROGRESS_Unregister [Internal] | 
|  | * | 
|  | * Unregisters the progress bar window class. | 
|  | */ | 
|  | void PROGRESS_Unregister (void) | 
|  | { | 
|  | UnregisterClassW (PROGRESS_CLASSW, NULL); | 
|  | } |