This keyword is not expanded by Git which means it's not replaced with the correct revision value in the releases made using git-based scripts and it's confusing to have lines with unexpanded "$Id$" in the released files. As expanding them with Git is not that simple (it could be done with git archive and export-subst attribute) and there are not many benefits in having them in the first place, just remove all these lines. If nothing else, this will make an eventual transition to Git simpler. Closes #14487. git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@74602 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
344 lines
9.5 KiB
C++
344 lines
9.5 KiB
C++
///////////////////////////////////////////////////////////////////////////////
|
|
// Name: src/common/lboxcmn.cpp
|
|
// Purpose: wxListBox class methods common to all platforms
|
|
// Author: Vadim Zeitlin
|
|
// Modified by:
|
|
// Created: 22.10.99
|
|
// Copyright: (c) wxWidgets team
|
|
// Licence: wxWindows licence
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// ============================================================================
|
|
// declarations
|
|
// ============================================================================
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// headers
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// For compilers that support precompilation, includes "wx.h".
|
|
#include "wx/wxprec.h"
|
|
|
|
#ifdef __BORLANDC__
|
|
#pragma hdrstop
|
|
#endif
|
|
|
|
#if wxUSE_LISTBOX
|
|
|
|
#include "wx/listbox.h"
|
|
|
|
#ifndef WX_PRECOMP
|
|
#include "wx/dynarray.h"
|
|
#include "wx/arrstr.h"
|
|
#include "wx/log.h"
|
|
#endif
|
|
|
|
extern WXDLLEXPORT_DATA(const char) wxListBoxNameStr[] = "listBox";
|
|
|
|
// ============================================================================
|
|
// implementation
|
|
// ============================================================================
|
|
|
|
wxListBoxBase::~wxListBoxBase()
|
|
{
|
|
// this destructor is required for Darwin
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// XTI
|
|
// ----------------------------------------------------------------------------
|
|
|
|
wxDEFINE_FLAGS( wxListBoxStyle )
|
|
wxBEGIN_FLAGS( wxListBoxStyle )
|
|
// new style border flags, we put them first to
|
|
// use them for streaming out
|
|
wxFLAGS_MEMBER(wxBORDER_SIMPLE)
|
|
wxFLAGS_MEMBER(wxBORDER_SUNKEN)
|
|
wxFLAGS_MEMBER(wxBORDER_DOUBLE)
|
|
wxFLAGS_MEMBER(wxBORDER_RAISED)
|
|
wxFLAGS_MEMBER(wxBORDER_STATIC)
|
|
wxFLAGS_MEMBER(wxBORDER_NONE)
|
|
|
|
// old style border flags
|
|
wxFLAGS_MEMBER(wxSIMPLE_BORDER)
|
|
wxFLAGS_MEMBER(wxSUNKEN_BORDER)
|
|
wxFLAGS_MEMBER(wxDOUBLE_BORDER)
|
|
wxFLAGS_MEMBER(wxRAISED_BORDER)
|
|
wxFLAGS_MEMBER(wxSTATIC_BORDER)
|
|
wxFLAGS_MEMBER(wxBORDER)
|
|
|
|
// standard window styles
|
|
wxFLAGS_MEMBER(wxTAB_TRAVERSAL)
|
|
wxFLAGS_MEMBER(wxCLIP_CHILDREN)
|
|
wxFLAGS_MEMBER(wxTRANSPARENT_WINDOW)
|
|
wxFLAGS_MEMBER(wxWANTS_CHARS)
|
|
wxFLAGS_MEMBER(wxFULL_REPAINT_ON_RESIZE)
|
|
wxFLAGS_MEMBER(wxALWAYS_SHOW_SB )
|
|
wxFLAGS_MEMBER(wxVSCROLL)
|
|
wxFLAGS_MEMBER(wxHSCROLL)
|
|
|
|
wxFLAGS_MEMBER(wxLB_SINGLE)
|
|
wxFLAGS_MEMBER(wxLB_MULTIPLE)
|
|
wxFLAGS_MEMBER(wxLB_EXTENDED)
|
|
wxFLAGS_MEMBER(wxLB_HSCROLL)
|
|
wxFLAGS_MEMBER(wxLB_ALWAYS_SB)
|
|
wxFLAGS_MEMBER(wxLB_NEEDED_SB)
|
|
wxFLAGS_MEMBER(wxLB_SORT)
|
|
wxEND_FLAGS( wxListBoxStyle )
|
|
|
|
wxIMPLEMENT_DYNAMIC_CLASS_XTI(wxListBox, wxControl, "wx/listbox.h")
|
|
|
|
wxBEGIN_PROPERTIES_TABLE(wxListBox)
|
|
wxEVENT_PROPERTY( Select, wxEVT_LISTBOX, wxCommandEvent )
|
|
wxEVENT_PROPERTY( DoubleClick, wxEVT_LISTBOX_DCLICK, wxCommandEvent )
|
|
|
|
wxPROPERTY( Font, wxFont, SetFont, GetFont , wxEMPTY_PARAMETER_VALUE, 0 /*flags*/, \
|
|
wxT("Helpstring"), wxT("group"))
|
|
wxPROPERTY_COLLECTION( Choices, wxArrayString, wxString, AppendString, \
|
|
GetStrings, 0 /*flags*/, wxT("Helpstring"), wxT("group") )
|
|
wxPROPERTY( Selection, int, SetSelection, GetSelection, wxEMPTY_PARAMETER_VALUE, \
|
|
0 /*flags*/, wxT("Helpstring"), wxT("group") )
|
|
|
|
wxPROPERTY_FLAGS( WindowStyle, wxListBoxStyle, long, SetWindowStyleFlag, \
|
|
GetWindowStyleFlag, wxEMPTY_PARAMETER_VALUE, 0 /*flags*/, \
|
|
wxT("Helpstring"), wxT("group")) // style
|
|
wxEND_PROPERTIES_TABLE()
|
|
|
|
wxEMPTY_HANDLERS_TABLE(wxListBox)
|
|
|
|
wxCONSTRUCTOR_4( wxListBox, wxWindow*, Parent, wxWindowID, Id, \
|
|
wxPoint, Position, wxSize, Size )
|
|
|
|
/*
|
|
TODO PROPERTIES
|
|
selection
|
|
content
|
|
item
|
|
*/
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// selection
|
|
// ----------------------------------------------------------------------------
|
|
|
|
bool wxListBoxBase::SetStringSelection(const wxString& s, bool select)
|
|
{
|
|
const int sel = FindString(s);
|
|
if ( sel == wxNOT_FOUND )
|
|
return false;
|
|
|
|
SetSelection(sel, select);
|
|
|
|
return true;
|
|
}
|
|
|
|
void wxListBoxBase::SetSelection(int n)
|
|
{
|
|
if ( !HasMultipleSelection() )
|
|
DoChangeSingleSelection(n);
|
|
|
|
DoSetSelection(n, true);
|
|
}
|
|
|
|
void wxListBoxBase::DeselectAll(int itemToLeaveSelected)
|
|
{
|
|
if ( HasMultipleSelection() )
|
|
{
|
|
wxArrayInt selections;
|
|
GetSelections(selections);
|
|
|
|
size_t count = selections.GetCount();
|
|
for ( size_t n = 0; n < count; n++ )
|
|
{
|
|
int item = selections[n];
|
|
if ( item != itemToLeaveSelected )
|
|
Deselect(item);
|
|
}
|
|
}
|
|
else // single selection
|
|
{
|
|
int sel = GetSelection();
|
|
if ( sel != wxNOT_FOUND && sel != itemToLeaveSelected )
|
|
{
|
|
Deselect(sel);
|
|
}
|
|
}
|
|
}
|
|
|
|
void wxListBoxBase::UpdateOldSelections()
|
|
{
|
|
// When the control becomes empty, any previously remembered selections are
|
|
// invalid anyhow, so just forget them.
|
|
if ( IsEmpty() )
|
|
{
|
|
m_oldSelections.clear();
|
|
return;
|
|
}
|
|
|
|
// We need to remember the selection even in single-selection case on
|
|
// Windows, so that we don't send an event when the user clicks on an
|
|
// already selected item.
|
|
#ifndef __WXMSW__
|
|
if (HasFlag(wxLB_MULTIPLE) || HasFlag(wxLB_EXTENDED))
|
|
#endif
|
|
{
|
|
GetSelections( m_oldSelections );
|
|
}
|
|
}
|
|
|
|
bool wxListBoxBase::SendEvent(wxEventType evtType, int item, bool selected)
|
|
{
|
|
wxCommandEvent event(evtType, GetId());
|
|
event.SetEventObject(this);
|
|
|
|
event.SetInt(item);
|
|
event.SetString(GetString(item));
|
|
event.SetExtraLong(selected);
|
|
|
|
if ( HasClientObjectData() )
|
|
event.SetClientObject(GetClientObject(item));
|
|
else if ( HasClientUntypedData() )
|
|
event.SetClientData(GetClientData(item));
|
|
|
|
return HandleWindowEvent(event);
|
|
}
|
|
|
|
bool wxListBoxBase::DoChangeSingleSelection(int item)
|
|
{
|
|
// As we don't use m_oldSelections in single selection mode, we store the
|
|
// last item that we notified the user about in it in this case because we
|
|
// need to remember it to be able to filter out the dummy selection changes
|
|
// that we get when the user clicks on an already selected item.
|
|
if ( !m_oldSelections.empty() && *m_oldSelections.begin() == item )
|
|
{
|
|
// Same item as the last time.
|
|
return false;
|
|
}
|
|
|
|
m_oldSelections.clear();
|
|
m_oldSelections.push_back(item);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool wxListBoxBase::CalcAndSendEvent()
|
|
{
|
|
wxArrayInt selections;
|
|
GetSelections(selections);
|
|
bool selected = true;
|
|
|
|
if ( selections.empty() && m_oldSelections.empty() )
|
|
{
|
|
// nothing changed, just leave
|
|
return false;
|
|
}
|
|
|
|
const size_t countSel = selections.size(),
|
|
countSelOld = m_oldSelections.size();
|
|
if ( countSel == countSelOld )
|
|
{
|
|
bool changed = false;
|
|
for ( size_t idx = 0; idx < countSel; idx++ )
|
|
{
|
|
if (selections[idx] != m_oldSelections[idx])
|
|
{
|
|
changed = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// nothing changed, just leave
|
|
if ( !changed )
|
|
return false;
|
|
}
|
|
|
|
int item = wxNOT_FOUND;
|
|
if ( selections.empty() )
|
|
{
|
|
selected = false;
|
|
item = m_oldSelections[0];
|
|
}
|
|
else // we [still] have some selections
|
|
{
|
|
// Now test if any new item is selected
|
|
bool any_new_selected = false;
|
|
for ( size_t idx = 0; idx < countSel; idx++ )
|
|
{
|
|
item = selections[idx];
|
|
if ( m_oldSelections.Index(item) == wxNOT_FOUND )
|
|
{
|
|
any_new_selected = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ( !any_new_selected )
|
|
{
|
|
// No new items selected, now test if any new item is deselected
|
|
bool any_new_deselected = false;
|
|
for ( size_t idx = 0; idx < countSelOld; idx++ )
|
|
{
|
|
item = m_oldSelections[idx];
|
|
if ( selections.Index(item) == wxNOT_FOUND )
|
|
{
|
|
any_new_deselected = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ( any_new_deselected )
|
|
{
|
|
// indicate that this is a selection
|
|
selected = false;
|
|
}
|
|
else
|
|
{
|
|
item = wxNOT_FOUND; // this should be impossible
|
|
}
|
|
}
|
|
}
|
|
|
|
wxASSERT_MSG( item != wxNOT_FOUND,
|
|
"Logic error in wxListBox selection event generation code" );
|
|
|
|
m_oldSelections = selections;
|
|
|
|
return SendEvent(wxEVT_LISTBOX, item, selected);
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// misc
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void wxListBoxBase::Command(wxCommandEvent& event)
|
|
{
|
|
SetSelection(event.GetInt(), event.GetExtraLong() != 0);
|
|
(void)GetEventHandler()->ProcessEvent(event);
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// SetFirstItem() and such
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void wxListBoxBase::SetFirstItem(const wxString& s)
|
|
{
|
|
int n = FindString(s);
|
|
|
|
wxCHECK_RET( n != wxNOT_FOUND, wxT("invalid string in wxListBox::SetFirstItem") );
|
|
|
|
DoSetFirstItem(n);
|
|
}
|
|
|
|
void wxListBoxBase::AppendAndEnsureVisible(const wxString& s)
|
|
{
|
|
Append(s);
|
|
EnsureVisible(GetCount() - 1);
|
|
}
|
|
|
|
void wxListBoxBase::EnsureVisible(int WXUNUSED(n))
|
|
{
|
|
// the base class version does nothing (the only alternative would be to
|
|
// call SetFirstItem() but this is probably even more stupid)
|
|
}
|
|
|
|
#endif // wxUSE_LISTBOX
|