1<?php
2/* vim: set expandtab tabstop=4 shiftwidth=4 softtabstop=4: */
3
4/**
5 * PHP Version 4
6 *
7 * Copyright (c) 2002-2005, Sebastian Bergmann <sb@sebastian-bergmann.de>.
8 * All rights reserved.
9 *
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
12 * are met:
13 *
14 *   * Redistributions of source code must retain the above copyright
15 *     notice, this list of conditions and the following disclaimer.
16 *
17 *   * Redistributions in binary form must reproduce the above copyright
18 *     notice, this list of conditions and the following disclaimer in
19 *     the documentation and/or other materials provided with the
20 *     distribution.
21 *
22 *   * Neither the name of Sebastian Bergmann nor the names of his
23 *     contributors may be used to endorse or promote products derived
24 *     from this software without specific prior written permission.
25 *
26 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
27 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
28 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
29 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
30 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
31 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
32 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
33 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
34 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRIC
35 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
36 * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
37 * POSSIBILITY OF SUCH DAMAGE.
38 *
39 * @category   Testing
40 * @package    PHPUnit
41 * @author     Scott Mattocks <scott@crisscott.com>
42 * @copyright  2002-2005 Sebastian Bergmann <sb@sebastian-bergmann.de>
43 * @license    http://www.opensource.org/licenses/bsd-license.php  BSD License
44 * @version    CVS: $Id: Gtk.php,v 1.6 2005/11/10 09:47:15 sebastian Exp $
45 * @link       http://pear.php.net/package/PHPUnit
46 * @since      File available since Release 1.2.0
47 */
48
49if (!function_exists('is_a')) {
50    require_once 'PHP/Compat/Function/is_a.php';
51}
52
53/**
54 * GTK GUI interface for PHPUnit.
55 *
56 * This class is a PHP port of junit.awtui.testrunner. Documentation
57 * for junit.awtui.testrunner can be found at
58 * http://junit.sourceforge.net
59 *
60 * Due to the limitations of PHP4 and PHP-Gtk, this class can not
61 * duplicate all of the functionality of the JUnit GUI. Some of the
62 * things this class cannot do include:
63 * - Reloading the class for each run
64 * - Stopping the test in progress
65 *
66 * To use simply intantiate the class and call main()
67 * $gtk =& new PHPUnit_GUI_Gtk;
68 * $gtk->main();
69 *
70 * Once the window has finished loading, you can enter the name of
71 * a class that has been loaded (include/require some where in your
72 * code, or you can pass the name of the file containing the class.
73 *
74 * You can also load classes using the SetupDecorator class.
75 * require_once 'PHPUnit/GUI/SetupDecorator.php';
76 * require_once 'PHPUnit/GUI/Gtk.php';
77 * $gui = new PHPUnit_GUI_SetupDecorator(new PHPUnit_GUI_Gtk());
78 * $gui->getSuitesFromDir('/path/to/test','.*\.php$',array('index.php','sql.php'));
79 * $gui->show();
80 *
81 *
82 * @category   Testing
83 * @package    PHPUnit
84 * @author     Scott Mattocks <scott@crisscott.com>
85 * @copyright  2002-2005 Sebastian Bergmann <sb@sebastian-bergmann.de>
86 * @license    http://www.opensource.org/licenses/bsd-license.php  BSD License
87 * @version    Release: 1.3.2
88 * @link       http://pear.php.net/package/PHPUnit
89 * @since      Class available since Release 1.2.0
90 * @todo       Allow file drop. (Gtk_FileDrop)
91 */
92class PHPUnit_GUI_Gtk {
93
94    /**
95     * The main gtk window
96     * @var object
97     */
98    var $gui;
99    /**
100     * The text entry that contains the name of the
101     * file that holds the test(s)/suite(s).
102     * @var object
103     */
104    var $suiteField;
105    /**
106     * The label that shows the number of tests that
107     * were run.
108     * @var object
109     */
110    var $numberOfRuns;
111    /**
112     * The label that shows the number of errors that
113     * were encountered.
114     * @var object
115     */
116    var $numberOfErrors;
117    /**
118     * The label that shows the number of failures
119     * that were encountered.
120     * @var object
121     */
122    var $numberOfFailures;
123    /**
124     * The label for reporting user messages.
125     * @var object
126     */
127    var $statusLine;
128    /**
129     * The text area for reporting messages from successful
130     * test runs. (not necessarily successful tests)
131     * @var object
132     */
133    var $reportArea;
134    /**
135     * The text area for reporting errors encountered when
136     * running tests.
137     * @var object
138     */
139    var $dumpArea;
140    /**
141     * The progress bar indicator. Shows the percentage of
142     * passed tests.
143     * @var object
144     */
145    var $progress;
146    /**
147     * A checkbox for the user to indicate whether or not they
148     * would like to see results from all tests or just failures.
149     * @object
150     */
151    var $showPassed;
152
153    /**
154     * Constructor.
155     *
156     * The constructor checks for the gtk extension and loads it
157     * if needed. Then it creates the GUI. The GUI is not shown
158     * nor is the main gtk loop started until main() is called.
159     *
160     * @access public
161     * @param  none
162     * @return void
163     */
164    function PHPUnit_GUI_Gtk()
165    {
166        // Check for php-gtk extension.
167        if (!extension_loaded('gtk')) {
168            dl( 'php_gtk.' . PHP_SHLIB_SUFFIX);
169        }
170
171        // Create the interface but don't start the loop
172        $this->_createUI();
173    }
174    /**
175     * Start the main gtk loop.
176     *
177     * main() first sets the default state of the showPassed
178     * check box. Next all widgets that are part of the GUI
179     * are shown. Finally the main gtk loop is started.
180     *
181     * @access public
182     * @param  boolean $showPassed
183     * @return void
184     */
185    function main($showPassed = true)
186    {
187        $this->showPassed->set_active($showPassed);
188        $this->gui->show_all();
189
190        gtk::main();
191    }
192    /**
193     * Create the user interface.
194     *
195     * The user interface is pretty simple. It consists of a
196     * menu, text entry, run button, some labels, a progress
197     * indicator, and a couple of areas for notification of
198     * any messages.
199     *
200     * @access private
201     * @param  none
202     * @return void
203     */
204    function _createUI()
205    {
206        // Create a window.
207        $window =& new GtkWindow;
208        $window->set_title('PHPUnit Gtk');
209        $window->set_usize(400, -1);
210
211        // Create the main box.
212        $mainBox =& new GtkVBox;
213        $window->add($mainBox);
214
215        // Start with the menu.
216        $mainBox->pack_start($this->_createMenu());
217
218        // Then add the suite field entry.
219        $mainBox->pack_start($this->_createSuiteEntry());
220
221        // Then add the report labels.
222        $mainBox->pack_start($this->_createReportLabels());
223
224        // Next add the progress bar.
225        $mainBox->pack_start($this->_createProgressBar());
226
227        // Then add the report area and the dump area.
228        $mainBox->pack_start($this->_createReportAreas());
229
230        // Finish off with the status line.
231        $mainBox->pack_start($this->_createStatusLine());
232
233        // Connect the destroy signal.
234        $window->connect_object('destroy', array('gtk', 'main_quit'));
235
236        // Assign the member.
237        $this->gui =& $window;
238    }
239    /**
240     * Create the menu.
241     *
242     * The menu is very simple. It an exit menu item, which exits
243     * the application, and an about menu item, which shows some
244     * basic information about the application itself.
245     *
246     * @access private
247     * @param  none
248     * @return &object The GtkMenuBar
249     */
250    function &_createMenu()
251    {
252        // Create the menu bar.
253        $menuBar =& new GtkMenuBar;
254
255        // Create the main (only) menu item.
256        $phpHeader =& new GtkMenuItem('PHPUnit');
257
258        // Add the menu item to the menu bar.
259        $menuBar->append($phpHeader);
260
261        // Create the PHPUnit menu.
262        $phpMenu =& new GtkMenu;
263
264        // Add the menu items
265        $about =& new GtkMenuItem('About...');
266        $about->connect('activate', array(&$this, 'about'));
267        $phpMenu->append($about);
268
269        $exit =& new GtkMenuItem('Exit');
270        $exit->connect_object('activate', array('gtk', 'main_quit'));
271        $phpMenu->append($exit);
272
273        // Complete the menu.
274        $phpHeader->set_submenu($phpMenu);
275
276        return $menuBar;
277    }
278    /**
279     * Create the suite entry and related widgets.
280     *
281     * The suite entry has some supporting components such as a
282     * label, the show passed check box and the run button. All
283     * of these items are packed into two nested boxes.
284     *
285     * @access private
286     * @param  none
287     * @return &object A box that contains all of the suite entry pieces.
288     */
289    function &_createSuiteEntry()
290    {
291        // Create the outermost box.
292        $outerBox         =& new GtkVBox;
293
294        // Create the suite label, box, and field.
295        $suiteLabel       =& new GtkLabel('Test class name:');
296        $suiteBox         =& new GtkHBox;
297        $this->suiteField =& new GtkEntry;
298        $this->suiteField->set_text($suiteName != NULL ? $suiteName : '');
299
300        // Create the button the user will use to start the test.
301        $runButton =& new GtkButton('Run');
302        $runButton->connect_object('clicked', array(&$this, 'run'));
303
304        // Create the check box that lets the user show only failures.
305        $this->showPassed =& new GtkCheckButton('Show passed tests');
306
307        // Add the components to their respective boxes.
308        $suiteLabel->set_alignment(0, 0);
309        $outerBox->pack_start($suiteLabel);
310        $outerBox->pack_start($suiteBox);
311        $outerBox->pack_start($this->showPassed);
312
313        $suiteBox->pack_start($this->suiteField);
314        $suiteBox->pack_start($runButton);
315
316        return $outerBox;
317    }
318
319    /**
320     * Create the labels that tell the user what has happened.
321     *
322     * There are three labels, one each for total runs, errors and
323     * failures. There is also one label for each of these that
324     * describes what the label is. It could be done with one label
325     * instead of two but that would make updates much harder.
326     *
327     * @access private
328     * @param  none
329     * @return &object A box containing the labels.
330     */
331    function &_createReportLabels()
332    {
333        // Create a box to hold everything.
334        $labelBox         =& new GtkHBox;
335
336        // Create the non-updated labels.
337        $numberOfRuns     =& new GtkLabel('Runs:');
338        $numberOfErrors   =& new GtkLabel('Errors:');
339        $numberOfFailures =& new GtkLabel('Failures:');
340
341        // Create the labels that will be updated.
342        // These are asssigned to members to make it easier to
343        // set their values later.
344        $this->numberOfRuns     =& new GtkLabel(0);
345        $this->numberOfErrors   =& new GtkLabel(0);
346        $this->numberOfFailures =& new GtkLabel(0);
347
348        // Pack everything in.
349        $labelBox->pack_start($numberOfRuns);
350        $labelBox->pack_start($this->numberOfRuns);
351        $labelBox->pack_start($numberOfErrors);
352        $labelBox->pack_start($this->numberOfErrors);
353        $labelBox->pack_start($numberOfFailures);
354        $labelBox->pack_start($this->numberOfFailures);
355
356        return $labelBox;
357    }
358
359    /**
360     * Create the success/failure indicator.
361     *
362     * A GtkProgressBar is used to visually indicate how many
363     * tests were successful compared to how many were not. The
364     * progress bar shows the percentage of success and will
365     * change from green to red if there are any failures.
366     *
367     * @access private
368     * @param  none
369     * @return &object The progress bar
370     */
371    function &_createProgressBar()
372    {
373        // Create the progress bar.
374        $this->progress =& new GtkProgressBar(new GtkAdjustment(0, 0, 1, .1, 1, 0));
375
376        // Set the progress bar to print the percentage.
377        $this->progress->set_show_text(true);
378
379        return $this->progress;
380    }
381
382    /**
383     * Create the report text areas.
384     *
385     * The report area consists of one text area for failures, one
386     * text area for errors and one label for identification purposes.
387     * All three widgets are packed into a box.
388     *
389     * @access private
390     * @param  none
391     * @return &object The box containing the report areas.
392     */
393    function &_createReportAreas()
394    {
395        // Create the containing box.
396        $reportBox =& new GtkVBox;
397
398        // Create the identification label
399        $reportLabel =& new GtkLabel('Errors and Failures:');
400        $reportLabel->set_alignment(0, 0);
401
402        // Create the scrolled windows for the text areas.
403        $reportScroll =& new GtkScrolledWindow;
404        $dumpScroll   =& new GtkScrolledWindow;
405
406        // Make the scroll areas big enough.
407        $reportScroll->set_usize(-1, 150);
408        $dumpScroll->set_usize(-1, 150);
409
410        // Only show the vertical scroll bar when needed.
411        // Never show the horizontal scroll bar.
412        $reportScroll->set_policy(GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);
413        $dumpScroll->set_policy(GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);
414
415        // Create the text areas.
416        $this->reportArea =& new GtkText;
417        $this->dumpArea =& new GtkText;
418
419        // Don't let words get broken.
420        $this->reportArea->set_word_wrap(true);
421        $this->dumpArea->set_word_wrap(true);
422
423        // Pack everything in.
424        $reportBox->pack_start($reportLabel);
425        $reportScroll->add($this->reportArea);
426        $reportBox->pack_start($reportScroll);
427        $dumpScroll->add($this->dumpArea);
428        $reportBox->pack_start($dumpScroll);
429
430        return $reportBox;
431    }
432
433    /**
434     * Create a status label.
435     *
436     * A status line at the bottom of the application is used
437     * to notify the user of non-test related messages such as
438     * failures loading a test suite.
439     *
440     * @access private
441     * @param  none
442     * @return &object The status label.
443     */
444    function &_createStatusLine()
445    {
446        // Create the status label.
447        $this->statusLine =& new GtkLabel('');
448        $this->statusLine->set_alignment(0, 0);
449
450        return $this->statusLine;
451    }
452
453    /**
454     * Show a popup with information about the application.
455     *
456     * The popup should show information about the version,
457     * the author, the license, where to get the latest
458     * version and a short description.
459     *
460     * @access public
461     * @param  none
462     * @return void
463     */
464    function about()
465    {
466        // Create the new window.
467        $about =& new GtkWindow;
468        $about->set_title('About PHPUnit GUI Gtk');
469        $about->set_usize(250, -1);
470
471        // Put two vboxes in the hbox.
472        $vBox =& new GtkVBox;
473        $about->add($vBox);
474
475        // Create the labels.
476        $version     =& new GtkLabel(" Version: 1.0");
477        $license     =& new GtkLabel(" License: PHP License v3.0");
478        $where       =& new GtkLabel(" Download from: http://pear.php.net/PHPUnit/");
479        $unitAuth    =& new GtkLabel(" PHPUnit Author: Sebastian Bergman");
480        $gtkAuth     =& new GtkLabel(" Gtk GUI Author: Scott Mattocks");
481
482        // Align everything to the left
483        $where->set_alignment(0, .5);
484        $version->set_alignment(0, .5);
485        $license->set_alignment(0, .5);
486        $gtkAuth->set_alignment(0, .5);
487        $unitAuth->set_alignment(0, .5);
488
489        // Pack everything into the vBox;
490        $vBox->pack_start($version);
491        $vBox->pack_start($license);
492        $vBox->pack_start($where);
493        $vBox->pack_start($unitAuth);
494        $vBox->pack_start($gtkAuth);
495
496        // Connect the destroy signal.
497        $about->connect('destroy', array('gtk', 'true'));
498
499        // Show the goods.
500        $about->show_all();
501    }
502
503    /**
504     * Load the test suite.
505     *
506     * This method tries to load test suite based on the user
507     * info. If the user passes the name of a tests suite, it
508     * is instantiated and a new object is returned. If the
509     * user passes a file that contains a test suite, the class
510     * is instantiated and a new object is returned. If the user
511     * passes a file that contains a test case, the test case is
512     * passed to a new test suite and the new suite object is
513     * returned.
514     *
515     * @access public
516     * @param  string  The file that contains a test case/suite or the classname.
517     * @return &object The new test suite.
518     */
519    function &loadTest(&$file)
520    {
521        // Check to see if a class name was given.
522        if (is_a($file, 'PHPUnit_TestSuite')) {
523            return $file;
524        } elseif (class_exists($file)) {
525            require_once 'PHPUnit/TestSuite.php';
526            return new PHPUnit_TestSuite($file);
527        }
528
529        // Check that the file exists.
530        if (!@is_readable($file)) {
531            $this->_showStatus('Cannot find file: ' . $file);
532            return false;
533        }
534
535        $this->_showStatus('Loading test suite...');
536
537        // Instantiate the class.
538        // If the path is /path/to/test/TestClass.php
539        // the class name should be test_TestClass
540        require_once $file;
541        $className = str_replace(DIRECTORY_SEPARATOR, '_', $file);
542        $className = substr($className, 0, strpos($className, '.'));
543
544        require_once 'PHPUnit/TestSuite.php';
545        return new PHPUnit_TestSuite($className);
546    }
547
548    /**
549     * Run the test suite.
550     *
551     * This method runs the test suite and updates the messages
552     * for the user. When finished it changes the status line
553     * to 'Test Complete'
554     *
555     * @access public
556     * @param  none
557     * @return void
558     */
559    function runTest()
560    {
561        // Notify the user that the test is running.
562        $this->_showStatus('Running Test...');
563
564        // Run the test.
565        $result = PHPUnit::run($this->suite);
566
567        // Update the labels.
568        $this->_setLabelValue($this->numberOfRuns,     $result->runCount());
569        $this->_setLabelValue($this->numberOfErrors,   $result->errorCount());
570        $this->_setLabelValue($this->numberOfFailures, $result->failureCount());
571
572        // Update the progress bar.
573        $this->_updateProgress($result->runCount(),
574                               $result->errorCount(),
575                               $result->failureCount()
576                               );
577
578        // Show the errors.
579        $this->_showFailures($result->errors(), $this->dumpArea);
580
581        // Show the messages from the tests.
582        if ($this->showPassed->get_active()) {
583            // Show failures and success.
584            $this->_showAll($result, $this->reportArea);
585        } else {
586            // Show only failures.
587            $this->_showFailures($result->failures(), $this->reportArea);
588        }
589
590        // Update the status message.
591        $this->_showStatus('Test complete');
592    }
593
594    /**
595     * Set the text of a label.
596     *
597     * Change the text of a given label.
598     *
599     * @access private
600     * @param  widget  &$label The label whose value is to be changed.
601     * @param  string  $value  The new text of the label.
602     * @return void
603     */
604    function _setLabelValue(&$label, $value)
605    {
606        $label->set_text($value);
607    }
608
609    /**
610     * The main work of the application.
611     *
612     * Load the test suite and then execute the tests.
613     *
614     * @access public
615     * @param  none
616     * @return void
617     */
618    function run()
619    {
620        // Load the test suite.
621        $this->suite =& $this->loadTest($this->suiteField->get_text());
622
623        // Check to make sure the suite was loaded properly.
624        if (!is_object($this->suite)) {
625            // Raise an error.
626            $this->_showStatus('Could not load test suite.');
627            return false;
628        }
629
630        // Run the tests.
631        $this->runTest();
632    }
633
634    /**
635     * Update the status message.
636     *
637     * @access private
638     * @param  string  $status The new message.
639     * @return void
640     */
641    function _showStatus($status)
642    {
643        $this->statusLine->set_text($status);
644    }
645
646    /**
647     * Alias for main()
648     *
649     * @see main
650     */
651    function show($showPassed = true)
652    {
653        $this->main($showPassed);
654    }
655
656    /**
657     * Add a suite to the tests.
658     *
659     * This method is require by SetupDecorator. It adds a
660     * suite to the the current set of suites.
661     *
662     * @access public
663     * @param  object $testSuite The suite to add.
664     * @return void
665     */
666    function addSuites($testSuite)
667    {
668        if (!is_array($testSuite)) {
669            settype($testSuite, 'array');
670        }
671
672        foreach ($testSuite as $suite) {
673
674            if (is_a($this->suite, 'PHPUnit_TestSuite')) {
675                $this->suite->addTestSuite($suite->getName());
676            } else {
677                $this->suite =& $this->loadTest($suite);
678            }
679
680            // Set the suite field.
681            $text = $this->suiteField->get_text();
682            if (empty($text)) {
683                $this->suiteField->set_text($this->suite->getName());
684            }
685        }
686    }
687
688    /**
689     * Show all test messages.
690     *
691     * @access private
692     * @param  object  The TestResult from the test suite.
693     * @return void
694     */
695    function _showAll(&$result)
696    {
697        // Clear the area first.
698        $this->reportArea->delete_text(0, -1);
699        $this->reportArea->insert_text($result->toString(), 0);
700    }
701
702    /**
703     * Show failure/error messages in the given text area.
704     *
705     * @access private
706     * @param  object  &$results The results of the test.
707     * @param  widget  &$area    The area to show the results in.
708     */
709    function _showFailures(&$results, &$area)
710    {
711        $area->delete_text(0, -1);
712        foreach (array_reverse($results, true) as $result) {
713            $area->insert_text($result->toString(), 0);
714        }
715    }
716
717    /**
718     * Update the progress indicator.
719     *
720     * @access private
721     * @param  integer $runs
722     * @param  integer $errors
723     * @param  integer $failures
724     * @return void
725     */
726    function _updateProgress($runs, $errors, $failures)
727    {
728        $percentage = 1 - (($errors + $failures) / $runs);
729        $this->progress->set_percentage($percentage);
730    }
731}
732
733/*
734 * Local variables:
735 * tab-width: 4
736 * c-basic-offset: 4
737 * c-hanging-comment-ender-p: nil
738 * End:
739 */
740?>
741