1 /* 2 * Copyright 2009, Stephan Aßmus <superstippi@gmx.de> 3 * Copyright 2012-2020 Haiku, Inc. All rights reserved. 4 * Distributed under the terms of the MIT License. 5 * 6 * Authors: 7 * Stephan Aßmus, superstippi@gmx.de 8 * John Scipione, jscipione@gmail.com 9 */ 10 11 12 #include <HaikuControlLook.h> 13 14 #include <algorithm> 15 16 #include <Bitmap.h> 17 #include <Control.h> 18 #include <GradientLinear.h> 19 #include <LayoutUtils.h> 20 #include <Region.h> 21 #include <Shape.h> 22 #include <String.h> 23 #include <TabView.h> 24 #include <View.h> 25 #include <Window.h> 26 #include <WindowPrivate.h> 27 28 29 namespace BPrivate { 30 31 static const float kEdgeBevelLightTint = 0.59; 32 static const float kEdgeBevelShadowTint = 1.0735; 33 static const float kHoverTintFactor = 0.85; 34 35 static const int32 kButtonPopUpIndicatorWidth = B_USE_ITEM_SPACING; 36 37 38 HaikuControlLook::HaikuControlLook() 39 : 40 fCachedOutline(false) 41 { 42 } 43 44 45 HaikuControlLook::~HaikuControlLook() 46 { 47 } 48 49 50 BAlignment 51 HaikuControlLook::DefaultLabelAlignment() const 52 { 53 return BAlignment(B_ALIGN_LEFT, B_ALIGN_VERTICAL_CENTER); 54 } 55 56 57 float 58 HaikuControlLook::DefaultLabelSpacing() const 59 { 60 return ceilf(be_plain_font->Size() / 2.0); 61 } 62 63 64 float 65 HaikuControlLook::DefaultItemSpacing() const 66 { 67 return ceilf(be_plain_font->Size() * 0.85); 68 } 69 70 71 uint32 72 HaikuControlLook::Flags(BControl* control) const 73 { 74 uint32 flags = B_IS_CONTROL; 75 76 if (!control->IsEnabled()) 77 flags |= B_DISABLED; 78 79 if (control->IsFocus() && control->Window() != NULL 80 && control->Window()->IsActive()) { 81 flags |= B_FOCUSED; 82 } 83 84 switch (control->Value()) { 85 case B_CONTROL_ON: 86 flags |= B_ACTIVATED; 87 break; 88 case B_CONTROL_PARTIALLY_ON: 89 flags |= B_PARTIALLY_ACTIVATED; 90 break; 91 } 92 93 if (control->Parent() != NULL 94 && (control->Parent()->Flags() & B_DRAW_ON_CHILDREN) != 0) { 95 // In this constellation, assume we want to render the control 96 // against the already existing view contents of the parent view. 97 flags |= B_BLEND_FRAME; 98 } 99 100 return flags; 101 } 102 103 104 // #pragma mark - 105 106 107 void 108 HaikuControlLook::DrawButtonFrame(BView* view, BRect& rect, const BRect& updateRect, 109 const rgb_color& base, const rgb_color& background, uint32 flags, 110 uint32 borders) 111 { 112 _DrawButtonFrame(view, rect, updateRect, 0.0f, 0.0f, 0.0f, 0.0f, base, 113 background, 1.0, 1.0, flags, borders); 114 } 115 116 117 void 118 HaikuControlLook::DrawButtonFrame(BView* view, BRect& rect, const BRect& updateRect, 119 float radius, const rgb_color& base, const rgb_color& background, uint32 flags, 120 uint32 borders) 121 { 122 _DrawButtonFrame(view, rect, updateRect, radius, radius, radius, radius, 123 base, background, 1.0, 1.0, flags, borders); 124 } 125 126 127 void 128 HaikuControlLook::DrawButtonFrame(BView* view, BRect& rect, 129 const BRect& updateRect, float leftTopRadius, float rightTopRadius, 130 float leftBottomRadius, float rightBottomRadius, const rgb_color& base, 131 const rgb_color& background, uint32 flags, 132 uint32 borders) 133 { 134 _DrawButtonFrame(view, rect, updateRect, leftTopRadius, rightTopRadius, 135 leftBottomRadius, rightBottomRadius, base, background, 136 1.0, 1.0, flags, borders); 137 } 138 139 140 void 141 HaikuControlLook::DrawButtonBackground(BView* view, BRect& rect, 142 const BRect& updateRect, const rgb_color& base, uint32 flags, 143 uint32 borders, orientation orientation) 144 { 145 _DrawButtonBackground(view, rect, updateRect, 0.0f, 0.0f, 0.0f, 0.0f, 146 base, false, flags, borders, orientation); 147 } 148 149 150 void 151 HaikuControlLook::DrawButtonBackground(BView* view, BRect& rect, 152 const BRect& updateRect, float radius, const rgb_color& base, uint32 flags, 153 uint32 borders, orientation orientation) 154 { 155 _DrawButtonBackground(view, rect, updateRect, radius, radius, radius, 156 radius, base, false, flags, borders, orientation); 157 } 158 159 160 void 161 HaikuControlLook::DrawButtonBackground(BView* view, BRect& rect, 162 const BRect& updateRect, float leftTopRadius, float rightTopRadius, 163 float leftBottomRadius, float rightBottomRadius, const rgb_color& base, 164 uint32 flags, uint32 borders, orientation orientation) 165 { 166 _DrawButtonBackground(view, rect, updateRect, leftTopRadius, 167 rightTopRadius, leftBottomRadius, rightBottomRadius, base, false, flags, 168 borders, orientation); 169 } 170 171 172 void 173 HaikuControlLook::DrawMenuBarBackground(BView* view, BRect& rect, 174 const BRect& updateRect, const rgb_color& base, uint32 flags, 175 uint32 borders) 176 { 177 if (!ShouldDraw(view, rect, updateRect)) 178 return; 179 180 // the surface edges 181 182 // colors 183 float topTint; 184 float bottomTint; 185 186 if ((flags & B_ACTIVATED) != 0) { 187 rgb_color bevelColor1 = tint_color(base, 1.40); 188 rgb_color bevelColor2 = tint_color(base, 1.25); 189 190 topTint = 1.25; 191 bottomTint = 1.20; 192 193 _DrawFrame(view, rect, 194 bevelColor1, bevelColor1, 195 bevelColor2, bevelColor2, 196 borders & B_TOP_BORDER); 197 } else { 198 rgb_color cornerColor = tint_color(base, 0.9); 199 rgb_color bevelColorTop = tint_color(base, 0.5); 200 rgb_color bevelColorLeft = tint_color(base, 0.7); 201 rgb_color bevelColorRightBottom = tint_color(base, 1.08); 202 203 topTint = 0.69; 204 bottomTint = 1.03; 205 206 _DrawFrame(view, rect, 207 bevelColorLeft, bevelColorTop, 208 bevelColorRightBottom, bevelColorRightBottom, 209 cornerColor, cornerColor, 210 borders); 211 } 212 213 // draw surface top 214 _FillGradient(view, rect, base, topTint, bottomTint); 215 } 216 217 218 void 219 HaikuControlLook::DrawMenuFieldFrame(BView* view, BRect& rect, 220 const BRect& updateRect, const rgb_color& base, 221 const rgb_color& background, uint32 flags, uint32 borders) 222 { 223 _DrawButtonFrame(view, rect, updateRect, 0.0f, 0.0f, 0.0f, 0.0f, base, 224 background, 0.6, 1.0, flags, borders); 225 } 226 227 228 void 229 HaikuControlLook::DrawMenuFieldFrame(BView* view, BRect& rect, 230 const BRect& updateRect, float radius, const rgb_color& base, 231 const rgb_color& background, uint32 flags, uint32 borders) 232 { 233 _DrawButtonFrame(view, rect, updateRect, radius, radius, radius, radius, 234 base, background, 0.6, 1.0, flags, borders); 235 } 236 237 238 void 239 HaikuControlLook::DrawMenuFieldFrame(BView* view, BRect& rect, 240 const BRect& updateRect, float leftTopRadius, 241 float rightTopRadius, float leftBottomRadius, 242 float rightBottomRadius, const rgb_color& base, 243 const rgb_color& background, uint32 flags, uint32 borders) 244 { 245 _DrawButtonFrame(view, rect, updateRect, leftTopRadius, rightTopRadius, 246 leftBottomRadius, rightBottomRadius, base, background, 0.6, 1.0, 247 flags, borders); 248 } 249 250 251 void 252 HaikuControlLook::DrawMenuFieldBackground(BView* view, BRect& rect, 253 const BRect& updateRect, const rgb_color& base, bool popupIndicator, 254 uint32 flags) 255 { 256 _DrawMenuFieldBackgroundOutside(view, rect, updateRect, 257 0.0f, 0.0f, 0.0f, 0.0f, base, popupIndicator, flags); 258 } 259 260 261 void 262 HaikuControlLook::DrawMenuFieldBackground(BView* view, BRect& rect, 263 const BRect& updateRect, const rgb_color& base, uint32 flags, 264 uint32 borders) 265 { 266 _DrawMenuFieldBackgroundInside(view, rect, updateRect, 267 0.0f, 0.0f, 0.0f, 0.0f, base, flags, borders); 268 } 269 270 271 void 272 HaikuControlLook::DrawMenuFieldBackground(BView* view, BRect& rect, 273 const BRect& updateRect, float radius, const rgb_color& base, 274 bool popupIndicator, uint32 flags) 275 { 276 _DrawMenuFieldBackgroundOutside(view, rect, updateRect, radius, radius, 277 radius, radius, base, popupIndicator, flags); 278 } 279 280 281 void 282 HaikuControlLook::DrawMenuFieldBackground(BView* view, BRect& rect, 283 const BRect& updateRect, float leftTopRadius, float rightTopRadius, 284 float leftBottomRadius, float rightBottomRadius, const rgb_color& base, 285 bool popupIndicator, uint32 flags) 286 { 287 _DrawMenuFieldBackgroundOutside(view, rect, updateRect, leftTopRadius, 288 rightTopRadius, leftBottomRadius, rightBottomRadius, base, 289 popupIndicator, flags); 290 } 291 292 293 void 294 HaikuControlLook::DrawMenuBackground(BView* view, BRect& rect, 295 const BRect& updateRect, const rgb_color& base, uint32 flags, 296 uint32 borders) 297 { 298 if (!ShouldDraw(view, rect, updateRect)) 299 return; 300 301 // inner bevel colors 302 rgb_color bevelLightColor; 303 rgb_color bevelShadowColor; 304 305 if ((flags & B_DISABLED) != 0) { 306 bevelLightColor = tint_color(base, 0.80); 307 bevelShadowColor = tint_color(base, 1.07); 308 } else { 309 bevelLightColor = tint_color(base, 0.6); 310 bevelShadowColor = tint_color(base, 1.12); 311 } 312 313 // draw inner bevel 314 _DrawFrame(view, rect, 315 bevelLightColor, bevelLightColor, 316 bevelShadowColor, bevelShadowColor, 317 borders); 318 319 // draw surface top 320 view->SetHighColor(base); 321 view->FillRect(rect); 322 } 323 324 325 void 326 HaikuControlLook::DrawMenuItemBackground(BView* view, BRect& rect, 327 const BRect& updateRect, const rgb_color& base, uint32 flags, 328 uint32 borders) 329 { 330 if (!ShouldDraw(view, rect, updateRect)) 331 return; 332 333 // surface edges 334 float topTint; 335 float bottomTint; 336 rgb_color selectedColor = base; 337 338 if ((flags & B_ACTIVATED) != 0) { 339 topTint = 0.9; 340 bottomTint = 1.05; 341 } else if ((flags & B_DISABLED) != 0) { 342 topTint = 0.80; 343 bottomTint = 1.07; 344 } else { 345 topTint = 0.6; 346 bottomTint = 1.12; 347 } 348 349 rgb_color bevelLightColor = tint_color(selectedColor, topTint); 350 rgb_color bevelShadowColor = tint_color(selectedColor, bottomTint); 351 352 // draw surface edges 353 _DrawFrame(view, rect, 354 bevelLightColor, bevelLightColor, 355 bevelShadowColor, bevelShadowColor, 356 borders); 357 358 // draw surface top 359 view->SetLowColor(selectedColor); 360 // _FillGradient(view, rect, selectedColor, topTint, bottomTint); 361 _FillGradient(view, rect, selectedColor, bottomTint, topTint); 362 } 363 364 365 void 366 HaikuControlLook::DrawStatusBar(BView* view, BRect& rect, const BRect& updateRect, 367 const rgb_color& base, const rgb_color& barColor, float progressPosition) 368 { 369 if (!ShouldDraw(view, rect, updateRect)) 370 return; 371 372 _DrawOuterResessedFrame(view, rect, base, 0.6); 373 374 // colors 375 rgb_color dark1BorderColor = tint_color(base, 1.3); 376 rgb_color dark2BorderColor = tint_color(base, 1.2); 377 rgb_color dark1FilledBorderColor = tint_color(barColor, 1.20); 378 rgb_color dark2FilledBorderColor = tint_color(barColor, 1.45); 379 380 BRect filledRect(rect); 381 filledRect.right = progressPosition - 1; 382 383 BRect nonfilledRect(rect); 384 nonfilledRect.left = progressPosition; 385 386 bool filledSurface = filledRect.Width() > 0; 387 bool nonfilledSurface = nonfilledRect.Width() > 0; 388 389 if (filledSurface) { 390 _DrawFrame(view, filledRect, 391 dark1FilledBorderColor, dark1FilledBorderColor, 392 dark2FilledBorderColor, dark2FilledBorderColor); 393 394 _FillGlossyGradient(view, filledRect, barColor, 0.55, 0.68, 0.76, 0.90); 395 } 396 397 if (nonfilledSurface) { 398 _DrawFrame(view, nonfilledRect, dark1BorderColor, dark1BorderColor, 399 dark2BorderColor, dark2BorderColor, 400 B_TOP_BORDER | B_BOTTOM_BORDER | B_RIGHT_BORDER); 401 402 if (nonfilledRect.left < nonfilledRect.right) { 403 // shadow from fill bar, or left border 404 rgb_color leftBorder = dark1BorderColor; 405 if (filledSurface) 406 leftBorder = tint_color(base, 0.50); 407 view->SetHighColor(leftBorder); 408 view->StrokeLine(nonfilledRect.LeftTop(), 409 nonfilledRect.LeftBottom()); 410 nonfilledRect.left++; 411 } 412 413 _FillGradient(view, nonfilledRect, base, 0.25, 0.06); 414 } 415 } 416 417 418 void 419 HaikuControlLook::DrawCheckBox(BView* view, BRect& rect, const BRect& updateRect, 420 const rgb_color& base, uint32 flags) 421 { 422 if (!ShouldDraw(view, rect, updateRect)) 423 return; 424 425 rgb_color dark1BorderColor; 426 rgb_color dark2BorderColor; 427 rgb_color navigationColor = ui_color(B_KEYBOARD_NAVIGATION_COLOR); 428 429 if ((flags & B_DISABLED) != 0) { 430 _DrawOuterResessedFrame(view, rect, base, 0.0, 1.0, flags); 431 432 dark1BorderColor = tint_color(base, 1.15); 433 dark2BorderColor = tint_color(base, 1.15); 434 } else if ((flags & B_CLICKED) != 0) { 435 dark1BorderColor = tint_color(base, 1.50); 436 dark2BorderColor = tint_color(base, 1.48); 437 438 _DrawFrame(view, rect, 439 dark1BorderColor, dark1BorderColor, 440 dark2BorderColor, dark2BorderColor); 441 442 dark2BorderColor = dark1BorderColor; 443 } else { 444 _DrawOuterResessedFrame(view, rect, base, 0.6, 1.0, flags); 445 446 dark1BorderColor = tint_color(base, 1.40); 447 dark2BorderColor = tint_color(base, 1.38); 448 } 449 450 if ((flags & B_FOCUSED) != 0) { 451 dark1BorderColor = navigationColor; 452 dark2BorderColor = navigationColor; 453 } 454 455 _DrawFrame(view, rect, 456 dark1BorderColor, dark1BorderColor, 457 dark2BorderColor, dark2BorderColor); 458 459 if ((flags & B_DISABLED) != 0) 460 _FillGradient(view, rect, base, 0.4, 0.2); 461 else 462 _FillGradient(view, rect, base, 0.15, 0.0); 463 464 rgb_color markColor; 465 if (_RadioButtonAndCheckBoxMarkColor(base, markColor, flags)) { 466 view->SetHighColor(markColor); 467 468 BFont font; 469 view->GetFont(&font); 470 float inset = std::max(2.0f, roundf(font.Size() / 6)); 471 rect.InsetBy(inset, inset); 472 473 float penSize = std::max(1.0f, ceilf(rect.Width() / 3.5f)); 474 if (penSize > 1.0f && fmodf(penSize, 2.0f) == 0.0f) { 475 // Tweak ends to "include" the pixel at the index, 476 // we need to do this in order to produce results like R5, 477 // where coordinates were inclusive 478 rect.right++; 479 rect.bottom++; 480 } 481 482 view->SetPenSize(penSize); 483 view->SetDrawingMode(B_OP_OVER); 484 view->StrokeLine(rect.LeftTop(), rect.RightBottom()); 485 view->StrokeLine(rect.LeftBottom(), rect.RightTop()); 486 } 487 } 488 489 490 void 491 HaikuControlLook::DrawRadioButton(BView* view, BRect& rect, const BRect& updateRect, 492 const rgb_color& base, uint32 flags) 493 { 494 if (!ShouldDraw(view, rect, updateRect)) 495 return; 496 497 rgb_color borderColor; 498 rgb_color bevelLight; 499 rgb_color bevelShadow; 500 rgb_color navigationColor = ui_color(B_KEYBOARD_NAVIGATION_COLOR); 501 502 if ((flags & B_DISABLED) != 0) { 503 borderColor = tint_color(base, 1.15); 504 bevelLight = base; 505 bevelShadow = base; 506 } else if ((flags & B_CLICKED) != 0) { 507 borderColor = tint_color(base, 1.50); 508 bevelLight = borderColor; 509 bevelShadow = borderColor; 510 } else { 511 borderColor = tint_color(base, 1.45); 512 bevelLight = tint_color(base, 0.55); 513 bevelShadow = tint_color(base, 1.11); 514 } 515 516 if ((flags & B_FOCUSED) != 0) { 517 borderColor = navigationColor; 518 } 519 520 BGradientLinear bevelGradient; 521 bevelGradient.AddColor(bevelShadow, 0); 522 bevelGradient.AddColor(bevelLight, 255); 523 bevelGradient.SetStart(rect.LeftTop()); 524 bevelGradient.SetEnd(rect.RightBottom()); 525 526 view->FillEllipse(rect, bevelGradient); 527 rect.InsetBy(1, 1); 528 529 bevelGradient.MakeEmpty(); 530 bevelGradient.AddColor(borderColor, 0); 531 bevelGradient.AddColor(tint_color(borderColor, 0.8), 255); 532 view->FillEllipse(rect, bevelGradient); 533 rect.InsetBy(1, 1); 534 535 float topTint; 536 float bottomTint; 537 if ((flags & B_DISABLED) != 0) { 538 topTint = 0.4; 539 bottomTint = 0.2; 540 } else { 541 topTint = 0.15; 542 bottomTint = 0.0; 543 } 544 545 BGradientLinear gradient; 546 _MakeGradient(gradient, rect, base, topTint, bottomTint); 547 view->FillEllipse(rect, gradient); 548 549 rgb_color markColor; 550 if (_RadioButtonAndCheckBoxMarkColor(base, markColor, flags)) { 551 view->SetHighColor(markColor); 552 BFont font; 553 view->GetFont(&font); 554 float inset = roundf(font.Size() / 4); 555 rect.InsetBy(inset, inset); 556 view->FillEllipse(rect); 557 } 558 } 559 560 561 void 562 HaikuControlLook::DrawScrollBarBorder(BView* view, BRect rect, 563 const BRect& updateRect, const rgb_color& base, uint32 flags, 564 orientation orientation) 565 { 566 if (!ShouldDraw(view, rect, updateRect)) 567 return; 568 569 view->PushState(); 570 571 // set clipping constraints to rect 572 view->ClipToRect(rect); 573 574 bool isEnabled = (flags & B_DISABLED) == 0; 575 bool isFocused = (flags & B_FOCUSED) != 0; 576 577 view->SetHighColor(tint_color(base, B_DARKEN_2_TINT)); 578 579 // stroke a line around the entire scrollbar 580 // take care of border highlighting, scroll target is focus view 581 if (isEnabled && isFocused) { 582 rgb_color borderColor = view->HighColor(); 583 rgb_color highlightColor = ui_color(B_KEYBOARD_NAVIGATION_COLOR); 584 585 view->BeginLineArray(4); 586 587 view->AddLine(BPoint(rect.left + 1, rect.bottom), 588 BPoint(rect.right, rect.bottom), borderColor); 589 view->AddLine(BPoint(rect.right, rect.top + 1), 590 BPoint(rect.right, rect.bottom - 1), borderColor); 591 592 if (orientation == B_HORIZONTAL) { 593 view->AddLine(BPoint(rect.left, rect.top + 1), 594 BPoint(rect.left, rect.bottom), borderColor); 595 } else { 596 view->AddLine(BPoint(rect.left, rect.top), 597 BPoint(rect.left, rect.bottom), highlightColor); 598 } 599 600 if (orientation == B_HORIZONTAL) { 601 view->AddLine(BPoint(rect.left, rect.top), 602 BPoint(rect.right, rect.top), highlightColor); 603 } else { 604 view->AddLine(BPoint(rect.left + 1, rect.top), 605 BPoint(rect.right, rect.top), borderColor); 606 } 607 608 view->EndLineArray(); 609 } else 610 view->StrokeRect(rect); 611 612 view->PopState(); 613 } 614 615 616 void 617 HaikuControlLook::DrawScrollBarButton(BView* view, BRect rect, 618 const BRect& updateRect, const rgb_color& base, uint32 flags, 619 int32 direction, orientation orientation, bool down) 620 { 621 if (!ShouldDraw(view, rect, updateRect)) 622 return; 623 624 view->PushState(); 625 626 // clip to button 627 view->ClipToRect(rect); 628 629 bool isEnabled = (flags & B_DISABLED) == 0; 630 631 rgb_color buttonColor = isEnabled ? base 632 : tint_color(base, B_LIGHTEN_1_TINT); 633 DrawButtonBackground(view, rect, updateRect, buttonColor, flags, 634 BControlLook::B_ALL_BORDERS, orientation); 635 636 rect.InsetBy(-1, -1); 637 DrawArrowShape(view, rect, updateRect, base, direction, flags, 1.9f); 638 // almost but not quite B_DARKEN_MAX_TINT 639 640 // revert clipping constraints 641 view->PopState(); 642 } 643 644 void 645 HaikuControlLook::DrawScrollBarBackground(BView* view, BRect& rect1, 646 BRect& rect2, const BRect& updateRect, const rgb_color& base, uint32 flags, 647 orientation orientation) 648 { 649 DrawScrollBarBackground(view, rect1, updateRect, base, flags, orientation); 650 DrawScrollBarBackground(view, rect2, updateRect, base, flags, orientation); 651 } 652 653 654 void 655 HaikuControlLook::DrawScrollBarBackground(BView* view, BRect& rect, 656 const BRect& updateRect, const rgb_color& base, uint32 flags, 657 orientation orientation) 658 { 659 if (!ShouldDraw(view, rect, updateRect)) 660 return; 661 662 view->PushState(); 663 664 // set clipping constraints to rect 665 view->ClipToRect(rect); 666 667 bool isEnabled = (flags & B_DISABLED) == 0; 668 669 // fill background, we'll draw arrows and thumb on top 670 view->SetDrawingMode(B_OP_COPY); 671 672 float gradient1Tint; 673 float gradient2Tint; 674 float darkEdge1Tint; 675 float darkEdge2Tint; 676 float shadowTint; 677 678 if (isEnabled) { 679 gradient1Tint = 1.10; 680 gradient2Tint = 1.05; 681 darkEdge1Tint = B_DARKEN_3_TINT; 682 darkEdge2Tint = B_DARKEN_2_TINT; 683 shadowTint = gradient1Tint; 684 } else { 685 gradient1Tint = 0.9; 686 gradient2Tint = 0.8; 687 darkEdge1Tint = B_DARKEN_2_TINT; 688 darkEdge2Tint = B_DARKEN_2_TINT; 689 shadowTint = gradient1Tint; 690 } 691 692 rgb_color darkEdge1 = tint_color(base, darkEdge1Tint); 693 rgb_color darkEdge2 = tint_color(base, darkEdge2Tint); 694 rgb_color shadow = tint_color(base, shadowTint); 695 696 if (orientation == B_HORIZONTAL) { 697 // dark vertical line on left edge 698 if (rect.Width() > 0) { 699 view->SetHighColor(darkEdge1); 700 view->StrokeLine(rect.LeftTop(), rect.LeftBottom()); 701 rect.left++; 702 } 703 // dark vertical line on right edge 704 if (rect.Width() >= 0) { 705 view->SetHighColor(darkEdge2); 706 view->StrokeLine(rect.RightTop(), rect.RightBottom()); 707 rect.right--; 708 } 709 // vertical shadow line after left edge 710 if (rect.Width() >= 0) { 711 view->SetHighColor(shadow); 712 view->StrokeLine(rect.LeftTop(), rect.LeftBottom()); 713 rect.left++; 714 } 715 // fill 716 if (rect.Width() >= 0) { 717 _FillGradient(view, rect, base, gradient1Tint, gradient2Tint, 718 orientation); 719 } 720 } else { 721 // dark vertical line on top edge 722 if (rect.Height() > 0) { 723 view->SetHighColor(darkEdge1); 724 view->StrokeLine(rect.LeftTop(), rect.RightTop()); 725 rect.top++; 726 } 727 // dark vertical line on bottom edge 728 if (rect.Height() >= 0) { 729 view->SetHighColor(darkEdge2); 730 view->StrokeLine(rect.LeftBottom(), rect.RightBottom()); 731 rect.bottom--; 732 } 733 // horizontal shadow line after top edge 734 if (rect.Height() >= 0) { 735 view->SetHighColor(shadow); 736 view->StrokeLine(rect.LeftTop(), rect.RightTop()); 737 rect.top++; 738 } 739 // fill 740 if (rect.Height() >= 0) { 741 _FillGradient(view, rect, base, gradient1Tint, gradient2Tint, 742 orientation); 743 } 744 } 745 746 view->PopState(); 747 } 748 749 750 void 751 HaikuControlLook::DrawScrollBarThumb(BView* view, BRect& rect, 752 const BRect& updateRect, const rgb_color& base, uint32 flags, 753 orientation orientation, uint32 knobStyle) 754 { 755 if (!ShouldDraw(view, rect, updateRect)) 756 return; 757 758 view->PushState(); 759 760 // set clipping constraints to rect 761 view->ClipToRect(rect); 762 763 // flags 764 bool isEnabled = (flags & B_DISABLED) == 0; 765 766 // colors 767 rgb_color thumbColor = ui_color(B_SCROLL_BAR_THUMB_COLOR); 768 const float bgTint = 1.06; 769 770 rgb_color light, dark, dark1, dark2; 771 if (isEnabled) { 772 light = tint_color(base, B_LIGHTEN_MAX_TINT); 773 dark = tint_color(base, B_DARKEN_3_TINT); 774 dark1 = tint_color(base, B_DARKEN_1_TINT); 775 dark2 = tint_color(base, B_DARKEN_2_TINT); 776 } else { 777 light = tint_color(base, B_LIGHTEN_MAX_TINT); 778 dark = tint_color(base, B_DARKEN_2_TINT); 779 dark1 = tint_color(base, B_LIGHTEN_2_TINT); 780 dark2 = tint_color(base, B_LIGHTEN_1_TINT); 781 } 782 783 // draw thumb over background 784 view->SetDrawingMode(B_OP_OVER); 785 view->SetHighColor(dark1); 786 787 // draw scroll thumb 788 if (isEnabled) { 789 // fill the clickable surface of the thumb 790 DrawButtonBackground(view, rect, updateRect, thumbColor, 0, 791 B_ALL_BORDERS, orientation); 792 } else { 793 // thumb bevel 794 view->BeginLineArray(4); 795 view->AddLine(BPoint(rect.left, rect.bottom), 796 BPoint(rect.left, rect.top), light); 797 view->AddLine(BPoint(rect.left + 1, rect.top), 798 BPoint(rect.right, rect.top), light); 799 view->AddLine(BPoint(rect.right, rect.top + 1), 800 BPoint(rect.right, rect.bottom), dark2); 801 view->AddLine(BPoint(rect.right - 1, rect.bottom), 802 BPoint(rect.left + 1, rect.bottom), dark2); 803 view->EndLineArray(); 804 805 // thumb fill 806 rect.InsetBy(1, 1); 807 view->SetHighColor(dark1); 808 view->FillRect(rect); 809 } 810 811 // draw knob style 812 if (knobStyle != B_KNOB_NONE) { 813 rgb_color knobLight = isEnabled 814 ? tint_color(thumbColor, B_LIGHTEN_MAX_TINT) 815 : tint_color(dark1, bgTint); 816 rgb_color knobDark = isEnabled 817 ? tint_color(thumbColor, 1.22) 818 : tint_color(knobLight, B_DARKEN_1_TINT); 819 820 if (knobStyle == B_KNOB_DOTS) { 821 // draw dots on the scroll bar thumb 822 float hcenter = rect.left + rect.Width() / 2; 823 float vmiddle = rect.top + rect.Height() / 2; 824 BRect knob(hcenter, vmiddle, hcenter, vmiddle); 825 826 if (orientation == B_HORIZONTAL) { 827 view->SetHighColor(knobDark); 828 view->FillRect(knob); 829 view->SetHighColor(knobLight); 830 view->FillRect(knob.OffsetByCopy(1, 1)); 831 832 float spacer = rect.Height(); 833 834 if (rect.left + 3 < hcenter - spacer) { 835 view->SetHighColor(knobDark); 836 view->FillRect(knob.OffsetByCopy(-spacer, 0)); 837 view->SetHighColor(knobLight); 838 view->FillRect(knob.OffsetByCopy(-spacer + 1, 1)); 839 } 840 841 if (rect.right - 3 > hcenter + spacer) { 842 view->SetHighColor(knobDark); 843 view->FillRect(knob.OffsetByCopy(spacer, 0)); 844 view->SetHighColor(knobLight); 845 view->FillRect(knob.OffsetByCopy(spacer + 1, 1)); 846 } 847 } else { 848 // B_VERTICAL 849 view->SetHighColor(knobDark); 850 view->FillRect(knob); 851 view->SetHighColor(knobLight); 852 view->FillRect(knob.OffsetByCopy(1, 1)); 853 854 float spacer = rect.Width(); 855 856 if (rect.top + 3 < vmiddle - spacer) { 857 view->SetHighColor(knobDark); 858 view->FillRect(knob.OffsetByCopy(0, -spacer)); 859 view->SetHighColor(knobLight); 860 view->FillRect(knob.OffsetByCopy(1, -spacer + 1)); 861 } 862 863 if (rect.bottom - 3 > vmiddle + spacer) { 864 view->SetHighColor(knobDark); 865 view->FillRect(knob.OffsetByCopy(0, spacer)); 866 view->SetHighColor(knobLight); 867 view->FillRect(knob.OffsetByCopy(1, spacer + 1)); 868 } 869 } 870 } else if (knobStyle == B_KNOB_LINES) { 871 // draw lines on the scroll bar thumb 872 if (orientation == B_HORIZONTAL) { 873 float middle = rect.Width() / 2; 874 875 view->BeginLineArray(6); 876 view->AddLine( 877 BPoint(rect.left + middle - 3, rect.top + 2), 878 BPoint(rect.left + middle - 3, rect.bottom - 2), 879 knobDark); 880 view->AddLine( 881 BPoint(rect.left + middle, rect.top + 2), 882 BPoint(rect.left + middle, rect.bottom - 2), 883 knobDark); 884 view->AddLine( 885 BPoint(rect.left + middle + 3, rect.top + 2), 886 BPoint(rect.left + middle + 3, rect.bottom - 2), 887 knobDark); 888 view->AddLine( 889 BPoint(rect.left + middle - 2, rect.top + 2), 890 BPoint(rect.left + middle - 2, rect.bottom - 2), 891 knobLight); 892 view->AddLine( 893 BPoint(rect.left + middle + 1, rect.top + 2), 894 BPoint(rect.left + middle + 1, rect.bottom - 2), 895 knobLight); 896 view->AddLine( 897 BPoint(rect.left + middle + 4, rect.top + 2), 898 BPoint(rect.left + middle + 4, rect.bottom - 2), 899 knobLight); 900 view->EndLineArray(); 901 } else { 902 // B_VERTICAL 903 float middle = rect.Height() / 2; 904 905 view->BeginLineArray(6); 906 view->AddLine( 907 BPoint(rect.left + 2, rect.top + middle - 3), 908 BPoint(rect.right - 2, rect.top + middle - 3), 909 knobDark); 910 view->AddLine( 911 BPoint(rect.left + 2, rect.top + middle), 912 BPoint(rect.right - 2, rect.top + middle), 913 knobDark); 914 view->AddLine( 915 BPoint(rect.left + 2, rect.top + middle + 3), 916 BPoint(rect.right - 2, rect.top + middle + 3), 917 knobDark); 918 view->AddLine( 919 BPoint(rect.left + 2, rect.top + middle - 2), 920 BPoint(rect.right - 2, rect.top + middle - 2), 921 knobLight); 922 view->AddLine( 923 BPoint(rect.left + 2, rect.top + middle + 1), 924 BPoint(rect.right - 2, rect.top + middle + 1), 925 knobLight); 926 view->AddLine( 927 BPoint(rect.left + 2, rect.top + middle + 4), 928 BPoint(rect.right - 2, rect.top + middle + 4), 929 knobLight); 930 view->EndLineArray(); 931 } 932 } 933 } 934 935 view->PopState(); 936 } 937 938 939 void 940 HaikuControlLook::DrawScrollViewFrame(BView* view, BRect& rect, 941 const BRect& updateRect, BRect verticalScrollBarFrame, 942 BRect horizontalScrollBarFrame, const rgb_color& base, 943 border_style borderStyle, uint32 flags, uint32 _borders) 944 { 945 // calculate scroll corner rect before messing with the "rect" 946 BRect scrollCornerFillRect(rect.right, rect.bottom, 947 rect.right, rect.bottom); 948 949 if (horizontalScrollBarFrame.IsValid()) 950 scrollCornerFillRect.left = horizontalScrollBarFrame.right + 1; 951 952 if (verticalScrollBarFrame.IsValid()) 953 scrollCornerFillRect.top = verticalScrollBarFrame.bottom + 1; 954 955 if (borderStyle == B_NO_BORDER) { 956 if (scrollCornerFillRect.IsValid()) { 957 view->SetHighColor(base); 958 view->FillRect(scrollCornerFillRect); 959 } 960 return; 961 } 962 963 bool excludeScrollCorner = borderStyle == B_FANCY_BORDER 964 && horizontalScrollBarFrame.IsValid() 965 && verticalScrollBarFrame.IsValid(); 966 967 uint32 borders = _borders; 968 if (excludeScrollCorner) { 969 rect.bottom = horizontalScrollBarFrame.top; 970 rect.right = verticalScrollBarFrame.left; 971 borders &= ~(B_RIGHT_BORDER | B_BOTTOM_BORDER); 972 } 973 974 rgb_color scrollbarFrameColor = tint_color(base, B_DARKEN_2_TINT); 975 976 if (borderStyle == B_FANCY_BORDER) 977 _DrawOuterResessedFrame(view, rect, base, 1.0, 1.0, flags, borders); 978 979 if ((flags & B_FOCUSED) != 0) { 980 rgb_color focusColor = ui_color(B_KEYBOARD_NAVIGATION_COLOR); 981 _DrawFrame(view, rect, focusColor, focusColor, focusColor, focusColor, 982 borders); 983 } else { 984 _DrawFrame(view, rect, scrollbarFrameColor, scrollbarFrameColor, 985 scrollbarFrameColor, scrollbarFrameColor, borders); 986 } 987 988 if (excludeScrollCorner) { 989 horizontalScrollBarFrame.InsetBy(-1, -1); 990 // do not overdraw the top edge 991 horizontalScrollBarFrame.top += 2; 992 borders = _borders; 993 borders &= ~B_TOP_BORDER; 994 _DrawOuterResessedFrame(view, horizontalScrollBarFrame, base, 995 1.0, 1.0, flags, borders); 996 _DrawFrame(view, horizontalScrollBarFrame, scrollbarFrameColor, 997 scrollbarFrameColor, scrollbarFrameColor, scrollbarFrameColor, 998 borders); 999 1000 verticalScrollBarFrame.InsetBy(-1, -1); 1001 // do not overdraw the left edge 1002 verticalScrollBarFrame.left += 2; 1003 borders = _borders; 1004 borders &= ~B_LEFT_BORDER; 1005 _DrawOuterResessedFrame(view, verticalScrollBarFrame, base, 1006 1.0, 1.0, flags, borders); 1007 _DrawFrame(view, verticalScrollBarFrame, scrollbarFrameColor, 1008 scrollbarFrameColor, scrollbarFrameColor, scrollbarFrameColor, 1009 borders); 1010 1011 // exclude recessed frame 1012 scrollCornerFillRect.top++; 1013 scrollCornerFillRect.left++; 1014 } 1015 1016 if (scrollCornerFillRect.IsValid()) { 1017 view->SetHighColor(base); 1018 view->FillRect(scrollCornerFillRect); 1019 } 1020 } 1021 1022 1023 void 1024 HaikuControlLook::DrawArrowShape(BView* view, BRect& rect, 1025 const BRect& updateRect, const rgb_color& base, uint32 direction, 1026 uint32 flags, float tint) 1027 { 1028 BPoint tri1, tri2, tri3; 1029 float hInset = rect.Width() / 3; 1030 float vInset = rect.Height() / 3; 1031 rect.InsetBy(hInset, vInset); 1032 1033 switch (direction) { 1034 case B_LEFT_ARROW: 1035 tri1.Set(rect.right, rect.top); 1036 tri2.Set(rect.right - rect.Width() / 1.33, 1037 (rect.top + rect.bottom + 1) / 2); 1038 tri3.Set(rect.right, rect.bottom + 1); 1039 break; 1040 case B_RIGHT_ARROW: 1041 tri1.Set(rect.left + 1, rect.bottom + 1); 1042 tri2.Set(rect.left + 1 + rect.Width() / 1.33, 1043 (rect.top + rect.bottom + 1) / 2); 1044 tri3.Set(rect.left + 1, rect.top); 1045 break; 1046 case B_UP_ARROW: 1047 tri1.Set(rect.left, rect.bottom); 1048 tri2.Set((rect.left + rect.right + 1) / 2, 1049 rect.bottom - rect.Height() / 1.33); 1050 tri3.Set(rect.right + 1, rect.bottom); 1051 break; 1052 case B_DOWN_ARROW: 1053 default: 1054 tri1.Set(rect.left, rect.top + 1); 1055 tri2.Set((rect.left + rect.right + 1) / 2, 1056 rect.top + 1 + rect.Height() / 1.33); 1057 tri3.Set(rect.right + 1, rect.top + 1); 1058 break; 1059 case B_LEFT_UP_ARROW: 1060 tri1.Set(rect.left, rect.bottom); 1061 tri2.Set(rect.left, rect.top); 1062 tri3.Set(rect.right - 1, rect.top); 1063 break; 1064 case B_RIGHT_UP_ARROW: 1065 tri1.Set(rect.left + 1, rect.top); 1066 tri2.Set(rect.right, rect.top); 1067 tri3.Set(rect.right, rect.bottom); 1068 break; 1069 case B_RIGHT_DOWN_ARROW: 1070 tri1.Set(rect.right, rect.top); 1071 tri2.Set(rect.right, rect.bottom); 1072 tri3.Set(rect.left + 1, rect.bottom); 1073 break; 1074 case B_LEFT_DOWN_ARROW: 1075 tri1.Set(rect.right - 1, rect.bottom); 1076 tri2.Set(rect.left, rect.bottom); 1077 tri3.Set(rect.left, rect.top); 1078 break; 1079 } 1080 1081 BShape arrowShape; 1082 arrowShape.MoveTo(tri1); 1083 arrowShape.LineTo(tri2); 1084 arrowShape.LineTo(tri3); 1085 1086 if ((flags & B_DISABLED) != 0) 1087 tint = (tint + B_NO_TINT + B_NO_TINT) / 3; 1088 1089 view->SetHighColor(tint_color(base, tint)); 1090 1091 float penSize = view->PenSize(); 1092 drawing_mode mode = view->DrawingMode(); 1093 1094 view->MovePenTo(BPoint(0, 0)); 1095 1096 view->SetPenSize(ceilf(hInset / 2.0)); 1097 view->SetDrawingMode(B_OP_OVER); 1098 view->StrokeShape(&arrowShape); 1099 1100 view->SetPenSize(penSize); 1101 view->SetDrawingMode(mode); 1102 } 1103 1104 1105 rgb_color 1106 HaikuControlLook::SliderBarColor(const rgb_color& base) 1107 { 1108 return tint_color(ui_color(B_PANEL_BACKGROUND_COLOR), B_DARKEN_1_TINT); 1109 } 1110 1111 1112 void 1113 HaikuControlLook::DrawSliderBar(BView* view, BRect rect, const BRect& updateRect, 1114 const rgb_color& base, rgb_color leftFillColor, rgb_color rightFillColor, 1115 float sliderScale, uint32 flags, orientation orientation) 1116 { 1117 if (!ShouldDraw(view, rect, updateRect)) 1118 return; 1119 1120 // separate the bar in two sides 1121 float sliderPosition; 1122 BRect leftBarSide = rect; 1123 BRect rightBarSide = rect; 1124 1125 if (orientation == B_HORIZONTAL) { 1126 sliderPosition = floorf(rect.left + 2 + (rect.Width() - 2) 1127 * sliderScale); 1128 leftBarSide.right = sliderPosition - 1; 1129 rightBarSide.left = sliderPosition; 1130 } else { 1131 // NOTE: position is reverse of coords 1132 sliderPosition = floorf(rect.top + 2 + (rect.Height() - 2) 1133 * (1.0 - sliderScale)); 1134 leftBarSide.top = sliderPosition; 1135 rightBarSide.bottom = sliderPosition - 1; 1136 } 1137 1138 view->PushState(); 1139 view->ClipToRect(leftBarSide); 1140 DrawSliderBar(view, rect, updateRect, base, leftFillColor, flags, 1141 orientation); 1142 view->PopState(); 1143 1144 view->PushState(); 1145 view->ClipToRect(rightBarSide); 1146 DrawSliderBar(view, rect, updateRect, base, rightFillColor, flags, 1147 orientation); 1148 view->PopState(); 1149 } 1150 1151 1152 void 1153 HaikuControlLook::DrawSliderBar(BView* view, BRect rect, const BRect& updateRect, 1154 const rgb_color& base, rgb_color fillColor, uint32 flags, 1155 orientation orientation) 1156 { 1157 if (!ShouldDraw(view, rect, updateRect)) 1158 return; 1159 1160 // separate the rect into corners 1161 BRect leftCorner(rect); 1162 BRect rightCorner(rect); 1163 BRect barRect(rect); 1164 1165 if (orientation == B_HORIZONTAL) { 1166 leftCorner.right = leftCorner.left + leftCorner.Height(); 1167 rightCorner.left = rightCorner.right - rightCorner.Height(); 1168 barRect.left += ceilf(barRect.Height() / 2); 1169 barRect.right -= ceilf(barRect.Height() / 2); 1170 } else { 1171 leftCorner.bottom = leftCorner.top + leftCorner.Width(); 1172 rightCorner.top = rightCorner.bottom - rightCorner.Width(); 1173 barRect.top += ceilf(barRect.Width() / 2); 1174 barRect.bottom -= ceilf(barRect.Width() / 2); 1175 } 1176 1177 // fill the background for the corners, exclude the middle bar for now 1178 view->PushState(); 1179 view->ClipToRect(rect); 1180 view->ClipToInverseRect(barRect); 1181 1182 if ((flags & B_BLEND_FRAME) == 0) { 1183 view->SetHighColor(base); 1184 view->FillRect(rect); 1185 } 1186 1187 // figure out the tints to be used 1188 float edgeLightTint; 1189 float edgeShadowTint; 1190 float frameLightTint; 1191 float frameShadowTint; 1192 float fillLightTint; 1193 float fillShadowTint; 1194 uint8 edgeLightAlpha; 1195 uint8 edgeShadowAlpha; 1196 uint8 frameLightAlpha; 1197 uint8 frameShadowAlpha; 1198 1199 if ((flags & B_DISABLED) != 0) { 1200 edgeLightTint = 1.0; 1201 edgeShadowTint = 1.0; 1202 frameLightTint = 1.20; 1203 frameShadowTint = 1.25; 1204 fillLightTint = 0.9; 1205 fillShadowTint = 1.05; 1206 edgeLightAlpha = 12; 1207 edgeShadowAlpha = 12; 1208 frameLightAlpha = 40; 1209 frameShadowAlpha = 45; 1210 1211 fillColor.red = uint8(fillColor.red * 0.4 + base.red * 0.6); 1212 fillColor.green = uint8(fillColor.green * 0.4 + base.green * 0.6); 1213 fillColor.blue = uint8(fillColor.blue * 0.4 + base.blue * 0.6); 1214 } else { 1215 edgeLightTint = 0.65; 1216 edgeShadowTint = 1.07; 1217 frameLightTint = 1.40; 1218 frameShadowTint = 1.50; 1219 fillLightTint = 0.8; 1220 fillShadowTint = 1.1; 1221 edgeLightAlpha = 15; 1222 edgeShadowAlpha = 15; 1223 frameLightAlpha = 92; 1224 frameShadowAlpha = 107; 1225 } 1226 1227 rgb_color edgeLightColor; 1228 rgb_color edgeShadowColor; 1229 rgb_color frameLightColor; 1230 rgb_color frameShadowColor; 1231 rgb_color fillLightColor = tint_color(fillColor, fillLightTint); 1232 rgb_color fillShadowColor = tint_color(fillColor, fillShadowTint); 1233 1234 drawing_mode oldMode = view->DrawingMode(); 1235 1236 if ((flags & B_BLEND_FRAME) != 0) { 1237 edgeLightColor = (rgb_color){ 255, 255, 255, edgeLightAlpha }; 1238 edgeShadowColor = (rgb_color){ 0, 0, 0, edgeShadowAlpha }; 1239 frameLightColor = (rgb_color){ 0, 0, 0, frameLightAlpha }; 1240 frameShadowColor = (rgb_color){ 0, 0, 0, frameShadowAlpha }; 1241 1242 view->SetDrawingMode(B_OP_ALPHA); 1243 } else { 1244 edgeLightColor = tint_color(base, edgeLightTint); 1245 edgeShadowColor = tint_color(base, edgeShadowTint); 1246 frameLightColor = tint_color(fillColor, frameLightTint); 1247 frameShadowColor = tint_color(fillColor, frameShadowTint); 1248 } 1249 1250 if (orientation == B_HORIZONTAL) { 1251 _DrawRoundBarCorner(view, leftCorner, updateRect, edgeLightColor, 1252 edgeShadowColor, frameLightColor, frameShadowColor, fillLightColor, 1253 fillShadowColor, 1.0, 1.0, 0.0, -1.0, orientation); 1254 1255 _DrawRoundBarCorner(view, rightCorner, updateRect, edgeLightColor, 1256 edgeShadowColor, frameLightColor, frameShadowColor, fillLightColor, 1257 fillShadowColor, 0.0, 1.0, -1.0, -1.0, orientation); 1258 } else { 1259 _DrawRoundBarCorner(view, leftCorner, updateRect, edgeLightColor, 1260 edgeShadowColor, frameLightColor, frameShadowColor, fillLightColor, 1261 fillShadowColor, 1.0, 1.0, -1.0, 0.0, orientation); 1262 1263 _DrawRoundBarCorner(view, rightCorner, updateRect, edgeLightColor, 1264 edgeShadowColor, frameLightColor, frameShadowColor, fillLightColor, 1265 fillShadowColor, 1.0, 0.0, -1.0, -1.0, orientation); 1266 } 1267 1268 view->PopState(); 1269 if ((flags & B_BLEND_FRAME) != 0) 1270 view->SetDrawingMode(B_OP_ALPHA); 1271 1272 view->BeginLineArray(4); 1273 if (orientation == B_HORIZONTAL) { 1274 view->AddLine(barRect.LeftTop(), barRect.RightTop(), 1275 edgeShadowColor); 1276 view->AddLine(barRect.LeftBottom(), barRect.RightBottom(), 1277 edgeLightColor); 1278 barRect.InsetBy(0, 1); 1279 view->AddLine(barRect.LeftTop(), barRect.RightTop(), 1280 frameShadowColor); 1281 view->AddLine(barRect.LeftBottom(), barRect.RightBottom(), 1282 frameLightColor); 1283 barRect.InsetBy(0, 1); 1284 } else { 1285 view->AddLine(barRect.LeftTop(), barRect.LeftBottom(), 1286 edgeShadowColor); 1287 view->AddLine(barRect.RightTop(), barRect.RightBottom(), 1288 edgeLightColor); 1289 barRect.InsetBy(1, 0); 1290 view->AddLine(barRect.LeftTop(), barRect.LeftBottom(), 1291 frameShadowColor); 1292 view->AddLine(barRect.RightTop(), barRect.RightBottom(), 1293 frameLightColor); 1294 barRect.InsetBy(1, 0); 1295 } 1296 view->EndLineArray(); 1297 1298 view->SetDrawingMode(oldMode); 1299 1300 _FillGradient(view, barRect, fillColor, fillShadowTint, fillLightTint, 1301 orientation); 1302 } 1303 1304 1305 void 1306 HaikuControlLook::DrawSliderThumb(BView* view, BRect& rect, const BRect& updateRect, 1307 const rgb_color& base, uint32 flags, orientation orientation) 1308 { 1309 if (!ShouldDraw(view, rect, updateRect)) 1310 return; 1311 1312 // figure out frame color 1313 rgb_color frameLightColor; 1314 rgb_color frameShadowColor; 1315 rgb_color shadowColor = (rgb_color){ 0, 0, 0, 60 }; 1316 1317 if ((flags & B_FOCUSED) != 0) { 1318 // focused 1319 frameLightColor = ui_color(B_KEYBOARD_NAVIGATION_COLOR); 1320 frameShadowColor = frameLightColor; 1321 } else { 1322 // figure out the tints to be used 1323 float frameLightTint; 1324 float frameShadowTint; 1325 1326 if ((flags & B_DISABLED) != 0) { 1327 frameLightTint = 1.30; 1328 frameShadowTint = 1.35; 1329 shadowColor.alpha = 30; 1330 } else { 1331 frameLightTint = 1.6; 1332 frameShadowTint = 1.65; 1333 } 1334 1335 frameLightColor = tint_color(base, frameLightTint); 1336 frameShadowColor = tint_color(base, frameShadowTint); 1337 } 1338 1339 BRect originalRect(rect); 1340 rect.right--; 1341 rect.bottom--; 1342 1343 _DrawFrame(view, rect, frameLightColor, frameLightColor, 1344 frameShadowColor, frameShadowColor); 1345 1346 flags &= ~B_ACTIVATED; 1347 DrawButtonBackground(view, rect, updateRect, base, flags); 1348 1349 // thumb shadow 1350 view->SetDrawingMode(B_OP_ALPHA); 1351 view->SetHighColor(shadowColor); 1352 originalRect.left++; 1353 originalRect.top++; 1354 view->StrokeLine(originalRect.LeftBottom(), originalRect.RightBottom()); 1355 originalRect.bottom--; 1356 view->StrokeLine(originalRect.RightTop(), originalRect.RightBottom()); 1357 1358 // thumb edge 1359 if (orientation == B_HORIZONTAL) { 1360 rect.InsetBy(0, floorf(rect.Height() / 4)); 1361 rect.left = floorf((rect.left + rect.right) / 2); 1362 rect.right = rect.left + 1; 1363 shadowColor = tint_color(base, B_DARKEN_2_TINT); 1364 shadowColor.alpha = 128; 1365 view->SetHighColor(shadowColor); 1366 view->StrokeLine(rect.LeftTop(), rect.LeftBottom()); 1367 rgb_color lightColor = tint_color(base, B_LIGHTEN_2_TINT); 1368 lightColor.alpha = 128; 1369 view->SetHighColor(lightColor); 1370 view->StrokeLine(rect.RightTop(), rect.RightBottom()); 1371 } else { 1372 rect.InsetBy(floorf(rect.Width() / 4), 0); 1373 rect.top = floorf((rect.top + rect.bottom) / 2); 1374 rect.bottom = rect.top + 1; 1375 shadowColor = tint_color(base, B_DARKEN_2_TINT); 1376 shadowColor.alpha = 128; 1377 view->SetHighColor(shadowColor); 1378 view->StrokeLine(rect.LeftTop(), rect.RightTop()); 1379 rgb_color lightColor = tint_color(base, B_LIGHTEN_2_TINT); 1380 lightColor.alpha = 128; 1381 view->SetHighColor(lightColor); 1382 view->StrokeLine(rect.LeftBottom(), rect.RightBottom()); 1383 } 1384 1385 view->SetDrawingMode(B_OP_COPY); 1386 } 1387 1388 1389 void 1390 HaikuControlLook::DrawSliderTriangle(BView* view, BRect& rect, 1391 const BRect& updateRect, const rgb_color& base, uint32 flags, 1392 orientation orientation) 1393 { 1394 DrawSliderTriangle(view, rect, updateRect, base, base, flags, orientation); 1395 } 1396 1397 1398 void 1399 HaikuControlLook::DrawSliderTriangle(BView* view, BRect& rect, 1400 const BRect& updateRect, const rgb_color& base, const rgb_color& fill, 1401 uint32 flags, orientation orientation) 1402 { 1403 if (!ShouldDraw(view, rect, updateRect)) 1404 return; 1405 1406 // figure out frame color 1407 rgb_color frameLightColor; 1408 rgb_color frameShadowColor; 1409 rgb_color shadowColor = (rgb_color){ 0, 0, 0, 60 }; 1410 1411 float topTint = 0.49; 1412 float middleTint1 = 0.62; 1413 float middleTint2 = 0.76; 1414 float bottomTint = 0.90; 1415 1416 if ((flags & B_DISABLED) != 0) { 1417 topTint = (topTint + B_NO_TINT) / 2; 1418 middleTint1 = (middleTint1 + B_NO_TINT) / 2; 1419 middleTint2 = (middleTint2 + B_NO_TINT) / 2; 1420 bottomTint = (bottomTint + B_NO_TINT) / 2; 1421 } else if ((flags & B_HOVER) != 0) { 1422 topTint *= kHoverTintFactor; 1423 middleTint1 *= kHoverTintFactor; 1424 middleTint2 *= kHoverTintFactor; 1425 bottomTint *= kHoverTintFactor; 1426 } 1427 1428 if ((flags & B_FOCUSED) != 0) { 1429 // focused 1430 frameLightColor = ui_color(B_KEYBOARD_NAVIGATION_COLOR); 1431 frameShadowColor = frameLightColor; 1432 } else { 1433 // figure out the tints to be used 1434 float frameLightTint; 1435 float frameShadowTint; 1436 1437 if ((flags & B_DISABLED) != 0) { 1438 frameLightTint = 1.30; 1439 frameShadowTint = 1.35; 1440 shadowColor.alpha = 30; 1441 } else { 1442 frameLightTint = 1.6; 1443 frameShadowTint = 1.65; 1444 } 1445 1446 frameLightColor = tint_color(base, frameLightTint); 1447 frameShadowColor = tint_color(base, frameShadowTint); 1448 } 1449 1450 // make room for the shadow 1451 rect.right--; 1452 rect.bottom--; 1453 1454 uint32 viewFlags = view->Flags(); 1455 view->SetFlags(viewFlags | B_SUBPIXEL_PRECISE); 1456 view->SetLineMode(B_ROUND_CAP, B_ROUND_JOIN); 1457 1458 float centerh = (rect.left + rect.right) / 2; 1459 float centerv = (rect.top + rect.bottom) / 2; 1460 1461 BShape shape; 1462 if (orientation == B_HORIZONTAL) { 1463 shape.MoveTo(BPoint(rect.left + 0.5, rect.bottom + 0.5)); 1464 shape.LineTo(BPoint(rect.right + 0.5, rect.bottom + 0.5)); 1465 shape.LineTo(BPoint(rect.right + 0.5, rect.bottom - 1 + 0.5)); 1466 shape.LineTo(BPoint(centerh + 0.5, rect.top + 0.5)); 1467 shape.LineTo(BPoint(rect.left + 0.5, rect.bottom - 1 + 0.5)); 1468 } else { 1469 shape.MoveTo(BPoint(rect.right + 0.5, rect.top + 0.5)); 1470 shape.LineTo(BPoint(rect.right + 0.5, rect.bottom + 0.5)); 1471 shape.LineTo(BPoint(rect.right - 1 + 0.5, rect.bottom + 0.5)); 1472 shape.LineTo(BPoint(rect.left + 0.5, centerv + 0.5)); 1473 shape.LineTo(BPoint(rect.right - 1 + 0.5, rect.top + 0.5)); 1474 } 1475 shape.Close(); 1476 1477 view->MovePenTo(BPoint(1, 1)); 1478 1479 view->SetDrawingMode(B_OP_ALPHA); 1480 view->SetHighColor(shadowColor); 1481 view->StrokeShape(&shape); 1482 1483 view->MovePenTo(B_ORIGIN); 1484 1485 view->SetDrawingMode(B_OP_COPY); 1486 view->SetHighColor(frameLightColor); 1487 view->StrokeShape(&shape); 1488 1489 rect.InsetBy(1, 1); 1490 shape.Clear(); 1491 if (orientation == B_HORIZONTAL) { 1492 shape.MoveTo(BPoint(rect.left, rect.bottom + 1)); 1493 shape.LineTo(BPoint(rect.right + 1, rect.bottom + 1)); 1494 shape.LineTo(BPoint(centerh + 0.5, rect.top)); 1495 } else { 1496 shape.MoveTo(BPoint(rect.right + 1, rect.top)); 1497 shape.LineTo(BPoint(rect.right + 1, rect.bottom + 1)); 1498 shape.LineTo(BPoint(rect.left, centerv + 0.5)); 1499 } 1500 shape.Close(); 1501 1502 BGradientLinear gradient; 1503 if ((flags & B_DISABLED) != 0) { 1504 _MakeGradient(gradient, rect, fill, topTint, bottomTint); 1505 } else { 1506 _MakeGlossyGradient(gradient, rect, fill, topTint, middleTint1, 1507 middleTint2, bottomTint); 1508 } 1509 1510 view->FillShape(&shape, gradient); 1511 1512 view->SetFlags(viewFlags); 1513 } 1514 1515 1516 void 1517 HaikuControlLook::DrawSliderHashMarks(BView* view, BRect& rect, 1518 const BRect& updateRect, const rgb_color& base, int32 count, 1519 hash_mark_location location, uint32 flags, orientation orientation) 1520 { 1521 if (!ShouldDraw(view, rect, updateRect)) 1522 return; 1523 1524 rgb_color lightColor; 1525 rgb_color darkColor; 1526 1527 if ((flags & B_DISABLED) != 0) { 1528 lightColor = tint_color(base, 0.9); 1529 darkColor = tint_color(base, 1.07); 1530 } else { 1531 lightColor = tint_color(base, 0.8); 1532 darkColor = tint_color(base, 1.14); 1533 } 1534 1535 int32 hashMarkCount = std::max(count, (int32)2); 1536 // draw at least two hashmarks at min/max if 1537 // fHashMarks != B_HASH_MARKS_NONE 1538 float factor; 1539 float startPos; 1540 if (orientation == B_HORIZONTAL) { 1541 factor = (rect.Width() - 2) / (hashMarkCount - 1); 1542 startPos = rect.left + 1; 1543 } else { 1544 factor = (rect.Height() - 2) / (hashMarkCount - 1); 1545 startPos = rect.top + 1; 1546 } 1547 1548 if (location & B_HASH_MARKS_TOP) { 1549 view->BeginLineArray(hashMarkCount * 2); 1550 1551 if (orientation == B_HORIZONTAL) { 1552 float pos = startPos; 1553 for (int32 i = 0; i < hashMarkCount; i++) { 1554 view->AddLine(BPoint(pos, rect.top), 1555 BPoint(pos, rect.top + 4), darkColor); 1556 view->AddLine(BPoint(pos + 1, rect.top), 1557 BPoint(pos + 1, rect.top + 4), lightColor); 1558 1559 pos += factor; 1560 } 1561 } else { 1562 float pos = startPos; 1563 for (int32 i = 0; i < hashMarkCount; i++) { 1564 view->AddLine(BPoint(rect.left, pos), 1565 BPoint(rect.left + 4, pos), darkColor); 1566 view->AddLine(BPoint(rect.left, pos + 1), 1567 BPoint(rect.left + 4, pos + 1), lightColor); 1568 1569 pos += factor; 1570 } 1571 } 1572 1573 view->EndLineArray(); 1574 } 1575 1576 if ((location & B_HASH_MARKS_BOTTOM) != 0) { 1577 view->BeginLineArray(hashMarkCount * 2); 1578 1579 if (orientation == B_HORIZONTAL) { 1580 float pos = startPos; 1581 for (int32 i = 0; i < hashMarkCount; i++) { 1582 view->AddLine(BPoint(pos, rect.bottom - 4), 1583 BPoint(pos, rect.bottom), darkColor); 1584 view->AddLine(BPoint(pos + 1, rect.bottom - 4), 1585 BPoint(pos + 1, rect.bottom), lightColor); 1586 1587 pos += factor; 1588 } 1589 } else { 1590 float pos = startPos; 1591 for (int32 i = 0; i < hashMarkCount; i++) { 1592 view->AddLine(BPoint(rect.right - 4, pos), 1593 BPoint(rect.right, pos), darkColor); 1594 view->AddLine(BPoint(rect.right - 4, pos + 1), 1595 BPoint(rect.right, pos + 1), lightColor); 1596 1597 pos += factor; 1598 } 1599 } 1600 1601 view->EndLineArray(); 1602 } 1603 } 1604 1605 1606 void 1607 HaikuControlLook::DrawTabFrame(BView* view, BRect& rect, 1608 const BRect& updateRect, const rgb_color& base, uint32 flags, 1609 uint32 borders, border_style borderStyle, uint32 side) 1610 { 1611 if (!ShouldDraw(view, rect, updateRect)) 1612 return; 1613 1614 if (side == BTabView::kTopSide || side == BTabView::kBottomSide) { 1615 // draw an inactive tab frame behind all tabs 1616 borders = B_TOP_BORDER | B_BOTTOM_BORDER; 1617 if (borderStyle != B_NO_BORDER) 1618 borders |= B_LEFT_BORDER | B_RIGHT_BORDER; 1619 1620 // DrawInactiveTab draws 2px border 1621 // draw tab frame wider to align B_PLAIN_BORDER with it 1622 if (borderStyle == B_PLAIN_BORDER) 1623 rect.InsetBy(-1, 0); 1624 } else if (side == BTabView::kLeftSide || side == BTabView::kRightSide) { 1625 // draw an inactive tab frame behind all tabs 1626 borders = B_LEFT_BORDER | B_RIGHT_BORDER; 1627 if (borderStyle != B_NO_BORDER) 1628 borders |= B_TOP_BORDER | B_BOTTOM_BORDER; 1629 1630 // DrawInactiveTab draws 2px border 1631 // draw tab frame wider to align B_PLAIN_BORDER with it 1632 if (borderStyle == B_PLAIN_BORDER) 1633 rect.InsetBy(0, -1); 1634 } 1635 1636 DrawInactiveTab(view, rect, rect, base, 0, borders, side); 1637 } 1638 1639 1640 void 1641 HaikuControlLook::DrawActiveTab(BView* view, BRect& rect, 1642 const BRect& updateRect, const rgb_color& base, uint32 flags, 1643 uint32 borders, uint32 side, int32, int32, int32, int32) 1644 { 1645 if (!ShouldDraw(view, rect, updateRect)) 1646 return; 1647 1648 // Snap the rectangle to pixels to avoid rounding errors. 1649 rect.left = floorf(rect.left); 1650 rect.right = floorf(rect.right); 1651 rect.top = floorf(rect.top); 1652 rect.bottom = floorf(rect.bottom); 1653 1654 // save the clipping constraints of the view 1655 view->PushState(); 1656 1657 // set clipping constraints to rect 1658 view->ClipToRect(rect); 1659 1660 rgb_color edgeShadowColor; 1661 rgb_color edgeLightColor; 1662 rgb_color frameShadowColor; 1663 rgb_color frameLightColor; 1664 rgb_color bevelShadowColor; 1665 rgb_color bevelLightColor; 1666 BGradientLinear fillGradient; 1667 fillGradient.SetStart(rect.LeftTop() + BPoint(3, 3)); 1668 fillGradient.SetEnd(rect.LeftBottom() + BPoint(3, -3)); 1669 1670 if ((flags & B_DISABLED) != 0) { 1671 edgeLightColor = base; 1672 edgeShadowColor = base; 1673 frameLightColor = tint_color(base, 1.25); 1674 frameShadowColor = tint_color(base, 1.30); 1675 bevelLightColor = tint_color(base, 0.8); 1676 bevelShadowColor = tint_color(base, 1.07); 1677 fillGradient.AddColor(tint_color(base, 0.85), 0); 1678 fillGradient.AddColor(base, 255); 1679 } else { 1680 edgeLightColor = tint_color(base, 0.80); 1681 edgeShadowColor = tint_color(base, 1.03); 1682 frameLightColor = tint_color(base, 1.30); 1683 frameShadowColor = tint_color(base, 1.30); 1684 bevelLightColor = tint_color(base, 0.6); 1685 bevelShadowColor = tint_color(base, 1.07); 1686 fillGradient.AddColor(tint_color(base, 0.75), 0); 1687 fillGradient.AddColor(tint_color(base, 1.03), 255); 1688 } 1689 1690 static const float kRoundCornerRadius = 4.0f; 1691 1692 // left top corner dimensions 1693 BRect leftTopCorner(rect); 1694 leftTopCorner.right = floorf(leftTopCorner.left + kRoundCornerRadius); 1695 leftTopCorner.bottom = floorf(rect.top + kRoundCornerRadius); 1696 1697 // right top corner dimensions 1698 BRect rightTopCorner(rect); 1699 rightTopCorner.left = floorf(rightTopCorner.right - kRoundCornerRadius); 1700 rightTopCorner.bottom = floorf(rect.top + kRoundCornerRadius); 1701 1702 // left bottom corner dimensions 1703 BRect leftBottomCorner(rect); 1704 leftBottomCorner.right = floorf(leftBottomCorner.left + kRoundCornerRadius); 1705 leftBottomCorner.top = floorf(rect.bottom - kRoundCornerRadius); 1706 1707 // right bottom corner dimensions 1708 BRect rightBottomCorner(rect); 1709 rightBottomCorner.left = floorf(rightBottomCorner.right 1710 - kRoundCornerRadius); 1711 rightBottomCorner.top = floorf(rect.bottom - kRoundCornerRadius); 1712 1713 BRect roundCorner[2]; 1714 1715 switch (side) { 1716 case B_TOP_BORDER: 1717 roundCorner[0] = leftTopCorner; 1718 roundCorner[1] = rightTopCorner; 1719 1720 // draw the left top corner 1721 _DrawRoundCornerLeftTop(view, leftTopCorner, updateRect, base, 1722 edgeShadowColor, frameLightColor, bevelLightColor, 1723 fillGradient); 1724 // draw the right top corner 1725 _DrawRoundCornerRightTop(view, rightTopCorner, updateRect, base, 1726 edgeShadowColor, edgeLightColor, frameLightColor, 1727 frameShadowColor, bevelLightColor, bevelShadowColor, 1728 fillGradient); 1729 break; 1730 case B_BOTTOM_BORDER: 1731 roundCorner[0] = leftBottomCorner; 1732 roundCorner[1] = rightBottomCorner; 1733 1734 // draw the left bottom corner 1735 _DrawRoundCornerLeftBottom(view, leftBottomCorner, updateRect, base, 1736 edgeShadowColor, edgeLightColor, frameLightColor, 1737 frameShadowColor, bevelLightColor, bevelShadowColor, 1738 fillGradient); 1739 // draw the right bottom corner 1740 _DrawRoundCornerRightBottom(view, rightBottomCorner, updateRect, 1741 base, edgeLightColor, frameShadowColor, bevelShadowColor, 1742 fillGradient); 1743 break; 1744 case B_LEFT_BORDER: 1745 roundCorner[0] = leftTopCorner; 1746 roundCorner[1] = leftBottomCorner; 1747 1748 // draw the left top corner 1749 _DrawRoundCornerLeftTop(view, leftTopCorner, updateRect, base, 1750 edgeShadowColor, frameLightColor, bevelLightColor, 1751 fillGradient); 1752 // draw the left bottom corner 1753 _DrawRoundCornerLeftBottom(view, leftBottomCorner, updateRect, base, 1754 edgeShadowColor, edgeLightColor, frameLightColor, 1755 frameShadowColor, bevelLightColor, bevelShadowColor, 1756 fillGradient); 1757 break; 1758 case B_RIGHT_BORDER: 1759 roundCorner[0] = rightTopCorner; 1760 roundCorner[1] = rightBottomCorner; 1761 1762 // draw the right top corner 1763 _DrawRoundCornerRightTop(view, rightTopCorner, updateRect, base, 1764 edgeShadowColor, edgeLightColor, frameLightColor, 1765 frameShadowColor, bevelLightColor, bevelShadowColor, 1766 fillGradient); 1767 // draw the right bottom corner 1768 _DrawRoundCornerRightBottom(view, rightBottomCorner, updateRect, 1769 base, edgeLightColor, frameShadowColor, bevelShadowColor, 1770 fillGradient); 1771 break; 1772 } 1773 1774 // clip out the corners 1775 view->ClipToInverseRect(roundCorner[0]); 1776 view->ClipToInverseRect(roundCorner[1]); 1777 1778 uint32 bordersToDraw = 0; 1779 switch (side) { 1780 case B_TOP_BORDER: 1781 bordersToDraw = (B_LEFT_BORDER | B_TOP_BORDER | B_RIGHT_BORDER); 1782 break; 1783 case B_BOTTOM_BORDER: 1784 bordersToDraw = (B_LEFT_BORDER | B_BOTTOM_BORDER | B_RIGHT_BORDER); 1785 break; 1786 case B_LEFT_BORDER: 1787 bordersToDraw = (B_LEFT_BORDER | B_BOTTOM_BORDER | B_TOP_BORDER); 1788 break; 1789 case B_RIGHT_BORDER: 1790 bordersToDraw = (B_RIGHT_BORDER | B_BOTTOM_BORDER | B_TOP_BORDER); 1791 break; 1792 } 1793 1794 // draw the rest of frame and fill 1795 _DrawFrame(view, rect, edgeShadowColor, edgeShadowColor, edgeLightColor, 1796 edgeLightColor, borders); 1797 if (side == B_TOP_BORDER || side == B_BOTTOM_BORDER) { 1798 if ((borders & B_LEFT_BORDER) == 0) 1799 rect.left++; 1800 if ((borders & B_RIGHT_BORDER) == 0) 1801 rect.right--; 1802 } else if (side == B_LEFT_BORDER || side == B_RIGHT_BORDER) { 1803 if ((borders & B_TOP_BORDER) == 0) 1804 rect.top++; 1805 if ((borders & B_BOTTOM_BORDER) == 0) 1806 rect.bottom--; 1807 } 1808 1809 _DrawFrame(view, rect, frameLightColor, frameLightColor, frameShadowColor, 1810 frameShadowColor, bordersToDraw); 1811 1812 _DrawFrame(view, rect, bevelLightColor, bevelLightColor, bevelShadowColor, 1813 bevelShadowColor); 1814 1815 view->FillRect(rect, fillGradient); 1816 1817 // restore the clipping constraints of the view 1818 view->PopState(); 1819 } 1820 1821 1822 void 1823 HaikuControlLook::DrawInactiveTab(BView* view, BRect& rect, 1824 const BRect& updateRect, const rgb_color& base, uint32 flags, 1825 uint32 borders, uint32 side, int32, int32, int32, int32) 1826 { 1827 if (!ShouldDraw(view, rect, updateRect)) 1828 return; 1829 1830 rgb_color edgeShadowColor; 1831 rgb_color edgeLightColor; 1832 rgb_color frameShadowColor; 1833 rgb_color frameLightColor; 1834 rgb_color bevelShadowColor; 1835 rgb_color bevelLightColor; 1836 BGradientLinear fillGradient; 1837 fillGradient.SetStart(rect.LeftTop() + BPoint(3, 3)); 1838 fillGradient.SetEnd(rect.LeftBottom() + BPoint(3, -3)); 1839 1840 if ((flags & B_DISABLED) != 0) { 1841 edgeLightColor = base; 1842 edgeShadowColor = base; 1843 frameLightColor = tint_color(base, 1.25); 1844 frameShadowColor = tint_color(base, 1.30); 1845 bevelLightColor = tint_color(base, 0.8); 1846 bevelShadowColor = tint_color(base, 1.07); 1847 fillGradient.AddColor(tint_color(base, 0.85), 0); 1848 fillGradient.AddColor(base, 255); 1849 } else { 1850 edgeLightColor = tint_color(base, 0.80); 1851 edgeShadowColor = tint_color(base, 1.03); 1852 frameLightColor = tint_color(base, 1.30); 1853 frameShadowColor = tint_color(base, 1.30); 1854 bevelLightColor = tint_color(base, 1.10); 1855 bevelShadowColor = tint_color(base, 1.17); 1856 fillGradient.AddColor(tint_color(base, 1.12), 0); 1857 fillGradient.AddColor(tint_color(base, 1.08), 255); 1858 } 1859 1860 BRect background = rect; 1861 bool isVertical; 1862 switch (side) { 1863 default: 1864 case BTabView::kTopSide: 1865 rect.top += 4; 1866 background.bottom = rect.top; 1867 isVertical = false; 1868 break; 1869 1870 case BTabView::kBottomSide: 1871 rect.bottom -= 4; 1872 background.top = rect.bottom; 1873 isVertical = false; 1874 break; 1875 1876 case BTabView::kLeftSide: 1877 rect.left += 4; 1878 background.right = rect.left; 1879 isVertical = true; 1880 break; 1881 1882 case BTabView::kRightSide: 1883 rect.right -= 4; 1884 background.left = rect.right; 1885 isVertical = true; 1886 break; 1887 } 1888 1889 // active tabs stand out at the top, but this is an inactive tab 1890 view->SetHighColor(base); 1891 view->FillRect(background); 1892 1893 // frame and fill 1894 // Note that _DrawFrame also insets the rect, so each of the calls here 1895 // operate on a smaller rect than the previous ones 1896 _DrawFrame(view, rect, edgeShadowColor, edgeShadowColor, edgeLightColor, 1897 edgeLightColor, borders); 1898 1899 _DrawFrame(view, rect, frameLightColor, frameLightColor, frameShadowColor, 1900 frameShadowColor, borders); 1901 1902 if (rect.IsValid()) { 1903 if (isVertical) { 1904 _DrawFrame(view, rect, bevelShadowColor, bevelShadowColor, 1905 bevelLightColor, bevelLightColor, B_TOP_BORDER & ~borders); 1906 } else { 1907 _DrawFrame(view, rect, bevelShadowColor, bevelShadowColor, 1908 bevelLightColor, bevelLightColor, B_LEFT_BORDER & ~borders); 1909 } 1910 } else { 1911 if (isVertical) { 1912 if ((B_LEFT_BORDER & ~borders) != 0) 1913 rect.left++; 1914 } else { 1915 if ((B_TOP_BORDER & ~borders) != 0) 1916 rect.top++; 1917 } 1918 } 1919 1920 view->FillRect(rect, fillGradient); 1921 } 1922 1923 1924 void 1925 HaikuControlLook::DrawSplitter(BView* view, BRect& rect, const BRect& updateRect, 1926 const rgb_color& base, orientation orientation, uint32 flags, 1927 uint32 borders) 1928 { 1929 if (!ShouldDraw(view, rect, updateRect)) 1930 return; 1931 1932 rgb_color background; 1933 if ((flags & (B_CLICKED | B_ACTIVATED)) != 0) 1934 background = tint_color(base, B_DARKEN_1_TINT); 1935 else 1936 background = base; 1937 1938 rgb_color light = tint_color(background, 0.6); 1939 rgb_color shadow = tint_color(background, 1.21); 1940 1941 // frame 1942 if (borders != 0 && rect.Width() > 3 && rect.Height() > 3) 1943 DrawRaisedBorder(view, rect, updateRect, background, flags, borders); 1944 1945 // dots and rest of background 1946 if (orientation == B_HORIZONTAL) { 1947 if (rect.Width() > 2) { 1948 // background on left/right 1949 BRegion region(rect); 1950 rect.left = floorf((rect.left + rect.right) / 2.0 - 0.5); 1951 rect.right = rect.left + 1; 1952 region.Exclude(rect); 1953 view->SetHighColor(background); 1954 view->FillRegion(®ion); 1955 } 1956 1957 BPoint dot = rect.LeftTop(); 1958 BPoint stop = rect.LeftBottom(); 1959 int32 num = 1; 1960 while (dot.y <= stop.y) { 1961 rgb_color col1; 1962 rgb_color col2; 1963 switch (num) { 1964 case 1: 1965 col1 = background; 1966 col2 = background; 1967 break; 1968 case 2: 1969 col1 = shadow; 1970 col2 = background; 1971 break; 1972 case 3: 1973 default: 1974 col1 = background; 1975 col2 = light; 1976 num = 0; 1977 break; 1978 } 1979 view->SetHighColor(col1); 1980 view->StrokeLine(dot, dot, B_SOLID_HIGH); 1981 view->SetHighColor(col2); 1982 dot.x++; 1983 view->StrokeLine(dot, dot, B_SOLID_HIGH); 1984 dot.x -= 1.0; 1985 // next pixel 1986 num++; 1987 dot.y++; 1988 } 1989 } else { 1990 if (rect.Height() > 2) { 1991 // background on left/right 1992 BRegion region(rect); 1993 rect.top = floorf((rect.top + rect.bottom) / 2.0 - 0.5); 1994 rect.bottom = rect.top + 1; 1995 region.Exclude(rect); 1996 view->SetHighColor(background); 1997 view->FillRegion(®ion); 1998 } 1999 2000 BPoint dot = rect.LeftTop(); 2001 BPoint stop = rect.RightTop(); 2002 int32 num = 1; 2003 while (dot.x <= stop.x) { 2004 rgb_color col1; 2005 rgb_color col2; 2006 switch (num) { 2007 case 1: 2008 col1 = background; 2009 col2 = background; 2010 break; 2011 case 2: 2012 col1 = shadow; 2013 col2 = background; 2014 break; 2015 case 3: 2016 default: 2017 col1 = background; 2018 col2 = light; 2019 num = 0; 2020 break; 2021 } 2022 view->SetHighColor(col1); 2023 view->StrokeLine(dot, dot, B_SOLID_HIGH); 2024 view->SetHighColor(col2); 2025 dot.y++; 2026 view->StrokeLine(dot, dot, B_SOLID_HIGH); 2027 dot.y -= 1.0; 2028 // next pixel 2029 num++; 2030 dot.x++; 2031 } 2032 } 2033 } 2034 2035 2036 // #pragma mark - 2037 2038 2039 void 2040 HaikuControlLook::DrawBorder(BView* view, BRect& rect, const BRect& updateRect, 2041 const rgb_color& base, border_style borderStyle, uint32 flags, 2042 uint32 borders) 2043 { 2044 if (borderStyle == B_NO_BORDER) 2045 return; 2046 2047 rgb_color scrollbarFrameColor = tint_color(base, B_DARKEN_2_TINT); 2048 if (base.red + base.green + base.blue <= 128 * 3) { 2049 scrollbarFrameColor = tint_color(base, B_LIGHTEN_1_TINT); 2050 } 2051 2052 if ((flags & B_FOCUSED) != 0) 2053 scrollbarFrameColor = ui_color(B_KEYBOARD_NAVIGATION_COLOR); 2054 2055 if (borderStyle == B_FANCY_BORDER) 2056 _DrawOuterResessedFrame(view, rect, base, 1.0, 1.0, flags, borders); 2057 2058 _DrawFrame(view, rect, scrollbarFrameColor, scrollbarFrameColor, 2059 scrollbarFrameColor, scrollbarFrameColor, borders); 2060 } 2061 2062 2063 void 2064 HaikuControlLook::DrawRaisedBorder(BView* view, BRect& rect, 2065 const BRect& updateRect, const rgb_color& base, uint32 flags, 2066 uint32 borders) 2067 { 2068 rgb_color lightColor; 2069 rgb_color shadowColor; 2070 2071 if ((flags & B_DISABLED) != 0) { 2072 lightColor = base; 2073 shadowColor = base; 2074 } else { 2075 lightColor = tint_color(base, 0.85); 2076 shadowColor = tint_color(base, 1.07); 2077 } 2078 2079 _DrawFrame(view, rect, lightColor, lightColor, shadowColor, shadowColor, 2080 borders); 2081 } 2082 2083 2084 void 2085 HaikuControlLook::DrawTextControlBorder(BView* view, BRect& rect, 2086 const BRect& updateRect, const rgb_color& base, uint32 flags, 2087 uint32 borders) 2088 { 2089 if (!ShouldDraw(view, rect, updateRect)) 2090 return; 2091 2092 rgb_color dark1BorderColor; 2093 rgb_color dark2BorderColor; 2094 rgb_color navigationColor = ui_color(B_KEYBOARD_NAVIGATION_COLOR); 2095 rgb_color invalidColor = ui_color(B_FAILURE_COLOR); 2096 2097 if ((flags & B_DISABLED) != 0) { 2098 _DrawOuterResessedFrame(view, rect, base, 0.0, 1.0, flags, borders); 2099 2100 if ((flags & B_BLEND_FRAME) != 0) 2101 dark1BorderColor = (rgb_color){ 0, 0, 0, 40 }; 2102 else 2103 dark1BorderColor = tint_color(base, 1.15); 2104 dark2BorderColor = dark1BorderColor; 2105 } else if ((flags & B_CLICKED) != 0) { 2106 dark1BorderColor = tint_color(base, 1.50); 2107 dark2BorderColor = tint_color(base, 1.49); 2108 2109 // BCheckBox uses this to indicate the clicked state... 2110 _DrawFrame(view, rect, 2111 dark1BorderColor, dark1BorderColor, 2112 dark2BorderColor, dark2BorderColor); 2113 2114 dark2BorderColor = dark1BorderColor; 2115 } else { 2116 _DrawOuterResessedFrame(view, rect, base, 0.6, 1.0, flags, borders); 2117 2118 if ((flags & B_BLEND_FRAME) != 0) { 2119 dark1BorderColor = (rgb_color){ 0, 0, 0, 102 }; 2120 dark2BorderColor = (rgb_color){ 0, 0, 0, 97 }; 2121 } else { 2122 dark1BorderColor = tint_color(base, 1.40); 2123 dark2BorderColor = tint_color(base, 1.38); 2124 } 2125 } 2126 2127 if ((flags & B_DISABLED) == 0 && (flags & B_FOCUSED) != 0) { 2128 dark1BorderColor = navigationColor; 2129 dark2BorderColor = navigationColor; 2130 } 2131 2132 if ((flags & B_DISABLED) == 0 && (flags & B_INVALID) != 0) { 2133 dark1BorderColor = invalidColor; 2134 dark2BorderColor = invalidColor; 2135 } 2136 2137 if ((flags & B_BLEND_FRAME) != 0) { 2138 drawing_mode oldMode = view->DrawingMode(); 2139 view->SetDrawingMode(B_OP_ALPHA); 2140 2141 _DrawFrame(view, rect, 2142 dark1BorderColor, dark1BorderColor, 2143 dark2BorderColor, dark2BorderColor, borders); 2144 2145 view->SetDrawingMode(oldMode); 2146 } else { 2147 _DrawFrame(view, rect, 2148 dark1BorderColor, dark1BorderColor, 2149 dark2BorderColor, dark2BorderColor, borders); 2150 } 2151 } 2152 2153 2154 void 2155 HaikuControlLook::DrawGroupFrame(BView* view, BRect& rect, const BRect& updateRect, 2156 const rgb_color& base, uint32 borders) 2157 { 2158 rgb_color frameColor = tint_color(base, 1.30); 2159 rgb_color bevelLight = tint_color(base, 0.8); 2160 rgb_color bevelShadow = tint_color(base, 1.03); 2161 2162 _DrawFrame(view, rect, bevelShadow, bevelShadow, bevelLight, bevelLight, 2163 borders); 2164 2165 _DrawFrame(view, rect, frameColor, frameColor, frameColor, frameColor, 2166 borders); 2167 2168 _DrawFrame(view, rect, bevelLight, bevelLight, bevelShadow, bevelShadow, 2169 borders); 2170 } 2171 2172 2173 void 2174 HaikuControlLook::DrawLabel(BView* view, const char* label, BRect rect, 2175 const BRect& updateRect, const rgb_color& base, uint32 flags, 2176 const rgb_color* textColor) 2177 { 2178 DrawLabel(view, label, NULL, rect, updateRect, base, flags, 2179 DefaultLabelAlignment(), textColor); 2180 } 2181 2182 2183 void 2184 HaikuControlLook::DrawLabel(BView* view, const char* label, BRect rect, 2185 const BRect& updateRect, const rgb_color& base, uint32 flags, 2186 const BAlignment& alignment, const rgb_color* textColor) 2187 { 2188 DrawLabel(view, label, NULL, rect, updateRect, base, flags, alignment, 2189 textColor); 2190 } 2191 2192 2193 void 2194 HaikuControlLook::DrawLabel(BView* view, const char* label, const rgb_color& base, 2195 uint32 flags, const BPoint& where, const rgb_color* textColor) 2196 { 2197 // setup the text color 2198 2199 BWindow* window = view->Window(); 2200 bool isDesktop = window 2201 && window->Feel() == kDesktopWindowFeel 2202 && window->Look() == kDesktopWindowLook 2203 && view->Parent() 2204 && view->Parent()->Parent() == NULL 2205 && (flags & B_IGNORE_OUTLINE) == 0; 2206 2207 rgb_color low; 2208 rgb_color color; 2209 rgb_color glowColor; 2210 2211 if (textColor != NULL) 2212 glowColor = *textColor; 2213 else if ((flags & B_IS_CONTROL) != 0) 2214 glowColor = ui_color(B_CONTROL_TEXT_COLOR); 2215 else 2216 glowColor = ui_color(B_PANEL_TEXT_COLOR); 2217 2218 color = glowColor; 2219 2220 if (isDesktop) 2221 low = view->Parent()->ViewColor(); 2222 else 2223 low = base; 2224 2225 if ((flags & B_DISABLED) != 0) { 2226 color.red = (uint8)(((int32)low.red + color.red + 1) / 2); 2227 color.green = (uint8)(((int32)low.green + color.green + 1) / 2); 2228 color.blue = (uint8)(((int32)low.blue + color.blue + 1) / 2); 2229 } 2230 2231 drawing_mode oldMode = view->DrawingMode(); 2232 2233 if (isDesktop) { 2234 // enforce proper use of desktop label colors 2235 if (low.IsDark()) { 2236 if (textColor == NULL) 2237 color = make_color(255, 255, 255); 2238 2239 glowColor = make_color(0, 0, 0); 2240 } else { 2241 if (textColor == NULL) 2242 color = make_color(0, 0, 0); 2243 2244 glowColor = make_color(255, 255, 255); 2245 } 2246 2247 // drawing occurs on the desktop 2248 if (fCachedWorkspace != current_workspace()) { 2249 int8 indice = 0; 2250 int32 mask; 2251 bool tmpOutline; 2252 while (fBackgroundInfo.FindInt32("be:bgndimginfoworkspaces", 2253 indice, &mask) == B_OK 2254 && fBackgroundInfo.FindBool("be:bgndimginfoerasetext", 2255 indice, &tmpOutline) == B_OK) { 2256 2257 if (((1 << current_workspace()) & mask) != 0) { 2258 fCachedOutline = tmpOutline; 2259 fCachedWorkspace = current_workspace(); 2260 break; 2261 } 2262 indice++; 2263 } 2264 } 2265 2266 if (fCachedOutline) { 2267 BFont font; 2268 view->GetFont(&font); 2269 2270 view->SetDrawingMode(B_OP_ALPHA); 2271 view->SetBlendingMode(B_CONSTANT_ALPHA, B_ALPHA_OVERLAY); 2272 // Draw glow or outline 2273 if (glowColor.IsLight()) { 2274 font.SetFalseBoldWidth(2.0); 2275 view->SetFont(&font, B_FONT_FALSE_BOLD_WIDTH); 2276 2277 glowColor.alpha = 30; 2278 view->SetHighColor(glowColor); 2279 view->DrawString(label, where); 2280 2281 font.SetFalseBoldWidth(1.0); 2282 view->SetFont(&font, B_FONT_FALSE_BOLD_WIDTH); 2283 2284 glowColor.alpha = 65; 2285 view->SetHighColor(glowColor); 2286 view->DrawString(label, where); 2287 2288 font.SetFalseBoldWidth(0.0); 2289 view->SetFont(&font, B_FONT_FALSE_BOLD_WIDTH); 2290 } else { 2291 font.SetFalseBoldWidth(1.0); 2292 view->SetFont(&font, B_FONT_FALSE_BOLD_WIDTH); 2293 2294 glowColor.alpha = 30; 2295 view->SetHighColor(glowColor); 2296 view->DrawString(label, where); 2297 2298 font.SetFalseBoldWidth(0.0); 2299 view->SetFont(&font, B_FONT_FALSE_BOLD_WIDTH); 2300 2301 glowColor.alpha = 200; 2302 view->SetHighColor(glowColor); 2303 view->DrawString(label, BPoint(where.x + 1, where.y + 1)); 2304 } 2305 } 2306 } 2307 2308 view->SetHighColor(color); 2309 view->SetDrawingMode(B_OP_OVER); 2310 view->DrawString(label, where); 2311 view->SetDrawingMode(oldMode); 2312 } 2313 2314 2315 void 2316 HaikuControlLook::DrawLabel(BView* view, const char* label, const BBitmap* icon, 2317 BRect rect, const BRect& updateRect, const rgb_color& base, uint32 flags, 2318 const BAlignment& alignment, const rgb_color* textColor) 2319 { 2320 if (!ShouldDraw(view, rect, updateRect)) 2321 return; 2322 2323 if (label == NULL && icon == NULL) 2324 return; 2325 2326 if (label == NULL) { 2327 // icon only 2328 BRect alignedRect = BLayoutUtils::AlignInFrame(rect, 2329 icon->Bounds().Size(), alignment); 2330 drawing_mode oldMode = view->DrawingMode(); 2331 view->SetDrawingMode(B_OP_OVER); 2332 view->DrawBitmap(icon, alignedRect.LeftTop()); 2333 view->SetDrawingMode(oldMode); 2334 return; 2335 } 2336 2337 // label, possibly with icon 2338 float availableWidth = rect.Width() + 1; 2339 float width = 0; 2340 float textOffset = 0; 2341 float height = 0; 2342 2343 if (icon != NULL) { 2344 width = icon->Bounds().Width() + DefaultLabelSpacing() + 1; 2345 height = icon->Bounds().Height() + 1; 2346 textOffset = width; 2347 availableWidth -= textOffset; 2348 } 2349 2350 // truncate the label if necessary and get the width and height 2351 BString truncatedLabel(label); 2352 2353 BFont font; 2354 view->GetFont(&font); 2355 2356 font.TruncateString(&truncatedLabel, B_TRUNCATE_END, availableWidth); 2357 width += ceilf(font.StringWidth(truncatedLabel.String())); 2358 2359 font_height fontHeight; 2360 font.GetHeight(&fontHeight); 2361 float textHeight = ceilf(fontHeight.ascent) + ceilf(fontHeight.descent); 2362 height = std::max(height, textHeight); 2363 2364 // handle alignment 2365 BRect alignedRect(BLayoutUtils::AlignOnRect(rect, 2366 BSize(width - 1, height - 1), alignment)); 2367 2368 if (icon != NULL) { 2369 BPoint location(alignedRect.LeftTop()); 2370 if (icon->Bounds().Height() + 1 < height) 2371 location.y += ceilf((height - icon->Bounds().Height() - 1) / 2); 2372 2373 drawing_mode oldMode = view->DrawingMode(); 2374 view->SetDrawingMode(B_OP_OVER); 2375 view->DrawBitmap(icon, location); 2376 view->SetDrawingMode(oldMode); 2377 } 2378 2379 BPoint location(alignedRect.left + textOffset, 2380 alignedRect.top + ceilf(fontHeight.ascent)); 2381 if (textHeight < height) 2382 location.y += ceilf((height - textHeight) / 2); 2383 2384 DrawLabel(view, truncatedLabel.String(), base, flags, location, textColor); 2385 } 2386 2387 2388 void 2389 HaikuControlLook::GetFrameInsets(frame_type frameType, uint32 flags, float& _left, 2390 float& _top, float& _right, float& _bottom) 2391 { 2392 // All frames have the same inset on each side. 2393 float inset = 0; 2394 2395 switch (frameType) { 2396 case B_BUTTON_FRAME: 2397 inset = (flags & B_DEFAULT_BUTTON) != 0 ? 5 : 2; 2398 break; 2399 case B_GROUP_FRAME: 2400 case B_MENU_FIELD_FRAME: 2401 inset = 3; 2402 break; 2403 case B_SCROLL_VIEW_FRAME: 2404 case B_TEXT_CONTROL_FRAME: 2405 inset = 2; 2406 break; 2407 } 2408 2409 inset = ceilf(inset * (be_plain_font->Size() / 12.0f)); 2410 2411 _left = inset; 2412 _top = inset; 2413 _right = inset; 2414 _bottom = inset; 2415 } 2416 2417 2418 void 2419 HaikuControlLook::GetBackgroundInsets(background_type backgroundType, 2420 uint32 flags, float& _left, float& _top, float& _right, float& _bottom) 2421 { 2422 // Most backgrounds have the same inset on each side. 2423 float inset = 0; 2424 2425 switch (backgroundType) { 2426 case B_BUTTON_BACKGROUND: 2427 case B_MENU_BACKGROUND: 2428 case B_MENU_BAR_BACKGROUND: 2429 case B_MENU_FIELD_BACKGROUND: 2430 case B_MENU_ITEM_BACKGROUND: 2431 inset = 1; 2432 break; 2433 case B_BUTTON_WITH_POP_UP_BACKGROUND: 2434 _left = 1; 2435 _top = 1; 2436 _right = 1 + ComposeSpacing(kButtonPopUpIndicatorWidth); 2437 _bottom = 1; 2438 return; 2439 case B_HORIZONTAL_SCROLL_BAR_BACKGROUND: 2440 _left = 2; 2441 _top = 0; 2442 _right = 1; 2443 _bottom = 0; 2444 return; 2445 case B_VERTICAL_SCROLL_BAR_BACKGROUND: 2446 _left = 0; 2447 _top = 2; 2448 _right = 0; 2449 _bottom = 1; 2450 return; 2451 } 2452 2453 _left = inset; 2454 _top = inset; 2455 _right = inset; 2456 _bottom = inset; 2457 } 2458 2459 2460 void 2461 HaikuControlLook::DrawButtonWithPopUpBackground(BView* view, BRect& rect, 2462 const BRect& updateRect, const rgb_color& base, uint32 flags, 2463 uint32 borders, orientation orientation) 2464 { 2465 _DrawButtonBackground(view, rect, updateRect, 0.0f, 0.0f, 0.0f, 0.0f, 2466 base, true, flags, borders, orientation); 2467 } 2468 2469 2470 void 2471 HaikuControlLook::DrawButtonWithPopUpBackground(BView* view, BRect& rect, 2472 const BRect& updateRect, float radius, const rgb_color& base, uint32 flags, 2473 uint32 borders, orientation orientation) 2474 { 2475 _DrawButtonBackground(view, rect, updateRect, radius, radius, radius, 2476 radius, base, true, flags, borders, orientation); 2477 } 2478 2479 2480 void 2481 HaikuControlLook::DrawButtonWithPopUpBackground(BView* view, BRect& rect, 2482 const BRect& updateRect, float leftTopRadius, float rightTopRadius, 2483 float leftBottomRadius, float rightBottomRadius, const rgb_color& base, 2484 uint32 flags, uint32 borders, orientation orientation) 2485 { 2486 _DrawButtonBackground(view, rect, updateRect, leftTopRadius, 2487 rightTopRadius, leftBottomRadius, rightBottomRadius, base, true, flags, 2488 borders, orientation); 2489 } 2490 2491 2492 // #pragma mark - 2493 2494 2495 void 2496 HaikuControlLook::_DrawButtonFrame(BView* view, BRect& rect, 2497 const BRect& updateRect, float leftTopRadius, float rightTopRadius, 2498 float leftBottomRadius, float rightBottomRadius, const rgb_color& base, 2499 const rgb_color& background, float contrast, float brightness, 2500 uint32 flags, uint32 borders) 2501 { 2502 if (!rect.IsValid()) 2503 return; 2504 2505 // save the clipping constraints of the view 2506 view->PushState(); 2507 2508 // set clipping constraints to rect 2509 view->ClipToRect(rect); 2510 2511 // If the button is flat and neither activated nor otherwise highlighted 2512 // (mouse hovering or focussed), draw it flat. 2513 if ((flags & B_FLAT) != 0 2514 && (flags & (B_ACTIVATED | B_PARTIALLY_ACTIVATED)) == 0 2515 && ((flags & (B_HOVER | B_FOCUSED)) == 0 2516 || (flags & B_DISABLED) != 0)) { 2517 _DrawFrame(view, rect, background, background, background, 2518 background, borders); 2519 _DrawFrame(view, rect, background, background, background, 2520 background, borders); 2521 view->PopState(); 2522 return; 2523 } 2524 2525 // outer edge colors 2526 rgb_color edgeLightColor; 2527 rgb_color edgeShadowColor; 2528 2529 // default button frame color 2530 rgb_color defaultIndicatorColor = ui_color(B_CONTROL_BORDER_COLOR); 2531 rgb_color cornerBgColor; 2532 2533 if ((flags & B_DISABLED) != 0) { 2534 defaultIndicatorColor = disable_color(defaultIndicatorColor, 2535 background); 2536 } 2537 2538 drawing_mode oldMode = view->DrawingMode(); 2539 2540 if ((flags & B_DEFAULT_BUTTON) != 0) { 2541 cornerBgColor = defaultIndicatorColor; 2542 edgeLightColor = _EdgeLightColor(defaultIndicatorColor, 2543 contrast * ((flags & B_DISABLED) != 0 ? 0.3 : 0.8), 2544 brightness * ((flags & B_DISABLED) != 0 ? 1.0 : 0.9), flags); 2545 edgeShadowColor = _EdgeShadowColor(defaultIndicatorColor, 2546 contrast * ((flags & B_DISABLED) != 0 ? 0.3 : 0.8), 2547 brightness * ((flags & B_DISABLED) != 0 ? 1.0 : 0.9), flags); 2548 2549 // draw default button indicator 2550 // Allow a 1-pixel border of the background to come through. 2551 rect.InsetBy(1, 1); 2552 2553 view->SetHighColor(defaultIndicatorColor); 2554 view->StrokeRoundRect(rect, leftTopRadius, leftTopRadius); 2555 rect.InsetBy(1, 1); 2556 2557 view->StrokeRoundRect(rect, leftTopRadius, leftTopRadius); 2558 rect.InsetBy(1, 1); 2559 } else { 2560 cornerBgColor = background; 2561 if ((flags & B_BLEND_FRAME) != 0) { 2562 // set the background color to transparent for the case 2563 // that we are on the desktop 2564 cornerBgColor.alpha = 0; 2565 view->SetDrawingMode(B_OP_ALPHA); 2566 } 2567 2568 edgeLightColor = _EdgeLightColor(background, 2569 contrast * ((flags & B_DISABLED) != 0 ? 0.0 : 1.0), 2570 brightness * 1.0, flags); 2571 edgeShadowColor = _EdgeShadowColor(background, 2572 contrast * (flags & B_DISABLED) != 0 ? 0.0 : 1.0, 2573 brightness * 1.0, flags); 2574 } 2575 2576 // frame colors 2577 rgb_color frameLightColor = _FrameLightColor(base, flags); 2578 rgb_color frameShadowColor = _FrameShadowColor(base, flags); 2579 2580 // rounded corners 2581 2582 if ((borders & B_LEFT_BORDER) != 0 && (borders & B_TOP_BORDER) != 0 2583 && leftTopRadius > 0) { 2584 // draw left top rounded corner 2585 BRect leftTopCorner(floorf(rect.left), floorf(rect.top), 2586 floorf(rect.left + leftTopRadius), 2587 floorf(rect.top + leftTopRadius)); 2588 BRect cornerRect(leftTopCorner); 2589 _DrawRoundCornerFrameLeftTop(view, leftTopCorner, updateRect, 2590 cornerBgColor, edgeShadowColor, frameLightColor); 2591 view->ClipToInverseRect(cornerRect); 2592 } 2593 2594 if ((borders & B_TOP_BORDER) != 0 && (borders & B_RIGHT_BORDER) != 0 2595 && rightTopRadius > 0) { 2596 // draw right top rounded corner 2597 BRect rightTopCorner(floorf(rect.right - rightTopRadius), 2598 floorf(rect.top), floorf(rect.right), 2599 floorf(rect.top + rightTopRadius)); 2600 BRect cornerRect(rightTopCorner); 2601 _DrawRoundCornerFrameRightTop(view, rightTopCorner, updateRect, 2602 cornerBgColor, edgeShadowColor, edgeLightColor, 2603 frameLightColor, frameShadowColor); 2604 view->ClipToInverseRect(cornerRect); 2605 } 2606 2607 if ((borders & B_LEFT_BORDER) != 0 && (borders & B_BOTTOM_BORDER) != 0 2608 && leftBottomRadius > 0) { 2609 // draw left bottom rounded corner 2610 BRect leftBottomCorner(floorf(rect.left), 2611 floorf(rect.bottom - leftBottomRadius), 2612 floorf(rect.left + leftBottomRadius), floorf(rect.bottom)); 2613 BRect cornerRect(leftBottomCorner); 2614 _DrawRoundCornerFrameLeftBottom(view, leftBottomCorner, updateRect, 2615 cornerBgColor, edgeShadowColor, edgeLightColor, 2616 frameLightColor, frameShadowColor); 2617 view->ClipToInverseRect(cornerRect); 2618 } 2619 2620 if ((borders & B_RIGHT_BORDER) != 0 && (borders & B_BOTTOM_BORDER) != 0 2621 && rightBottomRadius > 0) { 2622 // draw right bottom rounded corner 2623 BRect rightBottomCorner(floorf(rect.right - rightBottomRadius), 2624 floorf(rect.bottom - rightBottomRadius), floorf(rect.right), 2625 floorf(rect.bottom)); 2626 BRect cornerRect(rightBottomCorner); 2627 _DrawRoundCornerFrameRightBottom(view, rightBottomCorner, 2628 updateRect, cornerBgColor, edgeLightColor, frameShadowColor); 2629 view->ClipToInverseRect(cornerRect); 2630 } 2631 2632 // draw outer edge 2633 if ((flags & B_DEFAULT_BUTTON) != 0) { 2634 _DrawOuterResessedFrame(view, rect, defaultIndicatorColor, 2635 contrast * ((flags & B_DISABLED) != 0 ? 0.3 : 0.8), 2636 brightness * ((flags & B_DISABLED) != 0 ? 1.0 : 0.9), 2637 flags, borders); 2638 } else { 2639 _DrawOuterResessedFrame(view, rect, background, 2640 contrast * ((flags & B_DISABLED) != 0 ? 0.0 : 1.0), 2641 brightness * 1.0, flags, borders); 2642 } 2643 2644 view->SetDrawingMode(oldMode); 2645 2646 // draw frame 2647 if ((flags & B_BLEND_FRAME) != 0) { 2648 drawing_mode oldDrawingMode = view->DrawingMode(); 2649 view->SetDrawingMode(B_OP_ALPHA); 2650 2651 _DrawFrame(view, rect, frameLightColor, frameLightColor, 2652 frameShadowColor, frameShadowColor, borders); 2653 2654 view->SetDrawingMode(oldDrawingMode); 2655 } else { 2656 _DrawFrame(view, rect, frameLightColor, frameLightColor, 2657 frameShadowColor, frameShadowColor, borders); 2658 } 2659 2660 // restore the clipping constraints of the view 2661 view->PopState(); 2662 } 2663 2664 2665 void 2666 HaikuControlLook::_DrawOuterResessedFrame(BView* view, BRect& rect, 2667 const rgb_color& base, float contrast, float brightness, uint32 flags, 2668 uint32 borders) 2669 { 2670 rgb_color edgeLightColor = _EdgeLightColor(base, contrast, 2671 brightness, flags); 2672 rgb_color edgeShadowColor = _EdgeShadowColor(base, contrast, 2673 brightness, flags); 2674 2675 if ((flags & B_BLEND_FRAME) != 0) { 2676 // assumes the background has already been painted 2677 drawing_mode oldDrawingMode = view->DrawingMode(); 2678 view->SetDrawingMode(B_OP_ALPHA); 2679 2680 _DrawFrame(view, rect, edgeShadowColor, edgeShadowColor, 2681 edgeLightColor, edgeLightColor, borders); 2682 2683 view->SetDrawingMode(oldDrawingMode); 2684 } else { 2685 _DrawFrame(view, rect, edgeShadowColor, edgeShadowColor, 2686 edgeLightColor, edgeLightColor, borders); 2687 } 2688 } 2689 2690 2691 void 2692 HaikuControlLook::_DrawFrame(BView* view, BRect& rect, const rgb_color& left, 2693 const rgb_color& top, const rgb_color& right, const rgb_color& bottom, 2694 uint32 borders) 2695 { 2696 view->BeginLineArray(4); 2697 2698 if (borders & B_LEFT_BORDER) { 2699 view->AddLine( 2700 BPoint(rect.left, rect.bottom), 2701 BPoint(rect.left, rect.top), left); 2702 rect.left++; 2703 } 2704 if (borders & B_TOP_BORDER) { 2705 view->AddLine( 2706 BPoint(rect.left, rect.top), 2707 BPoint(rect.right, rect.top), top); 2708 rect.top++; 2709 } 2710 if (borders & B_RIGHT_BORDER) { 2711 view->AddLine( 2712 BPoint(rect.right, rect.top), 2713 BPoint(rect.right, rect.bottom), right); 2714 rect.right--; 2715 } 2716 if (borders & B_BOTTOM_BORDER) { 2717 view->AddLine( 2718 BPoint(rect.left, rect.bottom), 2719 BPoint(rect.right, rect.bottom), bottom); 2720 rect.bottom--; 2721 } 2722 2723 view->EndLineArray(); 2724 } 2725 2726 2727 void 2728 HaikuControlLook::_DrawFrame(BView* view, BRect& rect, const rgb_color& left, 2729 const rgb_color& top, const rgb_color& right, const rgb_color& bottom, 2730 const rgb_color& rightTop, const rgb_color& leftBottom, uint32 borders) 2731 { 2732 view->BeginLineArray(6); 2733 2734 if (borders & B_TOP_BORDER) { 2735 if (borders & B_RIGHT_BORDER) { 2736 view->AddLine( 2737 BPoint(rect.left, rect.top), 2738 BPoint(rect.right - 1, rect.top), top); 2739 view->AddLine( 2740 BPoint(rect.right, rect.top), 2741 BPoint(rect.right, rect.top), rightTop); 2742 } else { 2743 view->AddLine( 2744 BPoint(rect.left, rect.top), 2745 BPoint(rect.right, rect.top), top); 2746 } 2747 rect.top++; 2748 } 2749 2750 if (borders & B_LEFT_BORDER) { 2751 view->AddLine( 2752 BPoint(rect.left, rect.top), 2753 BPoint(rect.left, rect.bottom - 1), left); 2754 view->AddLine( 2755 BPoint(rect.left, rect.bottom), 2756 BPoint(rect.left, rect.bottom), leftBottom); 2757 rect.left++; 2758 } 2759 2760 if (borders & B_BOTTOM_BORDER) { 2761 view->AddLine( 2762 BPoint(rect.left, rect.bottom), 2763 BPoint(rect.right, rect.bottom), bottom); 2764 rect.bottom--; 2765 } 2766 2767 if (borders & B_RIGHT_BORDER) { 2768 view->AddLine( 2769 BPoint(rect.right, rect.bottom), 2770 BPoint(rect.right, rect.top), right); 2771 rect.right--; 2772 } 2773 2774 view->EndLineArray(); 2775 } 2776 2777 2778 void 2779 HaikuControlLook::_DrawButtonBackground(BView* view, BRect& rect, 2780 const BRect& updateRect, float leftTopRadius, float rightTopRadius, 2781 float leftBottomRadius, float rightBottomRadius, const rgb_color& base, 2782 bool popupIndicator, uint32 flags, uint32 borders, orientation orientation) 2783 { 2784 if (!rect.IsValid()) 2785 return; 2786 2787 // save the clipping constraints of the view 2788 view->PushState(); 2789 2790 // set clipping constraints to rect 2791 view->ClipToRect(rect); 2792 2793 // If the button is flat and neither activated nor otherwise highlighted 2794 // (mouse hovering or focussed), draw it flat. 2795 if ((flags & B_FLAT) != 0 2796 && (flags & (B_ACTIVATED | B_PARTIALLY_ACTIVATED)) == 0 2797 && ((flags & (B_HOVER | B_FOCUSED)) == 0 2798 || (flags & B_DISABLED) != 0)) { 2799 _DrawFlatButtonBackground(view, rect, updateRect, base, popupIndicator, 2800 flags, borders, orientation); 2801 } else { 2802 BRegion clipping(rect); 2803 _DrawNonFlatButtonBackground(view, rect, updateRect, clipping, 2804 leftTopRadius, rightTopRadius, leftBottomRadius, rightBottomRadius, 2805 base, popupIndicator, flags, borders, orientation); 2806 } 2807 2808 // restore the clipping constraints of the view 2809 view->PopState(); 2810 } 2811 2812 2813 void 2814 HaikuControlLook::_DrawFlatButtonBackground(BView* view, BRect& rect, 2815 const BRect& updateRect, const rgb_color& base, bool popupIndicator, 2816 uint32 flags, uint32 borders, orientation orientation) 2817 { 2818 _DrawFrame(view, rect, base, base, base, base, borders); 2819 // Not an actual frame, but the method insets our rect as needed. 2820 2821 view->SetHighColor(base); 2822 view->FillRect(rect); 2823 2824 if (popupIndicator) { 2825 BRect indicatorRect(rect); 2826 rect.right -= ComposeSpacing(kButtonPopUpIndicatorWidth); 2827 indicatorRect.left = rect.right + 3; 2828 // 2 pixels for the separator 2829 2830 view->SetHighColor(base); 2831 view->FillRect(indicatorRect); 2832 2833 _DrawPopUpMarker(view, indicatorRect, base, flags); 2834 } 2835 } 2836 2837 2838 void 2839 HaikuControlLook::_DrawNonFlatButtonBackground(BView* view, BRect& rect, 2840 const BRect& updateRect, BRegion& clipping, float leftTopRadius, 2841 float rightTopRadius, float leftBottomRadius, float rightBottomRadius, 2842 const rgb_color& base, bool popupIndicator, uint32 flags, uint32 borders, 2843 orientation orientation) 2844 { 2845 // inner bevel colors 2846 rgb_color bevelLightColor = _BevelLightColor(base, flags); 2847 rgb_color bevelShadowColor = _BevelShadowColor(base, flags); 2848 2849 // button background color 2850 rgb_color buttonBgColor; 2851 if ((flags & B_DISABLED) != 0) 2852 buttonBgColor = tint_color(base, 0.7); 2853 else 2854 buttonBgColor = tint_color(base, B_LIGHTEN_1_TINT); 2855 2856 // surface top gradient 2857 BGradientLinear fillGradient; 2858 _MakeButtonGradient(fillGradient, rect, base, flags, orientation); 2859 2860 // rounded corners 2861 2862 if ((borders & B_LEFT_BORDER) != 0 && (borders & B_TOP_BORDER) != 0 2863 && leftTopRadius > 0) { 2864 // draw left top rounded corner 2865 BRect leftTopCorner(floorf(rect.left), floorf(rect.top), 2866 floorf(rect.left + leftTopRadius - 2.0), 2867 floorf(rect.top + leftTopRadius - 2.0)); 2868 clipping.Exclude(leftTopCorner); 2869 BRect cornerRect(leftTopCorner); 2870 _DrawRoundCornerBackgroundLeftTop(view, leftTopCorner, updateRect, 2871 bevelLightColor, fillGradient); 2872 view->ClipToInverseRect(cornerRect); 2873 } 2874 2875 if ((borders & B_TOP_BORDER) != 0 && (borders & B_RIGHT_BORDER) != 0 2876 && rightTopRadius > 0) { 2877 // draw right top rounded corner 2878 BRect rightTopCorner(floorf(rect.right - rightTopRadius + 2.0), 2879 floorf(rect.top), floorf(rect.right), 2880 floorf(rect.top + rightTopRadius - 2.0)); 2881 clipping.Exclude(rightTopCorner); 2882 BRect cornerRect(rightTopCorner); 2883 _DrawRoundCornerBackgroundRightTop(view, rightTopCorner, 2884 updateRect, bevelLightColor, bevelShadowColor, fillGradient); 2885 view->ClipToInverseRect(cornerRect); 2886 } 2887 2888 if ((borders & B_LEFT_BORDER) != 0 && (borders & B_BOTTOM_BORDER) != 0 2889 && leftBottomRadius > 0) { 2890 // draw left bottom rounded corner 2891 BRect leftBottomCorner(floorf(rect.left), 2892 floorf(rect.bottom - leftBottomRadius + 2.0), 2893 floorf(rect.left + leftBottomRadius - 2.0), 2894 floorf(rect.bottom)); 2895 clipping.Exclude(leftBottomCorner); 2896 BRect cornerRect(leftBottomCorner); 2897 _DrawRoundCornerBackgroundLeftBottom(view, leftBottomCorner, 2898 updateRect, bevelLightColor, bevelShadowColor, fillGradient); 2899 view->ClipToInverseRect(cornerRect); 2900 } 2901 2902 if ((borders & B_RIGHT_BORDER) != 0 && (borders & B_BOTTOM_BORDER) != 0 2903 && rightBottomRadius > 0) { 2904 // draw right bottom rounded corner 2905 BRect rightBottomCorner(floorf(rect.right - rightBottomRadius + 2.0), 2906 floorf(rect.bottom - rightBottomRadius + 2.0), floorf(rect.right), 2907 floorf(rect.bottom)); 2908 clipping.Exclude(rightBottomCorner); 2909 BRect cornerRect(rightBottomCorner); 2910 _DrawRoundCornerBackgroundRightBottom(view, rightBottomCorner, 2911 updateRect, bevelShadowColor, fillGradient); 2912 view->ClipToInverseRect(cornerRect); 2913 } 2914 2915 // draw inner bevel 2916 2917 if ((flags & B_ACTIVATED) != 0) { 2918 view->BeginLineArray(4); 2919 2920 // shadow along left/top borders 2921 if (borders & B_LEFT_BORDER) { 2922 view->AddLine(BPoint(rect.left, rect.top), 2923 BPoint(rect.left, rect.bottom), bevelLightColor); 2924 rect.left++; 2925 } 2926 if (borders & B_TOP_BORDER) { 2927 view->AddLine(BPoint(rect.left, rect.top), 2928 BPoint(rect.right, rect.top), bevelLightColor); 2929 rect.top++; 2930 } 2931 2932 // softer shadow along left/top borders 2933 if (borders & B_LEFT_BORDER) { 2934 view->AddLine(BPoint(rect.left, rect.top), 2935 BPoint(rect.left, rect.bottom), bevelShadowColor); 2936 rect.left++; 2937 } 2938 if (borders & B_TOP_BORDER) { 2939 view->AddLine(BPoint(rect.left, rect.top), 2940 BPoint(rect.right, rect.top), bevelShadowColor); 2941 rect.top++; 2942 } 2943 2944 view->EndLineArray(); 2945 } else { 2946 _DrawFrame(view, rect, 2947 bevelLightColor, bevelLightColor, 2948 bevelShadowColor, bevelShadowColor, 2949 buttonBgColor, buttonBgColor, borders); 2950 } 2951 2952 if (popupIndicator) { 2953 BRect indicatorRect(rect); 2954 rect.right -= ComposeSpacing(kButtonPopUpIndicatorWidth); 2955 indicatorRect.left = rect.right + 3; 2956 // 2 pixels for the separator 2957 2958 // Even when depressed we want the pop-up indicator background and 2959 // separator to cover the area up to the top. 2960 if ((flags & B_ACTIVATED) != 0) 2961 indicatorRect.top--; 2962 2963 // draw the separator 2964 rgb_color separatorBaseColor = base; 2965 if ((flags & B_ACTIVATED) != 0) 2966 separatorBaseColor = tint_color(base, B_DARKEN_1_TINT); 2967 2968 rgb_color separatorLightColor = _EdgeLightColor(separatorBaseColor, 2969 (flags & B_DISABLED) != 0 ? 0.7 : 1.0, 1.0, flags); 2970 rgb_color separatorShadowColor = _EdgeShadowColor(separatorBaseColor, 2971 (flags & B_DISABLED) != 0 ? 0.7 : 1.0, 1.0, flags); 2972 2973 view->BeginLineArray(2); 2974 2975 view->AddLine(BPoint(indicatorRect.left - 2, indicatorRect.top), 2976 BPoint(indicatorRect.left - 2, indicatorRect.bottom), 2977 separatorShadowColor); 2978 view->AddLine(BPoint(indicatorRect.left - 1, indicatorRect.top), 2979 BPoint(indicatorRect.left - 1, indicatorRect.bottom), 2980 separatorLightColor); 2981 2982 view->EndLineArray(); 2983 2984 // draw background and pop-up marker 2985 _DrawMenuFieldBackgroundInside(view, indicatorRect, updateRect, 2986 0.0f, rightTopRadius, 0.0f, rightBottomRadius, base, flags, 0); 2987 2988 if ((flags & B_ACTIVATED) != 0) 2989 indicatorRect.top++; 2990 2991 _DrawPopUpMarker(view, indicatorRect, base, flags); 2992 } 2993 2994 // fill in the background 2995 view->FillRect(rect, fillGradient); 2996 } 2997 2998 2999 void 3000 HaikuControlLook::_DrawPopUpMarker(BView* view, const BRect& rect, 3001 const rgb_color& base, uint32 flags) 3002 { 3003 BPoint center(roundf((rect.left + rect.right) / 2.0), 3004 roundf((rect.top + rect.bottom) / 2.0)); 3005 const float metric = roundf(rect.Width() * 3.125f) / 10.0f, 3006 offset = ceilf((metric * 0.2f) * 10.0f) / 10.0f; 3007 BPoint triangle[3]; 3008 triangle[0] = center + BPoint(-metric, -offset); 3009 triangle[1] = center + BPoint(metric, -offset); 3010 triangle[2] = center + BPoint(0.0, metric * 0.8f); 3011 3012 const uint32 viewFlags = view->Flags(); 3013 view->SetFlags(viewFlags | B_SUBPIXEL_PRECISE); 3014 3015 rgb_color markColor; 3016 if ((flags & B_DISABLED) != 0) 3017 markColor = tint_color(base, 1.35); 3018 else 3019 markColor = tint_color(base, 1.65); 3020 3021 view->SetHighColor(markColor); 3022 view->FillTriangle(triangle[0], triangle[1], triangle[2]); 3023 3024 view->SetFlags(viewFlags); 3025 } 3026 3027 3028 void 3029 HaikuControlLook::_DrawMenuFieldBackgroundOutside(BView* view, BRect& rect, 3030 const BRect& updateRect, float leftTopRadius, float rightTopRadius, 3031 float leftBottomRadius, float rightBottomRadius, const rgb_color& base, 3032 bool popupIndicator, uint32 flags) 3033 { 3034 if (!ShouldDraw(view, rect, updateRect)) 3035 return; 3036 3037 if (popupIndicator) { 3038 const float indicatorWidth = ComposeSpacing(kButtonPopUpIndicatorWidth); 3039 const float spacing = (indicatorWidth <= 11.0f) ? 1.0f : roundf(indicatorWidth / 11.0f); 3040 3041 BRect leftRect(rect); 3042 leftRect.right -= indicatorWidth - spacing; 3043 3044 BRect rightRect(rect); 3045 rightRect.left = rightRect.right - (indicatorWidth - spacing * 2); 3046 3047 _DrawMenuFieldBackgroundInside(view, leftRect, updateRect, 3048 leftTopRadius, 0.0f, leftBottomRadius, 0.0f, base, flags, 3049 B_LEFT_BORDER | B_TOP_BORDER | B_BOTTOM_BORDER); 3050 3051 _DrawMenuFieldBackgroundInside(view, rightRect, updateRect, 3052 0.0f, rightTopRadius, 0.0f, rightBottomRadius, base, flags, 3053 B_TOP_BORDER | B_RIGHT_BORDER | B_BOTTOM_BORDER); 3054 3055 _DrawPopUpMarker(view, rightRect, base, flags); 3056 3057 // draw a line on the left of the popup frame 3058 rgb_color bevelShadowColor = _BevelShadowColor(base, flags); 3059 view->SetHighColor(bevelShadowColor); 3060 BPoint leftTopCorner(floorf(rightRect.left - spacing), 3061 floorf(rightRect.top - spacing)); 3062 BPoint leftBottomCorner(floorf(rightRect.left - spacing), 3063 floorf(rightRect.bottom + spacing)); 3064 for (float i = 0; i < spacing; i++) { 3065 view->StrokeLine(leftTopCorner + BPoint(i, 0), 3066 leftBottomCorner + BPoint(i, 0)); 3067 } 3068 3069 rect = leftRect; 3070 } else { 3071 _DrawMenuFieldBackgroundInside(view, rect, updateRect, leftTopRadius, 3072 rightTopRadius, leftBottomRadius, rightBottomRadius, base, flags); 3073 } 3074 } 3075 3076 3077 void 3078 HaikuControlLook::_DrawMenuFieldBackgroundInside(BView* view, BRect& rect, 3079 const BRect& updateRect, float leftTopRadius, float rightTopRadius, 3080 float leftBottomRadius, float rightBottomRadius, const rgb_color& base, 3081 uint32 flags, uint32 borders) 3082 { 3083 if (!ShouldDraw(view, rect, updateRect)) 3084 return; 3085 3086 // save the clipping constraints of the view 3087 view->PushState(); 3088 3089 // set clipping constraints to rect 3090 view->ClipToRect(rect); 3091 3092 // frame colors 3093 rgb_color frameLightColor = _FrameLightColor(base, flags); 3094 rgb_color frameShadowColor = _FrameShadowColor(base, flags); 3095 3096 // indicator background color 3097 rgb_color indicatorBase; 3098 if ((borders & B_LEFT_BORDER) != 0) 3099 indicatorBase = base; 3100 else { 3101 if ((flags & B_DISABLED) != 0) 3102 indicatorBase = tint_color(base, 1.05); 3103 else 3104 indicatorBase = tint_color(base, 1.12); 3105 } 3106 3107 // bevel colors 3108 rgb_color cornerColor = tint_color(indicatorBase, 0.85); 3109 rgb_color bevelColor1 = tint_color(indicatorBase, 0.3); 3110 rgb_color bevelColor2 = tint_color(indicatorBase, 0.5); 3111 rgb_color bevelColor3 = tint_color(indicatorBase, 1.03); 3112 3113 if ((flags & B_DISABLED) != 0) { 3114 cornerColor = tint_color(indicatorBase, 0.8); 3115 bevelColor1 = tint_color(indicatorBase, 0.7); 3116 bevelColor2 = tint_color(indicatorBase, 0.8); 3117 bevelColor3 = tint_color(indicatorBase, 1.01); 3118 } else { 3119 cornerColor = tint_color(indicatorBase, 0.85); 3120 bevelColor1 = tint_color(indicatorBase, 0.3); 3121 bevelColor2 = tint_color(indicatorBase, 0.5); 3122 bevelColor3 = tint_color(indicatorBase, 1.03); 3123 } 3124 3125 // surface top gradient 3126 BGradientLinear fillGradient; 3127 _MakeButtonGradient(fillGradient, rect, indicatorBase, flags); 3128 3129 // rounded corners 3130 3131 if ((borders & B_LEFT_BORDER) != 0 && (borders & B_TOP_BORDER) != 0 3132 && leftTopRadius > 0) { 3133 // draw left top rounded corner 3134 BRect leftTopCorner(floorf(rect.left), floorf(rect.top), 3135 floorf(rect.left + leftTopRadius - 2.0), 3136 floorf(rect.top + leftTopRadius - 2.0)); 3137 BRect cornerRect(leftTopCorner); 3138 3139 view->PushState(); 3140 view->ClipToRect(cornerRect); 3141 3142 BRect ellipseRect(leftTopCorner); 3143 ellipseRect.InsetBy(-1.0, -1.0); 3144 ellipseRect.right = ellipseRect.left + ellipseRect.Width() * 2; 3145 ellipseRect.bottom = ellipseRect.top + ellipseRect.Height() * 2; 3146 3147 // draw the frame (again) 3148 view->SetHighColor(frameLightColor); 3149 view->FillEllipse(ellipseRect); 3150 3151 // draw the bevel and background 3152 _DrawRoundCornerBackgroundLeftTop(view, leftTopCorner, updateRect, 3153 bevelColor1, fillGradient); 3154 3155 view->PopState(); 3156 view->ClipToInverseRect(cornerRect); 3157 } 3158 3159 if ((borders & B_TOP_BORDER) != 0 && (borders & B_RIGHT_BORDER) != 0 3160 && rightTopRadius > 0) { 3161 // draw right top rounded corner 3162 BRect rightTopCorner(floorf(rect.right - rightTopRadius + 2.0), 3163 floorf(rect.top), floorf(rect.right), 3164 floorf(rect.top + rightTopRadius - 2.0)); 3165 BRect cornerRect(rightTopCorner); 3166 3167 view->PushState(); 3168 view->ClipToRect(cornerRect); 3169 3170 BRect ellipseRect(rightTopCorner); 3171 ellipseRect.InsetBy(-1.0, -1.0); 3172 ellipseRect.left = ellipseRect.right - ellipseRect.Width() * 2; 3173 ellipseRect.bottom = ellipseRect.top + ellipseRect.Height() * 2; 3174 3175 // draw the frame (again) 3176 if (frameLightColor == frameShadowColor) { 3177 view->SetHighColor(frameLightColor); 3178 view->FillEllipse(ellipseRect); 3179 } else { 3180 BGradientLinear gradient; 3181 gradient.AddColor(frameLightColor, 0); 3182 gradient.AddColor(frameShadowColor, 255); 3183 gradient.SetStart(rightTopCorner.LeftTop()); 3184 gradient.SetEnd(rightTopCorner.RightBottom()); 3185 view->FillEllipse(ellipseRect, gradient); 3186 } 3187 3188 // draw the bevel and background 3189 _DrawRoundCornerBackgroundRightTop(view, rightTopCorner, updateRect, 3190 bevelColor1, bevelColor3, fillGradient); 3191 3192 view->PopState(); 3193 view->ClipToInverseRect(cornerRect); 3194 } 3195 3196 if ((borders & B_LEFT_BORDER) != 0 && (borders & B_BOTTOM_BORDER) != 0 3197 && leftBottomRadius > 0) { 3198 // draw left bottom rounded corner 3199 BRect leftBottomCorner(floorf(rect.left), 3200 floorf(rect.bottom - leftBottomRadius + 2.0), 3201 floorf(rect.left + leftBottomRadius - 2.0), 3202 floorf(rect.bottom)); 3203 BRect cornerRect(leftBottomCorner); 3204 3205 view->PushState(); 3206 view->ClipToRect(cornerRect); 3207 3208 BRect ellipseRect(leftBottomCorner); 3209 ellipseRect.InsetBy(-1.0, -1.0); 3210 ellipseRect.right = ellipseRect.left + ellipseRect.Width() * 2; 3211 ellipseRect.top = ellipseRect.bottom - ellipseRect.Height() * 2; 3212 3213 // draw the frame (again) 3214 if (frameLightColor == frameShadowColor) { 3215 view->SetHighColor(frameLightColor); 3216 view->FillEllipse(ellipseRect); 3217 } else { 3218 BGradientLinear gradient; 3219 gradient.AddColor(frameLightColor, 0); 3220 gradient.AddColor(frameShadowColor, 255); 3221 gradient.SetStart(leftBottomCorner.LeftTop()); 3222 gradient.SetEnd(leftBottomCorner.RightBottom()); 3223 view->FillEllipse(ellipseRect, gradient); 3224 } 3225 3226 // draw the bevel and background 3227 _DrawRoundCornerBackgroundLeftBottom(view, leftBottomCorner, 3228 updateRect, bevelColor2, bevelColor3, fillGradient); 3229 3230 view->PopState(); 3231 view->ClipToInverseRect(cornerRect); 3232 } 3233 3234 if ((borders & B_RIGHT_BORDER) != 0 && (borders & B_BOTTOM_BORDER) != 0 3235 && rightBottomRadius > 0) { 3236 // draw right bottom rounded corner 3237 BRect rightBottomCorner(floorf(rect.right - rightBottomRadius + 2.0), 3238 floorf(rect.bottom - rightBottomRadius + 2.0), floorf(rect.right), 3239 floorf(rect.bottom)); 3240 BRect cornerRect(rightBottomCorner); 3241 3242 view->PushState(); 3243 view->ClipToRect(cornerRect); 3244 3245 BRect ellipseRect(rightBottomCorner); 3246 ellipseRect.InsetBy(-1.0, -1.0); 3247 ellipseRect.left = ellipseRect.right - ellipseRect.Width() * 2; 3248 ellipseRect.top = ellipseRect.bottom - ellipseRect.Height() * 2; 3249 3250 // draw the frame (again) 3251 view->SetHighColor(frameShadowColor); 3252 view->FillEllipse(ellipseRect); 3253 3254 // draw the bevel and background 3255 _DrawRoundCornerBackgroundRightBottom(view, rightBottomCorner, 3256 updateRect, bevelColor3, fillGradient); 3257 3258 view->PopState(); 3259 view->ClipToInverseRect(cornerRect); 3260 } 3261 3262 // draw the bevel 3263 _DrawFrame(view, rect, 3264 bevelColor2, bevelColor1, 3265 bevelColor3, bevelColor3, 3266 cornerColor, cornerColor, 3267 borders); 3268 3269 // fill in the background 3270 view->FillRect(rect, fillGradient); 3271 3272 // restore the clipping constraints of the view 3273 view->PopState(); 3274 } 3275 3276 3277 void 3278 HaikuControlLook::_DrawRoundCornerLeftTop(BView* view, BRect& cornerRect, 3279 const BRect& updateRect, const rgb_color& background, 3280 const rgb_color& edgeColor, const rgb_color& frameColor, 3281 const rgb_color& bevelColor, const BGradientLinear& fillGradient) 3282 { 3283 _DrawRoundCornerFrameLeftTop(view, cornerRect, updateRect, 3284 background, edgeColor, frameColor); 3285 _DrawRoundCornerBackgroundLeftTop(view, cornerRect, updateRect, 3286 bevelColor, fillGradient); 3287 } 3288 3289 3290 void 3291 HaikuControlLook::_DrawRoundCornerFrameLeftTop(BView* view, BRect& cornerRect, 3292 const BRect& updateRect, const rgb_color& background, 3293 const rgb_color& edgeColor, const rgb_color& frameColor) 3294 { 3295 view->PushState(); 3296 3297 // constrain clipping region to corner 3298 view->ClipToRect(cornerRect); 3299 3300 // background 3301 view->SetHighColor(background); 3302 view->FillRect(cornerRect); 3303 3304 // outer edge 3305 BRect ellipseRect(cornerRect); 3306 ellipseRect.right = ellipseRect.left + ellipseRect.Width() * 2; 3307 ellipseRect.bottom = ellipseRect.top + ellipseRect.Height() * 2; 3308 3309 view->SetHighColor(edgeColor); 3310 view->FillEllipse(ellipseRect); 3311 3312 // frame 3313 ellipseRect.InsetBy(1, 1); 3314 cornerRect.left++; 3315 cornerRect.top++; 3316 view->SetHighColor(frameColor); 3317 view->FillEllipse(ellipseRect); 3318 3319 // prepare for bevel 3320 cornerRect.left++; 3321 cornerRect.top++; 3322 3323 view->PopState(); 3324 } 3325 3326 3327 void 3328 HaikuControlLook::_DrawRoundCornerBackgroundLeftTop(BView* view, BRect& cornerRect, 3329 const BRect& updateRect, const rgb_color& bevelColor, 3330 const BGradientLinear& fillGradient) 3331 { 3332 view->PushState(); 3333 3334 // constrain clipping region to corner 3335 view->ClipToRect(cornerRect); 3336 3337 BRect ellipseRect(cornerRect); 3338 ellipseRect.right = ellipseRect.left + ellipseRect.Width() * 2; 3339 ellipseRect.bottom = ellipseRect.top + ellipseRect.Height() * 2; 3340 3341 // bevel 3342 view->SetHighColor(bevelColor); 3343 view->FillEllipse(ellipseRect); 3344 3345 // gradient 3346 ellipseRect.InsetBy(1, 1); 3347 view->FillEllipse(ellipseRect, fillGradient); 3348 3349 view->PopState(); 3350 } 3351 3352 3353 void 3354 HaikuControlLook::_DrawRoundCornerRightTop(BView* view, BRect& cornerRect, 3355 const BRect& updateRect, const rgb_color& background, 3356 const rgb_color& edgeTopColor, const rgb_color& edgeRightColor, 3357 const rgb_color& frameTopColor, const rgb_color& frameRightColor, 3358 const rgb_color& bevelTopColor, const rgb_color& bevelRightColor, 3359 const BGradientLinear& fillGradient) 3360 { 3361 _DrawRoundCornerFrameRightTop(view, cornerRect, updateRect, 3362 background, edgeTopColor, edgeRightColor, frameTopColor, 3363 frameRightColor); 3364 _DrawRoundCornerBackgroundRightTop(view, cornerRect, updateRect, 3365 bevelTopColor, bevelRightColor, fillGradient); 3366 } 3367 3368 3369 void 3370 HaikuControlLook::_DrawRoundCornerFrameRightTop(BView* view, BRect& cornerRect, 3371 const BRect& updateRect, const rgb_color& background, 3372 const rgb_color& edgeTopColor, const rgb_color& edgeRightColor, 3373 const rgb_color& frameTopColor, const rgb_color& frameRightColor) 3374 { 3375 view->PushState(); 3376 3377 // constrain clipping region to corner 3378 view->ClipToRect(cornerRect); 3379 3380 // background 3381 view->SetHighColor(background); 3382 view->FillRect(cornerRect); 3383 3384 // outer edge 3385 BRect ellipseRect(cornerRect); 3386 ellipseRect.left = ellipseRect.right - ellipseRect.Width() * 2; 3387 ellipseRect.bottom = ellipseRect.top + ellipseRect.Height() * 2; 3388 3389 BGradientLinear gradient; 3390 gradient.AddColor(edgeTopColor, 0); 3391 gradient.AddColor(edgeRightColor, 255); 3392 gradient.SetStart(cornerRect.LeftTop()); 3393 gradient.SetEnd(cornerRect.RightBottom()); 3394 view->FillEllipse(ellipseRect, gradient); 3395 3396 // frame 3397 ellipseRect.InsetBy(1, 1); 3398 cornerRect.right--; 3399 cornerRect.top++; 3400 if (frameTopColor == frameRightColor) { 3401 view->SetHighColor(frameTopColor); 3402 view->FillEllipse(ellipseRect); 3403 } else { 3404 gradient.SetColor(0, frameTopColor); 3405 gradient.SetColor(1, frameRightColor); 3406 gradient.SetStart(cornerRect.LeftTop()); 3407 gradient.SetEnd(cornerRect.RightBottom()); 3408 view->FillEllipse(ellipseRect, gradient); 3409 } 3410 3411 // prepare for bevel 3412 cornerRect.right--; 3413 cornerRect.top++; 3414 3415 view->PopState(); 3416 } 3417 3418 3419 void 3420 HaikuControlLook::_DrawRoundCornerBackgroundRightTop(BView* view, BRect& cornerRect, 3421 const BRect& updateRect, const rgb_color& bevelTopColor, 3422 const rgb_color& bevelRightColor, const BGradientLinear& fillGradient) 3423 { 3424 view->PushState(); 3425 3426 // constrain clipping region to corner 3427 view->ClipToRect(cornerRect); 3428 3429 BRect ellipseRect(cornerRect); 3430 ellipseRect.left = ellipseRect.right - ellipseRect.Width() * 2; 3431 ellipseRect.bottom = ellipseRect.top + ellipseRect.Height() * 2; 3432 3433 // bevel 3434 BGradientLinear gradient; 3435 gradient.AddColor(bevelTopColor, 0); 3436 gradient.AddColor(bevelRightColor, 255); 3437 gradient.SetStart(cornerRect.LeftTop()); 3438 gradient.SetEnd(cornerRect.RightBottom()); 3439 view->FillEllipse(ellipseRect, gradient); 3440 3441 // gradient 3442 ellipseRect.InsetBy(1, 1); 3443 view->FillEllipse(ellipseRect, fillGradient); 3444 3445 view->PopState(); 3446 } 3447 3448 3449 void 3450 HaikuControlLook::_DrawRoundCornerLeftBottom(BView* view, BRect& cornerRect, 3451 const BRect& updateRect, const rgb_color& background, 3452 const rgb_color& edgeLeftColor, const rgb_color& edgeBottomColor, 3453 const rgb_color& frameLeftColor, const rgb_color& frameBottomColor, 3454 const rgb_color& bevelLeftColor, const rgb_color& bevelBottomColor, 3455 const BGradientLinear& fillGradient) 3456 { 3457 _DrawRoundCornerFrameLeftBottom(view, cornerRect, updateRect, 3458 background, edgeLeftColor, edgeBottomColor, frameLeftColor, 3459 frameBottomColor); 3460 _DrawRoundCornerBackgroundLeftBottom(view, cornerRect, updateRect, 3461 bevelLeftColor, bevelBottomColor, fillGradient); 3462 } 3463 3464 3465 void 3466 HaikuControlLook::_DrawRoundCornerFrameLeftBottom(BView* view, BRect& cornerRect, 3467 const BRect& updateRect, const rgb_color& background, 3468 const rgb_color& edgeLeftColor, const rgb_color& edgeBottomColor, 3469 const rgb_color& frameLeftColor, const rgb_color& frameBottomColor) 3470 { 3471 view->PushState(); 3472 3473 // constrain clipping region to corner 3474 view->ClipToRect(cornerRect); 3475 3476 // background 3477 view->SetHighColor(background); 3478 view->FillRect(cornerRect); 3479 3480 // outer edge 3481 BRect ellipseRect(cornerRect); 3482 ellipseRect.right = ellipseRect.left + ellipseRect.Width() * 2; 3483 ellipseRect.top = ellipseRect.bottom - ellipseRect.Height() * 2; 3484 3485 BGradientLinear gradient; 3486 gradient.AddColor(edgeLeftColor, 0); 3487 gradient.AddColor(edgeBottomColor, 255); 3488 gradient.SetStart(cornerRect.LeftTop()); 3489 gradient.SetEnd(cornerRect.RightBottom()); 3490 view->FillEllipse(ellipseRect, gradient); 3491 3492 // frame 3493 ellipseRect.InsetBy(1, 1); 3494 cornerRect.left++; 3495 cornerRect.bottom--; 3496 if (frameLeftColor == frameBottomColor) { 3497 view->SetHighColor(frameLeftColor); 3498 view->FillEllipse(ellipseRect); 3499 } else { 3500 gradient.SetColor(0, frameLeftColor); 3501 gradient.SetColor(1, frameBottomColor); 3502 gradient.SetStart(cornerRect.LeftTop()); 3503 gradient.SetEnd(cornerRect.RightBottom()); 3504 view->FillEllipse(ellipseRect, gradient); 3505 } 3506 3507 // prepare for bevel 3508 cornerRect.left++; 3509 cornerRect.bottom--; 3510 3511 view->PopState(); 3512 } 3513 3514 3515 void 3516 HaikuControlLook::_DrawRoundCornerBackgroundLeftBottom(BView* view, BRect& cornerRect, 3517 const BRect& updateRect, const rgb_color& bevelLeftColor, 3518 const rgb_color& bevelBottomColor, const BGradientLinear& fillGradient) 3519 { 3520 view->PushState(); 3521 3522 // constrain clipping region to corner 3523 view->ClipToRect(cornerRect); 3524 3525 BRect ellipseRect(cornerRect); 3526 ellipseRect.right = ellipseRect.left + ellipseRect.Width() * 2; 3527 ellipseRect.top = ellipseRect.bottom - ellipseRect.Height() * 2; 3528 3529 // bevel 3530 BGradientLinear gradient; 3531 gradient.AddColor(bevelLeftColor, 0); 3532 gradient.AddColor(bevelBottomColor, 255); 3533 gradient.SetStart(cornerRect.LeftTop()); 3534 gradient.SetEnd(cornerRect.RightBottom()); 3535 view->FillEllipse(ellipseRect, gradient); 3536 3537 // gradient 3538 ellipseRect.InsetBy(1, 1); 3539 view->FillEllipse(ellipseRect, fillGradient); 3540 3541 view->PopState(); 3542 } 3543 3544 3545 void 3546 HaikuControlLook::_DrawRoundCornerRightBottom(BView* view, BRect& cornerRect, 3547 const BRect& updateRect, const rgb_color& background, 3548 const rgb_color& edgeColor, const rgb_color& frameColor, 3549 const rgb_color& bevelColor, const BGradientLinear& fillGradient) 3550 { 3551 _DrawRoundCornerFrameRightBottom(view, cornerRect, updateRect, 3552 background, edgeColor, frameColor); 3553 _DrawRoundCornerBackgroundRightBottom(view, cornerRect, updateRect, 3554 bevelColor, fillGradient); 3555 } 3556 3557 3558 void 3559 HaikuControlLook::_DrawRoundCornerFrameRightBottom(BView* view, BRect& cornerRect, 3560 const BRect& updateRect, const rgb_color& background, 3561 const rgb_color& edgeColor, const rgb_color& frameColor) 3562 { 3563 view->PushState(); 3564 3565 // constrain clipping region to corner 3566 view->ClipToRect(cornerRect); 3567 3568 // background 3569 view->SetHighColor(background); 3570 view->FillRect(cornerRect); 3571 3572 // outer edge 3573 BRect ellipseRect(cornerRect); 3574 ellipseRect.left = ellipseRect.right - ellipseRect.Width() * 2; 3575 ellipseRect.top = ellipseRect.bottom - ellipseRect.Height() * 2; 3576 3577 view->SetHighColor(edgeColor); 3578 view->FillEllipse(ellipseRect); 3579 3580 // frame 3581 ellipseRect.InsetBy(1, 1); 3582 cornerRect.right--; 3583 cornerRect.bottom--; 3584 view->SetHighColor(frameColor); 3585 view->FillEllipse(ellipseRect); 3586 3587 // prepare for bevel 3588 cornerRect.right--; 3589 cornerRect.bottom--; 3590 3591 view->PopState(); 3592 } 3593 3594 3595 void 3596 HaikuControlLook::_DrawRoundCornerBackgroundRightBottom(BView* view, 3597 BRect& cornerRect, const BRect& updateRect, const rgb_color& bevelColor, 3598 const BGradientLinear& fillGradient) 3599 { 3600 view->PushState(); 3601 3602 // constrain clipping region to corner 3603 view->ClipToRect(cornerRect); 3604 3605 BRect ellipseRect(cornerRect); 3606 ellipseRect.left = ellipseRect.right - ellipseRect.Width() * 2; 3607 ellipseRect.top = ellipseRect.bottom - ellipseRect.Height() * 2; 3608 3609 // bevel 3610 view->SetHighColor(bevelColor); 3611 view->FillEllipse(ellipseRect); 3612 3613 // gradient 3614 ellipseRect.InsetBy(1, 1); 3615 view->FillEllipse(ellipseRect, fillGradient); 3616 3617 view->PopState(); 3618 } 3619 3620 3621 void 3622 HaikuControlLook::_DrawRoundBarCorner(BView* view, BRect& rect, 3623 const BRect& updateRect, 3624 const rgb_color& edgeLightColor, const rgb_color& edgeShadowColor, 3625 const rgb_color& frameLightColor, const rgb_color& frameShadowColor, 3626 const rgb_color& fillLightColor, const rgb_color& fillShadowColor, 3627 float leftInset, float topInset, float rightInset, float bottomInset, 3628 orientation orientation) 3629 { 3630 if (!ShouldDraw(view, rect, updateRect)) 3631 return; 3632 3633 BGradientLinear gradient; 3634 gradient.AddColor(edgeShadowColor, 0); 3635 gradient.AddColor(edgeLightColor, 255); 3636 gradient.SetStart(rect.LeftTop()); 3637 if (orientation == B_HORIZONTAL) 3638 gradient.SetEnd(rect.LeftBottom()); 3639 else 3640 gradient.SetEnd(rect.RightTop()); 3641 3642 view->FillEllipse(rect, gradient); 3643 3644 rect.left += leftInset; 3645 rect.top += topInset; 3646 rect.right += rightInset; 3647 rect.bottom += bottomInset; 3648 3649 gradient.MakeEmpty(); 3650 gradient.AddColor(frameShadowColor, 0); 3651 gradient.AddColor(frameLightColor, 255); 3652 gradient.SetStart(rect.LeftTop()); 3653 if (orientation == B_HORIZONTAL) 3654 gradient.SetEnd(rect.LeftBottom()); 3655 else 3656 gradient.SetEnd(rect.RightTop()); 3657 3658 view->FillEllipse(rect, gradient); 3659 3660 rect.left += leftInset; 3661 rect.top += topInset; 3662 rect.right += rightInset; 3663 rect.bottom += bottomInset; 3664 3665 gradient.MakeEmpty(); 3666 gradient.AddColor(fillShadowColor, 0); 3667 gradient.AddColor(fillLightColor, 255); 3668 gradient.SetStart(rect.LeftTop()); 3669 if (orientation == B_HORIZONTAL) 3670 gradient.SetEnd(rect.LeftBottom()); 3671 else 3672 gradient.SetEnd(rect.RightTop()); 3673 3674 view->FillEllipse(rect, gradient); 3675 } 3676 3677 3678 rgb_color 3679 HaikuControlLook::_EdgeLightColor(const rgb_color& base, float contrast, 3680 float brightness, uint32 flags) 3681 { 3682 rgb_color edgeLightColor; 3683 3684 if ((flags & B_BLEND_FRAME) != 0) { 3685 uint8 alpha = uint8(20 * contrast); 3686 uint8 white = uint8(255 * brightness); 3687 3688 edgeLightColor = (rgb_color){ white, white, white, alpha }; 3689 } else { 3690 // colors 3691 float tintLight = kEdgeBevelLightTint; 3692 3693 if (contrast == 0.0) 3694 tintLight = B_NO_TINT; 3695 else if (contrast != 1.0) 3696 tintLight = B_NO_TINT + (tintLight - B_NO_TINT) * contrast; 3697 3698 edgeLightColor = tint_color(base, tintLight); 3699 3700 if (brightness < 1.0) { 3701 edgeLightColor.red = uint8(edgeLightColor.red * brightness); 3702 edgeLightColor.green = uint8(edgeLightColor.green * brightness); 3703 edgeLightColor.blue = uint8(edgeLightColor.blue * brightness); 3704 } 3705 } 3706 3707 return edgeLightColor; 3708 } 3709 3710 3711 rgb_color 3712 HaikuControlLook::_EdgeShadowColor(const rgb_color& base, float contrast, 3713 float brightness, uint32 flags) 3714 { 3715 rgb_color edgeShadowColor; 3716 3717 if ((flags & B_BLEND_FRAME) != 0) { 3718 uint8 alpha = uint8(20 * contrast); 3719 edgeShadowColor = (rgb_color){ 0, 0, 0, alpha }; 3720 } else { 3721 float tintShadow = kEdgeBevelShadowTint; 3722 3723 if (contrast == 0.0) 3724 tintShadow = B_NO_TINT; 3725 else if (contrast != 1.0) 3726 tintShadow = B_NO_TINT + (tintShadow - B_NO_TINT) * contrast; 3727 3728 edgeShadowColor = tint_color(base, tintShadow); 3729 3730 if (brightness < 1.0) { 3731 edgeShadowColor.red = uint8(edgeShadowColor.red * brightness); 3732 edgeShadowColor.green = uint8(edgeShadowColor.green * brightness); 3733 edgeShadowColor.blue = uint8(edgeShadowColor.blue * brightness); 3734 } 3735 } 3736 3737 return edgeShadowColor; 3738 } 3739 3740 3741 rgb_color 3742 HaikuControlLook::_FrameLightColor(const rgb_color& base, uint32 flags) 3743 { 3744 if ((flags & B_FOCUSED) != 0) 3745 return ui_color(B_KEYBOARD_NAVIGATION_COLOR); 3746 3747 if ((flags & B_ACTIVATED) != 0) 3748 return _FrameShadowColor(base, flags & ~B_ACTIVATED); 3749 3750 rgb_color frameLightColor; 3751 3752 if ((flags & B_DISABLED) != 0) { 3753 // TODO: B_BLEND_FRAME 3754 frameLightColor = tint_color(base, 1.145); 3755 3756 if ((flags & B_DEFAULT_BUTTON) != 0) 3757 frameLightColor = tint_color(frameLightColor, 1.14); 3758 } else { 3759 if ((flags & B_BLEND_FRAME) != 0) 3760 frameLightColor = (rgb_color){ 0, 0, 0, 75 }; 3761 else 3762 frameLightColor = tint_color(base, 1.33); 3763 3764 if ((flags & B_DEFAULT_BUTTON) != 0) 3765 frameLightColor = tint_color(frameLightColor, 1.35); 3766 } 3767 3768 return frameLightColor; 3769 } 3770 3771 3772 rgb_color 3773 HaikuControlLook::_FrameShadowColor(const rgb_color& base, uint32 flags) 3774 { 3775 if ((flags & B_FOCUSED) != 0) 3776 return ui_color(B_KEYBOARD_NAVIGATION_COLOR); 3777 3778 if ((flags & B_ACTIVATED) != 0) 3779 return _FrameLightColor(base, flags & ~B_ACTIVATED); 3780 3781 rgb_color frameShadowColor; 3782 3783 if ((flags & B_DISABLED) != 0) { 3784 // TODO: B_BLEND_FRAME 3785 frameShadowColor = tint_color(base, 1.24); 3786 3787 if ((flags & B_DEFAULT_BUTTON) != 0) { 3788 frameShadowColor = tint_color(base, 1.145); 3789 frameShadowColor = tint_color(frameShadowColor, 1.12); 3790 } 3791 } else { 3792 if ((flags & B_DEFAULT_BUTTON) != 0) { 3793 if ((flags & B_BLEND_FRAME) != 0) 3794 frameShadowColor = (rgb_color){ 0, 0, 0, 75 }; 3795 else 3796 frameShadowColor = tint_color(base, 1.33); 3797 3798 frameShadowColor = tint_color(frameShadowColor, 1.5); 3799 } else { 3800 if ((flags & B_BLEND_FRAME) != 0) 3801 frameShadowColor = (rgb_color){ 0, 0, 0, 95 }; 3802 else 3803 frameShadowColor = tint_color(base, 1.47); 3804 } 3805 } 3806 3807 return frameShadowColor; 3808 } 3809 3810 3811 rgb_color 3812 HaikuControlLook::_BevelLightColor(const rgb_color& base, uint32 flags) 3813 { 3814 rgb_color bevelLightColor = tint_color(base, 0.2); 3815 3816 if ((flags & B_DISABLED) != 0) 3817 bevelLightColor = tint_color(base, B_LIGHTEN_1_TINT); 3818 3819 if ((flags & B_ACTIVATED) != 0) 3820 bevelLightColor = tint_color(base, B_DARKEN_1_TINT); 3821 3822 return bevelLightColor; 3823 } 3824 3825 3826 rgb_color 3827 HaikuControlLook::_BevelShadowColor(const rgb_color& base, uint32 flags) 3828 { 3829 rgb_color bevelShadowColor = tint_color(base, 1.08); 3830 3831 if ((flags & B_DISABLED) != 0) 3832 bevelShadowColor = base; 3833 3834 if ((flags & B_ACTIVATED) != 0) 3835 bevelShadowColor = tint_color(base, B_DARKEN_1_TINT); 3836 3837 return bevelShadowColor; 3838 } 3839 3840 3841 void 3842 HaikuControlLook::_FillGradient(BView* view, const BRect& rect, 3843 const rgb_color& base, float topTint, float bottomTint, 3844 orientation orientation) 3845 { 3846 BGradientLinear gradient; 3847 _MakeGradient(gradient, rect, base, topTint, bottomTint, orientation); 3848 view->FillRect(rect, gradient); 3849 } 3850 3851 3852 void 3853 HaikuControlLook::_FillGlossyGradient(BView* view, const BRect& rect, 3854 const rgb_color& base, float topTint, float middle1Tint, 3855 float middle2Tint, float bottomTint, orientation orientation) 3856 { 3857 BGradientLinear gradient; 3858 _MakeGlossyGradient(gradient, rect, base, topTint, middle1Tint, 3859 middle2Tint, bottomTint, orientation); 3860 view->FillRect(rect, gradient); 3861 } 3862 3863 3864 void 3865 HaikuControlLook::_MakeGradient(BGradientLinear& gradient, const BRect& rect, 3866 const rgb_color& base, float topTint, float bottomTint, 3867 orientation orientation) const 3868 { 3869 gradient.AddColor(tint_color(base, topTint), 0); 3870 gradient.AddColor(tint_color(base, bottomTint), 255); 3871 gradient.SetStart(rect.LeftTop()); 3872 if (orientation == B_HORIZONTAL) 3873 gradient.SetEnd(rect.LeftBottom()); 3874 else 3875 gradient.SetEnd(rect.RightTop()); 3876 } 3877 3878 3879 void 3880 HaikuControlLook::_MakeGlossyGradient(BGradientLinear& gradient, const BRect& rect, 3881 const rgb_color& base, float topTint, float middle1Tint, 3882 float middle2Tint, float bottomTint, 3883 orientation orientation) const 3884 { 3885 gradient.AddColor(tint_color(base, topTint), 0); 3886 gradient.AddColor(tint_color(base, middle1Tint), 132); 3887 gradient.AddColor(tint_color(base, middle2Tint), 136); 3888 gradient.AddColor(tint_color(base, bottomTint), 255); 3889 gradient.SetStart(rect.LeftTop()); 3890 if (orientation == B_HORIZONTAL) 3891 gradient.SetEnd(rect.LeftBottom()); 3892 else 3893 gradient.SetEnd(rect.RightTop()); 3894 } 3895 3896 3897 void 3898 HaikuControlLook::_MakeButtonGradient(BGradientLinear& gradient, BRect& rect, 3899 const rgb_color& base, uint32 flags, orientation orientation) const 3900 { 3901 float topTint = 0.49; 3902 float middleTint1 = 0.62; 3903 float middleTint2 = 0.76; 3904 float bottomTint = 0.90; 3905 3906 if ((flags & B_ACTIVATED) != 0) { 3907 topTint = 1.11; 3908 bottomTint = 1.08; 3909 } 3910 3911 if ((flags & B_DISABLED) != 0) { 3912 topTint = (topTint + B_NO_TINT) / 2; 3913 middleTint1 = (middleTint1 + B_NO_TINT) / 2; 3914 middleTint2 = (middleTint2 + B_NO_TINT) / 2; 3915 bottomTint = (bottomTint + B_NO_TINT) / 2; 3916 } else if ((flags & B_HOVER) != 0) { 3917 topTint *= kHoverTintFactor; 3918 middleTint1 *= kHoverTintFactor; 3919 middleTint2 *= kHoverTintFactor; 3920 bottomTint *= kHoverTintFactor; 3921 } 3922 3923 if ((flags & B_ACTIVATED) != 0) { 3924 _MakeGradient(gradient, rect, base, topTint, bottomTint, orientation); 3925 } else { 3926 _MakeGlossyGradient(gradient, rect, base, topTint, middleTint1, 3927 middleTint2, bottomTint, orientation); 3928 } 3929 } 3930 3931 3932 bool 3933 HaikuControlLook::_RadioButtonAndCheckBoxMarkColor(const rgb_color& base, 3934 rgb_color& color, uint32 flags) const 3935 { 3936 if ((flags & (B_ACTIVATED | B_PARTIALLY_ACTIVATED | B_CLICKED)) == 0) { 3937 // no mark to be drawn at all 3938 return false; 3939 } 3940 3941 color = ui_color(B_CONTROL_MARK_COLOR); 3942 3943 float mix = 1.0; 3944 3945 if ((flags & B_DISABLED) != 0) { 3946 // activated, but disabled 3947 mix = 0.4; 3948 } else if ((flags & B_CLICKED) != 0) { 3949 if ((flags & B_ACTIVATED) != 0) { 3950 // losing activation 3951 mix = 0.7; 3952 } else { 3953 // becoming activated (or losing partial activation) 3954 mix = 0.3; 3955 } 3956 } else if ((flags & B_PARTIALLY_ACTIVATED) != 0) { 3957 // partially activated 3958 mix = 0.5; 3959 } else { 3960 // simply activated 3961 } 3962 3963 color.red = uint8(color.red * mix + base.red * (1.0 - mix)); 3964 color.green = uint8(color.green * mix + base.green * (1.0 - mix)); 3965 color.blue = uint8(color.blue * mix + base.blue * (1.0 - mix)); 3966 3967 return true; 3968 } 3969 3970 } // namespace BPrivate 3971