(Merged by Marcus Meissner <marcus@jet.franken.de>)
Fixed problem with origin for DST_COMPLEX style.
Fixed handling of DSS_DISABLED and DSS_DEFAULT styles.

Added handling of BS_MULTILINE style, label alignment
styles (BS_RIGHT, etc.) and some exotic styles BS_FLAT (only
pushbuttons for now), and BS_PUSHLIKE.

Modified label drawing procedure: now all kinds of buttons
use common BUTTON_DrawLabel function. Actual label drawing is
performed by DrawStateW function.

GroupBox must use WM_CTLCOLORSTATIC instead of WM_CTLCOLORBTN message.

diff --git a/controls/button.c b/controls/button.c
index 88fd974..560746b 100644
--- a/controls/button.c
+++ b/controls/button.c
@@ -6,6 +6,7 @@
  */
 
 #include <string.h>
+#include <stdlib.h>	/* for abs() */
 #include "win.h"
 #include "button.h"
 #include "winbase.h"
@@ -14,9 +15,6 @@
 #include "wine/winuser16.h"
 #include "tweak.h"
 
-static void PaintGrayOnGray( HDC hDC,HFONT hFont,RECT *rc,
-			     LPCWSTR text, UINT format );
-
 static void PB_Paint( WND *wndPtr, HDC hDC, WORD action );
 static void CB_Paint( WND *wndPtr, HDC hDC, WORD action );
 static void GB_Paint( WND *wndPtr, HDC hDC, WORD action );
@@ -280,21 +278,37 @@
 	break;
 
     case BM_SETIMAGE:
-	oldHbitmap = infoPtr->hImage;
-	if ((wndPtr->dwStyle & BS_BITMAP) || (wndPtr->dwStyle & BS_ICON))
+	/* Check that image format confirm button style */
+	if ((wndPtr->dwStyle & (BS_BITMAP|BS_ICON)) == BS_BITMAP)
 	{
-	    infoPtr->hImage = (HANDLE) lParam;
-	    InvalidateRect( hWnd, NULL, FALSE );
+	    if (wParam != (WPARAM) IMAGE_BITMAP)
+		return (HICON)0;
 	}
+	else if ((wndPtr->dwStyle & (BS_BITMAP|BS_ICON)) == BS_ICON)
+	{
+	    if (wParam != (WPARAM) IMAGE_ICON)
+		return (HICON)0;
+	} else
+	    return (HICON)0;
+
+	oldHbitmap = infoPtr->hImage;
+	infoPtr->hImage = (HANDLE) lParam;
+	InvalidateRect( hWnd, NULL, FALSE );
 	return oldHbitmap;
 
     case BM_GETIMAGE:
-        if (wParam == IMAGE_BITMAP)
-	    return (HBITMAP)infoPtr->hImage;
-	else if (wParam == IMAGE_ICON)
-	    return (HICON)infoPtr->hImage;
-	else
+	if ((wndPtr->dwStyle & (BS_BITMAP|BS_ICON)) == BS_BITMAP)
+	{
+	    if (wParam != (WPARAM) IMAGE_BITMAP)
+		return (HICON)0;
+	}
+	else if ((wndPtr->dwStyle & (BS_BITMAP|BS_ICON)) == BS_ICON)
+	{
+	    if (wParam != (WPARAM) IMAGE_ICON)
+		return (HICON)0;
+	} else
 	    return (HICON)0;
+	return infoPtr->hImage;
 
     case BM_GETCHECK16:
     case BM_GETCHECK:
@@ -381,6 +395,213 @@
 }
 
 /**********************************************************************
+ * Convert button styles to flags used by DrawText.
+ * TODO: handle WS_EX_RIGHT extended style.
+ */
+static UINT BUTTON_BStoDT(DWORD style)
+{
+   UINT dtStyle = DT_NOCLIP;  /* We use SelectClipRgn to limit output */
+
+   /* "Convert" pushlike buttons to pushbuttons */
+   if (style & BS_PUSHLIKE)
+      style &= ~0x0F;
+
+   if (!(style & BS_MULTILINE))
+      dtStyle |= DT_SINGLELINE;
+   else
+      dtStyle |= DT_WORDBREAK;
+
+   switch (style & BS_CENTER)
+   {
+      case BS_LEFT:   /* DT_LEFT is 0 */    break;
+      case BS_RIGHT:  dtStyle |= DT_RIGHT;  break;
+      case BS_CENTER: dtStyle |= DT_CENTER; break;
+      default:
+         /* Pushbutton's text is centered by default */
+         if ((style & 0x0F) <= BS_DEFPUSHBUTTON)
+            dtStyle |= DT_CENTER;
+         /* all other flavours have left aligned text */
+   }
+
+   /* DrawText ignores vertical alignment for multiline text,
+    * but we use these flags to align label manualy.
+    */
+   if ((style & 0x0F) != BS_GROUPBOX)
+   {
+      switch (style & BS_VCENTER)
+      {
+         case BS_TOP:     /* DT_TOP is 0 */      break;
+         case BS_BOTTOM:  dtStyle |= DT_BOTTOM;  break;
+         case BS_VCENTER: /* fall through */
+         default:         dtStyle |= DT_VCENTER; break;
+      }
+   }
+   else
+      /* GroupBox's text is always single line and is top aligned. */
+      dtStyle |= DT_SINGLELINE;
+
+   return dtStyle;
+}
+
+/**********************************************************************
+ *       BUTTON_CalcLabelRect
+ *
+ *   Calculates label's rectangle depending on button style.
+ *
+ * Returns flags to be passed to DrawText.
+ * Calculated rectangle doesn't take into account button state
+ * (pushed, etc.). If there is nothing to draw (no text/image) output
+ * rectangle is empty, and return value is (UINT)-1.
+ */
+static UINT BUTTON_CalcLabelRect(WND *wndPtr, HDC hdc, RECT *rc)
+{
+   BUTTONINFO *infoPtr = (BUTTONINFO *)wndPtr->wExtra;
+   ICONINFO    iconInfo;
+   BITMAP      bm;
+   UINT        dtStyle = BUTTON_BStoDT(wndPtr->dwStyle);
+   RECT        r = *rc;
+   INT         n;
+
+   /* Calculate label rectangle according to label type */
+   switch (wndPtr->dwStyle & (BS_ICON|BS_BITMAP))
+   {
+      case BS_TEXT:
+         if (wndPtr->text && wndPtr->text[0])
+            DrawTextW(hdc, wndPtr->text, -1, &r, dtStyle | DT_CALCRECT);
+         else
+            goto empty_rect;
+         break;
+
+      case BS_ICON:
+         if (!GetIconInfo((HICON)infoPtr->hImage, &iconInfo))
+            goto empty_rect;
+
+         GetObjectA (iconInfo.hbmColor, sizeof(BITMAP), &bm);
+
+         r.right  = r.left + bm.bmWidth;
+         r.bottom = r.top  + bm.bmHeight;
+
+         DeleteObject(iconInfo.hbmColor);
+         DeleteObject(iconInfo.hbmMask);
+         break;
+
+      case BS_BITMAP:
+         if (!GetObjectA (infoPtr->hImage, sizeof(BITMAP), &bm))
+            goto empty_rect;
+
+         r.right  = r.left + bm.bmWidth;
+         r.bottom = r.top  + bm.bmHeight;
+         break;
+
+      default:
+      empty_rect:   
+         r.right = r.left;
+         r.bottom = r.top;
+         return (UINT)(LONG)-1;
+   }
+
+   /* Position label inside bounding rectangle according to
+    * alignment flags. (calculated rect is always left-top aligned).
+    * If label is aligned to any side - shift label in opposite
+    * direction to leave extra space for focus rectangle.
+    */
+   switch (dtStyle & (DT_CENTER|DT_RIGHT))
+   {
+      case DT_LEFT:    r.left++;  r.right++;  break;
+      case DT_CENTER:  n = r.right - r.left;
+                       r.left   = rc->left + ((rc->right - rc->left) - n) / 2;
+                       r.right  = r.left + n; break;
+      case DT_RIGHT:   n = r.right - r.left;
+                       r.right  = rc->right - 1;
+                       r.left   = r.right - n;
+                       break;
+   }
+
+   switch (dtStyle & (DT_VCENTER|DT_BOTTOM))
+   {
+      case DT_TOP:     r.top++;  r.bottom++;  break;
+      case DT_VCENTER: n = r.bottom - r.top;
+                       r.top    = rc->top + ((rc->bottom - rc->top) - n) / 2;
+                       r.bottom = r.top + n;  break;
+      case DT_BOTTOM:  n = r.bottom - r.top;
+                       r.bottom = rc->bottom - 1;
+                       r.top    = r.bottom - n;
+                       break;
+   }
+
+   *rc = r;
+   return dtStyle;
+}
+
+
+/**********************************************************************
+ *       BUTTON_DrawTextCallback
+ *
+ *   Callback function used be DrawStateA function.
+ */
+static BOOL CALLBACK BUTTON_DrawTextCallback(HDC hdc, LPARAM lp, WPARAM wp, int cx, int cy)
+{
+   RECT rc = {0, 0, cx, cy};
+
+   DrawTextW(hdc, (LPCWSTR)lp, -1, &rc, (UINT)wp);
+   return TRUE;
+}
+
+
+/**********************************************************************
+ *       BUTTON_DrawLabel
+ *
+ *   Common function for drawing button label.
+ */
+static void BUTTON_DrawLabel(WND *wndPtr, HDC hdc, UINT dtFlags,
+                              RECT *rc)
+{
+   BUTTONINFO *infoPtr = (BUTTONINFO *)wndPtr->wExtra;
+   DRAWSTATEPROC lpOutputProc = NULL;
+   LPARAM lp;
+   WPARAM wp = 0;
+   HBRUSH hbr = 0;
+   UINT flags = IsWindowEnabled(wndPtr->hwndSelf) ? DSS_NORMAL : DSS_DISABLED;
+
+   /* Fixme: To draw disabled label in Win31 look-and-feel, we probably
+    * must use DSS_MONO flag and COLOR_GRAYTEXT brush (or maybe DSS_UNION).
+    * I don't have Win31 on hand to verify that, so I leave it as is.
+    */
+
+   if ((wndPtr->dwStyle & BS_PUSHLIKE) && (infoPtr->state & BUTTON_3STATE))
+   {
+      hbr = GetSysColorBrush(COLOR_GRAYTEXT);
+      flags |= DSS_MONO;
+   }
+
+   switch (wndPtr->dwStyle & (BS_ICON|BS_BITMAP))
+   {
+      case BS_TEXT:
+         /* DST_COMPLEX -- is 0 */
+         lpOutputProc = BUTTON_DrawTextCallback;
+         lp = (LPARAM)wndPtr->text;
+         wp = (WPARAM)dtFlags;
+         break;
+
+      case BS_ICON:
+         flags |= DST_ICON;
+         lp = (LPARAM)infoPtr->hImage;
+         break;
+
+      case BS_BITMAP:
+         flags |= DST_BITMAP;
+         lp = (LPARAM)infoPtr->hImage;
+         break;
+
+      default:
+         return;
+   }
+
+   DrawStateW(hdc, hbr, lpOutputProc, lp, wp, rc->left, rc->top,
+              rc->right - rc->left, rc->bottom - rc->top, flags);
+}
+
+/**********************************************************************
  * This method will actually do the drawing of the pushbutton 
  * depending on it's state and the pushedState parameter.
  */
@@ -390,30 +611,33 @@
   WORD action, 
   BOOL pushedState )
 {
-    RECT rc, focus_rect;
-    HPEN hOldPen;
-    HBRUSH hOldBrush;
     BUTTONINFO *infoPtr = (BUTTONINFO *)wndPtr->wExtra;
-    int xBorderOffset, yBorderOffset;
-    xBorderOffset = yBorderOffset = 0;
+    RECT     rc, focus_rect, r;
+    UINT     dtFlags;
+    HRGN     hRgn;
+    HPEN     hOldPen;
+    HBRUSH   hOldBrush;
+    INT      oldBkMode;
+    COLORREF oldTxtColor;
 
     GetClientRect( wndPtr->hwndSelf, &rc );
 
-      /* Send WM_CTLCOLOR to allow changing the font (the colors are fixed) */
+    /* Send WM_CTLCOLOR to allow changing the font (the colors are fixed) */
     if (infoPtr->hFont) SelectObject( hDC, infoPtr->hFont );
     BUTTON_SEND_CTLCOLOR( wndPtr, hDC );
     hOldPen = (HPEN)SelectObject(hDC, GetSysColorPen(COLOR_WINDOWFRAME));
     hOldBrush =(HBRUSH)SelectObject(hDC,GetSysColorBrush(COLOR_BTNFACE));
-    SetBkMode(hDC, TRANSPARENT);
+    oldBkMode = SetBkMode(hDC, TRANSPARENT);
 
     if ( TWEAK_WineLook == WIN31_LOOK)
     {
+        COLORREF clr_wnd = GetSysColor(COLOR_WINDOW);
         Rectangle(hDC, rc.left, rc.top, rc.right, rc.bottom);
 
-        SetPixel( hDC, rc.left, rc.top, GetSysColor(COLOR_WINDOW) );
-        SetPixel( hDC, rc.left, rc.bottom-1, GetSysColor(COLOR_WINDOW) );
-        SetPixel( hDC, rc.right-1, rc.top, GetSysColor(COLOR_WINDOW) );
-        SetPixel( hDC, rc.right-1, rc.bottom-1, GetSysColor(COLOR_WINDOW));
+        SetPixel( hDC, rc.left, rc.top, clr_wnd);
+        SetPixel( hDC, rc.left, rc.bottom-1, clr_wnd);
+        SetPixel( hDC, rc.right-1, rc.top, clr_wnd);
+        SetPixel( hDC, rc.right-1, rc.bottom-1, clr_wnd);
 	InflateRect( &rc, -1, -1 );
     }
     
@@ -431,24 +655,19 @@
 	    SelectObject(hDC, GetSysColorBrush(COLOR_BTNSHADOW));
 	    PatBlt(hDC, rc.left, rc.top, 1, rc.bottom-rc.top, PATCOPY );
 	    PatBlt(hDC, rc.left, rc.top, rc.right-rc.left, 1, PATCOPY );
-	    rc.left += 2;  /* To position the text down and right */
-	    rc.top  += 2;
 	} else {
 	   rc.right++, rc.bottom++;
 	   DrawEdge( hDC, &rc, EDGE_RAISED, BF_RECT );
-
-	   /* To place the bitmap correctly */
-	   xBorderOffset += GetSystemMetrics(SM_CXEDGE);
-	   yBorderOffset += GetSystemMetrics(SM_CYEDGE);
-
 	   rc.right--, rc.bottom--;
 	}
     }
     else
     {
-        UINT uState = DFCS_BUTTONPUSH;
+        UINT uState = DFCS_BUTTONPUSH | DFCS_ADJUSTRECT;
 
-        if (pushedState)
+        if (wndPtr->dwStyle & BS_FLAT)
+            uState |= DFCS_MONO;
+        else if (pushedState)
 	{
 	    if ( (wndPtr->dwStyle & 0x000f) == BS_DEFPUSHBUTTON )
 	        uState |= DFCS_FLAT;
@@ -456,211 +675,54 @@
 	        uState |= DFCS_PUSHED;
 	}
 
+        if (infoPtr->state & (BUTTON_CHECKED | BUTTON_3STATE))
+            uState |= DFCS_CHECKED;
+
 	DrawFrameControl( hDC, &rc, DFC_BUTTON, uState );
-	InflateRect( &rc, -2, -2 );
 
 	focus_rect = rc;
-
-        if (pushedState)
-	{
-	    rc.left += 2;  /* To position the text down and right */
-	    rc.top  += 2;
-	}
     }
 
-    /* draw button label, if any:
-     *
-     * In win9x we don't show text if there is a bitmap or icon.
-     * I don't know about win31 so I leave it as it was for win31.
-     * Dennis Björklund 12 Jul, 99
-     */
-    if ( wndPtr->text && wndPtr->text[0]
-	 && (TWEAK_WineLook == WIN31_LOOK || !(wndPtr->dwStyle & (BS_ICON|BS_BITMAP))) )
+    /* draw button label */
+    r = rc;
+    dtFlags = BUTTON_CalcLabelRect(wndPtr, hDC, &r);
+
+    if (dtFlags == (UINT)-1L)
+       goto cleanup;
+
+    if (pushedState)
+       OffsetRect(&r, 1, 1);
+
+    if(TWEAK_WineLook == WIN31_LOOK)
     {
-        LOGBRUSH lb;
-        GetObjectA( GetSysColorBrush(COLOR_BTNFACE), sizeof(lb), &lb );
-        if (wndPtr->dwStyle & WS_DISABLED &&
-            GetSysColor(COLOR_GRAYTEXT)==lb.lbColor)
-            /* don't write gray text on gray background */
-            PaintGrayOnGray( hDC,infoPtr->hFont,&rc,wndPtr->text,
-			       DT_CENTER | DT_VCENTER );
-        else
-        {
-            SetTextColor( hDC, (wndPtr->dwStyle & WS_DISABLED) ?
-                                 GetSysColor(COLOR_GRAYTEXT) :
-                                 GetSysColor(COLOR_BTNTEXT) );
-            DrawTextW( hDC, wndPtr->text, -1, &rc,
-                         DT_SINGLELINE | DT_CENTER | DT_VCENTER );
-            /* do we have the focus?
-	     * Win9x draws focus last with a size prop. to the button
-	     */
-            if (TWEAK_WineLook == WIN31_LOOK
-		&& infoPtr->state & BUTTON_HASFOCUS)
-            {
-                RECT r = { 0, 0, 0, 0 };
-                INT xdelta, ydelta;
-
-                DrawTextW( hDC, wndPtr->text, -1, &r,
-                             DT_SINGLELINE | DT_CALCRECT );
-                xdelta = ((rc.right - rc.left) - (r.right - r.left) - 1) / 2;
-                ydelta = ((rc.bottom - rc.top) - (r.bottom - r.top) - 1) / 2;
-                if (xdelta < 0) xdelta = 0;
-                if (ydelta < 0) ydelta = 0;
-                InflateRect( &rc, -xdelta, -ydelta );
-                DrawFocusRect( hDC, &rc );
-            }
-        }   
-    }
-    if ( ((wndPtr->dwStyle & BS_ICON) || (wndPtr->dwStyle & BS_BITMAP) ) &&
-	 (infoPtr->hImage != 0) )
-    {
-	int yOffset, xOffset;
-	int imageWidth, imageHeight;
-
-	/*
-	 * We extract the size of the image from the handle.
-	 */
-	if (wndPtr->dwStyle & BS_ICON)
-	{
-	    ICONINFO iconInfo;
-	    BITMAP   bm;
-
-	    GetIconInfo((HICON)infoPtr->hImage, &iconInfo);
-	    GetObjectA (iconInfo.hbmColor, sizeof(BITMAP), &bm);
-	    
-	    imageWidth  = bm.bmWidth;
-	    imageHeight = bm.bmHeight;
-
-            DeleteObject(iconInfo.hbmColor);
-            DeleteObject(iconInfo.hbmMask);
-
-	}
-	else
-	{
-	    BITMAP   bm;
-
-	    GetObjectA (infoPtr->hImage, sizeof(BITMAP), &bm);
-	
-	    imageWidth  = bm.bmWidth;
-	    imageHeight = bm.bmHeight;
-	}
-
-	/* Center the bitmap */
-	xOffset = (((rc.right - rc.left) - 2*xBorderOffset) - imageWidth ) / 2;
-	yOffset = (((rc.bottom - rc.top) - 2*yBorderOffset) - imageHeight) / 2;
-
-	/* If the image is too big for the button then create a region */
-        if(xOffset < 0 || yOffset < 0)
-	{
-            HRGN hBitmapRgn = 0;
-            hBitmapRgn = CreateRectRgn(
-                rc.left + xBorderOffset, rc.top +yBorderOffset, 
-                rc.right - xBorderOffset, rc.bottom - yBorderOffset);
-            SelectClipRgn(hDC, hBitmapRgn);
-            DeleteObject(hBitmapRgn);
-	}
-
-	/* Let minimum 1 space from border */
-	xOffset++, yOffset++;
-
-	/*
-	 * Draw the image now.
-	 */
-	if (wndPtr->dwStyle & BS_ICON)
-	{
-  	    DrawIcon(hDC,
-                rc.left + xOffset, rc.top + yOffset,
-		     (HICON)infoPtr->hImage);
-	}
-	else
-        {
-	    HDC hdcMem;
-
-	    hdcMem = CreateCompatibleDC (hDC);
-	    SelectObject (hdcMem, (HBITMAP)infoPtr->hImage);
-	    BitBlt(hDC, 
-		   rc.left + xOffset, 
-		   rc.top + yOffset, 
-		   imageWidth, imageHeight,
-		   hdcMem, 0, 0, SRCCOPY);
-	    
-	    DeleteDC (hdcMem);
-	}
-
-        if(xOffset < 0 || yOffset < 0)
-        {
-            SelectClipRgn(hDC, 0);
-        }
+       focus_rect = r;
+       InflateRect(&focus_rect, 2, 0);
     }
 
-    if (TWEAK_WineLook != WIN31_LOOK
-	&& infoPtr->state & BUTTON_HASFOCUS)
+    hRgn = CreateRectRgn(rc.left, rc.top, rc.right, rc.bottom);
+    SelectClipRgn(hDC, hRgn);
+
+    oldTxtColor = SetTextColor( hDC, GetSysColor(COLOR_BTNTEXT) );
+
+    BUTTON_DrawLabel(wndPtr, hDC, dtFlags, &r);
+
+    SetTextColor( hDC, oldTxtColor );
+    SelectClipRgn(hDC, 0);
+    DeleteObject(hRgn);
+
+    if (infoPtr->state & BUTTON_HASFOCUS)
     {
         InflateRect( &focus_rect, -1, -1 );
+        IntersectRect(&focus_rect, &focus_rect, &rc);
         DrawFocusRect( hDC, &focus_rect );
     }
 
-    
+ cleanup:
     SelectObject( hDC, hOldPen );
     SelectObject( hDC, hOldBrush );
+    SetBkMode(hDC, oldBkMode);
 }
 
-
-/**********************************************************************
- *   PB_Paint & CB_Paint sub function                        [internal]
- *   Paint text using a raster brush to avoid gray text on gray 
- *   background. 'format' can be a combination of DT_CENTER and 
- *   DT_VCENTER to use this function in both PB_PAINT and 
- *   CB_PAINT.   - Dirk Thierbach
- *
- *   FIXME: This and TEXT_GrayString should be eventually combined,
- *   so calling one common function from PB_Paint, CB_Paint and
- *   TEXT_GrayString will be enough. Also note that this
- *   function ignores the CACHE_GetPattern funcs.
- */
-
-void PaintGrayOnGray(HDC hDC, HFONT hFont, RECT *rc, LPCWSTR text,
-			UINT format)
-{
-/*  This is the standard gray on gray pattern:
-    static const WORD Pattern[] = {0xAA,0x55,0xAA,0x55,0xAA,0x55,0xAA,0x55}; 
-*/
-/*  This pattern gives better readability with X Fonts.
-    FIXME: Maybe the user should be allowed to decide which he wants. */
-    static const WORD Pattern[] = {0x55,0xFF,0xAA,0xFF,0x55,0xFF,0xAA,0xFF}; 
-
-    HBITMAP hbm  = CreateBitmap( 8, 8, 1, 1, Pattern );
-    HDC hdcMem = CreateCompatibleDC(hDC);
-    HBITMAP hbmMem;
-    HBRUSH hBr;
-    RECT rect,rc2;
-
-    rect=*rc;
-    DrawTextW( hDC, text, -1, &rect, DT_SINGLELINE | DT_CALCRECT);
-    /* now text width and height are in rect.right and rect.bottom */
-    rc2=rect;
-    rect.left = rect.top = 0; /* drawing pos in hdcMem */
-    if (format & DT_CENTER) rect.left=(rc->right-rect.right)/2;
-    if (format & DT_VCENTER) rect.top=(rc->bottom-rect.bottom)/2;
-    hbmMem = CreateCompatibleBitmap( hDC,rect.right,rect.bottom );
-    SelectObject( hdcMem, hbmMem);
-    PatBlt( hdcMem,0,0,rect.right,rect.bottom,WHITENESS);
-      /* will be overwritten by DrawText, but just in case */
-    if (hFont) SelectObject( hdcMem, hFont);
-    DrawTextW( hdcMem, text, -1, &rc2, DT_SINGLELINE);
-      /* After draw: foreground = 0 bits, background = 1 bits */
-    hBr = SelectObject( hdcMem, CreatePatternBrush(hbm) );
-    DeleteObject( hbm );
-    PatBlt( hdcMem,0,0,rect.right,rect.bottom,0xAF0229); 
-      /* only keep the foreground bits where pattern is 1 */
-    DeleteObject( SelectObject( hdcMem,hBr) );
-    BitBlt(hDC,rect.left,rect.top,rect.right,rect.bottom,hdcMem,0,0,SRCAND);
-      /* keep the background of the dest */
-    DeleteDC( hdcMem);
-    DeleteObject( hbmMem );
-}
-
-
 /**********************************************************************
  *       Check Box & Radio Button Functions
  */
@@ -669,17 +731,14 @@
 {
     RECT rbox, rtext, client;
     HBRUSH hBrush;
-    int textlen, delta;
+    int delta;
     BUTTONINFO *infoPtr = (BUTTONINFO *)wndPtr->wExtra;
+    UINT dtFlags;
+    HRGN hRgn;
 
-    /* 
-     * if the button has a bitmap/icon, draw a normal pushbutton
-     * instead of a radion button.
-     */
-    if (infoPtr->hImage != 0)
+    if (wndPtr->dwStyle & BS_PUSHLIKE)
     {
-        BOOL bHighLighted = ((infoPtr->state & BUTTON_HIGHLIGHTED) ||
-			     (infoPtr->state & BUTTON_CHECKED));
+        BOOL bHighLighted = (infoPtr->state & BUTTON_HIGHLIGHTED);
 
         BUTTON_DrawPushButton(wndPtr,
 			      hDC,
@@ -688,18 +747,14 @@
 	return;
     }
 
-    textlen = 0;
     GetClientRect(wndPtr->hwndSelf, &client);
     rbox = rtext = client;
 
     if (infoPtr->hFont) SelectObject( hDC, infoPtr->hFont );
 
-    /* Something is still not right, checkboxes (and edit controls)
-     * in wsping32 have white backgrounds instead of dark grey.
-     * BUTTON_SEND_CTLCOLOR() is even worse since it returns 0 in this
-     * particular case and the background is not painted at all.
+    /* GetControlBrush16 sends WM_CTLCOLORBTN, plus it returns default brush
+     * if parent didn't return valid one. So we kill two hares at once
      */
-
     hBrush = GetControlBrush16( wndPtr->hwndSelf, hDC, CTLCOLOR_BTN );
 
     if (wndPtr->dwStyle & BS_LEFTTEXT) 
@@ -715,9 +770,7 @@
         rbox.right = checkBoxWidth;
     }
 
-      /* Draw the check-box bitmap */
-
-    if (wndPtr->text) textlen = lstrlenW( wndPtr->text );
+    /* Draw the check-box bitmap */
     if (action == ODA_DRAWENTIRE || action == ODA_SELECT)
     { 
         if( TWEAK_WineLook == WIN31_LOOK )
@@ -781,40 +834,31 @@
 
 	    DrawFrameControl( hDC, &rbox, DFC_BUTTON, state );
         }
-
-        if( textlen && action != ODA_SELECT )
-        {
-	  if (wndPtr->dwStyle & WS_DISABLED &&
-	      GetSysColor(COLOR_GRAYTEXT)==GetBkColor(hDC)) {
-            /* don't write gray text on gray background */
-            PaintGrayOnGray( hDC, infoPtr->hFont, &rtext, wndPtr->text,
-			     DT_VCENTER);
-	  } else {
-            if (wndPtr->dwStyle & WS_DISABLED)
-                SetTextColor( hDC, GetSysColor(COLOR_GRAYTEXT) );
-            DrawTextW( hDC, wndPtr->text, textlen, &rtext,
-			 DT_SINGLELINE | DT_VCENTER );
-	  }
-        }
     }
 
+    /* Draw label */
+    client = rtext;
+    dtFlags = BUTTON_CalcLabelRect(wndPtr, hDC, &rtext);
+
+    if (dtFlags == (UINT)-1L) /* Noting to draw */
+	return;
+    hRgn = CreateRectRgn(client.left, client.top, client.right, client.bottom);
+    SelectClipRgn(hDC, hRgn);
+    DeleteObject(hRgn);
+
+    if (action == ODA_DRAWENTIRE)
+	BUTTON_DrawLabel(wndPtr, hDC, dtFlags, &rtext);
+
+    /* ... and focus */
     if ((action == ODA_FOCUS) ||
         ((action == ODA_DRAWENTIRE) && (infoPtr->state & BUTTON_HASFOCUS)))
     {
-	/* again, this is what CTL3D expects */
-
-        SetRectEmpty(&rbox);
-        if( textlen )
-            DrawTextW( hDC, wndPtr->text, textlen, &rbox,
-			 DT_SINGLELINE | DT_CALCRECT );
-        textlen = rbox.bottom - rbox.top;
-        delta = ((rtext.bottom - rtext.top) - textlen)/2;
-        rbox.bottom = (rbox.top = rtext.top + delta - 1) + textlen + 2;
-        textlen = rbox.right - rbox.left;
-        rbox.right = (rbox.left += --rtext.left) + textlen + 2;
-        IntersectRect(&rbox, &rbox, &rtext);
-        DrawFocusRect( hDC, &rbox );
+	rtext.left--;
+	rtext.right++;
+	IntersectRect(&rtext, &rtext, &client);
+	DrawFocusRect( hDC, &rtext );
     }
+    SelectClipRgn(hDC, 0);
 }
 
 
@@ -852,10 +896,16 @@
 {
     RECT rc, rcFrame;
     BUTTONINFO *infoPtr = (BUTTONINFO *)wndPtr->wExtra;
+    HBRUSH hbr;
+    UINT dtFlags;
 
     if (action != ODA_DRAWENTIRE) return;
 
-    BUTTON_SEND_CTLCOLOR( wndPtr, hDC );
+    if (infoPtr->hFont)
+	SelectObject (hDC, infoPtr->hFont);
+    /* GroupBox acts like static control, so it sends CTLCOLORSTATIC */
+    hbr = GetControlBrush16( wndPtr->hwndSelf, hDC, CTLCOLOR_STATIC );
+
 
     GetClientRect( wndPtr->hwndSelf, &rc);
     if (TWEAK_WineLook == WIN31_LOOK) {
@@ -871,21 +921,29 @@
 	TEXTMETRICA tm;
 	rcFrame = rc;
 
-	if (infoPtr->hFont)
-	    SelectObject (hDC, infoPtr->hFont);
 	GetTextMetricsA (hDC, &tm);
 	rcFrame.top += (tm.tmHeight / 2) - 1;
-	DrawEdge (hDC, &rcFrame, EDGE_ETCHED, BF_RECT);
+	DrawEdge (hDC, &rcFrame, EDGE_ETCHED, BF_RECT |
+			   ((wndPtr->dwStyle & BS_FLAT) ? BF_FLAT : 0));
     }
 
-    if (wndPtr->text)
-    {
-	if (infoPtr->hFont) SelectObject( hDC, infoPtr->hFont );
-        if (wndPtr->dwStyle & WS_DISABLED)
-            SetTextColor( hDC, GetSysColor(COLOR_GRAYTEXT) );
-        rc.left += 10;
-        DrawTextW( hDC, wndPtr->text, -1, &rc, DT_SINGLELINE | DT_NOCLIP );
-    }
+    InflateRect(&rc, -7, 1);
+    dtFlags = BUTTON_CalcLabelRect(wndPtr, hDC, &rc);
+
+    if (dtFlags == (UINT)-1L)
+       return;
+
+    /* Because buttons have CS_PARENTDC class style, there is a chance
+     * that label will be drawn out of client rect.
+     * But Windows doesn't clip label's rect, so do I.
+     */
+
+    /* There is 1-pixel marging at the left, right, and bottom */
+    rc.left--; rc.right++; rc.bottom++;
+    FillRect(hDC, &rc, hbr);
+    rc.left++; rc.right--; rc.bottom--;
+
+    BUTTON_DrawLabel(wndPtr, hDC, dtFlags, &rc);   
 }
 
 
@@ -930,7 +988,7 @@
     dis.itemAction = action;
     dis.itemState  = ((infoPtr->state & BUTTON_HASFOCUS) ? ODS_FOCUS : 0) |
                      ((infoPtr->state & BUTTON_HIGHLIGHTED) ? ODS_SELECTED : 0) |
-                     ((wndPtr->dwStyle & WS_DISABLED) ? ODS_DISABLED : 0);
+                     (IsWindowEnabled(wndPtr->hwndSelf) ? 0: ODS_DISABLED);
     dis.hwndItem   = wndPtr->hwndSelf;
     dis.hDC        = hDC;
     dis.itemData   = 0;