bb87b19b16
git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@58694 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
396 lines
13 KiB
C++
396 lines
13 KiB
C++
///////////////////////////////////////////////////////////////////////////////
|
|
// Name: tests/events/evthandler.cpp
|
|
// Purpose: Test the new event types and wxEvtHandler-methods
|
|
// Author: Peter Most
|
|
// Created: 2009-01-24
|
|
// RCS-ID: $Id$
|
|
// Copyright: (c) 2009 Peter Most
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// headers
|
|
// ----------------------------------------------------------------------------
|
|
|
|
#include "testprec.h"
|
|
|
|
#ifdef __BORLANDC__
|
|
#pragma hdrstop
|
|
#endif
|
|
|
|
#include "wx/event.h"
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// test events and their handlers
|
|
// ----------------------------------------------------------------------------
|
|
|
|
const wxEventType LegacyEventType = wxNewEventType();
|
|
|
|
class MyEvent;
|
|
wxDEFINE_EVENT( MyEventType, MyEvent )
|
|
|
|
class MyEvent : public wxEvent
|
|
{
|
|
public:
|
|
MyEvent() : wxEvent(0, MyEventType) { }
|
|
|
|
virtual wxEvent *Clone() const { return new MyEvent; }
|
|
};
|
|
|
|
typedef void (wxEvtHandler::*MyEventFunction)(MyEvent&);
|
|
#define MyEventHandler(func) wxEVENT_HANDLER_CAST(MyEventFunction, func)
|
|
#define EVT_MYEVENT(func) \
|
|
wx__DECLARE_EVT0(MyEventType, MyEventHandler(func))
|
|
|
|
class AnotherEvent : public wxEvent
|
|
{
|
|
};
|
|
|
|
namespace
|
|
{
|
|
|
|
struct Called
|
|
{
|
|
Called() { Reset(); }
|
|
|
|
void Reset()
|
|
{
|
|
function =
|
|
functor =
|
|
method =
|
|
smethod = false;
|
|
}
|
|
|
|
bool function,
|
|
functor,
|
|
method,
|
|
smethod;
|
|
} g_called;
|
|
|
|
void GlobalOnMyEvent(MyEvent&)
|
|
{
|
|
g_called.function = true;
|
|
}
|
|
|
|
void GlobalOnAnotherEvent(AnotherEvent&);
|
|
|
|
void GlobalOnIdle(wxIdleEvent&)
|
|
{
|
|
g_called.function = true;
|
|
}
|
|
|
|
struct MyFunctor
|
|
{
|
|
void operator()(MyEvent &) { g_called.functor = true; }
|
|
};
|
|
|
|
struct IdleFunctor
|
|
{
|
|
void operator()(wxIdleEvent &) { g_called.functor = true; }
|
|
};
|
|
|
|
class MyHandler : public wxEvtHandler
|
|
{
|
|
public:
|
|
static void StaticOnMyEvent(MyEvent &) { g_called.smethod = true; }
|
|
static void StaticOnAnotherEvent(AnotherEvent &);
|
|
static void StaticOnIdle(wxIdleEvent&) { g_called.smethod = true; }
|
|
|
|
void OnMyEvent(MyEvent&) { g_called.method = true; }
|
|
void OnEvent(wxEvent&) { g_called.method = true; }
|
|
void OnAnotherEvent(AnotherEvent&);
|
|
void OnIdle(wxIdleEvent&) { g_called.method = true; }
|
|
};
|
|
|
|
// we can also handle events in classes not deriving from wxEvtHandler
|
|
struct MySink
|
|
{
|
|
void OnMyEvent(MyEvent&) { g_called.method = true; }
|
|
void OnEvent(wxEvent&) { g_called.method = true; }
|
|
void OnIdle(wxIdleEvent&) { g_called.method = true; }
|
|
};
|
|
|
|
// also test event table compilation
|
|
class MyClassWithEventTable : public wxEvtHandler
|
|
{
|
|
public:
|
|
void OnMyEvent(MyEvent&) { g_called.method = true; }
|
|
void OnEvent(wxEvent&) { g_called.method = true; }
|
|
void OnAnotherEvent(AnotherEvent&);
|
|
void OnIdle(wxIdleEvent&) { g_called.method = true; }
|
|
|
|
private:
|
|
DECLARE_EVENT_TABLE()
|
|
};
|
|
|
|
BEGIN_EVENT_TABLE(MyClassWithEventTable, wxEvtHandler)
|
|
EVT_IDLE(MyClassWithEventTable::OnIdle)
|
|
|
|
EVT_MYEVENT(MyClassWithEventTable::OnMyEvent)
|
|
#if !wxEVENTS_COMPATIBILITY_2_8
|
|
EVT_MYEVENT(MyClassWithEventTable::OnEvent)
|
|
#endif
|
|
|
|
// this shouldn't compile:
|
|
//EVT_MYEVENT(MyClassWithEventTable::OnIdle)
|
|
//EVT_IDLE(MyClassWithEventTable::OnAnotherEvent)
|
|
END_EVENT_TABLE()
|
|
|
|
} // anonymous namespace
|
|
|
|
|
|
// --------------------------------------------------------------------------
|
|
// test class
|
|
// --------------------------------------------------------------------------
|
|
|
|
class EvtHandlerTestCase : public CppUnit::TestCase
|
|
{
|
|
public:
|
|
EvtHandlerTestCase() {}
|
|
|
|
private:
|
|
CPPUNIT_TEST_SUITE( EvtHandlerTestCase );
|
|
CPPUNIT_TEST( BuiltinConnect );
|
|
CPPUNIT_TEST( LegacyConnect );
|
|
#if !wxEVENTS_COMPATIBILITY_2_8
|
|
CPPUNIT_TEST( ConnectFunction );
|
|
CPPUNIT_TEST( ConnectStaticMethod );
|
|
CPPUNIT_TEST( ConnectFunctor );
|
|
CPPUNIT_TEST( ConnectMethod );
|
|
CPPUNIT_TEST( ConnectMethodUsingBaseEvent );
|
|
CPPUNIT_TEST( ConnectMethodWithSink );
|
|
CPPUNIT_TEST( ConnectNonHandler );
|
|
CPPUNIT_TEST( StaticConnect );
|
|
CPPUNIT_TEST( InvalidConnect );
|
|
#endif // !wxEVENTS_COMPATIBILITY_2_8
|
|
CPPUNIT_TEST_SUITE_END();
|
|
|
|
void BuiltinConnect();
|
|
void LegacyConnect();
|
|
#if !wxEVENTS_COMPATIBILITY_2_8
|
|
void ConnectFunction();
|
|
void ConnectStaticMethod();
|
|
void ConnectFunctor();
|
|
void ConnectMethod();
|
|
void ConnectMethodUsingBaseEvent();
|
|
void ConnectMethodWithSink();
|
|
void ConnectNonHandler();
|
|
void StaticConnect();
|
|
void InvalidConnect();
|
|
#endif // !wxEVENTS_COMPATIBILITY_2_8
|
|
|
|
|
|
// these member variables exceptionally don't use "m_" prefix because
|
|
// they're used so many times
|
|
MyHandler handler;
|
|
MyEvent e;
|
|
|
|
DECLARE_NO_COPY_CLASS(EvtHandlerTestCase)
|
|
};
|
|
|
|
// register in the unnamed registry so that these tests are run by default
|
|
CPPUNIT_TEST_SUITE_REGISTRATION( EvtHandlerTestCase );
|
|
|
|
// also include in it's own registry so that these tests can be run alone
|
|
CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( EvtHandlerTestCase, "EvtHandlerTestCase" );
|
|
|
|
void EvtHandlerTestCase::BuiltinConnect()
|
|
{
|
|
handler.Connect(wxEVT_IDLE, wxIdleEventHandler(MyHandler::OnIdle));
|
|
handler.Disconnect(wxEVT_IDLE, wxIdleEventHandler(MyHandler::OnIdle));
|
|
|
|
handler.Connect(wxEVT_IDLE, wxIdleEventHandler(MyHandler::OnIdle), NULL, &handler);
|
|
handler.Disconnect(wxEVT_IDLE, wxIdleEventHandler(MyHandler::OnIdle), NULL, &handler);
|
|
|
|
// using casts like this is even uglier than using wxIdleEventHandler but
|
|
// it should still continue to work for compatibility
|
|
handler.Connect(wxEVT_IDLE, (wxObjectEventFunction)(wxEventFunction)&MyHandler::OnIdle);
|
|
handler.Disconnect(wxEVT_IDLE, (wxObjectEventFunction)(wxEventFunction)&MyHandler::OnIdle);
|
|
|
|
#if !wxEVENTS_COMPATIBILITY_2_8
|
|
handler.Connect(wxEVT_IDLE, GlobalOnIdle);
|
|
handler.Disconnect(wxEVT_IDLE, GlobalOnIdle);
|
|
|
|
IdleFunctor f;
|
|
handler.Connect(wxEVT_IDLE, f);
|
|
handler.Disconnect(wxEVT_IDLE, f);
|
|
|
|
handler.Connect(wxEVT_IDLE, &MyHandler::OnIdle);
|
|
handler.Disconnect(wxEVT_IDLE, &MyHandler::OnIdle);
|
|
|
|
handler.Connect(wxEVT_IDLE, &MyHandler::StaticOnIdle);
|
|
handler.Disconnect(wxEVT_IDLE, &MyHandler::StaticOnIdle);
|
|
#endif // !wxEVENTS_COMPATIBILITY_2_8
|
|
}
|
|
|
|
void EvtHandlerTestCase::LegacyConnect()
|
|
{
|
|
handler.Connect( LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent );
|
|
handler.Connect( 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent );
|
|
handler.Connect( 0, 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent );
|
|
|
|
handler.Disconnect( LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent );
|
|
handler.Disconnect( 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent );
|
|
handler.Disconnect( 0, 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent );
|
|
|
|
|
|
handler.Connect( LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent, NULL, &handler );
|
|
handler.Connect( 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent, NULL, &handler );
|
|
handler.Connect( 0, 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent, NULL, &handler );
|
|
|
|
handler.Disconnect( LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent, NULL, &handler );
|
|
handler.Disconnect( 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent, NULL, &handler );
|
|
handler.Disconnect( 0, 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent, NULL, &handler );
|
|
}
|
|
|
|
#if !wxEVENTS_COMPATIBILITY_2_8
|
|
|
|
void EvtHandlerTestCase::ConnectFunction()
|
|
{
|
|
// function tests
|
|
handler.Connect( MyEventType, GlobalOnMyEvent );
|
|
g_called.Reset();
|
|
handler.ProcessEvent(e);
|
|
CPPUNIT_ASSERT( g_called.function );
|
|
handler.Disconnect( MyEventType, GlobalOnMyEvent );
|
|
g_called.Reset();
|
|
handler.ProcessEvent(e);
|
|
CPPUNIT_ASSERT( !g_called.function ); // check that it was disconnected
|
|
|
|
handler.Connect( 0, MyEventType, GlobalOnMyEvent );
|
|
handler.Disconnect( 0, MyEventType, GlobalOnMyEvent );
|
|
|
|
handler.Connect( 0, 0, MyEventType, GlobalOnMyEvent );
|
|
handler.Disconnect( 0, 0, MyEventType, GlobalOnMyEvent );
|
|
}
|
|
|
|
void EvtHandlerTestCase::ConnectStaticMethod()
|
|
{
|
|
// static method tests (this is same as functions but still test it just in
|
|
// case we hit some strange compiler bugs)
|
|
handler.Connect( MyEventType, &MyHandler::StaticOnMyEvent );
|
|
g_called.Reset();
|
|
handler.ProcessEvent(e);
|
|
CPPUNIT_ASSERT( g_called.smethod );
|
|
handler.Disconnect( MyEventType, &MyHandler::StaticOnMyEvent );
|
|
g_called.Reset();
|
|
handler.ProcessEvent(e);
|
|
CPPUNIT_ASSERT( !g_called.smethod );
|
|
|
|
handler.Connect( 0, MyEventType, &MyHandler::StaticOnMyEvent );
|
|
handler.Disconnect( 0, MyEventType, &MyHandler::StaticOnMyEvent );
|
|
|
|
handler.Connect( 0, 0, MyEventType, &MyHandler::StaticOnMyEvent );
|
|
handler.Disconnect( 0, 0, MyEventType, &MyHandler::StaticOnMyEvent );
|
|
}
|
|
|
|
void EvtHandlerTestCase::ConnectFunctor()
|
|
{
|
|
// generalized functor tests
|
|
MyFunctor functor;
|
|
|
|
handler.Connect( MyEventType, functor );
|
|
g_called.Reset();
|
|
handler.ProcessEvent(e);
|
|
CPPUNIT_ASSERT( g_called.functor );
|
|
handler.Disconnect( MyEventType, functor );
|
|
g_called.Reset();
|
|
handler.ProcessEvent(e);
|
|
CPPUNIT_ASSERT( !g_called.functor );
|
|
|
|
handler.Connect( 0, MyEventType, functor );
|
|
handler.Disconnect( 0, MyEventType, functor );
|
|
|
|
handler.Connect( 0, 0, MyEventType, functor );
|
|
handler.Disconnect( 0, 0, MyEventType, functor );
|
|
}
|
|
|
|
void EvtHandlerTestCase::ConnectMethod()
|
|
{
|
|
// class method tests
|
|
handler.Connect( MyEventType, &MyHandler::OnMyEvent );
|
|
g_called.Reset();
|
|
handler.ProcessEvent(e);
|
|
CPPUNIT_ASSERT( g_called.method );
|
|
handler.Disconnect( MyEventType, &MyHandler::OnMyEvent );
|
|
g_called.Reset();
|
|
handler.ProcessEvent(e);
|
|
CPPUNIT_ASSERT( !g_called.method );
|
|
|
|
handler.Connect( 0, MyEventType, &MyHandler::OnMyEvent );
|
|
handler.Disconnect( 0, MyEventType, &MyHandler::OnMyEvent );
|
|
|
|
handler.Connect( 0, 0, MyEventType, &MyHandler::OnMyEvent );
|
|
handler.Disconnect( 0, 0, MyEventType, &MyHandler::OnMyEvent );
|
|
}
|
|
|
|
void EvtHandlerTestCase::ConnectMethodUsingBaseEvent()
|
|
{
|
|
// test connecting a method taking just wxEvent and not MyEvent: this
|
|
// should work too if we don't need any MyEvent-specific information in the
|
|
// handler
|
|
handler.Connect( MyEventType, &MyHandler::OnEvent );
|
|
g_called.Reset();
|
|
handler.ProcessEvent(e);
|
|
CPPUNIT_ASSERT( g_called.method );
|
|
handler.Disconnect( MyEventType, &MyHandler::OnEvent );
|
|
g_called.Reset();
|
|
handler.ProcessEvent(e);
|
|
CPPUNIT_ASSERT( !g_called.method );
|
|
|
|
handler.Connect( 0, MyEventType, &MyHandler::OnEvent );
|
|
handler.Disconnect( 0, MyEventType, &MyHandler::OnEvent );
|
|
|
|
handler.Connect( 0, 0, MyEventType, &MyHandler::OnEvent );
|
|
handler.Disconnect( 0, 0, MyEventType, &MyHandler::OnEvent );
|
|
}
|
|
|
|
void EvtHandlerTestCase::ConnectMethodWithSink()
|
|
{
|
|
handler.Connect( MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
|
|
handler.Connect( 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
|
|
handler.Connect( 0, 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
|
|
|
|
handler.Disconnect( MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
|
|
handler.Disconnect( 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
|
|
handler.Disconnect( 0, 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
|
|
}
|
|
|
|
void EvtHandlerTestCase::ConnectNonHandler()
|
|
{
|
|
// class method tests for class not derived from wxEvtHandler
|
|
MySink sink;
|
|
|
|
handler.Connect( MyEventType, &MySink::OnMyEvent, NULL, &sink );
|
|
g_called.Reset();
|
|
handler.ProcessEvent(e);
|
|
CPPUNIT_ASSERT( g_called.method );
|
|
handler.Disconnect( MyEventType, &MySink::OnMyEvent, NULL, &sink );
|
|
g_called.Reset();
|
|
handler.ProcessEvent(e);
|
|
CPPUNIT_ASSERT( !g_called.method );
|
|
}
|
|
|
|
void EvtHandlerTestCase::StaticConnect()
|
|
{
|
|
wxEvtHandler::Connect( &handler, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
|
|
wxEvtHandler::Connect( &handler, 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
|
|
wxEvtHandler::Connect( &handler, 0, 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
|
|
|
|
wxEvtHandler::Disconnect( &handler, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
|
|
wxEvtHandler::Disconnect( &handler, 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
|
|
wxEvtHandler::Disconnect( &handler, 0, 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
|
|
}
|
|
|
|
void EvtHandlerTestCase::InvalidConnect()
|
|
{
|
|
// these calls shouldn't compile but we unfortunately can't check this
|
|
// automatically, you need to uncomment them manually and test that
|
|
// compilation does indeed fail
|
|
//handler.Connect(MyEventType, GlobalOnAnotherEvent);
|
|
//IdleFunctor f; handler.Connect(MyEventType, f);
|
|
//handler.Connect(MyEventType, &MyHandler::StaticOnAnotherEvent);
|
|
//handler.Connect(MyEventType, &MyHandler::OnAnotherEvent);
|
|
}
|
|
|
|
#endif // !wxEVENTS_COMPATIBILITY_2_8
|