1 /* 2 * ReactOS User32 Library 3 * - Various drawing functions 4 * 5 * Copyright 2001 Casper S. Hournstroup 6 * Copyright 2003 Andrew Greenwood 7 * Copyright 2003 Filip Navara 8 * Copyright 2009 Matthias Kupfer 9 * Copyright 2017-2022 Katayama Hirofumi MZ 10 * 11 * Based on Wine code. 12 * 13 * Copyright 1993, 1994 Alexandre Julliard 14 * Copyright 2002 Bill Medland 15 * 16 * This program is free software; you can redistribute it and/or modify 17 * it under the terms of the GNU General Public License as published by 18 * the Free Software Foundation; either version 2 of the License, or 19 * (at your option) any later version. 20 * 21 * This program is distributed in the hope that it will be useful, 22 * but WITHOUT ANY WARRANTY; without even the implied warranty of 23 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 24 * GNU General Public License for more details. 25 * 26 * You should have received a copy of the GNU General Public License 27 * along with this program; if not, write to the Free Software 28 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. 29 */ 30 31 #include <user32.h> 32 33 WINE_DEFAULT_DEBUG_CHANNEL(user32); 34 35 /* GLOBALS *******************************************************************/ 36 37 /* These tables are used in: 38 * UITOOLS_DrawDiagEdge() 39 * UITOOLS_DrawRectEdge() 40 */ 41 static const signed char LTInnerNormal[] = { 42 -1, -1, -1, -1, 43 -1, COLOR_BTNHIGHLIGHT, COLOR_BTNHIGHLIGHT, -1, 44 -1, COLOR_3DDKSHADOW, COLOR_3DDKSHADOW, -1, 45 -1, -1, -1, -1 46 }; 47 48 static const signed char LTOuterNormal[] = { 49 -1, COLOR_3DLIGHT, COLOR_BTNSHADOW, -1, 50 COLOR_BTNHIGHLIGHT, COLOR_3DLIGHT, COLOR_BTNSHADOW, -1, 51 COLOR_3DDKSHADOW, COLOR_3DLIGHT, COLOR_BTNSHADOW, -1, 52 -1, COLOR_3DLIGHT, COLOR_BTNSHADOW, -1 53 }; 54 55 static const signed char RBInnerNormal[] = { 56 -1, -1, -1, -1, 57 -1, COLOR_BTNSHADOW, COLOR_BTNSHADOW, -1, 58 -1, COLOR_3DLIGHT, COLOR_3DLIGHT, -1, 59 -1, -1, -1, -1 60 }; 61 62 static const signed char RBOuterNormal[] = { 63 -1, COLOR_3DDKSHADOW, COLOR_BTNHIGHLIGHT, -1, 64 COLOR_BTNSHADOW, COLOR_3DDKSHADOW, COLOR_BTNHIGHLIGHT, -1, 65 COLOR_3DLIGHT, COLOR_3DDKSHADOW, COLOR_BTNHIGHLIGHT, -1, 66 -1, COLOR_3DDKSHADOW, COLOR_BTNHIGHLIGHT, -1 67 }; 68 69 static const signed char LTInnerSoft[] = { 70 -1, -1, -1, -1, 71 -1, COLOR_3DLIGHT, COLOR_3DLIGHT, -1, 72 -1, COLOR_BTNSHADOW, COLOR_BTNSHADOW, -1, 73 -1, -1, -1, -1 74 }; 75 76 static const signed char LTOuterSoft[] = { 77 -1, COLOR_BTNHIGHLIGHT, COLOR_3DDKSHADOW, -1, 78 COLOR_3DLIGHT, COLOR_BTNHIGHLIGHT, COLOR_3DDKSHADOW, -1, 79 COLOR_BTNSHADOW, COLOR_BTNHIGHLIGHT, COLOR_3DDKSHADOW, -1, 80 -1, COLOR_BTNHIGHLIGHT, COLOR_3DDKSHADOW, -1 81 }; 82 83 #define RBInnerSoft RBInnerNormal /* These are the same */ 84 #define RBOuterSoft RBOuterNormal 85 86 static const signed char LTRBOuterMono[] = { 87 -1, COLOR_WINDOWFRAME, COLOR_WINDOWFRAME, COLOR_WINDOWFRAME, 88 COLOR_WINDOW, COLOR_WINDOWFRAME, COLOR_WINDOWFRAME, COLOR_WINDOWFRAME, 89 COLOR_WINDOW, COLOR_WINDOWFRAME, COLOR_WINDOWFRAME, COLOR_WINDOWFRAME, 90 COLOR_WINDOW, COLOR_WINDOWFRAME, COLOR_WINDOWFRAME, COLOR_WINDOWFRAME, 91 }; 92 93 static const signed char LTRBInnerMono[] = { 94 -1, -1, -1, -1, 95 -1, COLOR_WINDOW, COLOR_WINDOW, COLOR_WINDOW, 96 -1, COLOR_WINDOW, COLOR_WINDOW, COLOR_WINDOW, 97 -1, COLOR_WINDOW, COLOR_WINDOW, COLOR_WINDOW, 98 }; 99 100 static const signed char LTRBOuterFlat[] = { 101 -1, COLOR_BTNSHADOW, COLOR_BTNSHADOW, COLOR_BTNSHADOW, 102 COLOR_BTNFACE, COLOR_BTNSHADOW, COLOR_BTNSHADOW, COLOR_BTNSHADOW, 103 COLOR_BTNFACE, COLOR_BTNSHADOW, COLOR_BTNSHADOW, COLOR_BTNSHADOW, 104 COLOR_BTNFACE, COLOR_BTNSHADOW, COLOR_BTNSHADOW, COLOR_BTNSHADOW, 105 }; 106 107 static const signed char LTRBInnerFlat[] = { 108 -1, -1, -1, -1, 109 -1, COLOR_BTNFACE, COLOR_BTNFACE, COLOR_BTNFACE, 110 -1, COLOR_BTNFACE, COLOR_BTNFACE, COLOR_BTNFACE, 111 -1, COLOR_BTNFACE, COLOR_BTNFACE, COLOR_BTNFACE, 112 }; 113 /* FUNCTIONS *****************************************************************/ 114 115 116 HBRUSH WINAPI GetSysColorBrush(int nIndex); 117 118 /* Ported from WINE20020904 */ 119 /* Same as DrawEdge invoked with BF_DIAGONAL */ 120 static BOOL IntDrawDiagEdge(HDC hdc, LPRECT rc, UINT uType, UINT uFlags) 121 { 122 POINT Points[4]; 123 signed char InnerI, OuterI; 124 HPEN InnerPen, OuterPen; 125 POINT SavePoint; 126 HPEN SavePen; 127 int spx, spy; 128 int epx, epy; 129 int Width = rc->right - rc->left; 130 int Height= rc->bottom - rc->top; 131 int SmallDiam = Width > Height ? Height : Width; 132 BOOL retval = !( ((uType & BDR_INNER) == BDR_INNER 133 || (uType & BDR_OUTER) == BDR_OUTER) 134 && !(uFlags & (BF_FLAT|BF_MONO)) ); 135 int add = (LTRBInnerMono[uType & (BDR_INNER|BDR_OUTER)] != -1 ? 1 : 0) 136 + (LTRBOuterMono[uType & (BDR_INNER|BDR_OUTER)] != -1 ? 1 : 0); 137 138 /* Init some vars */ 139 OuterPen = InnerPen = (HPEN)GetStockObject(NULL_PEN); 140 SavePen = (HPEN)SelectObject(hdc, InnerPen); 141 spx = spy = epx = epy = 0; /* Satisfy the compiler... */ 142 143 /* Determine the colors of the edges */ 144 if(uFlags & BF_MONO) 145 { 146 InnerI = LTRBInnerMono[uType & (BDR_INNER|BDR_OUTER)]; 147 OuterI = LTRBOuterMono[uType & (BDR_INNER|BDR_OUTER)]; 148 } 149 else if(uFlags & BF_FLAT) 150 { 151 InnerI = LTRBInnerFlat[uType & (BDR_INNER|BDR_OUTER)]; 152 OuterI = LTRBOuterFlat[uType & (BDR_INNER|BDR_OUTER)]; 153 } 154 else if(uFlags & BF_SOFT) 155 { 156 if(uFlags & BF_BOTTOM) 157 { 158 InnerI = RBInnerSoft[uType & (BDR_INNER|BDR_OUTER)]; 159 OuterI = RBOuterSoft[uType & (BDR_INNER|BDR_OUTER)]; 160 } 161 else 162 { 163 InnerI = LTInnerSoft[uType & (BDR_INNER|BDR_OUTER)]; 164 OuterI = LTOuterSoft[uType & (BDR_INNER|BDR_OUTER)]; 165 } 166 } 167 else 168 { 169 if(uFlags & BF_BOTTOM) 170 { 171 InnerI = RBInnerNormal[uType & (BDR_INNER|BDR_OUTER)]; 172 OuterI = RBOuterNormal[uType & (BDR_INNER|BDR_OUTER)]; 173 } 174 else 175 { 176 InnerI = LTInnerNormal[uType & (BDR_INNER|BDR_OUTER)]; 177 OuterI = LTOuterNormal[uType & (BDR_INNER|BDR_OUTER)]; 178 } 179 } 180 181 if(InnerI != -1) InnerPen = GetStockObject(DC_PEN); 182 if(OuterI != -1) OuterPen = GetStockObject(DC_PEN); 183 184 MoveToEx(hdc, 0, 0, &SavePoint); 185 186 /* Don't ask me why, but this is what is visible... */ 187 /* This must be possible to do much simpler, but I fail to */ 188 /* see the logic in the MS implementation (sigh...). */ 189 /* So, this might look a bit brute force here (and it is), but */ 190 /* it gets the job done;) */ 191 192 switch(uFlags & BF_RECT) 193 { 194 case 0: 195 case BF_LEFT: 196 case BF_BOTTOM: 197 case BF_BOTTOMLEFT: 198 /* Left bottom endpoint */ 199 epx = rc->left-1; 200 spx = epx + SmallDiam; 201 epy = rc->bottom; 202 spy = epy - SmallDiam; 203 break; 204 205 case BF_TOPLEFT: 206 case BF_BOTTOMRIGHT: 207 /* Left top endpoint */ 208 epx = rc->left-1; 209 spx = epx + SmallDiam; 210 epy = rc->top-1; 211 spy = epy + SmallDiam; 212 break; 213 214 case BF_TOP: 215 case BF_RIGHT: 216 case BF_TOPRIGHT: 217 case BF_RIGHT|BF_LEFT: 218 case BF_RIGHT|BF_LEFT|BF_TOP: 219 case BF_BOTTOM|BF_TOP: 220 case BF_BOTTOM|BF_TOP|BF_LEFT: 221 case BF_BOTTOMRIGHT|BF_LEFT: 222 case BF_BOTTOMRIGHT|BF_TOP: 223 case BF_RECT: 224 /* Right top endpoint */ 225 spx = rc->left; 226 epx = spx + SmallDiam; 227 spy = rc->bottom-1; 228 epy = spy - SmallDiam; 229 break; 230 } 231 232 MoveToEx(hdc, spx, spy, NULL); 233 SelectObject(hdc, OuterPen); 234 SetDCPenColor(hdc, GetSysColor(OuterI)); 235 LineTo(hdc, epx, epy); 236 237 SelectObject(hdc, InnerPen); 238 SetDCPenColor(hdc, GetSysColor(InnerI)); 239 240 switch(uFlags & (BF_RECT|BF_DIAGONAL)) 241 { 242 case BF_DIAGONAL_ENDBOTTOMLEFT: 243 case (BF_DIAGONAL|BF_BOTTOM): 244 case BF_DIAGONAL: 245 case (BF_DIAGONAL|BF_LEFT): 246 MoveToEx(hdc, spx-1, spy, NULL); 247 LineTo(hdc, epx, epy-1); 248 Points[0].x = spx-add; 249 Points[0].y = spy; 250 Points[1].x = rc->left; 251 Points[1].y = rc->top; 252 Points[2].x = epx+1; 253 Points[2].y = epy-1-add; 254 Points[3] = Points[2]; 255 break; 256 257 case BF_DIAGONAL_ENDBOTTOMRIGHT: 258 MoveToEx(hdc, spx-1, spy, NULL); 259 LineTo(hdc, epx, epy+1); 260 Points[0].x = spx-add; 261 Points[0].y = spy; 262 Points[1].x = rc->left; 263 Points[1].y = rc->bottom-1; 264 Points[2].x = epx+1; 265 Points[2].y = epy+1+add; 266 Points[3] = Points[2]; 267 break; 268 269 case (BF_DIAGONAL|BF_BOTTOM|BF_RIGHT|BF_TOP): 270 case (BF_DIAGONAL|BF_BOTTOM|BF_RIGHT|BF_TOP|BF_LEFT): 271 case BF_DIAGONAL_ENDTOPRIGHT: 272 case (BF_DIAGONAL|BF_RIGHT|BF_TOP|BF_LEFT): 273 MoveToEx(hdc, spx+1, spy, NULL); 274 LineTo(hdc, epx, epy+1); 275 Points[0].x = epx-1; 276 Points[0].y = epy+1+add; 277 Points[1].x = rc->right-1; 278 Points[1].y = rc->top+add; 279 Points[2].x = rc->right-1; 280 Points[2].y = rc->bottom-1; 281 Points[3].x = spx+add; 282 Points[3].y = spy; 283 break; 284 285 case BF_DIAGONAL_ENDTOPLEFT: 286 MoveToEx(hdc, spx, spy-1, NULL); 287 LineTo(hdc, epx+1, epy); 288 Points[0].x = epx+1+add; 289 Points[0].y = epy+1; 290 Points[1].x = rc->right-1; 291 Points[1].y = rc->top; 292 Points[2].x = rc->right-1; 293 Points[2].y = rc->bottom-1-add; 294 Points[3].x = spx; 295 Points[3].y = spy-add; 296 break; 297 298 case (BF_DIAGONAL|BF_TOP): 299 case (BF_DIAGONAL|BF_BOTTOM|BF_TOP): 300 case (BF_DIAGONAL|BF_BOTTOM|BF_TOP|BF_LEFT): 301 MoveToEx(hdc, spx+1, spy-1, NULL); 302 LineTo(hdc, epx, epy); 303 Points[0].x = epx-1; 304 Points[0].y = epy+1; 305 Points[1].x = rc->right-1; 306 Points[1].y = rc->top; 307 Points[2].x = rc->right-1; 308 Points[2].y = rc->bottom-1-add; 309 Points[3].x = spx+add; 310 Points[3].y = spy-add; 311 break; 312 313 case (BF_DIAGONAL|BF_RIGHT): 314 case (BF_DIAGONAL|BF_RIGHT|BF_LEFT): 315 case (BF_DIAGONAL|BF_RIGHT|BF_LEFT|BF_BOTTOM): 316 MoveToEx(hdc, spx, spy, NULL); 317 LineTo(hdc, epx-1, epy+1); 318 Points[0].x = spx; 319 Points[0].y = spy; 320 Points[1].x = rc->left; 321 Points[1].y = rc->top+add; 322 Points[2].x = epx-1-add; 323 Points[2].y = epy+1+add; 324 Points[3] = Points[2]; 325 break; 326 } 327 328 /* Fill the interior if asked */ 329 if((uFlags & BF_MIDDLE) && retval) 330 { 331 HBRUSH hbsave; 332 HPEN hpsave; 333 hbsave = (HBRUSH)SelectObject(hdc, GetStockObject(DC_BRUSH)); 334 hpsave = (HPEN)SelectObject(hdc, GetStockObject(DC_PEN)); 335 SetDCBrushColor(hdc, GetSysColor(uFlags & BF_MONO ? COLOR_WINDOW : COLOR_BTNFACE)); 336 SetDCPenColor(hdc, GetSysColor(uFlags & BF_MONO ? COLOR_WINDOW : COLOR_BTNFACE)); 337 Polygon(hdc, Points, 4); 338 SelectObject(hdc, hbsave); 339 SelectObject(hdc, hpsave); 340 } 341 342 /* Adjust rectangle if asked */ 343 if(uFlags & BF_ADJUST) 344 { 345 if(uFlags & BF_LEFT) rc->left += add; 346 if(uFlags & BF_RIGHT) rc->right -= add; 347 if(uFlags & BF_TOP) rc->top += add; 348 if(uFlags & BF_BOTTOM) rc->bottom -= add; 349 } 350 351 /* Cleanup */ 352 SelectObject(hdc, SavePen); 353 MoveToEx(hdc, SavePoint.x, SavePoint.y, NULL); 354 355 return retval; 356 } 357 358 /* Ported from WINE20020904 */ 359 /* Same as DrawEdge invoked without BF_DIAGONAL 360 * 361 * 23-Nov-1997: Changed by Bertho Stultiens 362 * The width parameter sets the width of each outer and inner edge. 363 * 364 * Well, I started testing this and found out that there are a few things 365 * that weren't quite as win95. The following rewrite should reproduce 366 * win95 results completely. 367 * The colorselection is table-driven to avoid awfull if-statements. 368 * The table below show the color settings. 369 * 370 * Pen selection table for uFlags = 0 371 * 372 * uType | LTI | LTO | RBI | RBO 373 * ------+-------+-------+-------+------- 374 * 0000 | x | x | x | x 375 * 0001 | x | 22 | x | 21 376 * 0010 | x | 16 | x | 20 377 * 0011 | x | x | x | x 378 * ------+-------+-------+-------+------- 379 * 0100 | x | 20 | x | 16 380 * 0101 | 20 | 22 | 16 | 21 381 * 0110 | 20 | 16 | 16 | 20 382 * 0111 | x | x | x | x 383 * ------+-------+-------+-------+------- 384 * 1000 | x | 21 | x | 22 385 * 1001 | 21 | 22 | 22 | 21 386 * 1010 | 21 | 16 | 22 | 20 387 * 1011 | x | x | x | x 388 * ------+-------+-------+-------+------- 389 * 1100 | x | x | x | x 390 * 1101 | x | x (22)| x | x (21) 391 * 1110 | x | x (16)| x | x (20) 392 * 1111 | x | x | x | x 393 * 394 * Pen selection table for uFlags = BF_SOFT 395 * 396 * uType | LTI | LTO | RBI | RBO 397 * ------+-------+-------+-------+------- 398 * 0000 | x | x | x | x 399 * 0001 | x | 20 | x | 21 400 * 0010 | x | 21 | x | 20 401 * 0011 | x | x | x | x 402 * ------+-------+-------+-------+------- 403 * 0100 | x | 22 | x | 16 404 * 0101 | 22 | 20 | 16 | 21 405 * 0110 | 22 | 21 | 16 | 20 406 * 0111 | x | x | x | x 407 * ------+-------+-------+-------+------- 408 * 1000 | x | 16 | x | 22 409 * 1001 | 16 | 20 | 22 | 21 410 * 1010 | 16 | 21 | 22 | 20 411 * 1011 | x | x | x | x 412 * ------+-------+-------+-------+------- 413 * 1100 | x | x | x | x 414 * 1101 | x | x (20)| x | x (21) 415 * 1110 | x | x (21)| x | x (20) 416 * 1111 | x | x | x | x 417 * 418 * x = don't care; (n) = is what win95 actually uses 419 * LTI = left Top Inner line 420 * LTO = left Top Outer line 421 * RBI = Right Bottom Inner line 422 * RBO = Right Bottom Outer line 423 * 15 = COLOR_BTNFACE 424 * 16 = COLOR_BTNSHADOW 425 * 20 = COLOR_BTNHIGHLIGHT 426 * 21 = COLOR_3DDKSHADOW 427 * 22 = COLOR_3DLIGHT 428 */ 429 static BOOL IntDrawRectEdge(HDC hdc, LPRECT rc, UINT uType, UINT uFlags, UINT width) 430 { 431 signed char LTInnerI, LTOuterI; 432 signed char RBInnerI, RBOuterI; 433 HBRUSH lti_brush, lto_brush, rbi_brush, rbo_brush; 434 RECT InnerRect = *rc, fill_rect; 435 int lbi_offset = 0, lti_offset = 0, rti_offset = 0, rbi_offset = 0; 436 BOOL retval = !( ((uType & BDR_INNER) == BDR_INNER 437 || (uType & BDR_OUTER) == BDR_OUTER) 438 && !(uFlags & (BF_FLAT|BF_MONO)) ); 439 440 lti_brush = lto_brush = rbi_brush = rbo_brush = GetStockObject(NULL_BRUSH); 441 442 /* Determine the colors of the edges */ 443 if(uFlags & BF_MONO) 444 { 445 LTInnerI = RBInnerI = LTRBInnerMono[uType & (BDR_INNER|BDR_OUTER)]; 446 LTOuterI = RBOuterI = LTRBOuterMono[uType & (BDR_INNER|BDR_OUTER)]; 447 } 448 else if(uFlags & BF_FLAT) 449 { 450 LTInnerI = RBInnerI = LTRBInnerFlat[uType & (BDR_INNER|BDR_OUTER)]; 451 LTOuterI = RBOuterI = LTRBOuterFlat[uType & (BDR_INNER|BDR_OUTER)]; 452 453 /* Bertho Stultiens states above that this function exactly matches win95 454 * In win98 BF_FLAT rectangles have an inner border same color as the 455 * middle (COLOR_BTNFACE). I believe it's the same for win95 but since 456 * I don't know I go with Bertho and just sets it for win98 until proven 457 * otherwise. 458 * Dennis Björklund, 10 June, 99 459 */ 460 if( LTInnerI != -1 ) LTInnerI = RBInnerI = COLOR_BTNFACE; 461 } 462 else if(uFlags & BF_SOFT) 463 { 464 LTInnerI = LTInnerSoft[uType & (BDR_INNER|BDR_OUTER)]; 465 LTOuterI = LTOuterSoft[uType & (BDR_INNER|BDR_OUTER)]; 466 RBInnerI = RBInnerSoft[uType & (BDR_INNER|BDR_OUTER)]; 467 RBOuterI = RBOuterSoft[uType & (BDR_INNER|BDR_OUTER)]; 468 } 469 else 470 { 471 LTInnerI = LTInnerNormal[uType & (BDR_INNER|BDR_OUTER)]; 472 LTOuterI = LTOuterNormal[uType & (BDR_INNER|BDR_OUTER)]; 473 RBInnerI = RBInnerNormal[uType & (BDR_INNER|BDR_OUTER)]; 474 RBOuterI = RBOuterNormal[uType & (BDR_INNER|BDR_OUTER)]; 475 } 476 477 if((uFlags & BF_BOTTOMLEFT) == BF_BOTTOMLEFT) lbi_offset = width; 478 if((uFlags & BF_TOPRIGHT) == BF_TOPRIGHT) rti_offset = width; 479 if((uFlags & BF_BOTTOMRIGHT) == BF_BOTTOMRIGHT) rbi_offset = width; 480 if((uFlags & BF_TOPLEFT) == BF_TOPLEFT) lti_offset = width; 481 482 if(LTInnerI != -1) lti_brush = GetSysColorBrush(LTInnerI); 483 if(LTOuterI != -1) lto_brush = GetSysColorBrush(LTOuterI); 484 if(RBInnerI != -1) rbi_brush = GetSysColorBrush(RBInnerI); 485 if(RBOuterI != -1) rbo_brush = GetSysColorBrush(RBOuterI); 486 487 /* Draw the outer edge */ 488 if(uFlags & BF_TOP) 489 { 490 fill_rect = InnerRect; 491 fill_rect.bottom = fill_rect.top + width; 492 FillRect( hdc, &fill_rect, lto_brush ); 493 } 494 if(uFlags & BF_LEFT) 495 { 496 fill_rect = InnerRect; 497 fill_rect.right = fill_rect.left + width; 498 FillRect( hdc, &fill_rect, lto_brush ); 499 } 500 if(uFlags & BF_BOTTOM) 501 { 502 fill_rect = InnerRect; 503 fill_rect.top = fill_rect.bottom - width; 504 FillRect( hdc, &fill_rect, rbo_brush ); 505 } 506 if(uFlags & BF_RIGHT) 507 { 508 fill_rect = InnerRect; 509 fill_rect.left = fill_rect.right - width; 510 FillRect( hdc, &fill_rect, rbo_brush ); 511 } 512 513 /* Draw the inner edge */ 514 if(uFlags & BF_TOP) 515 { 516 SetRect( &fill_rect, InnerRect.left + lti_offset, InnerRect.top + width, 517 InnerRect.right - rti_offset, InnerRect.top + 2 * width ); 518 FillRect( hdc, &fill_rect, lti_brush ); 519 } 520 if(uFlags & BF_LEFT) 521 { 522 SetRect( &fill_rect, InnerRect.left + width, InnerRect.top + lti_offset, 523 InnerRect.left + 2 * width, InnerRect.bottom - lbi_offset ); 524 FillRect( hdc, &fill_rect, lti_brush ); 525 } 526 if(uFlags & BF_BOTTOM) 527 { 528 SetRect( &fill_rect, InnerRect.left + lbi_offset, InnerRect.bottom - 2 * width, 529 InnerRect.right - rbi_offset, InnerRect.bottom - width ); 530 FillRect( hdc, &fill_rect, rbi_brush ); 531 } 532 if(uFlags & BF_RIGHT) 533 { 534 SetRect( &fill_rect, InnerRect.right - 2 * width, InnerRect.top + rti_offset, 535 InnerRect.right - width, InnerRect.bottom - rbi_offset ); 536 FillRect( hdc, &fill_rect, rbi_brush ); 537 } 538 539 if( ((uFlags & BF_MIDDLE) && retval) || (uFlags & BF_ADJUST) ) 540 { 541 int add = (LTRBInnerMono[uType & (BDR_INNER|BDR_OUTER)] != -1 ? width : 0) 542 + (LTRBOuterMono[uType & (BDR_INNER|BDR_OUTER)] != -1 ? width : 0); 543 544 if(uFlags & BF_LEFT) InnerRect.left += add; 545 if(uFlags & BF_RIGHT) InnerRect.right -= add; 546 if(uFlags & BF_TOP) InnerRect.top += add; 547 if(uFlags & BF_BOTTOM) InnerRect.bottom -= add; 548 549 if((uFlags & BF_MIDDLE) && retval) 550 { 551 FillRect(hdc, &InnerRect, GetSysColorBrush(uFlags & BF_MONO ? 552 COLOR_WINDOW : COLOR_BTNFACE)); 553 } 554 555 if(uFlags & BF_ADJUST) 556 *rc = InnerRect; 557 } 558 559 return retval; 560 } 561 562 /* Ported from WINE20020904 */ 563 /* Utility to create a square rectangle and returning the width */ 564 static int UITOOLS_MakeSquareRect(LPRECT src, LPRECT dst) 565 { 566 int Width = src->right - src->left; 567 int Height = src->bottom - src->top; 568 int SmallDiam = Width > Height ? Height : Width; 569 570 *dst = *src; 571 572 /* Make it a square box */ 573 if(Width < Height) /* SmallDiam == Width */ 574 { 575 dst->top += (Height-Width)/2; 576 dst->bottom = dst->top + SmallDiam; 577 } 578 else if(Width > Height) /* SmallDiam == Height */ 579 { 580 dst->left += (Width-Height)/2; 581 dst->right = dst->left + SmallDiam; 582 } 583 584 return SmallDiam; 585 } 586 587 /* Ported from WINE20020904 */ 588 static void UITOOLS_DrawCheckedRect( HDC dc, LPRECT rect ) 589 { 590 if(GetSysColor(COLOR_BTNHIGHLIGHT) == RGB(255, 255, 255)) 591 { 592 HBRUSH hbsave; 593 COLORREF bg; 594 595 FillRect(dc, rect, GetSysColorBrush(COLOR_BTNFACE)); 596 bg = SetBkColor(dc, RGB(255, 255, 255)); 597 hbsave = (HBRUSH)SelectObject(dc, gpsi->hbrGray); 598 PatBlt(dc, rect->left, rect->top, rect->right-rect->left, rect->bottom-rect->top, 0x00FA0089); 599 SelectObject(dc, hbsave); 600 SetBkColor(dc, bg); 601 } 602 else 603 { 604 FillRect(dc, rect, GetSysColorBrush(COLOR_BTNHIGHLIGHT)); 605 } 606 } 607 608 /* Ported from WINE20020904 */ 609 /* Draw a push button coming from DrawFrameControl() 610 * 611 * Does a pretty good job in emulating MS behavior. Some quirks are 612 * however there because MS uses a TrueType font (Marlett) to draw 613 * the buttons. 614 * 615 * FIXME: This looks a little bit strange, needs to be rewritten completely 616 * (several quirks with adjust, DFCS_CHECKED aso) 617 */ 618 static BOOL UITOOLS95_DFC_ButtonPush(HDC dc, LPRECT r, UINT uFlags) 619 { 620 UINT edge; 621 RECT myr = *r; 622 623 if(uFlags & (DFCS_PUSHED | DFCS_CHECKED | DFCS_FLAT)) 624 edge = EDGE_SUNKEN; 625 else 626 edge = EDGE_RAISED; 627 628 if(uFlags & DFCS_CHECKED) 629 { 630 if(uFlags & DFCS_MONO) 631 IntDrawRectEdge(dc, &myr, edge, BF_MONO|BF_RECT|BF_ADJUST, 1); 632 else 633 IntDrawRectEdge(dc, &myr, edge, (uFlags&DFCS_FLAT)|BF_RECT|BF_SOFT|BF_ADJUST, 1); 634 635 UITOOLS_DrawCheckedRect( dc, &myr ); 636 } 637 else 638 { 639 if(uFlags & DFCS_MONO) 640 { 641 IntDrawRectEdge(dc, &myr, edge, BF_MONO|BF_RECT|BF_ADJUST, 1); 642 FillRect(dc, &myr, GetSysColorBrush(COLOR_BTNFACE)); 643 } 644 else 645 { 646 IntDrawRectEdge(dc, r, edge, (uFlags&DFCS_FLAT) | BF_MIDDLE | BF_RECT | BF_SOFT, 1); 647 } 648 } 649 650 /* Adjust rectangle if asked */ 651 if(uFlags & DFCS_ADJUSTRECT) 652 InflateRect(r, -2, -2); 653 654 return TRUE; 655 } 656 657 static BOOL UITOOLS95_DFC_ButtonCheckRadio(HDC dc, LPRECT r, UINT uFlags, BOOL Radio) 658 { 659 LOGFONTW lf; 660 HFONT hFont, hOldFont; 661 int i; 662 TCHAR OutRight, OutLeft, InRight, InLeft, Center; 663 INT X, Y, Width, Height, Shorter; 664 INT BkMode = GetBkMode(dc); 665 COLORREF TextColor = GetTextColor(dc); 666 667 Width = r->right - r->left; 668 Height = r->bottom - r->top; 669 Shorter = (Width < Height) ? Width : Height; 670 X = r->left + (Width - Shorter) / 2; 671 Y = r->top + (Height - Shorter) / 2; 672 673 if (Radio) 674 { 675 OutRight = 'j'; // Outer right 676 OutLeft = 'k'; // Outer left 677 InRight = 'l'; // inner left 678 InLeft = 'm'; // inner right 679 Center = 'n'; // center 680 } 681 else 682 { 683 OutRight = 'c'; // Outer right 684 OutLeft = 'd'; // Outer left 685 InRight = 'e'; // inner left 686 InLeft = 'f'; // inner right 687 Center = 'g'; // center 688 } 689 690 ZeroMemory(&lf, sizeof(LOGFONTW)); 691 lf.lfHeight = Shorter; 692 lf.lfCharSet = DEFAULT_CHARSET; 693 lstrcpy(lf.lfFaceName, TEXT("Marlett")); 694 if (Radio && ((uFlags & 0xFF) == DFCS_BUTTONRADIOMASK)) 695 { 696 lf.lfQuality = NONANTIALIASED_QUALITY; 697 } 698 hFont = CreateFontIndirect(&lf); 699 hOldFont = SelectObject(dc, hFont); 700 701 if (Radio && ((uFlags & 0xFF) == DFCS_BUTTONRADIOMASK)) 702 { 703 #if 1 704 // FIXME: improve font rendering 705 RECT Rect; 706 HGDIOBJ hbrOld, hpenOld; 707 SetRect(&Rect, X, Y, X + Shorter, Y + Shorter); 708 InflateRect(&Rect, -(Shorter * 8) / 54, -(Shorter * 8) / 54); 709 hbrOld = SelectObject(dc, GetStockObject(BLACK_BRUSH)); 710 hpenOld = SelectObject(dc, GetStockObject(NULL_PEN)); 711 Ellipse(dc, Rect.left, Rect.top, Rect.right, Rect.bottom); 712 SelectObject(dc, hbrOld); 713 SelectObject(dc, hpenOld); 714 #else 715 SetBkMode(dc, OPAQUE); 716 SetBkColor(dc, RGB(255, 255, 255)); 717 SetTextColor(dc, RGB(0, 0, 0)); 718 TextOut(dc, X, Y, &Center, 1); 719 SetBkMode(dc, TRANSPARENT); 720 TextOut(dc, X, Y, &OutRight, 1); 721 TextOut(dc, X, Y, &OutLeft, 1); 722 TextOut(dc, X, Y, &InRight, 1); 723 TextOut(dc, X, Y, &InLeft, 1); 724 #endif 725 } 726 else 727 { 728 SetBkMode(dc, TRANSPARENT); 729 730 /* Center section, white for active, grey for inactive */ 731 if ((uFlags & (DFCS_INACTIVE | DFCS_PUSHED))) 732 i = COLOR_BTNFACE; 733 else 734 i = COLOR_WINDOW; 735 SetTextColor(dc, GetSysColor(i)); 736 TextOut(dc, X, Y, &Center, 1); 737 738 if (uFlags & (DFCS_FLAT | DFCS_MONO)) 739 { 740 SetTextColor(dc, GetSysColor(COLOR_WINDOWFRAME)); 741 TextOut(dc, X, Y, &OutRight, 1); 742 TextOut(dc, X, Y, &OutLeft, 1); 743 TextOut(dc, X, Y, &InRight, 1); 744 TextOut(dc, X, Y, &InLeft, 1); 745 } 746 else 747 { 748 SetTextColor(dc, GetSysColor(COLOR_BTNSHADOW)); 749 TextOut(dc, X, Y, &OutRight, 1); 750 SetTextColor(dc, GetSysColor(COLOR_BTNHIGHLIGHT)); 751 TextOut(dc, X, Y, &OutLeft, 1); 752 SetTextColor(dc, GetSysColor(COLOR_3DDKSHADOW)); 753 TextOut(dc, X, Y, &InRight, 1); 754 SetTextColor(dc, GetSysColor(COLOR_3DLIGHT)); 755 TextOut(dc, X, Y, &InLeft, 1); 756 } 757 758 if (uFlags & DFCS_CHECKED) 759 { 760 TCHAR Check = (Radio) ? 'i' : 'b'; 761 762 SetTextColor(dc, GetSysColor((uFlags & DFCS_INACTIVE) ? COLOR_BTNSHADOW : COLOR_WINDOWTEXT)); 763 TextOut(dc, X, Y, &Check, 1); 764 } 765 } 766 767 SelectObject(dc, hOldFont); 768 DeleteObject(hFont); 769 770 SetTextColor(dc, TextColor); 771 SetBkMode(dc, BkMode); 772 773 return TRUE; 774 } 775 776 /* Ported from WINE20020904 */ 777 static BOOL UITOOLS95_DrawFrameButton(HDC hdc, LPRECT rc, UINT uState) 778 { 779 switch(uState & 0x1f) 780 { 781 case DFCS_BUTTONPUSH: 782 return UITOOLS95_DFC_ButtonPush(hdc, rc, uState); 783 784 case DFCS_BUTTONCHECK: 785 case DFCS_BUTTON3STATE: 786 return UITOOLS95_DFC_ButtonCheckRadio(hdc, rc, uState, FALSE); 787 788 case DFCS_BUTTONRADIOIMAGE: 789 case DFCS_BUTTONRADIOMASK: 790 if (uState & DFCS_BUTTONRADIOIMAGE) 791 FillRect(hdc, rc, (HBRUSH)GetStockObject(BLACK_BRUSH)); /* Fill by black */ 792 else 793 FillRect(hdc, rc, (HBRUSH)GetStockObject(WHITE_BRUSH)); /* Fill by white */ 794 795 return UITOOLS95_DFC_ButtonCheckRadio(hdc, rc, uState, TRUE); 796 797 case DFCS_BUTTONRADIO: 798 return UITOOLS95_DFC_ButtonCheckRadio(hdc, rc, uState, TRUE); 799 800 default: 801 ERR("Invalid button state=0x%04x\n", uState); 802 } 803 804 return FALSE; 805 } 806 807 static BOOL UITOOLS95_DrawFrameCaption(HDC dc, LPRECT r, UINT uFlags) 808 { 809 LOGFONTW lf; 810 HFONT hFont, hOldFont; 811 COLORREF clrsave; 812 RECT myr; 813 INT bkmode; 814 TCHAR Symbol; 815 switch(uFlags & 0xf) 816 { 817 case DFCS_CAPTIONCLOSE: 818 Symbol = 'r'; 819 break; 820 case DFCS_CAPTIONHELP: 821 Symbol = 's'; 822 break; 823 case DFCS_CAPTIONMIN: 824 Symbol = '0'; 825 break; 826 case DFCS_CAPTIONMAX: 827 Symbol = '1'; 828 break; 829 case DFCS_CAPTIONRESTORE: 830 Symbol = '2'; 831 break; 832 default: 833 ERR("Invalid caption; flags=0x%04x\n", uFlags); 834 return FALSE; 835 } 836 IntDrawRectEdge(dc,r,(uFlags&DFCS_PUSHED) ? EDGE_SUNKEN : EDGE_RAISED, BF_RECT | BF_MIDDLE | BF_SOFT, 1); 837 ZeroMemory(&lf, sizeof(LOGFONTW)); 838 UITOOLS_MakeSquareRect(r, &myr); 839 myr.left += 1; 840 myr.top += 1; 841 myr.right -= 1; 842 myr.bottom -= 1; 843 if(uFlags & DFCS_PUSHED) 844 OffsetRect(&myr,1,1); 845 lf.lfHeight = myr.bottom - myr.top; 846 lf.lfWidth = 0; 847 lf.lfWeight = FW_NORMAL; 848 lf.lfCharSet = DEFAULT_CHARSET; 849 lstrcpy(lf.lfFaceName, TEXT("Marlett")); 850 hFont = CreateFontIndirect(&lf); 851 /* save font and text color */ 852 hOldFont = SelectObject(dc, hFont); 853 clrsave = GetTextColor(dc); 854 bkmode = GetBkMode(dc); 855 /* set color and drawing mode */ 856 SetBkMode(dc, TRANSPARENT); 857 if(uFlags & DFCS_INACTIVE) 858 { 859 /* draw shadow */ 860 SetTextColor(dc, GetSysColor(COLOR_BTNHIGHLIGHT)); 861 TextOut(dc, myr.left + 1, myr.top + 1, &Symbol, 1); 862 } 863 SetTextColor(dc, GetSysColor((uFlags & DFCS_INACTIVE) ? COLOR_BTNSHADOW : COLOR_BTNTEXT)); 864 /* draw selected symbol */ 865 TextOut(dc, myr.left, myr.top, &Symbol, 1); 866 /* restore previous settings */ 867 SetTextColor(dc, clrsave); 868 SelectObject(dc, hOldFont); 869 SetBkMode(dc, bkmode); 870 DeleteObject(hFont); 871 return TRUE; 872 } 873 874 static BOOL UITOOLS95_DrawFrameScroll(HDC dc, LPRECT r, UINT uFlags) 875 { 876 LOGFONTW lf; 877 HFONT hFont, hOldFont; 878 COLORREF clrsave; 879 RECT myr; 880 INT bkmode; 881 TCHAR Symbol; 882 switch(uFlags & 0x1f) 883 { 884 case DFCS_SCROLLCOMBOBOX: 885 case DFCS_SCROLLDOWN: 886 Symbol = '6'; 887 break; 888 889 case DFCS_SCROLLUP: 890 Symbol = '5'; 891 break; 892 893 case DFCS_SCROLLLEFT: 894 Symbol = '3'; 895 break; 896 897 case DFCS_SCROLLRIGHT: 898 Symbol = '4'; 899 break; 900 901 case DFCS_SCROLLSIZEGRIP: 902 case DFCS_SCROLLSIZEGRIPRIGHT: 903 ZeroMemory(&lf, sizeof(LOGFONTW)); 904 UITOOLS_MakeSquareRect(r, &myr); 905 lf.lfHeight = myr.bottom - myr.top; 906 lf.lfWidth = 0; 907 lf.lfWeight = FW_NORMAL; 908 lf.lfCharSet = DEFAULT_CHARSET; 909 lstrcpy(lf.lfFaceName, TEXT("Marlett")); 910 hFont = CreateFontIndirect(&lf); 911 /* save font and text color */ 912 hOldFont = SelectObject(dc, hFont); 913 clrsave = GetTextColor(dc); 914 bkmode = GetBkMode(dc); 915 /* set color and drawing mode */ 916 SetBkMode(dc, TRANSPARENT); 917 if (!(uFlags & (DFCS_MONO | DFCS_FLAT))) 918 { 919 SetTextColor(dc, GetSysColor(COLOR_BTNHIGHLIGHT)); 920 /* draw selected symbol */ 921 Symbol = ((uFlags & 0xff) == DFCS_SCROLLSIZEGRIP) ? 'o' : 'x'; 922 TextOut(dc, myr.left, myr.top, &Symbol, 1); 923 SetTextColor(dc, GetSysColor(COLOR_BTNSHADOW)); 924 } else 925 SetTextColor(dc, GetSysColor(COLOR_WINDOWFRAME)); 926 /* draw selected symbol */ 927 Symbol = ((uFlags & 0xff) == DFCS_SCROLLSIZEGRIP) ? 'p' : 'y'; 928 TextOut(dc, myr.left, myr.top, &Symbol, 1); 929 /* restore previous settings */ 930 SetTextColor(dc, clrsave); 931 SelectObject(dc, hOldFont); 932 SetBkMode(dc, bkmode); 933 DeleteObject(hFont); 934 return TRUE; 935 default: 936 ERR("Invalid scroll; flags=0x%04x\n", uFlags); 937 return FALSE; 938 } 939 IntDrawRectEdge(dc, r, (uFlags & DFCS_PUSHED) ? EDGE_SUNKEN : EDGE_RAISED, (uFlags&DFCS_FLAT) | BF_MIDDLE | BF_RECT, 1); 940 ZeroMemory(&lf, sizeof(LOGFONTW)); 941 UITOOLS_MakeSquareRect(r, &myr); 942 myr.left += 1; 943 myr.top += 1; 944 myr.right -= 1; 945 myr.bottom -= 1; 946 if(uFlags & DFCS_PUSHED) 947 OffsetRect(&myr,1,1); 948 lf.lfHeight = myr.bottom - myr.top; 949 lf.lfWidth = 0; 950 lf.lfWeight = FW_NORMAL; 951 lf.lfCharSet = DEFAULT_CHARSET; 952 lstrcpy(lf.lfFaceName, TEXT("Marlett")); 953 hFont = CreateFontIndirect(&lf); 954 /* save font and text color */ 955 hOldFont = SelectObject(dc, hFont); 956 clrsave = GetTextColor(dc); 957 bkmode = GetBkMode(dc); 958 /* set color and drawing mode */ 959 SetBkMode(dc, TRANSPARENT); 960 if(uFlags & DFCS_INACTIVE) 961 { 962 /* draw shadow */ 963 SetTextColor(dc, GetSysColor(COLOR_BTNHIGHLIGHT)); 964 TextOut(dc, myr.left + 1, myr.top + 1, &Symbol, 1); 965 } 966 SetTextColor(dc, GetSysColor((uFlags & DFCS_INACTIVE) ? COLOR_BTNSHADOW : COLOR_BTNTEXT)); 967 /* draw selected symbol */ 968 TextOut(dc, myr.left, myr.top, &Symbol, 1); 969 /* restore previous settings */ 970 SetTextColor(dc, clrsave); 971 SelectObject(dc, hOldFont); 972 SetBkMode(dc, bkmode); 973 DeleteObject(hFont); 974 return TRUE; 975 } 976 977 static BOOL UITOOLS95_DrawFrameMenu(HDC dc, LPRECT r, UINT uFlags) 978 { 979 LOGFONTW lf; 980 HFONT hFont, hOldFont; 981 TCHAR Symbol; 982 RECT myr; 983 INT cxy; 984 cxy = UITOOLS_MakeSquareRect(r, &myr); 985 switch(uFlags & 0x1f) 986 { 987 case DFCS_MENUARROWUP: 988 Symbol = '5'; 989 break; 990 991 case DFCS_MENUARROWDOWN: 992 Symbol = '6'; 993 break; 994 995 case DFCS_MENUARROW: 996 Symbol = '8'; 997 break; 998 999 case DFCS_MENUARROWRIGHT: 1000 Symbol = 'w'; // FIXME: needs to confirm 1001 break; 1002 1003 case DFCS_MENUBULLET: 1004 Symbol = 'h'; 1005 break; 1006 1007 case DFCS_MENUCHECK: 1008 case DFCS_MENUCHECK | DFCS_MENUBULLET: 1009 Symbol = 'a'; 1010 break; 1011 1012 default: 1013 ERR("Invalid menu; flags=0x%04x\n", uFlags); 1014 return FALSE; 1015 } 1016 /* acquire ressources only if valid menu */ 1017 ZeroMemory(&lf, sizeof(LOGFONTW)); 1018 lf.lfHeight = cxy; 1019 lf.lfWidth = 0; 1020 lf.lfWeight = FW_NORMAL; 1021 lf.lfCharSet = DEFAULT_CHARSET; 1022 lstrcpy(lf.lfFaceName, TEXT("Marlett")); 1023 hFont = CreateFontIndirect(&lf); 1024 /* save font */ 1025 hOldFont = SelectObject(dc, hFont); 1026 1027 if ((uFlags & 0x1f) == DFCS_MENUARROWUP || 1028 (uFlags & 0x1f) == DFCS_MENUARROWDOWN ) 1029 { 1030 #if 0 1031 if (uFlags & DFCS_INACTIVE) 1032 { 1033 /* draw shadow */ 1034 SetTextColor(dc, GetSysColor(COLOR_BTNHIGHLIGHT)); 1035 TextOut(dc, myr.left + 1, myr.top + 1, &Symbol, 1); 1036 } 1037 #endif 1038 SetTextColor(dc, GetSysColor((uFlags & DFCS_INACTIVE) ? COLOR_BTNSHADOW : COLOR_BTNTEXT)); 1039 } 1040 /* draw selected symbol */ 1041 TextOut(dc, myr.left, myr.top, &Symbol, 1); 1042 /* restore previous settings */ 1043 SelectObject(dc, hOldFont); 1044 DeleteObject(hFont); 1045 return TRUE; 1046 } 1047 1048 BOOL 1049 WINAPI 1050 IntGrayString( 1051 HDC hDC, 1052 HBRUSH hBrush, 1053 GRAYSTRINGPROC lpOutputFunc, 1054 LPARAM lpData, 1055 int nCount, 1056 int X, 1057 int Y, 1058 int nWidth, 1059 int nHeight, 1060 BOOL unicode) 1061 { 1062 // AG: Mostly implemented, but probably won't work properly or return 1063 // correct error codes. I doubt it grays strings either... Untested! 1064 1065 BOOL success = FALSE; 1066 HBRUSH hbsave; 1067 HDC MemDC = NULL; 1068 HBITMAP MemBMP = NULL, OldBMP = NULL; 1069 HBRUSH OldBrush = NULL; 1070 HFONT OldFont = NULL; 1071 RECT r; 1072 COLORREF ForeColor, BackColor; 1073 1074 ForeColor = SetTextColor(hDC, RGB(0, 0, 0)); 1075 BackColor = SetBkColor(hDC, RGB(255, 255, 255)); 1076 1077 if (! hBrush) 1078 { 1079 // The documentation is a little vague on what exactly should happen 1080 // here. Something about using the same brush for window text??? 1081 hBrush = (HBRUSH) GetCurrentObject(hDC, OBJ_BRUSH); 1082 } 1083 1084 if ((nCount == -1) && (! lpOutputFunc)) 1085 return FALSE; 1086 1087 if (! nCount) 1088 { 1089 // TODO: calculate the length (easy enough) 1090 1091 if (unicode) 1092 nCount = lstrlenW((WCHAR*)lpData); 1093 else 1094 nCount = lstrlenA((CHAR*)lpData); 1095 } 1096 1097 if (! nWidth || ! nHeight) 1098 { 1099 SIZE s; 1100 // TODO: calculate the rect 1101 1102 if (unicode) 1103 success = GetTextExtentPoint32W(hDC, (WCHAR*) lpData, nCount, &s); 1104 else 1105 success = GetTextExtentPoint32A(hDC, (CHAR*) lpData, nCount, &s); 1106 1107 if (! success) goto cleanup; 1108 1109 if (! nWidth) nWidth = s.cx; 1110 if (! nHeight) nHeight = s.cy; 1111 } 1112 1113 SetRect(&r, X, Y, X + nWidth, Y + nHeight); 1114 1115 MemDC = CreateCompatibleDC(hDC); 1116 if (! MemDC) goto cleanup; 1117 MemBMP = CreateBitmap(nWidth, nHeight, 1, 1, NULL); 1118 if (! MemBMP) goto cleanup; 1119 OldBMP = SelectObject(MemDC, MemBMP); 1120 if (! OldBMP) goto cleanup; 1121 OldFont = SelectObject(MemDC, GetCurrentObject(hDC, OBJ_FONT)); 1122 if (! OldFont) goto cleanup; 1123 OldBrush = SelectObject(MemDC, hBrush); 1124 if (! OldBrush) goto cleanup; 1125 1126 if (! BitBlt(MemDC, 0, 0, nWidth, nHeight, hDC, X, Y, SRCCOPY)) goto cleanup; 1127 1128 SetTextColor(MemDC, RGB(255, 255, 255)); 1129 SetBkColor(MemDC, RGB(0, 0, 0)); 1130 1131 if (lpOutputFunc) 1132 { 1133 success = lpOutputFunc(MemDC, lpData, nCount); // Set brush etc first? 1134 1135 if ((nCount == -1) && (! success)) 1136 { 1137 // Don't gray (documented behaviour) 1138 success = (BOOL) BitBlt(hDC, X, Y, nWidth, nHeight, MemDC, 0, 0, SRCCOPY); 1139 goto cleanup; 1140 } 1141 } 1142 else 1143 { 1144 if (unicode) 1145 success = TextOutW(MemDC, 0, 0, (WCHAR*) lpData, nCount); 1146 else 1147 success = TextOutA(MemDC, 0, 0, (CHAR*) lpData, nCount); 1148 1149 if (! success) goto cleanup; 1150 1151 hbsave = (HBRUSH)SelectObject(MemDC, gpsi->hbrGray); 1152 PatBlt(MemDC, 0, 0, nWidth, nHeight, 0x000A0329); 1153 SelectObject(MemDC, hbsave); 1154 } 1155 1156 if (! BitBlt(hDC, X, Y, nWidth, nHeight, MemDC, 0, 0, SRCCOPY)) goto cleanup; 1157 1158 cleanup: 1159 SetTextColor(hDC, ForeColor); 1160 SetBkColor(hDC, BackColor); 1161 1162 if (MemDC) 1163 { 1164 if (OldFont) SelectObject(MemDC, OldFont); 1165 if (OldBrush) SelectObject(MemDC, OldBrush); 1166 if (OldBMP) SelectObject(MemDC, OldBMP); 1167 if (MemBMP) DeleteObject(MemBMP); 1168 DeleteDC(MemDC); 1169 } 1170 1171 return success; 1172 } 1173 1174 /********************************************************************** 1175 * PAINTING_DrawStateJam 1176 * 1177 * Jams in the requested type in the dc 1178 */ 1179 static BOOL PAINTING_DrawStateJam(HDC hdc, UINT opcode, 1180 DRAWSTATEPROC func, LPARAM lp, WPARAM wp, 1181 LPRECT rc, UINT dtflags, BOOL unicode ) 1182 { 1183 HDC memdc; 1184 HBITMAP hbmsave; 1185 BOOL retval; 1186 INT cx = rc->right - rc->left; 1187 INT cy = rc->bottom - rc->top; 1188 1189 switch(opcode) 1190 { 1191 case DST_TEXT: 1192 case DST_PREFIXTEXT: 1193 if(unicode) 1194 return DrawTextW(hdc, (LPWSTR)lp, (INT)wp, rc, dtflags); 1195 else 1196 return DrawTextA(hdc, (LPSTR)lp, (INT)wp, rc, dtflags); 1197 1198 case DST_ICON: 1199 return DrawIconEx(hdc, rc->left, rc->top, (HICON)lp, cx, cy, 0, NULL, DI_NORMAL); 1200 1201 case DST_BITMAP: 1202 memdc = CreateCompatibleDC(hdc); 1203 if(!memdc) 1204 return FALSE; 1205 hbmsave = (HBITMAP)SelectObject(memdc, (HBITMAP)lp); 1206 if(!hbmsave) 1207 { 1208 DeleteDC(memdc); 1209 return FALSE; 1210 } 1211 retval = BitBlt(hdc, rc->left, rc->top, cx, cy, memdc, 0, 0, SRCCOPY); 1212 SelectObject(memdc, hbmsave); 1213 DeleteDC(memdc); 1214 return retval; 1215 1216 case DST_COMPLEX: 1217 if(func) 1218 { 1219 BOOL bRet; 1220 /* DRAWSTATEPROC assumes that it draws at the center of coordinates */ 1221 1222 OffsetViewportOrgEx(hdc, rc->left, rc->top, NULL); 1223 bRet = func(hdc, lp, wp, cx, cy); 1224 /* Restore origin */ 1225 OffsetViewportOrgEx(hdc, -rc->left, -rc->top, NULL); 1226 return bRet; 1227 } 1228 else 1229 { 1230 return FALSE; 1231 } 1232 } 1233 return FALSE; 1234 } 1235 1236 static BOOL 1237 IntDrawState(HDC hdc, HBRUSH hbr, DRAWSTATEPROC func, LPARAM lp, WPARAM wp, 1238 INT x, INT y, INT cx, INT cy, UINT flags, BOOL unicode) 1239 { 1240 HBITMAP hbm, hbmsave; 1241 HFONT hfsave; 1242 HBRUSH hbsave, hbrtmp = 0; 1243 HDC memdc; 1244 RECT rc; 1245 UINT dtflags = DT_NOCLIP; 1246 COLORREF fg, bg; 1247 UINT opcode = flags & 0xf; 1248 INT len = wp; 1249 BOOL retval, tmp; 1250 LOGFONTW lf; 1251 HFONT hFontOriginal, hNaaFont = NULL; 1252 1253 if((opcode == DST_TEXT || opcode == DST_PREFIXTEXT) && !len) /* The string is '\0' terminated */ 1254 { 1255 if(unicode) 1256 len = lstrlenW((LPWSTR)lp); 1257 else 1258 len = lstrlenA((LPSTR)lp); 1259 } 1260 1261 hFontOriginal = GetCurrentObject(hdc, OBJ_FONT); 1262 if (flags & (DSS_MONO | DSS_DISABLED)) 1263 { 1264 /* Create a non-antialiased font */ 1265 GetObjectW(hFontOriginal, sizeof(lf), &lf); 1266 lf.lfQuality = NONANTIALIASED_QUALITY; 1267 hNaaFont = CreateFontIndirectW(&lf); 1268 } 1269 1270 /* Find out what size the image has if not given by caller */ 1271 if(!cx || !cy) 1272 { 1273 SIZE s; 1274 BITMAP bm; 1275 1276 switch(opcode) 1277 { 1278 case DST_TEXT: 1279 case DST_PREFIXTEXT: 1280 if(unicode) 1281 retval = GetTextExtentPoint32W(hdc, (LPWSTR)lp, len, &s); 1282 else 1283 retval = GetTextExtentPoint32A(hdc, (LPSTR)lp, len, &s); 1284 if(!retval) 1285 return FALSE; 1286 break; 1287 1288 case DST_ICON: 1289 if(!get_icon_size((HICON)lp, &s)) 1290 return FALSE; 1291 break; 1292 1293 case DST_BITMAP: 1294 if(!GetObjectW((HBITMAP)lp, sizeof(bm), &bm)) 1295 return FALSE; 1296 s.cx = bm.bmWidth; 1297 s.cy = bm.bmHeight; 1298 break; 1299 1300 case DST_COMPLEX: /* cx and cy must be set in this mode */ 1301 return FALSE; 1302 1303 default: 1304 ERR("Invalid opcode: %u\n", opcode); 1305 return FALSE; 1306 } 1307 1308 if(!cx) cx = s.cx; 1309 if(!cy) cy = s.cy; 1310 } 1311 1312 SetRect(&rc, x, y, x + cx, y + cy); 1313 1314 if(flags & DSS_RIGHT) /* This one is not documented in the win32.hlp file */ 1315 dtflags |= DT_RIGHT; 1316 if(opcode == DST_TEXT) 1317 dtflags |= DT_NOPREFIX; 1318 else if(opcode == DST_PREFIXTEXT) 1319 { 1320 if (flags & DSS_HIDEPREFIX) 1321 dtflags |= DT_HIDEPREFIX; 1322 if (flags & DSS_PREFIXONLY) 1323 dtflags |= DT_PREFIXONLY; 1324 } 1325 1326 /* For DSS_NORMAL we just jam in the image and return */ 1327 if((flags & 0x79f0) == DSS_NORMAL) 1328 { 1329 return PAINTING_DrawStateJam(hdc, opcode, func, lp, len, &rc, dtflags, unicode); 1330 } 1331 1332 /* For all other states we need to convert the image to B/W in a local bitmap */ 1333 /* before it is displayed */ 1334 fg = SetTextColor(hdc, RGB(0, 0, 0)); 1335 bg = SetBkColor(hdc, RGB(255, 255, 255)); 1336 hbm = NULL; hbmsave = NULL; 1337 memdc = NULL; hbsave = NULL; 1338 retval = FALSE; /* assume failure */ 1339 1340 /* From here on we must use "goto cleanup" when something goes wrong */ 1341 hbm = CreateBitmap(cx, cy, 1, 1, NULL); 1342 if(!hbm) goto cleanup; 1343 memdc = CreateCompatibleDC(hdc); 1344 if(!memdc) goto cleanup; 1345 hbmsave = (HBITMAP)SelectObject(memdc, hbm); 1346 if(!hbmsave) goto cleanup; 1347 SetRect(&rc, 0, 0, cx, cy); 1348 if(!FillRect(memdc, &rc, (HBRUSH)GetStockObject(WHITE_BRUSH))) goto cleanup; 1349 SetBkColor(memdc, RGB(255, 255, 255)); 1350 SetTextColor(memdc, RGB(0, 0, 0)); 1351 if (hNaaFont) 1352 hfsave = (HFONT)SelectObject(memdc, hNaaFont); 1353 else 1354 hfsave = (HFONT)SelectObject(memdc, hFontOriginal); 1355 SetLayout( memdc, GetLayout( hdc )); 1356 1357 /* DST_COMPLEX may draw text as well, 1358 * so we must be sure that correct font is selected 1359 */ 1360 if(!hfsave && (opcode <= DST_PREFIXTEXT)) goto cleanup; 1361 tmp = PAINTING_DrawStateJam(memdc, opcode, func, lp, len, &rc, dtflags, unicode); 1362 if(hfsave) SelectObject(memdc, hfsave); 1363 if (hNaaFont) DeleteObject(hNaaFont); 1364 if(!tmp) goto cleanup; 1365 1366 /* This state cause the image to be dithered */ 1367 if(flags & DSS_UNION) 1368 { 1369 hbsave = (HBRUSH)SelectObject(memdc, gpsi->hbrGray); 1370 if(!hbsave) goto cleanup; 1371 tmp = PatBlt(memdc, 0, 0, cx, cy, 0x00FA0089); 1372 SelectObject(memdc, hbsave); 1373 if(!tmp) goto cleanup; 1374 } 1375 1376 if (flags & DSS_DISABLED) 1377 hbrtmp = GetSysColorBrush(COLOR_3DHILIGHT); 1378 else if (flags & DSS_DEFAULT) 1379 hbrtmp = GetSysColorBrush(COLOR_3DSHADOW); 1380 1381 /* Draw light or dark shadow */ 1382 if (flags & (DSS_DISABLED|DSS_DEFAULT)) 1383 { 1384 if(!hbrtmp) goto cleanup; 1385 hbsave = (HBRUSH)SelectObject(hdc, hbrtmp); 1386 if(!hbsave) goto cleanup; 1387 if(!BitBlt(hdc, x+1, y+1, cx, cy, memdc, 0, 0, 0x00B8074A)) goto cleanup; 1388 SelectObject(hdc, hbsave); 1389 } 1390 1391 if (flags & DSS_DISABLED) 1392 { 1393 hbr = hbrtmp = GetSysColorBrush(COLOR_3DSHADOW); 1394 if(!hbrtmp) goto cleanup; 1395 } 1396 else if (!hbr) 1397 { 1398 hbr = (HBRUSH)GetStockObject(BLACK_BRUSH); 1399 } 1400 1401 hbsave = (HBRUSH)SelectObject(hdc, hbr); 1402 1403 if(!BitBlt(hdc, x, y, cx, cy, memdc, 0, 0, 0x00B8074A)) goto cleanup; 1404 1405 retval = TRUE; /* We succeeded */ 1406 1407 cleanup: 1408 SetTextColor(hdc, fg); 1409 SetBkColor(hdc, bg); 1410 1411 if(hbsave) SelectObject(hdc, hbsave); 1412 if(hbmsave) SelectObject(memdc, hbmsave); 1413 if(hbm) DeleteObject(hbm); 1414 if(memdc) DeleteDC(memdc); 1415 1416 return retval; 1417 } 1418 1419 /* 1420 * @implemented 1421 */ 1422 BOOL WINAPI 1423 RealDrawFrameControl(HDC hDC, LPRECT rc, UINT uType, UINT uState) 1424 { 1425 if (GetMapMode(hDC) != MM_TEXT) 1426 return FALSE; 1427 1428 switch(uType) 1429 { 1430 case DFC_BUTTON: 1431 return UITOOLS95_DrawFrameButton(hDC, rc, uState); 1432 case DFC_CAPTION: 1433 return UITOOLS95_DrawFrameCaption(hDC, rc, uState); 1434 case DFC_MENU: 1435 { 1436 BOOL ret; 1437 COLORREF rgbOldText; 1438 INT iOldBackMode; 1439 1440 if (uState & (DFCS_MENUARROWUP | DFCS_MENUARROWDOWN)) 1441 { 1442 if (!(uState & DFCS_TRANSPARENT)) 1443 FillRect(hDC, rc, (HBRUSH)(COLOR_MENU + 1)); /* Fill by menu color */ 1444 } 1445 else 1446 { 1447 FillRect(hDC, rc, (HBRUSH)GetStockObject(WHITE_BRUSH)); /* Fill by white */ 1448 } 1449 1450 rgbOldText = SetTextColor(hDC, RGB(0, 0, 0)); /* Draw by black */ 1451 iOldBackMode = SetBkMode(hDC, TRANSPARENT); 1452 ret = UITOOLS95_DrawFrameMenu(hDC, rc, uState); 1453 SetBkMode(hDC, iOldBackMode); 1454 SetTextColor(hDC, rgbOldText); 1455 return ret; 1456 } 1457 #if 0 1458 case DFC_POPUPMENU: 1459 UNIMPLEMENTED; 1460 break; 1461 #endif 1462 case DFC_SCROLL: 1463 return UITOOLS95_DrawFrameScroll(hDC, rc, uState); 1464 } 1465 return FALSE; 1466 } 1467 1468 BOOL WINAPI 1469 DrawFrameControl(HDC hDC, LPRECT rc, UINT uType, UINT uState) 1470 { 1471 BOOL Hook, Ret = FALSE; 1472 1473 LoadUserApiHook(); 1474 1475 Hook = BeginIfHookedUserApiHook(); 1476 1477 /* Bypass SEH and go direct. */ 1478 if (!Hook) return RealDrawFrameControl(hDC, rc, uType, uState); 1479 1480 _SEH2_TRY 1481 { 1482 Ret = guah.DrawFrameControl(hDC, rc, uType, uState); 1483 } 1484 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1485 { 1486 ERR("Got exception in hooked DrawFrameControl!\n"); 1487 } 1488 _SEH2_END; 1489 1490 EndUserApiHook(); 1491 1492 return Ret; 1493 } 1494 1495 /* 1496 * @implemented 1497 */ 1498 BOOL WINAPI 1499 DrawEdge(HDC hDC, LPRECT rc, UINT edge, UINT flags) 1500 { 1501 if (flags & BF_DIAGONAL) 1502 return IntDrawDiagEdge(hDC, rc, edge, flags); 1503 else 1504 return IntDrawRectEdge(hDC, rc, edge, flags, 1); 1505 } 1506 1507 /* 1508 * @implemented 1509 */ 1510 BOOL WINAPI 1511 GrayStringA(HDC hDC, HBRUSH hBrush, GRAYSTRINGPROC lpOutputFunc, LPARAM lpData, 1512 int nCount, int X, int Y, int nWidth, int nHeight) 1513 { 1514 return IntGrayString(hDC, hBrush, lpOutputFunc, lpData, nCount, X, Y, nWidth, nHeight, FALSE); 1515 } 1516 1517 /* 1518 * @implemented 1519 */ 1520 BOOL WINAPI 1521 GrayStringW(HDC hDC, HBRUSH hBrush, GRAYSTRINGPROC lpOutputFunc, LPARAM lpData, 1522 int nCount, int X, int Y, int nWidth, int nHeight) 1523 { 1524 return IntGrayString(hDC, hBrush, lpOutputFunc, lpData, nCount, X, Y, nWidth, nHeight, TRUE); 1525 } 1526 1527 /* 1528 * @implemented 1529 */ 1530 BOOL WINAPI 1531 InvertRect(HDC hDC, CONST RECT *lprc) 1532 { 1533 return PatBlt(hDC, lprc->left, lprc->top, lprc->right - lprc->left, 1534 lprc->bottom - lprc->top, DSTINVERT); 1535 } 1536 1537 /* 1538 * @implemented 1539 */ 1540 INT WINAPI 1541 FrameRect(HDC hDC, CONST RECT *lprc, HBRUSH hbr) 1542 { 1543 HBRUSH oldbrush; 1544 RECT r = *lprc; 1545 1546 if (IsRectEmpty(&r)) return 0; 1547 if (!(oldbrush = SelectObject(hDC, hbr))) return 0; 1548 1549 PatBlt(hDC, r.left, r.top, 1, r.bottom - r.top, PATCOPY); 1550 PatBlt(hDC, r.right - 1, r.top, 1, r.bottom - r.top, PATCOPY); 1551 PatBlt(hDC, r.left, r.top, r.right - r.left, 1, PATCOPY); 1552 PatBlt(hDC, r.left, r.bottom - 1, r.right - r.left, 1, PATCOPY); 1553 1554 SelectObject(hDC, oldbrush); 1555 return TRUE; 1556 } 1557 1558 /* 1559 * @implemented 1560 */ 1561 BOOL WINAPI 1562 FlashWindow(HWND hWnd, BOOL bInvert) 1563 { 1564 FLASHWINFO FlashWndInfo; 1565 1566 FlashWndInfo.cbSize = sizeof(FLASHWINFO); 1567 FlashWndInfo.hwnd = hWnd; 1568 FlashWndInfo.dwFlags = !bInvert ? 0 : (FLASHW_TRAY | FLASHW_CAPTION); 1569 FlashWndInfo.uCount = 1; 1570 FlashWndInfo.dwTimeout = 0; 1571 1572 return NtUserFlashWindowEx(&FlashWndInfo); 1573 } 1574 1575 /* 1576 * @implemented 1577 */ 1578 INT WINAPI 1579 FillRect(HDC hDC, CONST RECT *lprc, HBRUSH hbr) 1580 { 1581 BOOL Ret; 1582 HBRUSH prevhbr = NULL; 1583 1584 /* Select brush if specified */ 1585 if (hbr) 1586 { 1587 /* Handle system colors */ 1588 if (hbr <= (HBRUSH)(COLOR_MENUBAR + 1)) 1589 hbr = GetSysColorBrush(PtrToUlong(hbr) - 1); 1590 1591 prevhbr = SelectObject(hDC, hbr); 1592 if (prevhbr == NULL) 1593 return (INT)FALSE; 1594 } 1595 1596 Ret = PatBlt(hDC, lprc->left, lprc->top, lprc->right - lprc->left, 1597 lprc->bottom - lprc->top, PATCOPY); 1598 1599 /* Select old brush */ 1600 if (prevhbr) 1601 SelectObject(hDC, prevhbr); 1602 1603 return (INT)Ret; 1604 } 1605 1606 /* 1607 * @implemented 1608 */ 1609 BOOL WINAPI 1610 DrawFocusRect(HDC hdc, CONST RECT *rect) 1611 { 1612 HGDIOBJ OldObj; 1613 UINT cx, cy; 1614 1615 NtUserSystemParametersInfo(SPI_GETFOCUSBORDERWIDTH, 0, &cx, 0); 1616 NtUserSystemParametersInfo(SPI_GETFOCUSBORDERHEIGHT, 0, &cy, 0); 1617 1618 OldObj = SelectObject(hdc, gpsi->hbrGray); 1619 1620 /* top */ 1621 PatBlt(hdc, rect->left, rect->top, rect->right - rect->left, cy, PATINVERT); 1622 /* bottom */ 1623 PatBlt(hdc, rect->left, rect->bottom - cy, rect->right - rect->left, cy, PATINVERT); 1624 /* left */ 1625 PatBlt(hdc, rect->left, rect->top + cy, cx, rect->bottom - rect->top - (2 * cy), PATINVERT); 1626 /* right */ 1627 PatBlt(hdc, rect->right - cx, rect->top + cy, cx, rect->bottom - rect->top - (2 * cy), PATINVERT); 1628 1629 SelectObject(hdc, OldObj); 1630 return TRUE; 1631 } 1632 1633 /* 1634 * @implemented 1635 */ 1636 BOOL WINAPI 1637 DrawStateA(HDC hDC, HBRUSH hBrush, DRAWSTATEPROC lpOutputFunc, LPARAM lData, 1638 WPARAM wData, int x, int y, int cx, int cy, UINT fuFlags) 1639 { 1640 return IntDrawState(hDC, hBrush, lpOutputFunc, lData, wData, x, y, cx, cy, fuFlags, FALSE); 1641 } 1642 1643 /* 1644 * @implemented 1645 */ 1646 BOOL WINAPI 1647 DrawStateW(HDC hDC, HBRUSH hBrush, DRAWSTATEPROC lpOutputFunc, LPARAM lData, 1648 WPARAM wData, int x, int y, int cx, int cy, UINT fuFlags) 1649 { 1650 return IntDrawState(hDC, hBrush, lpOutputFunc, lData, wData, x, y, cx, cy, fuFlags, TRUE); 1651 } 1652