]> git.saurik.com Git - wxWidgets.git/blame_incremental - tests/events/evthandler.cpp
fixed test failures under wxMSW when using our wxVsnprintf implementation (remains...
[wxWidgets.git] / tests / events / evthandler.cpp
... / ...
CommitLineData
1///////////////////////////////////////////////////////////////////////////////
2// Name: tests/events/evthandler.cpp
3// Purpose: Test the new event types and wxEvtHandler-methods
4// Author: Peter Most
5// Created: 2009-01-24
6// RCS-ID: $Id$
7// Copyright: (c) 2009 Peter Most
8///////////////////////////////////////////////////////////////////////////////
9
10// ----------------------------------------------------------------------------
11// headers
12// ----------------------------------------------------------------------------
13
14#include "testprec.h"
15
16#ifdef __BORLANDC__
17 #pragma hdrstop
18#endif
19
20#include "wx/event.h"
21
22// ----------------------------------------------------------------------------
23// test events and their handlers
24// ----------------------------------------------------------------------------
25
26const wxEventType LegacyEventType = wxNewEventType();
27
28class MyEvent;
29wxDEFINE_EVENT(MyEventType, MyEvent);
30
31class MyEvent : public wxEvent
32{
33public:
34 MyEvent() : wxEvent(0, MyEventType) { }
35
36 virtual wxEvent *Clone() const { return new MyEvent; }
37};
38
39typedef void (wxEvtHandler::*MyEventFunction)(MyEvent&);
40#define MyEventHandler(func) wxEVENT_HANDLER_CAST(MyEventFunction, func)
41#define EVT_MYEVENT(func) \
42 wx__DECLARE_EVT0(MyEventType, MyEventHandler(func))
43
44class AnotherEvent : public wxEvent
45{
46};
47
48namespace
49{
50
51struct Called
52{
53 Called() { Reset(); }
54
55 void Reset()
56 {
57 function =
58 functor =
59 method =
60 smethod = false;
61 }
62
63 bool function,
64 functor,
65 method,
66 smethod;
67} g_called;
68
69void GlobalOnMyEvent(MyEvent&)
70{
71 g_called.function = true;
72}
73
74void GlobalOnAnotherEvent(AnotherEvent&);
75
76void GlobalOnIdle(wxIdleEvent&)
77{
78 g_called.function = true;
79}
80
81struct MyFunctor
82{
83 void operator()(MyEvent &) { g_called.functor = true; }
84};
85
86struct IdleFunctor
87{
88 void operator()(wxIdleEvent &) { g_called.functor = true; }
89};
90
91class MyHandler : public wxEvtHandler
92{
93public:
94 static void StaticOnMyEvent(MyEvent &) { g_called.smethod = true; }
95 static void StaticOnAnotherEvent(AnotherEvent &);
96 static void StaticOnIdle(wxIdleEvent&) { g_called.smethod = true; }
97
98 void OnMyEvent(MyEvent&) { g_called.method = true; }
99 void OnEvent(wxEvent&) { g_called.method = true; }
100 void OnAnotherEvent(AnotherEvent&);
101 void OnIdle(wxIdleEvent&) { g_called.method = true; }
102};
103
104// we can also handle events in classes not deriving from wxEvtHandler
105struct MySink
106{
107 void OnMyEvent(MyEvent&) { g_called.method = true; }
108 void OnEvent(wxEvent&) { g_called.method = true; }
109 void OnIdle(wxIdleEvent&) { g_called.method = true; }
110};
111
112// also test event table compilation
113class MyClassWithEventTable : public wxEvtHandler
114{
115public:
116 void OnMyEvent(MyEvent&) { g_called.method = true; }
117 void OnEvent(wxEvent&) { g_called.method = true; }
118 void OnAnotherEvent(AnotherEvent&);
119 void OnIdle(wxIdleEvent&) { g_called.method = true; }
120
121private:
122 DECLARE_EVENT_TABLE()
123};
124
125BEGIN_EVENT_TABLE(MyClassWithEventTable, wxEvtHandler)
126 EVT_IDLE(MyClassWithEventTable::OnIdle)
127
128 EVT_MYEVENT(MyClassWithEventTable::OnMyEvent)
129#if !wxEVENTS_COMPATIBILITY_2_8
130 EVT_MYEVENT(MyClassWithEventTable::OnEvent)
131#endif
132
133 // this shouldn't compile:
134 //EVT_MYEVENT(MyClassWithEventTable::OnIdle)
135 //EVT_IDLE(MyClassWithEventTable::OnAnotherEvent)
136END_EVENT_TABLE()
137
138} // anonymous namespace
139
140
141// --------------------------------------------------------------------------
142// test class
143// --------------------------------------------------------------------------
144
145class EvtHandlerTestCase : public CppUnit::TestCase
146{
147public:
148 EvtHandlerTestCase() {}
149
150private:
151 CPPUNIT_TEST_SUITE( EvtHandlerTestCase );
152 CPPUNIT_TEST( BuiltinConnect );
153 CPPUNIT_TEST( LegacyConnect );
154#if !wxEVENTS_COMPATIBILITY_2_8
155 CPPUNIT_TEST( ConnectFunction );
156 CPPUNIT_TEST( ConnectStaticMethod );
157 CPPUNIT_TEST( ConnectFunctor );
158 CPPUNIT_TEST( ConnectMethod );
159 CPPUNIT_TEST( ConnectMethodUsingBaseEvent );
160 CPPUNIT_TEST( ConnectMethodWithSink );
161 CPPUNIT_TEST( ConnectNonHandler );
162 CPPUNIT_TEST( StaticConnect );
163 CPPUNIT_TEST( InvalidConnect );
164#endif // !wxEVENTS_COMPATIBILITY_2_8
165 CPPUNIT_TEST_SUITE_END();
166
167 void BuiltinConnect();
168 void LegacyConnect();
169#if !wxEVENTS_COMPATIBILITY_2_8
170 void ConnectFunction();
171 void ConnectStaticMethod();
172 void ConnectFunctor();
173 void ConnectMethod();
174 void ConnectMethodUsingBaseEvent();
175 void ConnectMethodWithSink();
176 void ConnectNonHandler();
177 void StaticConnect();
178 void InvalidConnect();
179#endif // !wxEVENTS_COMPATIBILITY_2_8
180
181
182 // these member variables exceptionally don't use "m_" prefix because
183 // they're used so many times
184 MyHandler handler;
185 MyEvent e;
186
187 DECLARE_NO_COPY_CLASS(EvtHandlerTestCase)
188};
189
190// register in the unnamed registry so that these tests are run by default
191CPPUNIT_TEST_SUITE_REGISTRATION( EvtHandlerTestCase );
192
193// also include in it's own registry so that these tests can be run alone
194CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( EvtHandlerTestCase, "EvtHandlerTestCase" );
195
196void EvtHandlerTestCase::BuiltinConnect()
197{
198 handler.Connect(wxEVT_IDLE, wxIdleEventHandler(MyHandler::OnIdle));
199 handler.Disconnect(wxEVT_IDLE, wxIdleEventHandler(MyHandler::OnIdle));
200
201 handler.Connect(wxEVT_IDLE, wxIdleEventHandler(MyHandler::OnIdle), NULL, &handler);
202 handler.Disconnect(wxEVT_IDLE, wxIdleEventHandler(MyHandler::OnIdle), NULL, &handler);
203
204 // using casts like this is even uglier than using wxIdleEventHandler but
205 // it should still continue to work for compatibility
206 handler.Connect(wxEVT_IDLE, (wxObjectEventFunction)(wxEventFunction)&MyHandler::OnIdle);
207 handler.Disconnect(wxEVT_IDLE, (wxObjectEventFunction)(wxEventFunction)&MyHandler::OnIdle);
208
209#if !wxEVENTS_COMPATIBILITY_2_8
210 handler.Connect(wxEVT_IDLE, GlobalOnIdle);
211 handler.Disconnect(wxEVT_IDLE, GlobalOnIdle);
212
213 IdleFunctor f;
214 handler.Connect(wxEVT_IDLE, f);
215 handler.Disconnect(wxEVT_IDLE, f);
216
217 handler.Connect(wxEVT_IDLE, &MyHandler::OnIdle);
218 handler.Disconnect(wxEVT_IDLE, &MyHandler::OnIdle);
219
220 handler.Connect(wxEVT_IDLE, &MyHandler::StaticOnIdle);
221 handler.Disconnect(wxEVT_IDLE, &MyHandler::StaticOnIdle);
222#endif // !wxEVENTS_COMPATIBILITY_2_8
223}
224
225void EvtHandlerTestCase::LegacyConnect()
226{
227 handler.Connect( LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent );
228 handler.Connect( 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent );
229 handler.Connect( 0, 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent );
230
231 handler.Disconnect( LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent );
232 handler.Disconnect( 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent );
233 handler.Disconnect( 0, 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent );
234
235
236 handler.Connect( LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent, NULL, &handler );
237 handler.Connect( 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent, NULL, &handler );
238 handler.Connect( 0, 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent, NULL, &handler );
239
240 handler.Disconnect( LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent, NULL, &handler );
241 handler.Disconnect( 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent, NULL, &handler );
242 handler.Disconnect( 0, 0, LegacyEventType, (wxObjectEventFunction)&MyHandler::OnEvent, NULL, &handler );
243}
244
245#if !wxEVENTS_COMPATIBILITY_2_8
246
247void EvtHandlerTestCase::ConnectFunction()
248{
249 // function tests
250 handler.Connect( MyEventType, GlobalOnMyEvent );
251 g_called.Reset();
252 handler.ProcessEvent(e);
253 CPPUNIT_ASSERT( g_called.function );
254 handler.Disconnect( MyEventType, GlobalOnMyEvent );
255 g_called.Reset();
256 handler.ProcessEvent(e);
257 CPPUNIT_ASSERT( !g_called.function ); // check that it was disconnected
258
259 handler.Connect( 0, MyEventType, GlobalOnMyEvent );
260 handler.Disconnect( 0, MyEventType, GlobalOnMyEvent );
261
262 handler.Connect( 0, 0, MyEventType, GlobalOnMyEvent );
263 handler.Disconnect( 0, 0, MyEventType, GlobalOnMyEvent );
264}
265
266void EvtHandlerTestCase::ConnectStaticMethod()
267{
268 // static method tests (this is same as functions but still test it just in
269 // case we hit some strange compiler bugs)
270 handler.Connect( MyEventType, &MyHandler::StaticOnMyEvent );
271 g_called.Reset();
272 handler.ProcessEvent(e);
273 CPPUNIT_ASSERT( g_called.smethod );
274 handler.Disconnect( MyEventType, &MyHandler::StaticOnMyEvent );
275 g_called.Reset();
276 handler.ProcessEvent(e);
277 CPPUNIT_ASSERT( !g_called.smethod );
278
279 handler.Connect( 0, MyEventType, &MyHandler::StaticOnMyEvent );
280 handler.Disconnect( 0, MyEventType, &MyHandler::StaticOnMyEvent );
281
282 handler.Connect( 0, 0, MyEventType, &MyHandler::StaticOnMyEvent );
283 handler.Disconnect( 0, 0, MyEventType, &MyHandler::StaticOnMyEvent );
284}
285
286void EvtHandlerTestCase::ConnectFunctor()
287{
288 // generalized functor tests
289 MyFunctor functor;
290
291 handler.Connect( MyEventType, functor );
292 g_called.Reset();
293 handler.ProcessEvent(e);
294 CPPUNIT_ASSERT( g_called.functor );
295 handler.Disconnect( MyEventType, functor );
296 g_called.Reset();
297 handler.ProcessEvent(e);
298 CPPUNIT_ASSERT( !g_called.functor );
299
300 handler.Connect( 0, MyEventType, functor );
301 handler.Disconnect( 0, MyEventType, functor );
302
303 handler.Connect( 0, 0, MyEventType, functor );
304 handler.Disconnect( 0, 0, MyEventType, functor );
305}
306
307void EvtHandlerTestCase::ConnectMethod()
308{
309 // class method tests
310 handler.Connect( MyEventType, &MyHandler::OnMyEvent );
311 g_called.Reset();
312 handler.ProcessEvent(e);
313 CPPUNIT_ASSERT( g_called.method );
314 handler.Disconnect( MyEventType, &MyHandler::OnMyEvent );
315 g_called.Reset();
316 handler.ProcessEvent(e);
317 CPPUNIT_ASSERT( !g_called.method );
318
319 handler.Connect( 0, MyEventType, &MyHandler::OnMyEvent );
320 handler.Disconnect( 0, MyEventType, &MyHandler::OnMyEvent );
321
322 handler.Connect( 0, 0, MyEventType, &MyHandler::OnMyEvent );
323 handler.Disconnect( 0, 0, MyEventType, &MyHandler::OnMyEvent );
324}
325
326void EvtHandlerTestCase::ConnectMethodUsingBaseEvent()
327{
328 // test connecting a method taking just wxEvent and not MyEvent: this
329 // should work too if we don't need any MyEvent-specific information in the
330 // handler
331 handler.Connect( MyEventType, &MyHandler::OnEvent );
332 g_called.Reset();
333 handler.ProcessEvent(e);
334 CPPUNIT_ASSERT( g_called.method );
335 handler.Disconnect( MyEventType, &MyHandler::OnEvent );
336 g_called.Reset();
337 handler.ProcessEvent(e);
338 CPPUNIT_ASSERT( !g_called.method );
339
340 handler.Connect( 0, MyEventType, &MyHandler::OnEvent );
341 handler.Disconnect( 0, MyEventType, &MyHandler::OnEvent );
342
343 handler.Connect( 0, 0, MyEventType, &MyHandler::OnEvent );
344 handler.Disconnect( 0, 0, MyEventType, &MyHandler::OnEvent );
345}
346
347void EvtHandlerTestCase::ConnectMethodWithSink()
348{
349 handler.Connect( MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
350 handler.Connect( 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
351 handler.Connect( 0, 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
352
353 handler.Disconnect( MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
354 handler.Disconnect( 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
355 handler.Disconnect( 0, 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
356}
357
358void EvtHandlerTestCase::ConnectNonHandler()
359{
360 // class method tests for class not derived from wxEvtHandler
361 MySink sink;
362
363 handler.Connect( MyEventType, &MySink::OnMyEvent, NULL, &sink );
364 g_called.Reset();
365 handler.ProcessEvent(e);
366 CPPUNIT_ASSERT( g_called.method );
367 handler.Disconnect( MyEventType, &MySink::OnMyEvent, NULL, &sink );
368 g_called.Reset();
369 handler.ProcessEvent(e);
370 CPPUNIT_ASSERT( !g_called.method );
371}
372
373void EvtHandlerTestCase::StaticConnect()
374{
375 wxEvtHandler::Connect( &handler, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
376 wxEvtHandler::Connect( &handler, 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
377 wxEvtHandler::Connect( &handler, 0, 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
378
379 wxEvtHandler::Disconnect( &handler, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
380 wxEvtHandler::Disconnect( &handler, 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
381 wxEvtHandler::Disconnect( &handler, 0, 0, MyEventType, &MyHandler::OnMyEvent, NULL, &handler );
382}
383
384void EvtHandlerTestCase::InvalidConnect()
385{
386 // these calls shouldn't compile but we unfortunately can't check this
387 // automatically, you need to uncomment them manually and test that
388 // compilation does indeed fail
389 //handler.Connect(MyEventType, GlobalOnAnotherEvent);
390 //IdleFunctor f; handler.Connect(MyEventType, f);
391 //handler.Connect(MyEventType, &MyHandler::StaticOnAnotherEvent);
392 //handler.Connect(MyEventType, &MyHandler::OnAnotherEvent);
393}
394
395#endif // !wxEVENTS_COMPATIBILITY_2_8