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"
25 #pragma warning(disable:4100)
27 #include <cppunit/TestListener.h>
29 #pragma warning(default:4100)
31 #include <cppunit/Test.h>
32 #include <cppunit/TestResult.h>
33 #include "wx/afterstd.h"
35 #include "wx/cmdline.h"
39 using CppUnit::TestSuite
;
40 using CppUnit::TestFactoryRegistry
;
43 // Displays the test name before starting to execute it: this helps with
44 // diagnosing where exactly does a test crash or hang when/if it does.
45 class DetailListener
: public CppUnit::TestListener
48 DetailListener(bool doTiming
= false):
49 CppUnit::TestListener(),
54 virtual void startTest(CppUnit::Test
*test
)
56 std::cout
<< test
->getName () << " ";
60 virtual void endTest(CppUnit::Test
* WXUNUSED(test
))
64 std::cout
<< " (in "<< m_watch
.Time() << " ms )";
76 typedef wxApp TestAppBase
;
78 typedef wxAppConsole TestAppBase
;
81 // The application class
83 class TestApp
: public TestAppBase
89 virtual void OnInitCmdLine(wxCmdLineParser
& parser
);
90 virtual bool OnCmdLineParsed(wxCmdLineParser
& parser
);
91 virtual bool OnInit();
96 void List(Test
*test
, const string
& parent
= "") const;
98 // command lines options/parameters
103 vector
<string
> m_registries
;
106 IMPLEMENT_APP_CONSOLE(TestApp
)
116 bool TestApp::OnInit()
118 if ( !TestAppBase::OnInit() )
121 cout
<< "Test program for wxWidgets\n"
122 << "build: " << WX_BUILD_OPTIONS_SIGNATURE
<< std::endl
;
125 // create a hidden parent window to be used as parent for the GUI controls
126 new wxFrame(NULL
, wxID_ANY
, "Hidden wx test frame");
132 // The table of command line options
134 void TestApp::OnInitCmdLine(wxCmdLineParser
& parser
)
136 TestAppBase::OnInitCmdLine(parser
);
138 static const wxCmdLineEntryDesc cmdLineDesc
[] = {
139 { wxCMD_LINE_SWITCH
, "l", "list",
140 "list the test suites, do not run them",
141 wxCMD_LINE_VAL_NONE
, 0 },
142 { wxCMD_LINE_SWITCH
, "L", "longlist",
143 "list the test cases, do not run them",
144 wxCMD_LINE_VAL_NONE
, 0 },
145 { wxCMD_LINE_SWITCH
, "d", "detail",
146 "print the test case names, run them",
147 wxCMD_LINE_VAL_NONE
, 0 },
148 { wxCMD_LINE_SWITCH
, "t", "timing",
149 "print names and mesure running time of individual test, run them",
150 wxCMD_LINE_VAL_NONE
, 0 },
151 { wxCMD_LINE_PARAM
, NULL
, NULL
, "REGISTRY", wxCMD_LINE_VAL_STRING
,
152 wxCMD_LINE_PARAM_OPTIONAL
| wxCMD_LINE_PARAM_MULTIPLE
},
156 parser
.SetDesc(cmdLineDesc
);
159 // Handle command line options
161 bool TestApp::OnCmdLineParsed(wxCmdLineParser
& parser
)
163 if (parser
.GetParamCount())
164 for (size_t i
= 0; i
< parser
.GetParamCount(); i
++)
165 m_registries
.push_back(string(parser
.GetParam(i
).mb_str()));
167 m_registries
.push_back("");
169 m_longlist
= parser
.Found(_T("longlist"));
170 m_list
= m_longlist
|| parser
.Found(_T("list"));
171 m_timing
= parser
.Found(_T("timing"));
172 m_detail
= !m_timing
&& parser
.Found(_T("detail"));
174 return TestAppBase::OnCmdLineParsed(parser
);
181 CppUnit::TextTestRunner runner
;
183 for (size_t i
= 0; i
< m_registries
.size(); i
++) {
184 auto_ptr
<Test
> test(m_registries
[i
].empty() ?
185 TestFactoryRegistry::getRegistry().makeTest() :
186 TestFactoryRegistry::getRegistry(m_registries
[i
]).makeTest());
188 TestSuite
*suite
= dynamic_cast<TestSuite
*>(test
.get());
190 if (suite
&& suite
->countTestCases() == 0)
191 wxLogError(_T("No such test suite: %s"),
192 wxString(m_registries
[i
].c_str(), wxConvUTF8
).c_str());
196 runner
.addTest(test
.release());
202 runner
.setOutputter(new CppUnit::CompilerOutputter(&runner
.result(), cout
));
205 // Switch off logging unless --verbose
206 bool verbose
= wxLog::GetVerbose();
207 wxLog::EnableLogging(verbose
);
209 bool verbose
= false;
213 // (http://sf.net/tracker/index.php?func=detail&aid=1649369&group_id=11795&atid=111795)
214 // in some versions of cppunit: they write progress dots to cout (and not
215 // cerr) and don't flush it so all the dots appear at once at the end which
216 // is not very useful so unbuffer cout to work around this
217 cout
.setf(ios::unitbuf
);
219 // add detail listener if needed
220 DetailListener
detailListener(m_timing
);
221 if ( m_detail
|| m_timing
)
222 runner
.eventManager().addListener(&detailListener
);
224 return runner
.run("", false, true, !verbose
) ? EXIT_SUCCESS
: EXIT_FAILURE
;
227 int TestApp::OnExit()
230 delete GetTopWindow();
238 void TestApp::List(Test
*test
, const string
& parent
/*=""*/) const
240 TestSuite
*suite
= dynamic_cast<TestSuite
*>(test
);
244 // take the last component of the name and append to the parent
245 name
= test
->getName();
246 string::size_type i
= name
.find_last_of(".:");
247 if (i
!= string::npos
)
248 name
= name
.substr(i
+ 1);
249 name
= parent
+ "." + name
;
251 // drop the 1st component from the display and indent
253 string::size_type j
= i
= name
.find('.', 1);
254 while ((j
= name
.find('.', j
+ 1)) != string::npos
)
256 cout
<< " " << name
.substr(i
+ 1) << "\n";
259 typedef vector
<Test
*> Tests
;
260 typedef Tests::const_iterator Iter
;
262 const Tests
& tests
= suite
->getTests();
264 for (Iter it
= tests
.begin(); it
!= tests
.end(); ++it
)
267 else if (m_longlist
) {
268 string::size_type i
= 0;
269 while ((i
= parent
.find('.', i
+ 1)) != string::npos
)
271 cout
<< " " << test
->getName() << "\n";