aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/linden/indra/llui/llcombobox.cpp
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--linden/indra/llui/llcombobox.cpp292
1 files changed, 107 insertions, 185 deletions
diff --git a/linden/indra/llui/llcombobox.cpp b/linden/indra/llui/llcombobox.cpp
index 7c3755a..983dd43 100644
--- a/linden/indra/llui/llcombobox.cpp
+++ b/linden/indra/llui/llcombobox.cpp
@@ -4,6 +4,7 @@
4 * 4 *
5 * Copyright (c) 2001-2007, Linden Research, Inc. 5 * Copyright (c) 2001-2007, Linden Research, Inc.
6 * 6 *
7 * Second Life Viewer Source Code
7 * The source code in this file ("Source Code") is provided by Linden Lab 8 * The source code in this file ("Source Code") is provided by Linden Lab
8 * to you under the terms of the GNU General Public License, version 2.0 9 * to you under the terms of the GNU General Public License, version 2.0
9 * ("GPL"), unless you have obtained a separate licensing agreement 10 * ("GPL"), unless you have obtained a separate licensing agreement
@@ -59,7 +60,7 @@ LLComboBox::LLComboBox( const LLString& name, const LLRect &rect, const LLString
59 ) 60 )
60: LLUICtrl(name, rect, TRUE, commit_callback, callback_userdata, 61: LLUICtrl(name, rect, TRUE, commit_callback, callback_userdata,
61 FOLLOWS_LEFT | FOLLOWS_TOP), 62 FOLLOWS_LEFT | FOLLOWS_TOP),
62 mDrawButton(TRUE), 63 mDrawArrow(TRUE),
63 mTextEntry(NULL), 64 mTextEntry(NULL),
64 mArrowImage(NULL), 65 mArrowImage(NULL),
65 mArrowImageWidth(8), 66 mArrowImageWidth(8),
@@ -67,8 +68,8 @@ LLComboBox::LLComboBox( const LLString& name, const LLRect &rect, const LLString
67 mMaxChars(20), 68 mMaxChars(20),
68 mTextEntryTentative(TRUE), 69 mTextEntryTentative(TRUE),
69 mPrearrangeCallback( NULL ), 70 mPrearrangeCallback( NULL ),
70 mTextEntryCallback( NULL ), 71 mListPosition(BELOW),
71 mListWidth(list_width) 72 mTextEntryCallback( NULL )
72{ 73{
73 // For now, all comboboxes don't take keyboard focus when clicked. 74 // For now, all comboboxes don't take keyboard focus when clicked.
74 // This might change if it is part of a modal dialog. 75 // This might change if it is part of a modal dialog.
@@ -76,7 +77,7 @@ LLComboBox::LLComboBox( const LLString& name, const LLRect &rect, const LLString
76 77
77 // Revert to standard behavior. When this control's parent is hidden, it needs to 78 // Revert to standard behavior. When this control's parent is hidden, it needs to
78 // hide this ctrl--which won't just happen automatically since when LLComboBox is 79 // hide this ctrl--which won't just happen automatically since when LLComboBox is
79 // showing its list, it's also set to TopView. When keyboard focus is cleared all 80 // showing its list, it's also set to TopCtrl. When keyboard focus is cleared all
80 // controls (including this one) know that they are no longer editing. 81 // controls (including this one) know that they are no longer editing.
81 mKeyboardFocusOnClick = TRUE; 82 mKeyboardFocusOnClick = TRUE;
82 83
@@ -87,7 +88,8 @@ LLComboBox::LLComboBox( const LLString& name, const LLRect &rect, const LLString
87 // Text label button 88 // Text label button
88 mButton = new LLSquareButton("comboxbox button", 89 mButton = new LLSquareButton("comboxbox button",
89 r, label, NULL, LLString::null, 90 r, label, NULL, LLString::null,
90 &LLComboBox::onButtonClick, this); 91 NULL, this);
92 mButton->setMouseDownCallback(onButtonDown);
91 mButton->setFont(LLFontGL::sSansSerifSmall); 93 mButton->setFont(LLFontGL::sSansSerifSmall);
92 mButton->setFollows(FOLLOWS_LEFT | FOLLOWS_BOTTOM | FOLLOWS_RIGHT); 94 mButton->setFollows(FOLLOWS_LEFT | FOLLOWS_BOTTOM | FOLLOWS_RIGHT);
93 mButton->setHAlign( LLFontGL::LEFT ); 95 mButton->setHAlign( LLFontGL::LEFT );
@@ -110,6 +112,7 @@ LLComboBox::LLComboBox( const LLString& name, const LLRect &rect, const LLString
110 mList->setVisible(FALSE); 112 mList->setVisible(FALSE);
111 mList->setBgWriteableColor( LLColor4(1,1,1,1) ); 113 mList->setBgWriteableColor( LLColor4(1,1,1,1) );
112 mList->setCommitOnKeyboardMovement(FALSE); 114 mList->setCommitOnKeyboardMovement(FALSE);
115 mList->setFocusChangedCallback(onListFocusChanged);
113 addChild(mList); 116 addChild(mList);
114 117
115 LLRect border_rect(0, mRect.getHeight(), mRect.getWidth(), 0); 118 LLRect border_rect(0, mRect.getHeight(), mRect.getWidth(), 0);
@@ -119,7 +122,7 @@ LLComboBox::LLComboBox( const LLString& name, const LLRect &rect, const LLString
119 122
120 LLUUID arrow_image_id( LLUI::sAssetsGroup->getString("combobox_arrow.tga") ); 123 LLUUID arrow_image_id( LLUI::sAssetsGroup->getString("combobox_arrow.tga") );
121 mArrowImage = LLUI::sImageProvider->getUIImageByID(arrow_image_id); 124 mArrowImage = LLUI::sImageProvider->getUIImageByID(arrow_image_id);
122 mArrowImageWidth = llmax(8,mArrowImage->getWidth()); // In case image hasn't loaded yet 125 mArrowImageWidth = llmax(8,mArrowImage->getWidth(0)); // In case image hasn't loaded yet
123} 126}
124 127
125 128
@@ -219,115 +222,10 @@ LLView* LLComboBox::fromXML(LLXMLNodePtr node, LLView *parent, LLUICtrlFactory *
219 222
220void LLComboBox::setEnabled(BOOL enabled) 223void LLComboBox::setEnabled(BOOL enabled)
221{ 224{
222 LLUICtrl::setEnabled(enabled); 225 LLView::setEnabled(enabled);
223 mButton->setEnabled(enabled); 226 mButton->setEnabled(enabled);
224} 227}
225 228
226// *HACK: these are all hacks to support the fact that the combobox
227// has mouse capture so we can hide the list when we don't handle the
228// mouse up event
229BOOL LLComboBox::handleHover(S32 x, S32 y, MASK mask)
230{
231 if (mList->getVisible())
232 {
233 S32 local_x, local_y;
234 LLView::localPointToOtherView(x, y, &local_x, &local_y, mList);
235 if (mList->pointInView(local_x, local_y))
236 {
237 return mList->handleHover(local_x, local_y, mask);
238 }
239 }
240 return LLUICtrl::handleHover(x, y, mask);
241}
242
243BOOL LLComboBox::handleMouseDown(S32 x, S32 y, MASK mask)
244{
245 if (mList->getVisible())
246 {
247 S32 local_x, local_y;
248 LLView::localPointToOtherView(x, y, &local_x, &local_y, mList);
249 if (mList->pointInView(local_x, local_y))
250 {
251 return mList->handleMouseDown(local_x, local_y, mask);
252 }
253 }
254 BOOL has_focus_now = hasFocus();
255 BOOL handled = LLUICtrl::handleMouseDown(x, y, mask);
256 if (handled && !has_focus_now)
257 {
258 onFocusReceived();
259 }
260
261 return handled;
262}
263
264BOOL LLComboBox::handleRightMouseDown(S32 x, S32 y, MASK mask)
265{
266 if (mList->getVisible())
267 {
268 S32 local_x, local_y;
269 LLView::localPointToOtherView(x, y, &local_x, &local_y, mList);
270 if (mList->pointInView(local_x, local_y))
271 {
272 return mList->handleRightMouseDown(local_x, local_y, mask);
273 }
274 }
275 return LLUICtrl::handleRightMouseDown(x, y, mask);
276}
277
278BOOL LLComboBox::handleRightMouseUp(S32 x, S32 y, MASK mask)
279{
280 if (mList->getVisible())
281 {
282 S32 local_x, local_y;
283 LLView::localPointToOtherView(x, y, &local_x, &local_y, mList);
284 if (mList->pointInView(local_x, local_y))
285 {
286 return mList->handleRightMouseUp(local_x, local_y, mask);
287 }
288 }
289 return LLUICtrl::handleRightMouseUp(x, y, mask);
290}
291
292BOOL LLComboBox::handleDoubleClick(S32 x, S32 y, MASK mask)
293{
294 if (mList->getVisible())
295 {
296 S32 local_x, local_y;
297 LLView::localPointToOtherView(x, y, &local_x, &local_y, mList);
298 if (mList->pointInView(local_x, local_y))
299 {
300 return mList->handleDoubleClick(local_x, local_y, mask);
301 }
302 }
303 return LLUICtrl::handleDoubleClick(x, y, mask);
304}
305
306BOOL LLComboBox::handleMouseUp(S32 x, S32 y, MASK mask)
307{
308 BOOL handled = childrenHandleMouseUp(x, y, mask) != NULL;
309
310 if (!handled && mList->getVisible())
311 {
312 S32 local_x, local_y;
313 LLView::localPointToOtherView(x, y, &local_x, &local_y, mList);
314 if (mList->pointInView(local_x, local_y))
315 {
316 handled = mList->handleMouseUp(local_x, local_y, mask);
317 }
318 }
319
320 if( !handled && gFocusMgr.getMouseCapture() == this )
321 {
322 // Mouse events that we didn't handle cause the list to be hidden.
323 // Eat mouse event, regardless of where on the screen it happens.
324 hideList();
325 handled = TRUE;
326 }
327
328 return handled;
329}
330
331void LLComboBox::clear() 229void LLComboBox::clear()
332{ 230{
333 if (mTextEntry) 231 if (mTextEntry)
@@ -512,12 +410,13 @@ void LLComboBox::onFocusLost()
512 { 410 {
513 mTextEntry->selectAll(); 411 mTextEntry->selectAll();
514 } 412 }
413 LLUICtrl::onFocusLost();
515} 414}
516 415
517void LLComboBox::setButtonVisible(BOOL visible) 416void LLComboBox::setButtonVisible(BOOL visible)
518{ 417{
519 mButton->setVisible(visible); 418 mButton->setVisible(visible);
520 mDrawButton = visible; 419 mDrawArrow = visible;
521 if (mTextEntry) 420 if (mTextEntry)
522 { 421 {
523 LLRect text_entry_rect(0, mRect.getHeight(), mRect.getWidth(), 0); 422 LLRect text_entry_rect(0, mRect.getHeight(), mRect.getWidth(), 0);
@@ -541,15 +440,17 @@ void LLComboBox::draw()
541 // Draw children 440 // Draw children
542 LLUICtrl::draw(); 441 LLUICtrl::draw();
543 442
544 if (mDrawButton) 443 if (mDrawArrow)
545 { 444 {
546 // Paste the graphic on the right edge 445 // Paste the graphic on the right edge
547 if (!mArrowImage.isNull()) 446 if (!mArrowImage.isNull())
548 { 447 {
549 S32 left = mRect.getWidth() - mArrowImageWidth - LLUI::sConfigGroup->getS32("DropShadowButton"); 448 S32 arrow_height = llmin(mRect.getHeight(), mArrowImage->getHeight());
449 S32 arrow_width = llround((F32)mArrowImage->getWidth() * ((F32)arrow_height / (F32)mArrowImage->getHeight()));
450
451 S32 left = mRect.getWidth() - mArrowImage->getWidth() - LLUI::sConfigGroup->getS32("DropShadowButton");
550 452
551 gl_draw_image( left, 0, mArrowImage, 453 gl_draw_scaled_image( left, 0, arrow_width, arrow_height, mArrowImage, LLColor4::white);
552 LLColor4::white);
553 } 454 }
554 } 455 }
555 } 456 }
@@ -595,18 +496,61 @@ void LLComboBox::showList()
595 //HACK: shouldn't have to know about scale here 496 //HACK: shouldn't have to know about scale here
596 mList->arrange( 192, llfloor((F32)window_size.mY / LLUI::sGLScaleFactor.mV[VY]) - 50 ); 497 mList->arrange( 192, llfloor((F32)window_size.mY / LLUI::sGLScaleFactor.mV[VY]) - 50 );
597 498
598 // Move rect so it hangs off the bottom of this view 499 // Make sure that we can see the whole list
500 LLRect root_view_local;
501 LLView* root_view = getRootView();
502 root_view->localRectToOtherView(root_view->getLocalRect(), &root_view_local, this);
503
599 LLRect rect = mList->getRect(); 504 LLRect rect = mList->getRect();
600 505
601 rect.setLeftTopAndSize(0, 0, rect.getWidth(), rect.getHeight() ); 506 if (mListPosition == BELOW)
602 mList->setRect(rect); 507 {
508 if (rect.getHeight() <= -root_view_local.mBottom)
509 {
510 // Move rect so it hangs off the bottom of this view
511 rect.setLeftTopAndSize(0, 0, rect.getWidth(), rect.getHeight() );
512 }
513 else
514 {
515 // stack on top or bottom, depending on which has more room
516 if (-root_view_local.mBottom > root_view_local.mTop - mRect.getHeight())
517 {
518 // Move rect so it hangs off the bottom of this view
519 rect.setLeftTopAndSize(0, 0, rect.getWidth(), llmin(-root_view_local.mBottom, rect.getHeight()));
520 }
521 else
522 {
523 // move rect so it stacks on top of this view (clipped to size of screen)
524 rect.setOriginAndSize(0, mRect.getHeight(), rect.getWidth(), llmin(root_view_local.mTop - mRect.getHeight(), rect.getHeight()));
525 }
526 }
527 }
528 else // ABOVE
529 {
530 if (rect.getHeight() <= root_view_local.mTop - mRect.getHeight())
531 {
532 // move rect so it stacks on top of this view (clipped to size of screen)
533 rect.setOriginAndSize(0, mRect.getHeight(), rect.getWidth(), llmin(root_view_local.mTop - mRect.getHeight(), rect.getHeight()));
534 }
535 else
536 {
537 // stack on top or bottom, depending on which has more room
538 if (-root_view_local.mBottom > root_view_local.mTop - mRect.getHeight())
539 {
540 // Move rect so it hangs off the bottom of this view
541 rect.setLeftTopAndSize(0, 0, rect.getWidth(), llmin(-root_view_local.mBottom, rect.getHeight()));
542 }
543 else
544 {
545 // move rect so it stacks on top of this view (clipped to size of screen)
546 rect.setOriginAndSize(0, mRect.getHeight(), rect.getWidth(), llmin(root_view_local.mTop - mRect.getHeight(), rect.getHeight()));
547 }
548 }
603 549
604 // Make sure that we can see the whole list 550 }
605 LLRect floater_area_screen; 551 mList->setOrigin(rect.mLeft, rect.mBottom);
606 LLRect floater_area_local; 552 mList->reshape(rect.getWidth(), rect.getHeight());
607 gFloaterView->getParent()->localRectToScreen( gFloaterView->getRect(), &floater_area_screen ); 553 mList->translateIntoRect(root_view_local, FALSE);
608 screenRectToLocal( floater_area_screen, &floater_area_local );
609 mList->translateIntoRect( floater_area_local, FALSE );
610 554
611 // Make sure we didn't go off bottom of screen 555 // Make sure we didn't go off bottom of screen
612 S32 x, y; 556 S32 x, y;
@@ -617,7 +561,12 @@ void LLComboBox::showList()
617 mList->translate(0, -y); 561 mList->translate(0, -y);
618 } 562 }
619 563
620 gFocusMgr.setMouseCapture( this, LLComboBox::onMouseCaptureLost ); 564 // pass mouse capture on to list if button is depressed
565 if (mButton->hasMouseCapture())
566 {
567 gFocusMgr.setMouseCapture(mList);
568 }
569
621 // NB: this call will trigger the focuslost callback which will hide the list, so do it first 570 // NB: this call will trigger the focuslost callback which will hide the list, so do it first
622 // before finally showing the list 571 // before finally showing the list
623 572
@@ -627,14 +576,13 @@ void LLComboBox::showList()
627 // so that the callback is not immediately triggered on setFocus() 576 // so that the callback is not immediately triggered on setFocus()
628 mList->selectFirstItem(); 577 mList->selectFirstItem();
629 } 578 }
630 gFocusMgr.setKeyboardFocus(mList, onListFocusLost); 579 mList->setFocus(TRUE);
631 580
632 // Show the list and push the button down 581 // Show the list and push the button down
633 mButton->setToggleState(TRUE); 582 mButton->setToggleState(TRUE);
634 mList->setVisible(TRUE); 583 mList->setVisible(TRUE);
635 584
636 gFocusMgr.setTopView(mList, LLComboBox::onTopViewLost ); 585 gFocusMgr.setTopCtrl(mList);
637
638} 586}
639 587
640void LLComboBox::hideList() 588void LLComboBox::hideList()
@@ -643,37 +591,21 @@ void LLComboBox::hideList()
643 mList->setVisible(FALSE); 591 mList->setVisible(FALSE);
644 mList->highlightNthItem(-1); 592 mList->highlightNthItem(-1);
645 593
646 if( gFocusMgr.getTopView() == mList ) 594 if( gFocusMgr.getTopCtrl() == mList )
647 {
648 gFocusMgr.setTopView(NULL, NULL);
649 }
650
651 if( gFocusMgr.getMouseCapture() == this )
652 { 595 {
653 gFocusMgr.setMouseCapture( NULL, NULL ); 596 gFocusMgr.setTopCtrl(NULL);
654 } 597 }
655 598
656 if( gFocusMgr.getKeyboardFocus() == mList ) 599 //mList->setFocus(FALSE);
657 {
658 if (mAllowTextEntry)
659 {
660 mTextEntry->setFocus(TRUE);
661 }
662 else
663 {
664 setFocus(TRUE);
665 }
666 }
667} 600}
668 601
669 602
670
671//------------------------------------------------------------------ 603//------------------------------------------------------------------
672// static functions 604// static functions
673//------------------------------------------------------------------ 605//------------------------------------------------------------------
674 606
675// static 607// static
676void LLComboBox::onButtonClick(void *userdata) 608void LLComboBox::onButtonDown(void *userdata)
677{ 609{
678 LLComboBox *self = (LLComboBox *)userdata; 610 LLComboBox *self = (LLComboBox *)userdata;
679 611
@@ -720,50 +652,47 @@ void LLComboBox::onItemSelected(LLUICtrl* item, void *userdata)
720 652
721 const LLString& name = self->mList->getSimpleSelectedItem(); 653 const LLString& name = self->mList->getSimpleSelectedItem();
722 654
723 self->hideList();
724
725 S32 cur_id = self->getCurrentIndex(); 655 S32 cur_id = self->getCurrentIndex();
726 if (cur_id != -1) 656 if (cur_id != -1)
727 { 657 {
728 self->setLabel(self->mList->getSimpleSelectedItem()); 658 self->setLabel(name);
729 659
730 if (self->mAllowTextEntry) 660 if (self->mAllowTextEntry)
731 { 661 {
732 self->mTextEntry->setText(name);
733 self->mTextEntry->setTentative(FALSE);
734 gFocusMgr.setKeyboardFocus(self->mTextEntry, NULL); 662 gFocusMgr.setKeyboardFocus(self->mTextEntry, NULL);
735 self->mTextEntry->selectAll(); 663 self->mTextEntry->selectAll();
736 } 664 }
737 else 665 }
738 { 666 else
739 self->mButton->setLabelUnselected( name ); 667 {
740 self->mButton->setLabelSelected( name ); 668 // invalid selection, just restore existing value
741 self->mButton->setDisabledLabel( name ); 669 self->mList->selectSimpleItem(self->mButton->getLabelSelected());
742 self->mButton->setDisabledSelectedLabel( name );
743 }
744 } 670 }
745 self->onCommit(); 671 self->onCommit();
746}
747 672
748// static
749void LLComboBox::onTopViewLost(LLView* old_focus)
750{
751 LLComboBox *self = (LLComboBox *) old_focus->getParent();
752 self->hideList(); 673 self->hideList();
753} 674}
754 675
755
756// static 676// static
757void LLComboBox::onMouseCaptureLost(LLMouseHandler*) 677void LLComboBox::onListFocusChanged(LLUICtrl* list, void* user_data)
758{ 678{
759 // Can't hide the list here. If the list scrolls off the screen, 679 LLComboBox *self = (LLComboBox *) list->getParent();
760 // and you click in the arrow buttons of the scroll bar, they must capture 680 // user not manipulating list or clicking on drop down button
761 // the mouse to handle scrolling-while-mouse-down. 681 if (!self->mList->hasFocus() && !self->mButton->hasMouseCapture())
682 {
683 //*HACK: store the original value explicitly somewhere, not just in label
684 LLString orig_selection = self->mAllowTextEntry ? self->mTextEntry->getText() : self->mButton->getLabelSelected();
685
686 self->hideList();
687
688 // reassert original selection
689 self->mList->selectSimpleItem(orig_selection, FALSE);
690 }
762} 691}
763 692
764BOOL LLComboBox::handleToolTip(S32 x, S32 y, LLString& msg, LLRect* sticky_rect_screen) 693BOOL LLComboBox::handleToolTip(S32 x, S32 y, LLString& msg, LLRect* sticky_rect_screen)
765{ 694{
766 695
767 LLString tool_tip; 696 LLString tool_tip;
768 697
769 if (LLUI::sShowXUINames) 698 if (LLUI::sShowXUINames)
@@ -875,6 +804,7 @@ void LLComboBox::setAllowTextEntry(BOOL allow, S32 max_chars, BOOL set_tentative
875 mTextEntry->setCommitOnFocusLost(FALSE); 804 mTextEntry->setCommitOnFocusLost(FALSE);
876 mTextEntry->setText(cur_label); 805 mTextEntry->setText(cur_label);
877 mTextEntry->setIgnoreTab(TRUE); 806 mTextEntry->setIgnoreTab(TRUE);
807 mTextEntry->setFollowsAll();
878 addChild(mTextEntry); 808 addChild(mTextEntry);
879 mMaxChars = max_chars; 809 mMaxChars = max_chars;
880 } 810 }
@@ -882,6 +812,8 @@ void LLComboBox::setAllowTextEntry(BOOL allow, S32 max_chars, BOOL set_tentative
882 { 812 {
883 mTextEntry->setVisible(TRUE); 813 mTextEntry->setVisible(TRUE);
884 } 814 }
815
816 mButton->setFollows(FOLLOWS_BOTTOM | FOLLOWS_TOP | FOLLOWS_RIGHT);
885 } 817 }
886 else if (!allow && mAllowTextEntry) 818 else if (!allow && mAllowTextEntry)
887 { 819 {
@@ -892,6 +824,7 @@ void LLComboBox::setAllowTextEntry(BOOL allow, S32 max_chars, BOOL set_tentative
892 { 824 {
893 mTextEntry->setVisible(FALSE); 825 mTextEntry->setVisible(FALSE);
894 } 826 }
827 mButton->setFollowsAll();
895 } 828 }
896 mAllowTextEntry = allow; 829 mAllowTextEntry = allow;
897 mTextEntryTentative = set_tentative; 830 mTextEntryTentative = set_tentative;
@@ -927,6 +860,7 @@ void LLComboBox::onTextEntry(LLLineEditor* line_editor, void* user_data)
927 else 860 else
928 { 861 {
929 line_editor->setTentative(self->mTextEntryTentative); 862 line_editor->setTentative(self->mTextEntryTentative);
863 self->mList->deselectAllItems();
930 } 864 }
931 return; 865 return;
932 } 866 }
@@ -1141,15 +1075,3 @@ BOOL LLComboBox::operateOnAll(EOperation op)
1141 } 1075 }
1142 return FALSE; 1076 return FALSE;
1143} 1077}
1144
1145//static
1146void LLComboBox::onListFocusLost(LLUICtrl* old_focus)
1147{
1148 // if focus is going to nothing (user hit ESC), take it back
1149 LLComboBox* combo = (LLComboBox*)old_focus->getParent();
1150 combo->hideList();
1151 if (gFocusMgr.getKeyboardFocus() == NULL)
1152 {
1153 combo->focusFirstItem();
1154 }
1155}