1 /* 2 * Copyright 2001-2008, Haiku. 3 * Distributed under the terms of the MIT License. 4 * 5 * Authors: 6 * Erik Jaesler (erik@cgsoftware.com) 7 * Axel Dörfler, axeld@pinc-software.de 8 */ 9 10 //! BAlert displays a modal alert window. 11 12 #include <new> 13 #include <stdio.h> 14 #include <string.h> 15 16 #include <Alert.h> 17 #include <Autolock.h> 18 #include <Beep.h> 19 #include <Bitmap.h> 20 #include <Button.h> 21 #include <File.h> 22 #include <FindDirectory.h> 23 #include <Font.h> 24 #include <IconUtils.h> 25 #include <Invoker.h> 26 #include <Looper.h> 27 #include <Message.h> 28 #include <MessageFilter.h> 29 #include <Path.h> 30 #include <Resources.h> 31 #include <Screen.h> 32 #include <TextView.h> 33 #include <View.h> 34 35 #include <binary_compatibility/Interface.h> 36 37 38 //#define DEBUG_ALERT 39 #ifdef DEBUG_ALERT 40 # define FTRACE(x) fprintf(x) 41 #else 42 # define FTRACE(x) /* nothing */ 43 #endif 44 45 // Default size of the Alert window. 46 #define DEFAULT_RECT BRect(0, 0, 310, 75) 47 #define max(LHS, RHS) ((LHS) > (RHS) ? (LHS) : (RHS)) 48 49 static const unsigned int kAlertButtonMsg = 'ALTB'; 50 static const int kSemTimeOut = 50000; 51 52 static const int kLeftOffset = 10; 53 static const int kTopOffset = 6; 54 static const int kBottomOffset = 8; 55 static const int kRightOffset = 8; 56 57 static const int kButtonSpacing = 6; 58 static const int kButtonOffsetSpacing = 62; 59 static const int kButtonUsualWidth = 75; 60 61 static const int kWindowIconOffset = 27; 62 static const int kWindowMinOffset = 12; 63 static const int kWindowMinWidth = 310; 64 static const int kWindowOffsetMinWidth = 335; 65 66 static const int kIconStripeWidth = 30; 67 68 static const int kTextButtonOffset = 10; 69 70 static inline int32 71 icon_layout_scale() 72 { 73 #ifdef __HAIKU__ 74 return max_c(1, ((int32)be_plain_font->Size() + 15) / 16); 75 #endif 76 return 1; 77 } 78 79 80 class TAlertView : public BView { 81 public: 82 TAlertView(BRect frame); 83 TAlertView(BMessage* archive); 84 ~TAlertView(); 85 86 static TAlertView* Instantiate(BMessage* archive); 87 virtual status_t Archive(BMessage* archive, bool deep = true) const; 88 89 virtual void Draw(BRect updateRect); 90 91 // These functions (or something analogous) are missing from libbe.so's 92 // dump. I can only assume that the bitmap is a public var in the 93 // original implementation -- or BAlert is a friend of TAlertView. 94 // Neither one is necessary, since I can just add these. 95 void SetBitmap(BBitmap* Icon) { fIconBitmap = Icon; } 96 BBitmap* Bitmap() { return fIconBitmap; } 97 98 private: 99 BBitmap* fIconBitmap; 100 }; 101 102 class _BAlertFilter_ : public BMessageFilter { 103 public: 104 _BAlertFilter_(BAlert* Alert); 105 ~_BAlertFilter_(); 106 107 virtual filter_result Filter(BMessage* msg, BHandler** target); 108 109 private: 110 BAlert* fAlert; 111 }; 112 113 114 // #pragma mark - BAlert 115 116 117 BAlert::BAlert(const char *title, const char *text, const char *button1, 118 const char *button2, const char *button3, button_width width, 119 alert_type type) 120 : BWindow(DEFAULT_RECT, title, B_MODAL_WINDOW, 121 B_NOT_CLOSABLE | B_NOT_RESIZABLE | B_ASYNCHRONOUS_CONTROLS) 122 { 123 _InitObject(text, button1, button2, button3, width, B_EVEN_SPACING, type); 124 } 125 126 127 BAlert::BAlert(const char *title, const char *text, const char *button1, 128 const char *button2, const char *button3, button_width width, 129 button_spacing spacing, alert_type type) 130 : BWindow(DEFAULT_RECT, title, B_MODAL_WINDOW, 131 B_NOT_CLOSABLE | B_NOT_RESIZABLE | B_ASYNCHRONOUS_CONTROLS) 132 { 133 _InitObject(text, button1, button2, button3, width, spacing, type); 134 } 135 136 137 BAlert::~BAlert() 138 { 139 // Probably not necessary, but it makes me feel better. 140 if (fAlertSem >= B_OK) 141 delete_sem(fAlertSem); 142 } 143 144 145 BAlert::BAlert(BMessage* data) 146 : BWindow(data) 147 { 148 fInvoker = NULL; 149 fAlertSem = -1; 150 fAlertValue = -1; 151 152 fTextView = (BTextView*)FindView("_tv_"); 153 154 fButtons[0] = (BButton*)FindView("_b0_"); 155 fButtons[1] = (BButton*)FindView("_b1_"); 156 fButtons[2] = (BButton*)FindView("_b2_"); 157 158 if (fButtons[2]) 159 SetDefaultButton(fButtons[2]); 160 else if (fButtons[1]) 161 SetDefaultButton(fButtons[1]); 162 else if (fButtons[0]) 163 SetDefaultButton(fButtons[0]); 164 165 TAlertView* view = (TAlertView*)FindView("_master_"); 166 if (view) 167 view->SetBitmap(_InitIcon()); 168 169 // Get keys 170 char key; 171 for (int32 i = 0; i < 3; ++i) { 172 if (data->FindInt8("_but_key", i, (int8*)&key) == B_OK) 173 fKeys[i] = key; 174 } 175 176 int32 temp; 177 // Get alert type 178 if (data->FindInt32("_atype", &temp) == B_OK) 179 fMsgType = (alert_type)temp; 180 181 // Get button width 182 if (data->FindInt32("_but_width", &temp) == B_OK) 183 fButtonWidth = (button_width)temp; 184 185 AddCommonFilter(new(std::nothrow) _BAlertFilter_(this)); 186 } 187 188 189 BArchivable* 190 BAlert::Instantiate(BMessage* data) 191 { 192 if (!validate_instantiation(data, "BAlert")) 193 return NULL; 194 195 return new(std::nothrow) BAlert(data); 196 } 197 198 199 status_t 200 BAlert::Archive(BMessage* data, bool deep) const 201 { 202 status_t ret = BWindow::Archive(data, deep); 203 204 // Stow the text 205 if (ret == B_OK) 206 ret = data->AddString("_text", fTextView->Text()); 207 208 // Stow the alert type 209 if (ret == B_OK) 210 ret = data->AddInt32("_atype", fMsgType); 211 212 // Stow the button width 213 if (ret == B_OK) 214 ret = data->AddInt32("_but_width", fButtonWidth); 215 216 // Stow the shortcut keys 217 if (fKeys[0] || fKeys[1] || fKeys[2]) { 218 // If we have any to save, we must save something for everyone so it 219 // doesn't get confusing on the unarchive. 220 if (ret == B_OK) 221 ret = data->AddInt8("_but_key", fKeys[0]); 222 if (ret == B_OK) 223 ret = data->AddInt8("_but_key", fKeys[1]); 224 if (ret == B_OK) 225 ret = data->AddInt8("_but_key", fKeys[2]); 226 } 227 228 return ret; 229 } 230 231 232 void 233 BAlert::SetShortcut(int32 index, char key) 234 { 235 if (index >= 0 && index < 3) 236 fKeys[index] = key; 237 } 238 239 240 char 241 BAlert::Shortcut(int32 index) const 242 { 243 if (index >= 0 && index < 3) 244 return fKeys[index]; 245 246 return 0; 247 } 248 249 250 int32 251 BAlert::Go() 252 { 253 fAlertSem = create_sem(0, "AlertSem"); 254 if (fAlertSem < B_OK) { 255 Quit(); 256 return -1; 257 } 258 259 // Get the originating window, if it exists 260 BWindow* window = 261 dynamic_cast<BWindow*>(BLooper::LooperForThread(find_thread(NULL))); 262 263 Show(); 264 265 // Heavily modified from TextEntryAlert code; the original didn't let the 266 // blocked window ever draw. 267 if (window) { 268 status_t err; 269 for (;;) { 270 do { 271 err = acquire_sem_etc(fAlertSem, 1, B_RELATIVE_TIMEOUT, 272 kSemTimeOut); 273 // We've (probably) had our time slice taken away from us 274 } while (err == B_INTERRUPTED); 275 276 if (err == B_BAD_SEM_ID) { 277 // Semaphore was finally nuked in MessageReceived 278 break; 279 } 280 window->UpdateIfNeeded(); 281 } 282 } else { 283 // No window to update, so just hang out until we're done. 284 while (acquire_sem(fAlertSem) == B_INTERRUPTED) { 285 } 286 } 287 288 // Have to cache the value since we delete on Quit() 289 int32 value = fAlertValue; 290 if (Lock()) 291 Quit(); 292 293 return value; 294 } 295 296 297 status_t 298 BAlert::Go(BInvoker* invoker) 299 { 300 fInvoker = invoker; 301 Show(); 302 return B_OK; 303 } 304 305 306 void 307 BAlert::MessageReceived(BMessage* msg) 308 { 309 if (msg->what != kAlertButtonMsg) 310 return BWindow::MessageReceived(msg); 311 312 int32 which; 313 if (msg->FindInt32("which", &which) == B_OK) { 314 if (fAlertSem < B_OK) { 315 // Semaphore hasn't been created; we're running asynchronous 316 if (fInvoker) { 317 BMessage* out = fInvoker->Message(); 318 if (out && (out->ReplaceInt32("which", which) == B_OK 319 || out->AddInt32("which", which) == B_OK)) 320 fInvoker->Invoke(); 321 } 322 PostMessage(B_QUIT_REQUESTED); 323 } else { 324 // Created semaphore means were running synchronously 325 fAlertValue = which; 326 327 // TextAlertVar does release_sem() below, and then sets the 328 // member var. That doesn't make much sense to me, since we 329 // want to be able to clean up at some point. Better to just 330 // nuke the semaphore now; we don't need it any more and this 331 // lets synchronous Go() continue just as well. 332 delete_sem(fAlertSem); 333 fAlertSem = -1; 334 } 335 } 336 } 337 338 339 void 340 BAlert::FrameResized(float newWidth, float newHeight) 341 { 342 BWindow::FrameResized(newWidth, newHeight); 343 } 344 345 346 BButton* 347 BAlert::ButtonAt(int32 index) const 348 { 349 if (index >= 0 && index < 3) 350 return fButtons[index]; 351 352 return NULL; 353 } 354 355 356 BTextView* 357 BAlert::TextView() const 358 { 359 return fTextView; 360 } 361 362 363 BHandler* 364 BAlert::ResolveSpecifier(BMessage* msg, int32 index, 365 BMessage* specifier, int32 form, const char* property) 366 { 367 return BWindow::ResolveSpecifier(msg, index, specifier, form, property); 368 } 369 370 371 status_t 372 BAlert::GetSupportedSuites(BMessage* data) 373 { 374 return BWindow::GetSupportedSuites(data); 375 } 376 377 378 void 379 BAlert::DispatchMessage(BMessage* msg, BHandler* handler) 380 { 381 BWindow::DispatchMessage(msg, handler); 382 } 383 384 385 void 386 BAlert::Quit() 387 { 388 BWindow::Quit(); 389 } 390 391 392 bool 393 BAlert::QuitRequested() 394 { 395 return BWindow::QuitRequested(); 396 } 397 398 399 BPoint 400 BAlert::AlertPosition(float width, float height) 401 { 402 BPoint result(100, 100); 403 404 BWindow* window = 405 dynamic_cast<BWindow*>(BLooper::LooperForThread(find_thread(NULL))); 406 407 BScreen screen(window); 408 BRect screenFrame(0, 0, 640, 480); 409 if (screen.IsValid()) 410 screenFrame = screen.Frame(); 411 412 // Horizontally, we're smack in the middle 413 result.x = screenFrame.left + (screenFrame.Width() / 2.0) - (width / 2.0); 414 415 // This is probably sooo wrong, but it looks right on 1024 x 768 416 result.y = screenFrame.top + (screenFrame.Height() / 4.0) - ceil(height / 3.0); 417 418 return result; 419 } 420 421 422 status_t 423 BAlert::Perform(perform_code code, void* _data) 424 { 425 switch (code) { 426 case PERFORM_CODE_SET_LAYOUT: 427 { 428 perform_data_set_layout* data = (perform_data_set_layout*)_data; 429 BAlert::SetLayout(data->layout); 430 return B_OK; 431 } 432 } 433 434 return BWindow::Perform(code, _data); 435 } 436 437 438 void BAlert::_ReservedAlert1() {} 439 void BAlert::_ReservedAlert2() {} 440 void BAlert::_ReservedAlert3() {} 441 442 443 void 444 BAlert::_InitObject(const char* text, const char* button0, const char* button1, 445 const char* button2, button_width buttonWidth, button_spacing spacing, 446 alert_type type) 447 { 448 fInvoker = NULL; 449 fAlertSem = -1; 450 fAlertValue = -1; 451 fButtons[0] = fButtons[1] = fButtons[2] = NULL; 452 fTextView = NULL; 453 fKeys[0] = fKeys[1] = fKeys[2] = 0; 454 fMsgType = type; 455 fButtonWidth = buttonWidth; 456 457 // Set up the "_master_" view 458 TAlertView* view = new(std::nothrow) TAlertView(Bounds()); 459 if (view == NULL) 460 return; 461 462 AddChild(view); 463 view->SetBitmap(_InitIcon()); 464 465 // Must have at least one button 466 if (button0 == NULL) { 467 debugger("BAlerts must have at least one button."); 468 button0 = ""; 469 } 470 471 // Set up the buttons 472 473 int32 buttonCount = 1; 474 view->AddChild(fButtons[0] = _CreateButton(0, button0)); 475 476 if (button1 != NULL) { 477 view->AddChild(fButtons[1] = _CreateButton(1, button1)); 478 buttonCount++; 479 } 480 481 if (button2 != NULL) { 482 view->AddChild(fButtons[2] = _CreateButton(2, button2)); 483 buttonCount++; 484 } 485 486 // Find the widest button only if the widest value needs to be known. 487 488 if (fButtonWidth == B_WIDTH_FROM_WIDEST) { 489 float maxWidth = 0; 490 for (int i = 0; i < buttonCount; i++) { 491 float width = fButtons[i]->Bounds().Width(); 492 if (width > maxWidth) 493 maxWidth = width; 494 } 495 496 // resize buttons 497 for (int i = 0; i < buttonCount; i++) { 498 fButtons[i]->ResizeTo(maxWidth, fButtons[i]->Bounds().Height()); 499 } 500 } 501 502 float defaultButtonFrameWidth = -fButtons[buttonCount - 1]->Bounds().Width() / 2.0f; 503 SetDefaultButton(fButtons[buttonCount - 1]); 504 defaultButtonFrameWidth += fButtons[buttonCount - 1]->Bounds().Width() / 2.0f; 505 506 // Layout buttons 507 508 float fontFactor = be_plain_font->Size() / 11.0f; 509 510 for (int i = buttonCount - 1; i >= 0; --i) { 511 float x = -fButtons[i]->Bounds().Width(); 512 if (i + 1 == buttonCount) 513 x += Bounds().right - kRightOffset + defaultButtonFrameWidth; 514 else 515 x += fButtons[i + 1]->Frame().left - kButtonSpacing; 516 517 if (buttonCount > 1 && i == 0 && spacing == B_OFFSET_SPACING) 518 x -= kButtonOffsetSpacing * fontFactor; 519 520 fButtons[i]->MoveTo(x, fButtons[i]->Frame().top); 521 } 522 523 // Adjust the window's width, if necessary 524 525 int32 iconLayoutScale = icon_layout_scale(); 526 float totalWidth = kRightOffset + fButtons[buttonCount - 1]->Frame().right 527 - defaultButtonFrameWidth - fButtons[0]->Frame().left; 528 if (view->Bitmap()) { 529 totalWidth += (kIconStripeWidth + kWindowIconOffset) * iconLayoutScale; 530 } else 531 totalWidth += kWindowMinOffset; 532 533 float width = (spacing == B_OFFSET_SPACING 534 ? kWindowOffsetMinWidth : kWindowMinWidth) * fontFactor; 535 536 ResizeTo(max_c(totalWidth, width), Bounds().Height()); 537 538 // Set up the text view 539 540 BRect textViewRect(kLeftOffset, kTopOffset, 541 Bounds().right - kRightOffset, 542 fButtons[0]->Frame().top - kTextButtonOffset); 543 if (view->Bitmap()) 544 textViewRect.left = (kWindowIconOffset 545 + kIconStripeWidth) * iconLayoutScale - 2; 546 547 fTextView = new(std::nothrow) BTextView(textViewRect, "_tv_", 548 textViewRect.OffsetByCopy(B_ORIGIN), 549 B_FOLLOW_LEFT | B_FOLLOW_TOP, B_WILL_DRAW); 550 if (fTextView == NULL) 551 return; 552 553 fTextView->SetViewColor(ui_color(B_PANEL_BACKGROUND_COLOR)); 554 rgb_color textColor = ui_color(B_PANEL_TEXT_COLOR); 555 fTextView->SetFontAndColor(be_plain_font, B_FONT_ALL, &textColor); 556 fTextView->SetText(text, strlen(text)); 557 fTextView->MakeEditable(false); 558 fTextView->MakeSelectable(false); 559 fTextView->SetWordWrap(true); 560 view->AddChild(fTextView); 561 562 // Now resize the TextView vertically so that all the text is visible 563 float textHeight = fTextView->TextHeight(0, fTextView->CountLines()); 564 textViewRect.OffsetTo(0, 0); 565 textHeight -= textViewRect.Height(); 566 ResizeBy(0, textHeight); 567 fTextView->ResizeBy(0, textHeight); 568 textViewRect.bottom += textHeight; 569 fTextView->SetTextRect(textViewRect); 570 571 AddCommonFilter(new(std::nothrow) _BAlertFilter_(this)); 572 573 MoveTo(AlertPosition(Frame().Width(), Frame().Height())); 574 } 575 576 577 BBitmap* 578 BAlert::_InitIcon() 579 { 580 // Save the desired alert type and set it to "empty" until 581 // loading the icon was successful 582 alert_type alertType = fMsgType; 583 fMsgType = B_EMPTY_ALERT; 584 585 // After a bit of a search, I found the icons in app_server. =P 586 BBitmap* icon = NULL; 587 BPath path; 588 status_t status = find_directory(B_BEOS_SERVERS_DIRECTORY, &path); 589 if (status < B_OK) { 590 FTRACE((stderr, "BAlert::_InitIcon() - find_directory failed: %s\n", 591 strerror(status))); 592 return NULL; 593 } 594 595 path.Append("app_server"); 596 BFile file; 597 status = file.SetTo(path.Path(), B_READ_ONLY); 598 if (status < B_OK) { 599 FTRACE((stderr, "BAlert::_InitIcon() - BFile init failed: %s\n", 600 strerror(status))); 601 return NULL; 602 } 603 604 BResources resources; 605 status = resources.SetTo(&file); 606 if (status < B_OK) { 607 FTRACE((stderr, "BAlert::_InitIcon() - BResources init failed: %s\n", 608 strerror(status))); 609 return NULL; 610 } 611 612 // Which icon are we trying to load? 613 const char* iconName = ""; // Don't want any seg faults 614 switch (alertType) { 615 case B_INFO_ALERT: 616 iconName = "info"; 617 break; 618 case B_IDEA_ALERT: 619 iconName = "idea"; 620 break; 621 case B_WARNING_ALERT: 622 iconName = "warn"; 623 break; 624 case B_STOP_ALERT: 625 iconName = "stop"; 626 break; 627 628 default: 629 // Alert type is either invalid or B_EMPTY_ALERT; 630 // either way, we're not going to load an icon 631 return NULL; 632 } 633 634 int32 iconSize = 32 * icon_layout_scale(); 635 // Allocate the icon bitmap 636 icon = new(std::nothrow) BBitmap(BRect(0, 0, iconSize - 1, iconSize - 1), 637 0, B_RGBA32); 638 if (icon == NULL || icon->InitCheck() < B_OK) { 639 FTRACE((stderr, "BAlert::_InitIcon() - No memory for bitmap\n")); 640 delete icon; 641 return NULL; 642 } 643 644 // Load the raw icon data 645 size_t size = 0; 646 const uint8* rawIcon; 647 648 #ifdef __HAIKU__ 649 // Try to load vector icon 650 rawIcon = (const uint8*)resources.LoadResource(B_VECTOR_ICON_TYPE, 651 iconName, &size); 652 if (rawIcon != NULL 653 && BIconUtils::GetVectorIcon(rawIcon, size, icon) == B_OK) { 654 // We have an icon, restore the saved alert type 655 fMsgType = alertType; 656 return icon; 657 } 658 #endif 659 660 // Fall back to bitmap icon 661 rawIcon = (const uint8*)resources.LoadResource(B_LARGE_ICON_TYPE, 662 iconName, &size); 663 if (rawIcon == NULL) { 664 FTRACE((stderr, "BAlert::_InitIcon() - Icon resource not found\n")); 665 delete icon; 666 return NULL; 667 } 668 669 // Handle color space conversion 670 #ifdef __HAIKU__ 671 if (icon->ColorSpace() != B_CMAP8) { 672 BIconUtils::ConvertFromCMAP8(rawIcon, iconSize, iconSize, 673 iconSize, icon); 674 } 675 #else 676 icon->SetBits(rawIcon, iconSize, 0, B_CMAP8); 677 #endif 678 679 // We have an icon, restore the saved alert type 680 fMsgType = alertType; 681 682 return icon; 683 } 684 685 686 BButton* 687 BAlert::_CreateButton(int32 which, const char* label) 688 { 689 BMessage* message = new BMessage(kAlertButtonMsg); 690 if (message == NULL) 691 return NULL; 692 693 message->AddInt32("which", which); 694 695 BRect rect; 696 rect.top = Bounds().bottom - kBottomOffset; 697 rect.bottom = rect.top; 698 699 char name[32]; 700 snprintf(name, sizeof(name), "_b%" B_PRId32 "_", which); 701 702 BButton* button = new(std::nothrow) BButton(rect, name, label, message, 703 B_FOLLOW_RIGHT | B_FOLLOW_BOTTOM); 704 if (button == NULL) 705 return NULL; 706 707 float width, height; 708 button->GetPreferredSize(&width, &height); 709 710 if (fButtonWidth == B_WIDTH_AS_USUAL) { 711 float fontFactor = be_plain_font->Size() / 11.0f; 712 width = max_c(width, kButtonUsualWidth * fontFactor); 713 } 714 715 button->ResizeTo(width, height); 716 button->MoveBy(0.0f, -height); 717 return button; 718 } 719 720 721 // #pragma mark - TAlertView 722 723 724 TAlertView::TAlertView(BRect frame) 725 : BView(frame, "TAlertView", B_FOLLOW_ALL_SIDES, B_WILL_DRAW), 726 fIconBitmap(NULL) 727 { 728 SetViewColor(ui_color(B_PANEL_BACKGROUND_COLOR)); 729 } 730 731 732 TAlertView::TAlertView(BMessage* archive) 733 : BView(archive), 734 fIconBitmap(NULL) 735 { 736 } 737 738 739 TAlertView::~TAlertView() 740 { 741 delete fIconBitmap; 742 } 743 744 745 TAlertView* 746 TAlertView::Instantiate(BMessage* archive) 747 { 748 if (!validate_instantiation(archive, "TAlertView")) 749 return NULL; 750 751 return new(std::nothrow) TAlertView(archive); 752 } 753 754 755 status_t 756 TAlertView::Archive(BMessage* archive, bool deep) const 757 { 758 return BView::Archive(archive, deep); 759 } 760 761 762 void 763 TAlertView::Draw(BRect updateRect) 764 { 765 if (!fIconBitmap) 766 return; 767 768 // Here's the fun stuff 769 BRect stripeRect = Bounds(); 770 int32 iconLayoutScale = icon_layout_scale(); 771 stripeRect.right = kIconStripeWidth * iconLayoutScale; 772 SetHighColor(tint_color(ViewColor(), B_DARKEN_1_TINT)); 773 FillRect(stripeRect); 774 775 SetDrawingMode(B_OP_ALPHA); 776 SetBlendingMode(B_PIXEL_ALPHA, B_ALPHA_OVERLAY); 777 DrawBitmapAsync(fIconBitmap, BPoint(18 * iconLayoutScale, 778 6 * iconLayoutScale)); 779 780 } 781 782 783 //------------------------------------------------------------------------------ 784 // #pragma mark - _BAlertFilter_ 785 786 787 _BAlertFilter_::_BAlertFilter_(BAlert* alert) 788 : BMessageFilter(B_KEY_DOWN), 789 fAlert(alert) 790 { 791 } 792 793 794 _BAlertFilter_::~_BAlertFilter_() 795 { 796 } 797 798 799 filter_result 800 _BAlertFilter_::Filter(BMessage* msg, BHandler** target) 801 { 802 if (msg->what == B_KEY_DOWN) { 803 char byte; 804 if (msg->FindInt8("byte", (int8*)&byte) == B_OK) { 805 for (int i = 0; i < 3; ++i) { 806 if (byte == fAlert->Shortcut(i) && fAlert->ButtonAt(i)) { 807 char space = ' '; 808 fAlert->ButtonAt(i)->KeyDown(&space, 1); 809 810 return B_SKIP_MESSAGE; 811 } 812 } 813 } 814 } 815 816 return B_DISPATCH_MESSAGE; 817 } 818 819