Allow to not create wxPaintDC in EVT_PAINT handler in wxMSW.
Failure to create a wxPaintDC in EVT_PAINT handler resulted in many serious and difficult to debug problems under wxMSW. We used to document that the user shouldn't do it but this wasn't enough (see #11648). We could also assert if we detected that a handler didn't create a wxPaintDC but it seems better to just handle this case gracefully for consistency with the other platforms. Add wxDidCreatePaintDC global variable which is reset before generating wxPaintEvent and set to true when ::BeginPaint() is called from wxPaintDC ctor and validate the update region of the window ourselves if it wasn't set (meaning that wxPaintDC wasn't created). Update the documentation to emphasize the link between EVT_PAINT handlers and wxPaintDC but without saying that wxPaintDC object must always be created in the handler as this is not true any more. git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@63229 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
parent
975fb32b5c
commit
7ca106e860
@ -517,6 +517,7 @@ MSW:
|
|||||||
- Worked around child window and caret positioning bug (in Windows) when using
|
- Worked around child window and caret positioning bug (in Windows) when using
|
||||||
wxBORDER_THEME in a container window.
|
wxBORDER_THEME in a container window.
|
||||||
- Suppressed spurious character event for decimal key in numeric keypad.
|
- Suppressed spurious character event for decimal key in numeric keypad.
|
||||||
|
- Allow to not create wxPaintDC in EVT_PAINT handler.
|
||||||
|
|
||||||
i18n:
|
i18n:
|
||||||
|
|
||||||
|
@ -1547,16 +1547,8 @@ public:
|
|||||||
|
|
||||||
A paint event is sent when a window's contents needs to be repainted.
|
A paint event is sent when a window's contents needs to be repainted.
|
||||||
|
|
||||||
Please notice that in general it is impossible to change the drawing of a
|
The handler of this event must create a wxPaintDC object and use it for
|
||||||
standard control (such as wxButton) and so you shouldn't attempt to handle
|
painting the window contents. For example:
|
||||||
paint events for them as even if it might work on some platforms, this is
|
|
||||||
inherently not portable and won't work everywhere.
|
|
||||||
|
|
||||||
@remarks
|
|
||||||
Note that in a paint event handler, the application must always create a
|
|
||||||
wxPaintDC object, even if you do not use it. Otherwise, under MS Windows,
|
|
||||||
refreshing for this and other windows will go wrong.
|
|
||||||
For example:
|
|
||||||
@code
|
@code
|
||||||
void MyWindow::OnPaint(wxPaintEvent& event)
|
void MyWindow::OnPaint(wxPaintEvent& event)
|
||||||
{
|
{
|
||||||
@ -1565,6 +1557,12 @@ public:
|
|||||||
DrawMyDocument(dc);
|
DrawMyDocument(dc);
|
||||||
}
|
}
|
||||||
@endcode
|
@endcode
|
||||||
|
|
||||||
|
Notice that you must @e not create other kinds of wxDC (e.g. wxClientDC or
|
||||||
|
wxWindowDC) in EVT_PAINT handlers and also don't create wxPaintDC outside
|
||||||
|
of this event handlers.
|
||||||
|
|
||||||
|
|
||||||
You can optimize painting by retrieving the rectangles that have been damaged
|
You can optimize painting by retrieving the rectangles that have been damaged
|
||||||
and only repainting these. The rectangles are in terms of the client area,
|
and only repainting these. The rectangles are in terms of the client area,
|
||||||
and are unscrolled, so you will need to do some calculations using the current
|
and are unscrolled, so you will need to do some calculations using the current
|
||||||
@ -1601,6 +1599,12 @@ public:
|
|||||||
}
|
}
|
||||||
@endcode
|
@endcode
|
||||||
|
|
||||||
|
@remarks
|
||||||
|
Please notice that in general it is impossible to change the drawing of a
|
||||||
|
standard control (such as wxButton) and so you shouldn't attempt to handle
|
||||||
|
paint events for them as even if it might work on some platforms, this is
|
||||||
|
inherently not portable and won't work everywhere.
|
||||||
|
|
||||||
|
|
||||||
@beginEventTable{wxPaintEvent}
|
@beginEventTable{wxPaintEvent}
|
||||||
@event{EVT_PAINT(func)}
|
@event{EVT_PAINT(func)}
|
||||||
|
@ -236,6 +236,11 @@ wxPaintDCImpl::wxPaintDCImpl( wxDC *owner, wxWindow *window ) :
|
|||||||
}
|
}
|
||||||
else // not in cache, create a new one
|
else // not in cache, create a new one
|
||||||
{
|
{
|
||||||
|
// see comments in src/msw/window.cpp where this is defined
|
||||||
|
extern bool wxDidCreatePaintDC;
|
||||||
|
|
||||||
|
wxDidCreatePaintDC = true;
|
||||||
|
|
||||||
m_hDC = (WXHDC)::BeginPaint(GetHwndOf(m_window), &g_paintStruct);
|
m_hDC = (WXHDC)::BeginPaint(GetHwndOf(m_window), &g_paintStruct);
|
||||||
if (m_hDC)
|
if (m_hDC)
|
||||||
ms_cache.Add(new wxPaintDCInfo(m_window, this));
|
ms_cache.Add(new wxPaintDCInfo(m_window, this));
|
||||||
|
@ -4796,6 +4796,14 @@ wxColour wxWindowMSW::MSWGetThemeColour(const wchar_t *themeName,
|
|||||||
// painting
|
// painting
|
||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// this variable is used to check that a paint event handler which processed
|
||||||
|
// the event did create a wxPaintDC inside its code and called BeginPaint() to
|
||||||
|
// validate the invalidated window area as otherwise we'd keep getting an
|
||||||
|
// endless stream of WM_PAINT messages for this window resulting in a lot of
|
||||||
|
// difficult to debug problems (e.g. impossibility to repaint other windows,
|
||||||
|
// lack of timer and idle events and so on)
|
||||||
|
extern bool wxDidCreatePaintDC = false;
|
||||||
|
|
||||||
bool wxWindowMSW::HandlePaint()
|
bool wxWindowMSW::HandlePaint()
|
||||||
{
|
{
|
||||||
HRGN hRegion = ::CreateRectRgn(0, 0, 0, 0); // Dummy call to get a handle
|
HRGN hRegion = ::CreateRectRgn(0, 0, 0, 0); // Dummy call to get a handle
|
||||||
@ -4810,11 +4818,20 @@ bool wxWindowMSW::HandlePaint()
|
|||||||
|
|
||||||
m_updateRegion = wxRegion((WXHRGN) hRegion);
|
m_updateRegion = wxRegion((WXHRGN) hRegion);
|
||||||
|
|
||||||
|
wxDidCreatePaintDC = false;
|
||||||
|
|
||||||
wxPaintEvent event(m_windowId);
|
wxPaintEvent event(m_windowId);
|
||||||
event.SetEventObject(this);
|
event.SetEventObject(this);
|
||||||
|
|
||||||
bool processed = HandleWindowEvent(event);
|
bool processed = HandleWindowEvent(event);
|
||||||
|
|
||||||
|
if ( processed && !wxDidCreatePaintDC )
|
||||||
|
{
|
||||||
|
// do call MSWDefWindowProc() to validate the update region to avoid
|
||||||
|
// the problems mentioned above
|
||||||
|
processed = false;
|
||||||
|
}
|
||||||
|
|
||||||
// note that we must generate NC event after the normal one as otherwise
|
// note that we must generate NC event after the normal one as otherwise
|
||||||
// BeginPaint() will happily overwrite our decorations with the background
|
// BeginPaint() will happily overwrite our decorations with the background
|
||||||
// colour
|
// colour
|
||||||
|
Loading…
Reference in New Issue
Block a user