1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #include <vcl/builder.hxx>
21 #include <vcl/commandevent.hxx>
22 #include <vcl/event.hxx>
23 #include <vcl/toolkit/lstbox.hxx>
24 #include <vcl/settings.hxx>
25 #include <vcl/uitest/uiobject.hxx>
26 #include <sal/log.hxx>
29 #include <listbox.hxx>
30 #include <dndeventdispatcher.hxx>
31 #include <comphelper/lok.hxx>
33 #include <com/sun/star/datatransfer/dnd/XDropTarget.hpp>
34 #include <boost/property_tree/ptree.hpp>
35 #include <tools/json_writer.hxx>
37 ListBox::ListBox(WindowType nType
)
41 ImplInitListBoxData();
44 ListBox::ListBox( vcl::Window
* pParent
, WinBits nStyle
) : Control( WindowType::LISTBOX
)
46 ImplInitListBoxData();
47 ImplInit( pParent
, nStyle
);
55 void ListBox::dispose()
57 CallEventListeners( VclEventId::ObjectDying
);
59 mpImplLB
.disposeAndClear();
60 mpFloatWin
.disposeAndClear();
61 mpImplWin
.disposeAndClear();
62 mpBtn
.disposeAndClear();
67 void ListBox::ImplInitListBoxData()
74 m_nMaxWidthChars
= -1;
78 void ListBox::ImplInit( vcl::Window
* pParent
, WinBits nStyle
)
80 nStyle
= ImplInitStyle( nStyle
);
81 if ( !(nStyle
& WB_NOBORDER
) && ( nStyle
& WB_DROPDOWN
) )
84 Control::ImplInit( pParent
, nStyle
, nullptr );
86 css::uno::Reference
< css::datatransfer::dnd::XDropTargetListener
> xDrop
= new DNDEventDispatcher(this);
88 if( nStyle
& WB_DROPDOWN
)
90 sal_Int32 nLeft
, nTop
, nRight
, nBottom
;
91 GetBorder( nLeft
, nTop
, nRight
, nBottom
);
92 mnDDHeight
= static_cast<sal_uInt16
>(GetTextHeight() + nTop
+ nBottom
+ 4);
94 if( IsNativeWidgetEnabled() &&
95 IsNativeControlSupported( ControlType::Listbox
, ControlPart::Entire
) )
97 ImplControlValue aControlValue
;
98 tools::Rectangle
aCtrlRegion( Point( 0, 0 ), Size( 20, mnDDHeight
) );
99 tools::Rectangle
aBoundingRgn( aCtrlRegion
);
100 tools::Rectangle
aContentRgn( aCtrlRegion
);
101 if( GetNativeControlRegion( ControlType::Listbox
, ControlPart::Entire
, aCtrlRegion
,
102 ControlState::ENABLED
, aControlValue
,
103 aBoundingRgn
, aContentRgn
) )
105 sal_Int32 nHeight
= aBoundingRgn
.GetHeight();
106 if( nHeight
> mnDDHeight
)
107 mnDDHeight
= static_cast<sal_uInt16
>(nHeight
);
111 mpFloatWin
= VclPtr
<ImplListBoxFloatingWindow
>::Create( this );
112 if (!IsNativeControlSupported(ControlType::Pushbutton
, ControlPart::Focus
))
113 mpFloatWin
->RequestDoubleBuffering(true);
114 mpFloatWin
->SetAutoWidth( true );
115 mpFloatWin
->SetPopupModeEndHdl( LINK( this, ListBox
, ImplPopupModeEndHdl
) );
116 mpFloatWin
->GetDropTarget()->addDropTargetListener(xDrop
);
118 mpImplWin
= VclPtr
<ImplWin
>::Create( this, (nStyle
& (WB_LEFT
|WB_RIGHT
|WB_CENTER
))|WB_NOBORDER
);
119 mpImplWin
->SetMBDownHdl( LINK( this, ListBox
, ImplClickBtnHdl
) );
121 mpImplWin
->GetDropTarget()->addDropTargetListener(xDrop
);
122 mpImplWin
->SetEdgeBlending(false);
124 mpBtn
= VclPtr
<ImplBtn
>::Create( this, WB_NOLIGHTBORDER
| WB_RECTSTYLE
);
125 ImplInitDropDownButton( mpBtn
);
126 mpBtn
->SetMBDownHdl( LINK( this, ListBox
, ImplClickBtnHdl
) );
128 mpBtn
->GetDropTarget()->addDropTargetListener(xDrop
);
131 vcl::Window
* pLBParent
= this;
133 pLBParent
= mpFloatWin
;
134 mpImplLB
= VclPtr
<ImplListBox
>::Create( pLBParent
, nStyle
&(~WB_BORDER
) );
135 mpImplLB
->SetSelectHdl( LINK( this, ListBox
, ImplSelectHdl
) );
136 mpImplLB
->SetScrollHdl( LINK( this, ListBox
, ImplScrollHdl
) );
137 mpImplLB
->SetCancelHdl( LINK( this, ListBox
, ImplCancelHdl
) );
138 mpImplLB
->SetDoubleClickHdl( LINK( this, ListBox
, ImplDoubleClickHdl
) );
139 mpImplLB
->SetFocusHdl( LINK( this, ListBox
, ImplFocusHdl
) );
140 mpImplLB
->SetListItemSelectHdl( LINK( this, ListBox
, ImplListItemSelectHdl
) );
141 mpImplLB
->SetPosPixel( Point() );
142 mpImplLB
->SetEdgeBlending(false);
145 mpImplLB
->GetDropTarget()->addDropTargetListener(xDrop
);
149 mpFloatWin
->SetImplListBox( mpImplLB
);
150 mpImplLB
->SetSelectionChangedHdl( LINK( this, ListBox
, ImplSelectionChangedHdl
) );
153 mpImplLB
->GetMainWindow()->AllowGrabFocus( true );
155 SetCompoundControl( true );
158 WinBits
ListBox::ImplInitStyle( WinBits nStyle
)
160 if ( !(nStyle
& WB_NOTABSTOP
) )
161 nStyle
|= WB_TABSTOP
;
162 if ( !(nStyle
& WB_NOGROUP
) )
167 IMPL_LINK_NOARG(ListBox
, ImplSelectHdl
, LinkParamNone
*, void)
169 bool bPopup
= IsInDropDown();
170 if( IsDropDownBox() )
172 if( !mpImplLB
->IsTravelSelect() )
174 mpFloatWin
->EndPopupMode();
175 mpImplWin
->GrabFocus();
178 mpImplWin
->SetItemPos( GetSelectedEntryPos() );
179 mpImplWin
->SetString( GetSelectedEntry() );
180 if( mpImplLB
->GetEntryList().HasImages() )
182 Image aImage
= mpImplLB
->GetEntryList().GetEntryImage( GetSelectedEntryPos() );
183 mpImplWin
->SetImage( aImage
);
185 mpImplWin
->Invalidate();
188 if ( ( !IsTravelSelect() || mpImplLB
->IsSelectionChanged() ) || ( bPopup
&& !IsMultiSelectionEnabled() ) )
192 IMPL_LINK( ListBox
, ImplFocusHdl
, sal_Int32
, nPos
, void )
194 CallEventListeners( VclEventId::ListboxFocus
, reinterpret_cast<void*>(nPos
) );
197 IMPL_LINK_NOARG( ListBox
, ImplListItemSelectHdl
, LinkParamNone
*, void )
199 CallEventListeners( VclEventId::DropdownSelect
);
202 IMPL_LINK_NOARG(ListBox
, ImplScrollHdl
, ImplListBox
*, void)
204 CallEventListeners( VclEventId::ListboxScrolled
);
207 IMPL_LINK_NOARG(ListBox
, ImplCancelHdl
, LinkParamNone
*, void)
210 mpFloatWin
->EndPopupMode();
213 IMPL_LINK( ListBox
, ImplSelectionChangedHdl
, sal_Int32
, nChanged
, void )
215 if ( mpImplLB
->IsTrackingSelect() )
218 const ImplEntryList
& rEntryList
= mpImplLB
->GetEntryList();
219 if ( rEntryList
.IsEntryPosSelected( nChanged
) )
221 // FIXME? This should've been turned into an ImplPaintEntry some time ago...
222 if ( nChanged
< rEntryList
.GetMRUCount() )
223 nChanged
= rEntryList
.FindEntry( rEntryList
.GetEntryText( nChanged
) );
224 mpImplWin
->SetItemPos( nChanged
);
225 mpImplWin
->SetString( rEntryList
.GetEntryText( nChanged
) );
226 if( rEntryList
.HasImages() )
228 Image aImage
= rEntryList
.GetEntryImage( nChanged
);
229 mpImplWin
->SetImage( aImage
);
234 mpImplWin
->SetItemPos( LISTBOX_ENTRY_NOTFOUND
);
235 mpImplWin
->SetString( OUString() );
237 mpImplWin
->SetImage( aImage
);
239 mpImplWin
->Invalidate();
242 IMPL_LINK_NOARG(ListBox
, ImplDoubleClickHdl
, ImplListBoxWindow
*, void)
247 IMPL_LINK_NOARG(ListBox
, ImplClickBtnHdl
, void*, void)
249 if( mpFloatWin
->IsInPopupMode() )
252 CallEventListeners( VclEventId::DropdownPreOpen
);
253 mpImplWin
->GrabFocus();
254 mpBtn
->SetPressed( true );
255 mpFloatWin
->StartFloat( true );
256 CallEventListeners( VclEventId::DropdownOpen
);
258 ImplClearLayoutData();
260 mpImplLB
->GetMainWindow()->ImplClearLayoutData();
262 mpImplWin
->ImplClearLayoutData();
265 IMPL_LINK_NOARG(ListBox
, ImplPopupModeEndHdl
, FloatingWindow
*, void)
267 if( mpFloatWin
->IsPopupModeCanceled() )
269 if ( ( mpFloatWin
->GetPopupModeStartSaveSelection() != LISTBOX_ENTRY_NOTFOUND
)
270 && !IsEntryPosSelected( mpFloatWin
->GetPopupModeStartSaveSelection() ) )
272 mpImplLB
->SelectEntry( mpFloatWin
->GetPopupModeStartSaveSelection(), true );
273 bool bTravelSelect
= mpImplLB
->IsTravelSelect();
274 mpImplLB
->SetTravelSelect( true );
276 VclPtr
<vcl::Window
> xWindow
= this;
278 if ( xWindow
->isDisposed() )
281 mpImplLB
->SetTravelSelect( bTravelSelect
);
285 ImplClearLayoutData();
287 mpImplLB
->GetMainWindow()->ImplClearLayoutData();
289 mpImplWin
->ImplClearLayoutData();
291 mpBtn
->SetPressed( false );
292 CallEventListeners( VclEventId::DropdownClose
);
295 void ListBox::ToggleDropDown()
297 if( !IsDropDownBox() )
300 if( mpFloatWin
->IsInPopupMode() )
301 mpFloatWin
->EndPopupMode();
304 CallEventListeners( VclEventId::DropdownPreOpen
);
305 mpImplWin
->GrabFocus();
306 mpBtn
->SetPressed( true );
307 mpFloatWin
->StartFloat( true );
308 CallEventListeners( VclEventId::DropdownOpen
);
312 void ListBox::ApplySettings(vcl::RenderContext
& rRenderContext
)
314 rRenderContext
.SetBackground();
317 void ListBox::Draw( OutputDevice
* pDev
, const Point
& rPos
, SystemTextColorFlags nFlags
)
319 mpImplLB
->GetMainWindow()->ApplySettings(*pDev
);
321 Point aPos
= pDev
->LogicToPixel( rPos
);
322 Size aSize
= GetSizePixel();
323 vcl::Font aFont
= mpImplLB
->GetMainWindow()->GetDrawPixelFont( pDev
);
327 pDev
->SetFont( aFont
);
328 pDev
->SetTextFillColor();
331 pDev
->SetLineColor();
332 pDev
->SetFillColor();
333 bool bBorder
= (GetStyle() & WB_BORDER
);
334 bool bBackground
= IsControlBackground();
335 if ( bBorder
|| bBackground
)
337 tools::Rectangle
aRect( aPos
, aSize
);
340 ImplDrawFrame( pDev
, aRect
);
344 pDev
->SetFillColor( GetControlBackground() );
345 pDev
->DrawRect( aRect
);
350 if ( nFlags
& SystemTextColorFlags::Mono
)
352 pDev
->SetTextColor( COL_BLACK
);
358 const StyleSettings
& rStyleSettings
= GetSettings().GetStyleSettings();
359 pDev
->SetTextColor( rStyleSettings
.GetDisableColor() );
363 pDev
->SetTextColor( GetTextColor() );
367 const tools::Long nOnePixel
= GetDrawPixel( pDev
, 1 );
368 const tools::Long nOffX
= 3*nOnePixel
;
369 DrawTextFlags nTextStyle
= DrawTextFlags::VCenter
;
370 tools::Rectangle
aTextRect( aPos
, aSize
);
372 if ( GetStyle() & WB_CENTER
)
373 nTextStyle
|= DrawTextFlags::Center
;
374 else if ( GetStyle() & WB_RIGHT
)
375 nTextStyle
|= DrawTextFlags::Right
;
377 nTextStyle
|= DrawTextFlags::Left
;
379 aTextRect
.AdjustLeft(nOffX
);
380 aTextRect
.AdjustRight( -nOffX
);
382 if ( IsDropDownBox() )
384 OUString aText
= GetSelectedEntry();
385 tools::Long nTextHeight
= pDev
->GetTextHeight();
386 tools::Long nTextWidth
= pDev
->GetTextWidth( aText
);
387 tools::Long nOffY
= (aSize
.Height()-nTextHeight
) / 2;
391 ((nOffY
+nTextHeight
) > aSize
.Height()) ||
392 ((nOffX
+nTextWidth
) > aSize
.Width()) )
394 tools::Rectangle
aClip( aPos
, aSize
);
395 if ( nTextHeight
> aSize
.Height() )
396 aClip
.AdjustBottom(nTextHeight
-aSize
.Height()+1 ); // So that HP Printers don't optimize this away
397 pDev
->IntersectClipRegion( aClip
);
400 pDev
->DrawText( aTextRect
, aText
, nTextStyle
);
404 tools::Long nTextHeight
= pDev
->GetTextHeight();
405 sal_uInt16 nLines
= ( nTextHeight
> 0 ) ? static_cast<sal_uInt16
>(aSize
.Height() / nTextHeight
) : 1;
406 tools::Rectangle
aClip( aPos
, aSize
);
408 pDev
->IntersectClipRegion( aClip
);
413 for ( sal_uInt16 n
= 0; n
< nLines
; n
++ )
415 sal_Int32 nEntry
= n
+mpImplLB
->GetTopEntry();
416 bool bSelected
= mpImplLB
->GetEntryList().IsEntryPosSelected( nEntry
);
419 pDev
->SetFillColor( COL_BLACK
);
420 pDev
->DrawRect( tools::Rectangle( Point( aPos
.X(), aPos
.Y() + n
*nTextHeight
),
421 Point( aPos
.X() + aSize
.Width(), aPos
.Y() + (n
+1)*nTextHeight
+ 2*nOnePixel
) ) );
422 pDev
->SetFillColor();
423 pDev
->SetTextColor( COL_WHITE
);
426 aTextRect
.SetTop( aPos
.Y() + n
*nTextHeight
);
427 aTextRect
.SetBottom( aTextRect
.Top() + nTextHeight
);
429 pDev
->DrawText( aTextRect
, mpImplLB
->GetEntryList().GetEntryText( nEntry
), nTextStyle
);
432 pDev
->SetTextColor( COL_BLACK
);
439 void ListBox::GetFocus()
443 if( IsDropDownBox() )
444 mpImplWin
->GrabFocus();
446 mpImplLB
->GrabFocus();
452 void ListBox::LoseFocus()
454 if( IsDropDownBox() )
457 mpImplWin
->HideFocus();
462 mpImplLB
->HideFocus();
465 Control::LoseFocus();
468 void ListBox::DataChanged( const DataChangedEvent
& rDCEvt
)
470 Control::DataChanged( rDCEvt
);
472 if ( !((rDCEvt
.GetType() == DataChangedEventType::FONTS
) ||
473 (rDCEvt
.GetType() == DataChangedEventType::FONTSUBSTITUTION
) ||
474 ((rDCEvt
.GetType() == DataChangedEventType::SETTINGS
) &&
475 (rDCEvt
.GetFlags() & AllSettingsFlags::STYLE
))) )
478 SetBackground(); // Due to a hack in Window::UpdateSettings the background must be reset
479 // otherwise it will overpaint NWF drawn listboxes
481 mpImplLB
->Resize(); // Is not called by ListBox::Resize() if the ImplLB does not change
485 mpImplWin
->GetOutDev()->SetSettings( GetSettings() ); // If not yet set...
486 mpImplWin
->ApplySettings(*mpImplWin
->GetOutDev());
488 mpBtn
->GetOutDev()->SetSettings( GetSettings() );
489 ImplInitDropDownButton( mpBtn
);
492 if ( IsDropDownBox() )
496 void ListBox::EnableAutoSize( bool bAuto
)
498 mbDDAutoSize
= bAuto
;
501 if ( bAuto
&& !mpFloatWin
->GetDropDownLineCount() )
503 // use GetListBoxMaximumLineCount here; before, was on fixed number of five
504 AdaptDropDownLineCountToMaximum();
508 mpFloatWin
->SetDropDownLineCount( 0 );
513 void ListBox::SetDropDownLineCount( sal_uInt16 nLines
)
515 mnLineCount
= nLines
;
517 mpFloatWin
->SetDropDownLineCount( mnLineCount
);
520 void ListBox::AdaptDropDownLineCountToMaximum()
522 // Adapt to maximum allowed number.
523 // Limit for LOK as we can't render outside of the dialog canvas.
524 if (comphelper::LibreOfficeKit::isActive())
525 SetDropDownLineCount(11);
527 SetDropDownLineCount(GetSettings().GetStyleSettings().GetListBoxMaximumLineCount());
530 sal_uInt16
ListBox::GetDropDownLineCount() const
533 return mpFloatWin
->GetDropDownLineCount();
537 void ListBox::setPosSizePixel( tools::Long nX
, tools::Long nY
, tools::Long nWidth
, tools::Long nHeight
, PosSizeFlags nFlags
)
539 if( IsDropDownBox() && ( nFlags
& PosSizeFlags::Size
) )
541 Size aPrefSz
= mpFloatWin
->GetPrefSize();
542 if ( ( nFlags
& PosSizeFlags::Height
) && ( nHeight
>= 2*mnDDHeight
) )
543 aPrefSz
.setHeight( nHeight
-mnDDHeight
);
544 if ( nFlags
& PosSizeFlags::Width
)
545 aPrefSz
.setWidth( nWidth
);
546 mpFloatWin
->SetPrefSize( aPrefSz
);
548 if (IsAutoSizeEnabled())
549 nHeight
= mnDDHeight
;
552 Control::setPosSizePixel( nX
, nY
, nWidth
, nHeight
, nFlags
);
555 void ListBox::Resize()
557 Size aOutSz
= GetOutputSizePixel();
558 if( IsDropDownBox() )
560 // Initialize the dropdown button size with the standard scrollbar width
561 tools::Long nSBWidth
= GetSettings().GetStyleSettings().GetScrollBarSize();
562 tools::Long nBottom
= aOutSz
.Height();
564 // Note: in case of no border, pBorder will actually be this
565 vcl::Window
*pBorder
= GetWindow( GetWindowType::Border
);
566 ImplControlValue aControlValue
;
568 tools::Rectangle aContent
, aBound
;
570 // Use the full extent of the control
571 tools::Rectangle
aArea( aPoint
, pBorder
->GetOutputSizePixel() );
573 if ( GetNativeControlRegion( ControlType::Listbox
, ControlPart::ButtonDown
,
574 aArea
, ControlState::NONE
, aControlValue
, aBound
, aContent
) )
576 // Convert back from border space to local coordinates
577 aPoint
= pBorder
->ScreenToOutputPixel( OutputToScreenPixel( aPoint
) );
578 aContent
.Move( -aPoint
.X(), -aPoint
.Y() );
580 // Use the themes drop down size for the button
581 aOutSz
.setWidth( aContent
.Left() );
582 mpBtn
->setPosSizePixel( aContent
.Left(), 0, aContent
.GetWidth(), nBottom
);
584 // Adjust the size of the edit field
585 if ( GetNativeControlRegion( ControlType::Listbox
, ControlPart::SubEdit
,
586 aArea
, ControlState::NONE
, aControlValue
, aBound
, aContent
) )
588 // Convert back from border space to local coordinates
589 aContent
.Move( -aPoint
.X(), -aPoint
.Y() );
591 // Use the themes drop down size
592 if( ! (GetStyle() & WB_BORDER
) && ImplGetSVData()->maNWFData
.mbNoFocusRects
)
594 // No border but focus ring behavior -> we have a problem; the
595 // native rect relies on the border to draw the focus
596 // let's do the best we can and center vertically, so it doesn't look
598 Size
aSz( GetOutputSizePixel() );
599 tools::Long nDiff
= aContent
.Top() - (aSz
.Height() - aContent
.GetHeight())/2;
600 aContent
.AdjustTop( -nDiff
);
601 aContent
.AdjustBottom( -nDiff
);
603 mpImplWin
->SetPosSizePixel( aContent
.TopLeft(), aContent
.GetSize() );
606 mpImplWin
->SetSizePixel( aOutSz
);
610 nSBWidth
= CalcZoom( nSBWidth
);
611 mpImplWin
->setPosSizePixel( 0, 0, aOutSz
.Width() - nSBWidth
, aOutSz
.Height() );
612 mpBtn
->setPosSizePixel( aOutSz
.Width() - nSBWidth
, 0, nSBWidth
, aOutSz
.Height() );
617 mpImplLB
->SetSizePixel( aOutSz
);
620 // Retain FloatingWindow size even when it's invisible, as we still process KEY_PGUP/DOWN ...
622 mpFloatWin
->SetSizePixel( mpFloatWin
->CalcFloatSize() );
627 void ListBox::FillLayoutData() const
629 mxLayoutData
.emplace();
630 const ImplListBoxWindow
* rMainWin
= mpImplLB
->GetMainWindow();
634 AppendLayoutData( *mpImplWin
);
635 mpImplWin
->SetLayoutDataParent( this );
636 if( mpFloatWin
->IsReallyVisible() )
638 AppendLayoutData( *rMainWin
);
639 rMainWin
->SetLayoutDataParent( this );
644 AppendLayoutData( *rMainWin
);
645 rMainWin
->SetLayoutDataParent( this );
649 tools::Long
ListBox::GetIndexForPoint( const Point
& rPoint
, sal_Int32
& rPos
) const
651 if( !HasLayoutData() )
654 // Check whether rPoint fits at all
655 tools::Long nIndex
= Control::GetIndexForPoint( rPoint
);
658 // Point must be either in main list window
659 // or in impl window (dropdown case)
660 ImplListBoxWindow
* rMain
= mpImplLB
->GetMainWindow();
662 // Convert coordinates to ImplListBoxWindow pixel coordinate space
663 Point aConvPoint
= LogicToPixel( rPoint
);
664 aConvPoint
= OutputToAbsoluteScreenPixel( aConvPoint
);
665 aConvPoint
= rMain
->AbsoluteScreenToOutputPixel( aConvPoint
);
666 aConvPoint
= rMain
->PixelToLogic( aConvPoint
);
669 sal_Int32 nEntry
= rMain
->GetEntryPosForPoint( aConvPoint
);
670 if( nEntry
== LISTBOX_ENTRY_NOTFOUND
)
672 // Not found, maybe dropdown case
673 if( mpImplWin
&& mpImplWin
->IsReallyVisible() )
675 // Convert to impl window pixel coordinates
676 aConvPoint
= LogicToPixel( rPoint
);
677 aConvPoint
= OutputToAbsoluteScreenPixel( aConvPoint
);
678 aConvPoint
= mpImplWin
->AbsoluteScreenToOutputPixel( aConvPoint
);
680 // Check whether converted point is inside impl window
681 Size aImplWinSize
= mpImplWin
->GetOutputSizePixel();
682 if( aConvPoint
.X() >= 0 && aConvPoint
.Y() >= 0 && aConvPoint
.X() < aImplWinSize
.Width() && aConvPoint
.Y() < aImplWinSize
.Height() )
684 // Inside the impl window, the position is the current item pos
685 rPos
= mpImplWin
->GetItemPos();
696 SAL_WARN_IF( nIndex
== -1, "vcl", "found index for point, but relative index failed" );
699 // Get line relative index
701 nIndex
= ToRelativeLineIndex( nIndex
);
706 void ListBox::StateChanged( StateChangedType nType
)
708 if( nType
== StateChangedType::ReadOnly
)
711 mpImplWin
->Enable( !IsReadOnly() );
713 mpBtn
->Enable( !IsReadOnly() );
715 else if( nType
== StateChangedType::Enable
)
717 mpImplLB
->Enable( IsEnabled() );
720 mpImplWin
->Enable( IsEnabled() );
721 if ( IsNativeControlSupported(ControlType::Listbox
, ControlPart::Entire
)
722 && ! IsNativeControlSupported(ControlType::Listbox
, ControlPart::ButtonDown
) )
724 GetWindow( GetWindowType::Border
)->Invalidate( InvalidateFlags::NoErase
);
727 mpImplWin
->Invalidate();
730 mpBtn
->Enable( IsEnabled() );
732 else if( nType
== StateChangedType::UpdateMode
)
734 mpImplLB
->SetUpdateMode( IsUpdateMode() );
736 else if ( nType
== StateChangedType::Zoom
)
738 mpImplLB
->SetZoom( GetZoom() );
741 mpImplWin
->SetZoom( GetZoom() );
742 mpImplWin
->SetFont( mpImplLB
->GetMainWindow()->GetFont() );
743 mpImplWin
->Invalidate();
747 else if ( nType
== StateChangedType::ControlFont
)
749 mpImplLB
->SetControlFont( GetControlFont() );
752 mpImplWin
->SetControlFont( GetControlFont() );
753 mpImplWin
->SetFont( mpImplLB
->GetMainWindow()->GetFont() );
754 mpImplWin
->Invalidate();
758 else if ( nType
== StateChangedType::ControlForeground
)
760 mpImplLB
->SetControlForeground( GetControlForeground() );
763 mpImplWin
->SetControlForeground( GetControlForeground() );
764 mpImplWin
->SetTextColor( GetControlForeground() );
765 mpImplWin
->SetFont( mpImplLB
->GetMainWindow()->GetFont() );
766 mpImplWin
->Invalidate();
769 else if ( nType
== StateChangedType::ControlBackground
)
771 mpImplLB
->SetControlBackground( GetControlBackground() );
775 mpImplWin
->SetBackground( GetControlBackground() );
776 mpImplWin
->SetControlBackground( GetControlBackground() );
777 mpImplWin
->SetFont( mpImplLB
->GetMainWindow()->GetFont() );
778 mpImplWin
->Invalidate();
781 else if ( nType
== StateChangedType::Style
)
783 SetStyle( ImplInitStyle( GetStyle() ) );
784 mpImplLB
->GetMainWindow()->EnableSort( ( GetStyle() & WB_SORT
) != 0 );
785 bool bSimpleMode
= ( GetStyle() & WB_SIMPLEMODE
) != 0;
786 mpImplLB
->SetMultiSelectionSimpleMode( bSimpleMode
);
788 else if( nType
== StateChangedType::Mirroring
)
792 mpBtn
->EnableRTL( IsRTLEnabled() );
793 ImplInitDropDownButton( mpBtn
);
795 mpImplLB
->EnableRTL( IsRTLEnabled() );
797 mpImplWin
->EnableRTL( IsRTLEnabled() );
801 Control::StateChanged( nType
);
804 bool ListBox::PreNotify( NotifyEvent
& rNEvt
)
809 if( ( rNEvt
.GetType() == NotifyEventType::KEYINPUT
) && ( rNEvt
.GetWindow() == mpImplWin
) )
811 KeyEvent aKeyEvt
= *rNEvt
.GetKeyEvent();
812 switch( aKeyEvt
.GetKeyCode().GetCode() )
816 if( mpFloatWin
&& !mpFloatWin
->IsInPopupMode() &&
817 aKeyEvt
.GetKeyCode().IsMod2() )
819 CallEventListeners( VclEventId::DropdownPreOpen
);
820 mpBtn
->SetPressed( true );
821 mpFloatWin
->StartFloat( false );
822 CallEventListeners( VclEventId::DropdownOpen
);
827 bDone
= mpImplLB
->ProcessKeyInput( aKeyEvt
);
833 if( mpFloatWin
&& mpFloatWin
->IsInPopupMode() &&
834 aKeyEvt
.GetKeyCode().IsMod2() )
836 mpFloatWin
->EndPopupMode();
841 bDone
= mpImplLB
->ProcessKeyInput( aKeyEvt
);
849 mpImplLB
->ProcessKeyInput( aKeyEvt
);
857 bDone
= mpImplLB
->ProcessKeyInput( aKeyEvt
);
861 else if ( rNEvt
.GetType() == NotifyEventType::LOSEFOCUS
)
863 if ( IsInDropDown() && !HasChildPathFocus( true ) )
864 mpFloatWin
->EndPopupMode();
866 else if ( (rNEvt
.GetType() == NotifyEventType::COMMAND
) &&
867 (rNEvt
.GetCommandEvent()->GetCommand() == CommandEventId::Wheel
) &&
868 (rNEvt
.GetWindow() == mpImplWin
) )
870 MouseWheelBehaviour
nWheelBehavior( GetSettings().GetMouseSettings().GetWheelBehavior() );
871 if ( ( nWheelBehavior
== MouseWheelBehaviour::ALWAYS
)
872 || ( ( nWheelBehavior
== MouseWheelBehaviour::FocusOnly
)
873 && HasChildPathFocus()
877 bDone
= mpImplLB
->HandleWheelAsCursorTravel(*rNEvt
.GetCommandEvent(), *this);
881 bDone
= false; // Don't consume this event, let the default handling take it (i.e. scroll the context)
886 if (rNEvt
.GetType() == NotifyEventType::MOUSEMOVE
)
888 const MouseEvent
* pMouseEvt
= rNEvt
.GetMouseEvent();
889 if (pMouseEvt
&& (pMouseEvt
->IsEnterWindow() || pMouseEvt
->IsLeaveWindow()))
891 // trigger redraw as mouse over state has changed
892 if (IsNativeControlSupported(ControlType::Listbox
, ControlPart::Entire
)
893 && !IsNativeControlSupported(ControlType::Listbox
, ControlPart::ButtonDown
))
895 GetWindow(GetWindowType::Border
)->Invalidate(InvalidateFlags::NoErase
);
900 return bDone
|| Control::PreNotify( rNEvt
);
903 void ListBox::Select()
905 ImplCallEventListenersAndHandler( VclEventId::ListboxSelect
, [this] () { maSelectHdl
.Call(*this); } );
908 void ListBox::DoubleClick()
910 ImplCallEventListenersAndHandler( VclEventId::ListboxDoubleClick
, {} );
913 void ListBox::Clear()
918 if( IsDropDownBox() )
920 mpImplWin
->SetItemPos( LISTBOX_ENTRY_NOTFOUND
);
921 mpImplWin
->SetString( OUString() );
923 mpImplWin
->SetImage( aImage
);
924 mpImplWin
->Invalidate();
926 CallEventListeners( VclEventId::ListboxItemRemoved
, reinterpret_cast<void*>(-1) );
929 void ListBox::SetNoSelection()
931 mpImplLB
->SetNoSelection();
932 if( IsDropDownBox() )
934 mpImplWin
->SetItemPos( LISTBOX_ENTRY_NOTFOUND
);
935 mpImplWin
->SetString( OUString() );
937 mpImplWin
->SetImage( aImage
);
938 mpImplWin
->Invalidate();
942 sal_Int32
ListBox::InsertEntry( const OUString
& rStr
, sal_Int32 nPos
)
944 sal_Int32 nRealPos
= mpImplLB
->InsertEntry( nPos
+ mpImplLB
->GetEntryList().GetMRUCount(), rStr
);
945 nRealPos
= sal::static_int_cast
<sal_Int32
>(nRealPos
- mpImplLB
->GetEntryList().GetMRUCount());
946 CallEventListeners( VclEventId::ListboxItemAdded
, reinterpret_cast<void*>(nRealPos
) );
950 sal_Int32
ListBox::InsertEntry( const OUString
& rStr
, const Image
& rImage
, sal_Int32 nPos
)
952 sal_Int32 nRealPos
= mpImplLB
->InsertEntry( nPos
+ mpImplLB
->GetEntryList().GetMRUCount(), rStr
, rImage
);
953 nRealPos
= sal::static_int_cast
<sal_Int32
>(nRealPos
- mpImplLB
->GetEntryList().GetMRUCount());
954 CallEventListeners( VclEventId::ListboxItemAdded
, reinterpret_cast<void*>(nRealPos
) );
958 void ListBox::RemoveEntry( sal_Int32 nPos
)
960 mpImplLB
->RemoveEntry( nPos
+ mpImplLB
->GetEntryList().GetMRUCount() );
961 CallEventListeners( VclEventId::ListboxItemRemoved
, reinterpret_cast<void*>(nPos
) );
964 Image
ListBox::GetEntryImage( sal_Int32 nPos
) const
966 if ( mpImplLB
&& mpImplLB
->GetEntryList().HasEntryImage( nPos
) )
967 return mpImplLB
->GetEntryList().GetEntryImage( nPos
);
971 sal_Int32
ListBox::GetEntryPos( std::u16string_view rStr
) const
974 return LISTBOX_ENTRY_NOTFOUND
;
975 sal_Int32 nPos
= mpImplLB
->GetEntryList().FindEntry( rStr
);
976 if ( nPos
!= LISTBOX_ENTRY_NOTFOUND
)
977 nPos
= nPos
- mpImplLB
->GetEntryList().GetMRUCount();
981 OUString
ListBox::GetEntry( sal_Int32 nPos
) const
985 return mpImplLB
->GetEntryList().GetEntryText( nPos
+ mpImplLB
->GetEntryList().GetMRUCount() );
988 sal_Int32
ListBox::GetEntryCount() const
992 return mpImplLB
->GetEntryList().GetEntryCount() - mpImplLB
->GetEntryList().GetMRUCount();
995 OUString
ListBox::GetSelectedEntry(sal_Int32 nIndex
) const
997 return GetEntry( GetSelectedEntryPos( nIndex
) );
1000 sal_Int32
ListBox::GetSelectedEntryCount() const
1004 return mpImplLB
->GetEntryList().GetSelectedEntryCount();
1007 sal_Int32
ListBox::GetSelectedEntryPos( sal_Int32 nIndex
) const
1010 return LISTBOX_ENTRY_NOTFOUND
;
1012 sal_Int32 nPos
= mpImplLB
->GetEntryList().GetSelectedEntryPos( nIndex
);
1013 if ( nPos
!= LISTBOX_ENTRY_NOTFOUND
)
1015 if ( nPos
< mpImplLB
->GetEntryList().GetMRUCount() )
1016 nPos
= mpImplLB
->GetEntryList().FindEntry( mpImplLB
->GetEntryList().GetEntryText( nPos
) );
1017 nPos
= nPos
- mpImplLB
->GetEntryList().GetMRUCount();
1022 bool ListBox::IsEntryPosSelected( sal_Int32 nPos
) const
1024 return mpImplLB
->GetEntryList().IsEntryPosSelected( nPos
+ mpImplLB
->GetEntryList().GetMRUCount() );
1027 void ListBox::SelectEntry( std::u16string_view rStr
, bool bSelect
)
1029 SelectEntryPos( GetEntryPos( rStr
), bSelect
);
1032 void ListBox::SelectEntryPos( sal_Int32 nPos
, bool bSelect
)
1037 if ( 0 <= nPos
&& nPos
< mpImplLB
->GetEntryList().GetEntryCount() )
1039 sal_Int32 nCurrentPos
= mpImplLB
->GetCurrentPos();
1040 mpImplLB
->SelectEntry( nPos
+ mpImplLB
->GetEntryList().GetMRUCount(), bSelect
);
1041 //Only when bSelect == true, send both Selection & Focus events
1042 if (nCurrentPos
!= nPos
&& bSelect
)
1044 CallEventListeners( VclEventId::ListboxSelect
, reinterpret_cast<void*>(nPos
));
1046 CallEventListeners( VclEventId::ListboxFocus
, reinterpret_cast<void*>(nPos
));
1051 void ListBox::SelectEntriesPos( const std::vector
<sal_Int32
>& rPositions
, bool bSelect
)
1056 bool bCallListeners
= false;
1058 const sal_Int32 nCurrentPos
= mpImplLB
->GetCurrentPos();
1059 const auto nEntryCount
= mpImplLB
->GetEntryList().GetEntryCount();
1060 const auto nMRUCount
= mpImplLB
->GetEntryList().GetMRUCount();
1062 for (auto nPos
: rPositions
)
1064 if (0 <= nPos
&& nPos
< nEntryCount
)
1066 mpImplLB
->SelectEntry(nPos
+ nMRUCount
, bSelect
);
1067 if (nCurrentPos
!= nPos
&& bSelect
)
1068 bCallListeners
= true;
1072 //Only when bSelect == true, send both Selection & Focus events
1075 CallEventListeners(VclEventId::ListboxSelect
);
1077 CallEventListeners(VclEventId::ListboxFocus
);
1081 void ListBox::SetEntryData( sal_Int32 nPos
, void* pNewData
)
1083 mpImplLB
->SetEntryData( nPos
+ mpImplLB
->GetEntryList().GetMRUCount(), pNewData
);
1086 void* ListBox::GetEntryData( sal_Int32 nPos
) const
1088 return mpImplLB
->GetEntryList().GetEntryData( nPos
+ mpImplLB
->GetEntryList().GetMRUCount() );
1091 void ListBox::SetEntryFlags( sal_Int32 nPos
, ListBoxEntryFlags nFlags
)
1093 mpImplLB
->SetEntryFlags( nPos
+ mpImplLB
->GetEntryList().GetMRUCount(), nFlags
);
1096 void ListBox::SetTopEntry( sal_Int32 nPos
)
1098 mpImplLB
->SetTopEntry( nPos
+ mpImplLB
->GetEntryList().GetMRUCount() );
1101 sal_Int32
ListBox::GetTopEntry() const
1103 sal_Int32 nPos
= GetEntryCount() ? mpImplLB
->GetTopEntry() : LISTBOX_ENTRY_NOTFOUND
;
1104 if ( nPos
< mpImplLB
->GetEntryList().GetMRUCount() )
1109 bool ListBox::IsTravelSelect() const
1111 return mpImplLB
->IsTravelSelect();
1114 bool ListBox::IsInDropDown() const
1116 // when the dropdown is dismissed, first mbInPopupMode is set to false, and on the next event iteration then
1117 // mbPopupMode is set to false
1118 return mpFloatWin
&& mpFloatWin
->IsInPopupMode() && mpFloatWin
->ImplIsInPrivatePopupMode();
1121 tools::Rectangle
ListBox::GetBoundingRectangle( sal_Int32 nItem
) const
1123 tools::Rectangle aRect
= mpImplLB
->GetMainWindow()->GetBoundingRectangle( nItem
);
1124 tools::Rectangle aOffset
= mpImplLB
->GetMainWindow()->GetWindowExtentsRelative( static_cast<vcl::Window
*>(const_cast<ListBox
*>(this)) );
1125 aRect
.Move( aOffset
.Left(), aOffset
.Top() );
1129 void ListBox::EnableMultiSelection( bool bMulti
)
1131 mpImplLB
->EnableMultiSelection( bMulti
);
1134 // The MultiListBox behaves just like a normal ListBox
1135 // MultiSelection is possible via corresponding additional keys
1136 bool bSimpleMode
= ( GetStyle() & WB_SIMPLEMODE
) != 0;
1137 mpImplLB
->SetMultiSelectionSimpleMode( bSimpleMode
);
1139 // In a MultiSelection, we can't see us travelling without focus
1141 mpImplLB
->GetMainWindow()->AllowGrabFocus( bMulti
);
1144 bool ListBox::IsMultiSelectionEnabled() const
1146 return mpImplLB
->IsMultiSelectionEnabled();
1149 void ListBox::SetHighlightColor(const Color
& rColor
)
1151 AllSettings
aSettings(GetSettings());
1152 StyleSettings
aStyle(aSettings
.GetStyleSettings());
1153 aStyle
.SetHighlightColor(rColor
);
1154 aSettings
.SetStyleSettings(aStyle
);
1155 SetSettings(aSettings
);
1157 mpImplLB
->SetHighlightColor(rColor
);
1160 void ListBox::SetHighlightTextColor(const Color
& rColor
)
1162 AllSettings
aSettings(GetSettings());
1163 StyleSettings
aStyle(aSettings
.GetStyleSettings());
1164 aStyle
.SetHighlightTextColor(rColor
);
1165 aSettings
.SetStyleSettings(aStyle
);
1166 SetSettings(aSettings
);
1168 mpImplLB
->SetHighlightTextColor(rColor
);
1171 Size
ListBox::CalcMinimumSize() const
1178 aSz
= CalcSubEditSize();
1180 bool bAddScrollWidth
= false;
1182 if (IsDropDownBox())
1184 aSz
.AdjustHeight(4 ); // add a space between entry and border
1185 aSz
.AdjustWidth(4 ); // add a little breathing space
1186 bAddScrollWidth
= true;
1189 bAddScrollWidth
= (GetStyle() & WB_VSCROLL
) == WB_VSCROLL
;
1191 if (bAddScrollWidth
)
1193 // Try native borders; scrollbar size may not be a good indicator
1194 // See how large the edit area inside is to estimate what is needed for the dropdown
1195 ImplControlValue aControlValue
;
1196 tools::Rectangle aContent
, aBound
;
1197 Size
aTestSize( 100, 20 );
1198 tools::Rectangle
aArea( Point(), aTestSize
);
1199 if( GetNativeControlRegion( ControlType::Listbox
, ControlPart::SubEdit
, aArea
, ControlState::NONE
,
1200 aControlValue
, aBound
, aContent
) )
1202 // use the themes drop down size
1203 aSz
.AdjustWidth(aTestSize
.Width() - aContent
.GetWidth() );
1206 aSz
.AdjustWidth(GetSettings().GetStyleSettings().GetScrollBarSize() );
1209 aSz
= CalcWindowSize( aSz
);
1211 if (IsDropDownBox()) // Check minimum height of dropdown box
1213 ImplControlValue aControlValue
;
1214 tools::Rectangle
aRect( Point( 0, 0 ), aSz
);
1215 tools::Rectangle aContent
, aBound
;
1216 if( GetNativeControlRegion( ControlType::Listbox
, ControlPart::Entire
, aRect
, ControlState::NONE
,
1217 aControlValue
, aBound
, aContent
) )
1219 if( aBound
.GetHeight() > aSz
.Height() )
1220 aSz
.setHeight( aBound
.GetHeight() );
1227 Size
ListBox::CalcSubEditSize() const
1234 if ( !IsDropDownBox() )
1235 aSz
= mpImplLB
->CalcSize (mnLineCount
? mnLineCount
: mpImplLB
->GetEntryList().GetEntryCount());
1238 aSz
.setHeight( mpImplLB
->GetEntryHeight() );
1239 // Size to maximum entry width
1240 aSz
.setWidth( mpImplLB
->GetMaxEntryWidth() );
1242 if (m_nMaxWidthChars
!= -1)
1244 tools::Long nMaxWidth
= m_nMaxWidthChars
* approximate_char_width();
1245 aSz
.setWidth( std::min(aSz
.Width(), nMaxWidth
) );
1248 // Do not create ultrathin ListBoxes, it doesn't look good
1249 if( aSz
.Width() < GetSettings().GetStyleSettings().GetScrollBarSize() )
1250 aSz
.setWidth( GetSettings().GetStyleSettings().GetScrollBarSize() );
1256 Size
ListBox::GetOptimalSize() const
1258 return CalcMinimumSize();
1261 Size
ListBox::CalcAdjustedSize( const Size
& rPrefSize
) const
1263 Size aSz
= rPrefSize
;
1264 sal_Int32 nLeft
, nTop
, nRight
, nBottom
;
1265 static_cast<vcl::Window
*>(const_cast<ListBox
*>(this))->GetBorder( nLeft
, nTop
, nRight
, nBottom
);
1266 aSz
.AdjustHeight( -(nTop
+nBottom
) );
1267 if ( !IsDropDownBox() )
1269 tools::Long nEntryHeight
= CalcBlockSize( 1, 1 ).Height();
1270 tools::Long nLines
= aSz
.Height() / nEntryHeight
;
1273 aSz
.setHeight( nLines
* nEntryHeight
);
1277 aSz
.setHeight( mnDDHeight
);
1279 aSz
.AdjustHeight(nTop
+nBottom
);
1281 aSz
= CalcWindowSize( aSz
);
1285 Size
ListBox::CalcBlockSize( sal_uInt16 nColumns
, sal_uInt16 nLines
) const
1287 // ScrollBars are shown if needed
1288 Size aMinSz
= CalcMinimumSize();
1289 // aMinSz = ImplCalcOutSz( aMinSz );
1296 if ( !IsDropDownBox() )
1297 aSz
.setHeight( mpImplLB
->CalcSize( nLines
).Height() );
1299 aSz
.setHeight( mnDDHeight
);
1302 aSz
.setHeight( aMinSz
.Height() );
1306 aSz
.setWidth( nColumns
* GetTextWidth( OUString('X') ) );
1308 aSz
.setWidth( aMinSz
.Width() );
1310 if ( IsDropDownBox() )
1311 aSz
.AdjustWidth(GetSettings().GetStyleSettings().GetScrollBarSize() );
1313 if ( !IsDropDownBox() )
1315 if ( aSz
.Width() < aMinSz
.Width() )
1316 aSz
.AdjustHeight(GetSettings().GetStyleSettings().GetScrollBarSize() );
1317 if ( aSz
.Height() < aMinSz
.Height() )
1318 aSz
.AdjustWidth(GetSettings().GetStyleSettings().GetScrollBarSize() );
1321 aSz
= CalcWindowSize( aSz
);
1325 void ListBox::GetMaxVisColumnsAndLines( sal_uInt16
& rnCols
, sal_uInt16
& rnLines
) const
1327 float nCharWidth
= approximate_char_width();
1328 if ( !IsDropDownBox() )
1330 Size aOutSz
= mpImplLB
->GetMainWindow()->GetOutputSizePixel();
1331 rnCols
= static_cast<sal_uInt16
>(aOutSz
.Width()/nCharWidth
);
1332 rnLines
= static_cast<sal_uInt16
>(aOutSz
.Height()/mpImplLB
->GetEntryHeightWithMargin());
1336 Size aOutSz
= mpImplWin
->GetOutputSizePixel();
1337 rnCols
= static_cast<sal_uInt16
>(aOutSz
.Width()/nCharWidth
);
1342 void ListBox::SetReadOnly( bool bReadOnly
)
1344 if ( mpImplLB
->IsReadOnly() != bReadOnly
)
1346 mpImplLB
->SetReadOnly( bReadOnly
);
1347 CompatStateChanged( StateChangedType::ReadOnly
);
1351 bool ListBox::IsReadOnly() const
1353 return mpImplLB
->IsReadOnly();
1356 void ListBox::SetSeparatorPos( sal_Int32 n
)
1358 mpImplLB
->SetSeparatorPos( n
);
1361 sal_Int32
ListBox::GetSeparatorPos() const
1363 return mpImplLB
->GetSeparatorPos();
1366 void ListBox::AddSeparator( sal_Int32 n
)
1368 mpImplLB
->AddSeparator( n
);
1371 sal_uInt16
ListBox::GetDisplayLineCount() const
1373 return mpImplLB
->GetDisplayLineCount();
1376 tools::Rectangle
ListBox::GetDropDownPosSizePixel() const
1378 return mpFloatWin
? mpFloatWin
->GetWindowExtentsRelative(this) : tools::Rectangle();
1381 const Wallpaper
& ListBox::GetDisplayBackground() const
1383 // !!! Recursion does not occur because the ImplListBox is initialized by default
1384 // to a non-transparent color in Window::ImplInitData
1385 return mpImplLB
->GetDisplayBackground();
1388 void ListBox::setMaxWidthChars(sal_Int32 nWidth
)
1390 if (nWidth
!= m_nMaxWidthChars
)
1392 m_nMaxWidthChars
= nWidth
;
1397 bool ListBox::set_property(const OUString
&rKey
, const OUString
&rValue
)
1399 if (rKey
== "active")
1400 SelectEntryPos(rValue
.toInt32());
1401 else if (rKey
== "max-width-chars")
1402 setMaxWidthChars(rValue
.toInt32());
1403 else if (rKey
== "can-focus")
1405 // as far as I can see in Gtk, setting a ComboBox as can.focus means
1406 // the focus gets stuck in it, so try here to behave like gtk does
1407 // with the settings that work, i.e. can.focus of false doesn't
1408 // set the hard WB_NOTABSTOP
1409 WinBits nBits
= GetStyle();
1410 nBits
&= ~(WB_TABSTOP
|WB_NOTABSTOP
);
1412 nBits
|= WB_TABSTOP
;
1416 return Control::set_property(rKey
, rValue
);
1420 FactoryFunction
ListBox::GetUITestFactory() const
1422 return ListBoxUIObject::create
;
1425 void ListBox::DumpAsPropertyTree(tools::JsonWriter
& rJsonWriter
)
1427 Control::DumpAsPropertyTree(rJsonWriter
);
1430 auto entriesNode
= rJsonWriter
.startArray("entries");
1431 for (int i
= 0; i
< GetEntryCount(); ++i
)
1433 rJsonWriter
.putSimpleValue(GetEntry(i
));
1437 rJsonWriter
.put("selectedCount", GetSelectedEntryCount());
1440 auto entriesNode
= rJsonWriter
.startArray("selectedEntries");
1441 for (int i
= 0; i
< GetSelectedEntryCount(); ++i
)
1443 rJsonWriter
.putSimpleValue(OUString::number(GetSelectedEntryPos(i
)));
1448 MultiListBox::MultiListBox( vcl::Window
* pParent
, WinBits nStyle
) :
1449 ListBox( WindowType::MULTILISTBOX
)
1451 ImplInit( pParent
, nStyle
);
1452 EnableMultiSelection( true );
1455 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */