mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-02-27 22:43:40 +01:00

From wxWidgets master 81570ae070b35c9d52de47b1f14897f3ff1a66c7. include/wx/defs.h -- __w64 warning disable patch by comex brought forward. include/wx/msw/window.h -- added GetContentScaleFactor() which was not implemented on Windows but is necessary for wxBitmap scaling on Mac OS X so it needs to work to avoid #ifdef-ing the code. src/gtk/window.cpp -- Modified DoSetClientSize() to direct call wxWindowGTK::DoSetSize() instead of using public wxWindowBase::SetSize() which now prevents derived classes (like wxAuiToolbar) intercepting the call and breaking it. This matches Windows which does NOT need to call DoSetSize internally. End result is this fixes Dolphin's debug tools toolbars on Linux. src/osx/window_osx.cpp -- Same fix as for GTK since it has the same issue. src/msw/radiobox.cpp -- Hacked to fix display in HiDPI (was clipping off end of text). Updated CMakeLists for Linux and Mac OS X. Small code changes to Dolphin to fix debug error boxes, deprecation warnings, and retain previous UI behavior on Windows.
373 lines
10 KiB
C++
373 lines
10 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"
|
|
#include "wx/dcclient.h"
|
|
#endif
|
|
|
|
// the spacing between the lines (in report mode)
|
|
static const int LINE_SPACING = 0;
|
|
|
|
#ifdef __WXGTK__
|
|
static const int EXTRA_HEIGHT = 6;
|
|
#else
|
|
static const int EXTRA_HEIGHT = 4;
|
|
#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)
|
|
}
|
|
|
|
wxCoord wxListBoxBase::GetLineHeight() const
|
|
{
|
|
wxListBoxBase *self = wxConstCast(this, wxListBoxBase);
|
|
|
|
wxClientDC dc( self );
|
|
dc.SetFont( GetFont() );
|
|
|
|
wxCoord y;
|
|
dc.GetTextExtent(wxT("H"), NULL, &y);
|
|
|
|
y += EXTRA_HEIGHT;
|
|
|
|
return y + LINE_SPACING;
|
|
}
|
|
|
|
int wxListBoxBase::GetCountPerPage() const
|
|
{
|
|
return GetClientSize().y / GetLineHeight();
|
|
}
|
|
#endif // wxUSE_LISTBOX
|