1 ///////////////////////////////////////////////////////////////////////////////
3 // Purpose: Test program for wxWidgets
4 // Author: Mike Wetherell
6 // Copyright: (c) 2004 Mike Wetherell
7 // Licence: wxWidgets licence
8 ///////////////////////////////////////////////////////////////////////////////
10 // For compilers that support precompilation, includes "wx/wx.h"
18 // for all others, include the necessary headers
23 #include "wx/beforestd.h"
24 #include <cppunit/TestListener.h>
25 #include <cppunit/Test.h>
26 #include <cppunit/TestResult.h>
27 #include "wx/afterstd.h"
29 #include "wx/cmdline.h"
33 using CppUnit::TestSuite
;
34 using CppUnit::TestFactoryRegistry
;
37 // Displays the test name. This allow for quick investigation on which test hangs
38 class DetailListener
: public CppUnit::TestListener
41 DetailListener(bool doTiming
= false):
42 CppUnit::TestListener(),
47 virtual void startTest(CppUnit::Test
*test
)
49 CppUnit::stdCOut() << test
->getName () << " ";
53 virtual void endTest(CppUnit::Test
* WXUNUSED(test
))
57 CppUnit::stdCOut() << " (in "<< m_watch
.Time() << " ms )";
58 CppUnit::stdCOut() << "\n";
69 typedef wxApp TestAppBase
;
71 typedef wxAppConsole TestAppBase
;
74 // The application class
76 class TestApp
: public TestAppBase
82 virtual void OnInitCmdLine(wxCmdLineParser
& parser
);
83 virtual bool OnCmdLineParsed(wxCmdLineParser
& parser
);
84 virtual bool OnInit();
89 void List(Test
*test
, const string
& parent
= "") const;
91 // command lines options/parameters
96 vector
<string
> m_registries
;
99 IMPLEMENT_APP_CONSOLE(TestApp
)
109 bool TestApp::OnInit()
111 if ( !TestAppBase::OnInit() )
114 cout
<< "Test program for wxWidgets\n"
115 << "build: " << WX_BUILD_OPTIONS_SIGNATURE
<< std::endl
;
118 // create a hidden parent window to be used as parent for the GUI controls
119 new wxFrame(NULL
, wxID_ANY
, "Hidden wx test frame");
125 // The table of command line options
127 void TestApp::OnInitCmdLine(wxCmdLineParser
& parser
)
129 TestAppBase::OnInitCmdLine(parser
);
131 static const wxCmdLineEntryDesc cmdLineDesc
[] = {
132 { wxCMD_LINE_SWITCH
, "l", "list",
133 "list the test suites, do not run them",
134 wxCMD_LINE_VAL_NONE
, 0 },
135 { wxCMD_LINE_SWITCH
, "L", "longlist",
136 "list the test cases, do not run them",
137 wxCMD_LINE_VAL_NONE
, 0 },
138 { wxCMD_LINE_SWITCH
, "d", "detail",
139 "print the test case names, run them",
140 wxCMD_LINE_VAL_NONE
, 0 },
141 { wxCMD_LINE_SWITCH
, "t", "timing",
142 "print names and mesure running time of individual test, run them",
143 wxCMD_LINE_VAL_NONE
, 0 },
144 { wxCMD_LINE_PARAM
, NULL
, NULL
, "REGISTRY", wxCMD_LINE_VAL_STRING
,
145 wxCMD_LINE_PARAM_OPTIONAL
| wxCMD_LINE_PARAM_MULTIPLE
},
149 parser
.SetDesc(cmdLineDesc
);
152 // Handle command line options
154 bool TestApp::OnCmdLineParsed(wxCmdLineParser
& parser
)
156 if (parser
.GetParamCount())
157 for (size_t i
= 0; i
< parser
.GetParamCount(); i
++)
158 m_registries
.push_back(string(parser
.GetParam(i
).mb_str()));
160 m_registries
.push_back("");
162 m_longlist
= parser
.Found(_T("longlist"));
163 m_list
= m_longlist
|| parser
.Found(_T("list"));
164 m_timing
= parser
.Found(_T("timing"));
165 m_detail
= !m_timing
&& parser
.Found(_T("detail"));
167 return TestAppBase::OnCmdLineParsed(parser
);
174 CppUnit::TextTestRunner runner
;
176 for (size_t i
= 0; i
< m_registries
.size(); i
++) {
177 auto_ptr
<Test
> test(m_registries
[i
].empty() ?
178 TestFactoryRegistry::getRegistry().makeTest() :
179 TestFactoryRegistry::getRegistry(m_registries
[i
]).makeTest());
181 TestSuite
*suite
= dynamic_cast<TestSuite
*>(test
.get());
183 if (suite
&& suite
->countTestCases() == 0)
184 wxLogError(_T("No such test suite: %s"),
185 wxString(m_registries
[i
].c_str(), wxConvUTF8
).c_str());
189 runner
.addTest(test
.release());
195 runner
.setOutputter(new CppUnit::CompilerOutputter(&runner
.result(), cout
));
198 // Switch off logging unless --verbose
199 bool verbose
= wxLog::GetVerbose();
200 wxLog::EnableLogging(verbose
);
202 bool verbose
= false;
206 // (http://sf.net/tracker/index.php?func=detail&aid=1649369&group_id=11795&atid=111795)
207 // in some versions of cppunit: they write progress dots to cout (and not
208 // cerr) and don't flush it so all the dots appear at once at the end which
209 // is not very useful so unbuffer cout to work around this
210 cout
.setf(ios::unitbuf
);
212 // add detail listener if needed
213 DetailListener
detailListener(m_timing
);
214 if ( m_detail
|| m_timing
)
215 runner
.eventManager().addListener(&detailListener
);
217 return runner
.run("", false, true, !verbose
) ? EXIT_SUCCESS
: EXIT_FAILURE
;
220 int TestApp::OnExit()
223 delete GetTopWindow();
231 void TestApp::List(Test
*test
, const string
& parent
/*=""*/) const
233 TestSuite
*suite
= dynamic_cast<TestSuite
*>(test
);
237 // take the last component of the name and append to the parent
238 name
= test
->getName();
239 string::size_type i
= name
.find_last_of(".:");
240 if (i
!= string::npos
)
241 name
= name
.substr(i
+ 1);
242 name
= parent
+ "." + name
;
244 // drop the 1st component from the display and indent
246 string::size_type j
= i
= name
.find('.', 1);
247 while ((j
= name
.find('.', j
+ 1)) != string::npos
)
249 cout
<< " " << name
.substr(i
+ 1) << "\n";
252 typedef vector
<Test
*> Tests
;
253 typedef Tests::const_iterator Iter
;
255 const Tests
& tests
= suite
->getTests();
257 for (Iter it
= tests
.begin(); it
!= tests
.end(); ++it
)
260 else if (m_longlist
) {
261 string::size_type i
= 0;
262 while ((i
= parent
.find('.', i
+ 1)) != string::npos
)
264 cout
<< " " << test
->getName() << "\n";