 97a80f383c
			
		
	
	
		97a80f383c
		
			
		
	
	
	
	
		
			
			There are a number of situations where HTML write was producing
HTML which could not be validated. These include:
  - inconsistent use of backslash terminating META, IMG, and COL tags
  - @page style tags in body rather than header. Aside from being
    non-standard, HTML Reader treats those as spreadsheet data.
  - <div style="page-break-before:always" />, a construct which is
    usually better handled through css anyhow.
  - no alt tag for images (drawings and charts)
Other problems:
  - Windows file names not handled correctly for images
  - Memory drawings not handled in extendRowsForChartsAndImages
  - No handling of different values for showing gridlines
    for screen and print
  - Mpdf and Dompdf do not require the use of inline css.
    Tcpdf remains a holdout in the use of this inferior approach.
  - no need to chunk base64 encoding of embedded images
  - support for colors in number format was buggy (html tags
    run through htmlspecialchars)
Code has been refactored when practical to reduce the number of
very large functions.
Coverage is now 100% for the entire HTML Writer module,
from 75% lines and 39% methods beforehand.
All functions dealing only with charts
are bypassed for coverage because the version of Jpgraph available in
Composer is not suitable for PHP7. The code will, nevertheless,
run successfully, but with warning messages. I have confirmed that
the code is entirely covered, without warnings, when the current
version of Jpgraph is used in lieu of the one available in Composer.
I will be glad to revisit this when the Jpgraph problem is resolved.
Directory PhpSpreadsheetTests/Writer/Html was created to house
the new tests. It seemed logical to move HtmlCommentsTest to
the new directory from PhpSpreadsheetTests/Functional.
A function to generate all the HTML is useful, especially for testing,
but also in lieu of the multiple other generate* functions. I have
added and documented generateHTMLAll.
The documentation for the generate* functions (a) produces invalid html,
(b) produces html which cannot be handled correctly by HTML reader,
and (c) even if those were correct, does not actually affect
the display of the spreadsheet. The documentation has been replaced
by a valid, and more instructive, example.
The (undocumented) useEmbeddedCss property, and the functions
to test and set it are no longer needed. Rather than breaking
existing code by deleting them, I marked the functions deprecated.
This change borrows a change to LocaleFloatsTest from
pull request 1456, submitted a little over a week before this one.
## Improve NumberFormat Support
First phase of this change included correcting NumberFormat handling
in HTML Writer. Certain complex formats could not be handled without
changes to Style/NumberFormat, and I did not wish to combine those changes.
Once the original change had been pushed, I took this part of it back up.
HTML Writer can now handle conditions in formats like:
[Blue][>=3000.5]$#,##0.00;[Red][<0]$#,##0.00;$#,##0.00
In testing, I discovered several errors and omissions
in handling of some other formats.
These are now corrected, and tests added.
		
	
			
		
			
				
	
	
		
			200 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			200 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| namespace PhpOffice\PhpSpreadsheetTests\Writer\Html;
 | |
| 
 | |
| use PhpOffice\PhpSpreadsheet\Spreadsheet;
 | |
| use PhpOffice\PhpSpreadsheet\Style\Color;
 | |
| use PhpOffice\PhpSpreadsheet\Writer\Html;
 | |
| use PhpOffice\PhpSpreadsheetTests\Functional;
 | |
| 
 | |
| class GridlinesTest extends Functional\AbstractFunctional
 | |
| {
 | |
|     public function testGridlines()
 | |
|     {
 | |
|         $spreadsheet = new Spreadsheet();
 | |
|         $sheet = $spreadsheet->getActiveSheet();
 | |
|         $sheet->setShowGridlines(true)->setPrintGridlines(true);
 | |
|         $sheet->setCellValue('A1', 1);
 | |
|         $sheet = $spreadsheet->createSheet();
 | |
|         $sheet->setShowGridlines(true)->setPrintGridlines(false);
 | |
|         $sheet->setCellValue('A1', 1);
 | |
|         $sheet = $spreadsheet->createSheet();
 | |
|         $sheet->setShowGridlines(false)->setPrintGridlines(true);
 | |
|         $sheet->setCellValue('A1', 1);
 | |
|         $sheet = $spreadsheet->createSheet();
 | |
|         $sheet->setShowGridlines(false)->setPrintGridlines(false);
 | |
|         $sheet->setCellValue('A1', 1);
 | |
| 
 | |
|         $writer = new Html($spreadsheet);
 | |
|         $writer->writeAllSheets();
 | |
| 
 | |
|         $html = $writer->generateHTMLAll();
 | |
|         $dom = new \DOMDocument();
 | |
|         $dom->loadHTML($html);
 | |
|         $body = $dom->getElementsByTagName('body')[0];
 | |
|         $divs = $body->getElementsByTagName('div');
 | |
|         self::assertCount(4, $divs);
 | |
| 
 | |
|         $tbl = $divs[0]->getElementsByTagName('table')[0];
 | |
|         $cls = $tbl->getAttribute('class');
 | |
|         self::assertEquals('sheet0 gridlines gridlinesp', $cls);
 | |
|         $tbl = $divs[1]->getElementsByTagName('table')[0];
 | |
|         $cls = $tbl->getAttribute('class');
 | |
|         self::assertEquals('sheet1 gridlines', $cls);
 | |
|         $tbl = $divs[2]->getElementsByTagName('table')[0];
 | |
|         $cls = $tbl->getAttribute('class');
 | |
|         self::assertEquals('sheet2 gridlinesp', $cls);
 | |
|         $tbl = $divs[3]->getElementsByTagName('table')[0];
 | |
|         $cls = $tbl->getAttribute('class');
 | |
|         self::assertEquals('sheet3', $cls);
 | |
| 
 | |
|         $this->writeAndReload($spreadsheet, 'Html');
 | |
|     }
 | |
| 
 | |
|     public function testGridlinesInline()
 | |
|     {
 | |
|         $spreadsheet = new Spreadsheet();
 | |
|         $sheet = $spreadsheet->getActiveSheet();
 | |
|         $sheet->setShowGridlines(true)->setPrintGridlines(true);
 | |
|         $sheet->setCellValue('A1', 1);
 | |
|         $sheet = $spreadsheet->createSheet();
 | |
|         $sheet->setShowGridlines(true)->setPrintGridlines(false);
 | |
|         $sheet->setCellValue('A1', 1);
 | |
|         $sheet = $spreadsheet->createSheet();
 | |
|         $sheet->setShowGridlines(false)->setPrintGridlines(true);
 | |
|         $sheet->setCellValue('A1', 1);
 | |
|         $sheet = $spreadsheet->createSheet();
 | |
|         $sheet->setShowGridlines(false)->setPrintGridlines(false);
 | |
|         $sheet->setCellValue('A1', 1);
 | |
| 
 | |
|         $writer = new Html($spreadsheet);
 | |
|         $writer->writeAllSheets();
 | |
|         $writer->setUseInlineCss(true);
 | |
| 
 | |
|         $html = $writer->generateHTMLAll();
 | |
|         $dom = new \DOMDocument();
 | |
|         $dom->loadHTML($html);
 | |
|         $body = $dom->getElementsByTagName('body')[0];
 | |
|         $divs = $body->getElementsByTagName('div');
 | |
|         self::assertCount(4, $divs);
 | |
| 
 | |
|         $tbl = $divs[0]->getElementsByTagName('table')[0];
 | |
|         $cls = $tbl->getAttribute('class');
 | |
|         self::assertEquals('gridlines gridlinesp', $cls);
 | |
|         $tbl = $divs[1]->getElementsByTagName('table')[0];
 | |
|         $cls = $tbl->getAttribute('class');
 | |
|         self::assertEquals('gridlines', $cls);
 | |
|         $tbl = $divs[2]->getElementsByTagName('table')[0];
 | |
|         $cls = $tbl->getAttribute('class');
 | |
|         self::assertEquals('gridlinesp', $cls);
 | |
|         $tbl = $divs[3]->getElementsByTagName('table')[0];
 | |
|         $cls = $tbl->getAttribute('class');
 | |
|         self::assertEquals('', $cls);
 | |
| 
 | |
|         $this->writeAndReload($spreadsheet, 'Html');
 | |
|     }
 | |
| 
 | |
|     public function testRichText()
 | |
|     {
 | |
|         $spreadsheet = new Spreadsheet();
 | |
|         $sheet = $spreadsheet->getActiveSheet();
 | |
|         $emc2 = new \PhpOffice\PhpSpreadsheet\RichText\RichText();
 | |
|         $part1 = $emc2->createTextRun('e=mc');
 | |
|         $part1->getFont()->getColor()->setARGB(Color::COLOR_BLUE);
 | |
|         $part2 = $emc2->createTextRun('2');
 | |
|         $font = $part2->getFont();
 | |
|         $font->getColor()->setARGB(Color::COLOR_DARKGREEN);
 | |
|         $font->setSuperScript(true);
 | |
|         $sheet->setCellValue('A1', $emc2);
 | |
|         $h2o = new \PhpOffice\PhpSpreadsheet\RichText\RichText();
 | |
|         $h2o->createTextRun('H');
 | |
|         $part2 = $h2o->createTextRun('2');
 | |
|         $font = $part2->getFont();
 | |
|         $font->setSubScript(true);
 | |
|         $font->getColor()->setARGB(Color::COLOR_RED);
 | |
|         $h2o->createTextRun('O');
 | |
|         $sheet->setCellValue('A2', $h2o);
 | |
|         $h2so4 = new \PhpOffice\PhpSpreadsheet\RichText\RichText();
 | |
|         $h2so4->createTextRun('H');
 | |
|         $part2 = $h2so4->createTextRun('2');
 | |
|         $font = $part2->getFont();
 | |
|         $font->setSubScript(true);
 | |
|         $h2so4->createTextRun('SO');
 | |
|         $part4 = $h2so4->createTextRun('4');
 | |
|         $part4->getFont()->setSubScript(true);
 | |
|         $sheet->setCellValue('A3', $h2so4);
 | |
|         $sheet->setCellValue('A4', '5');
 | |
|         $sheet->getCell('A4')->getStyle()->getFont()->setSuperScript(true);
 | |
|         $sheet->setCellValue('A5', '6');
 | |
|         $sheet->getCell('A5')->getStyle()->getFont()->setSubScript(true);
 | |
| 
 | |
|         $writer = new Html($spreadsheet);
 | |
|         $html = $writer->generateHTMLAll();
 | |
|         $dom = new \DOMDocument();
 | |
|         $dom->loadHTML($html);
 | |
|         $body = $dom->getElementsByTagName('body')[0];
 | |
|         $divs = $body->getElementsByTagName('div');
 | |
|         self::assertCount(1, $divs);
 | |
| 
 | |
|         $tabl = $divs[0]->getElementsByTagName('table');
 | |
|         $tbod = $tabl[0]->getElementsByTagName('tbody');
 | |
|         $rows = $tbod[0]->getElementsByTagName('tr');
 | |
|         self::assertCount(5, $rows);
 | |
|         $tds = $rows[0]->getElementsByTagName('td');
 | |
|         self::assertCount(1, $tds);
 | |
|         $spans = $tds[0]->getElementsByTagName('span');
 | |
|         self::assertCount(2, $spans);
 | |
|         self::assertEquals('e=mc', $spans[0]->textContent);
 | |
|         $style = $spans[0]->getAttribute('style');
 | |
|         self::assertEquals(1, preg_match('/color:#0000FF/', $style));
 | |
|         $style = $spans[1]->getAttribute('style');
 | |
|         self::assertEquals(1, preg_match('/color:#008000/', $style));
 | |
|         $sups = $spans[1]->getElementsByTagName('sup');
 | |
|         self::assertCount(1, $sups);
 | |
|         assert('2' == $sups[0]->textContent);
 | |
| 
 | |
|         $tds = $rows[1]->getElementsByTagName('td');
 | |
|         assert(1 == count($tds));
 | |
|         $spans = $tds[0]->getElementsByTagName('span');
 | |
|         assert(3 == count($spans));
 | |
|         assert('H' == $spans[0]->textContent);
 | |
|         $style = $spans[1]->getAttribute('style');
 | |
|         assert(1 == preg_match('/color:#FF0000/', $style));
 | |
|         $subs = $spans[1]->getElementsByTagName('sub');
 | |
|         assert(1 == count($subs));
 | |
|         assert('2' == $subs[0]->textContent);
 | |
|         assert('O' == $spans[2]->textContent);
 | |
| 
 | |
|         $tds = $rows[2]->getElementsByTagName('td');
 | |
|         self::assertCount(1, $tds);
 | |
|         $spans = $tds[0]->getElementsByTagName('span');
 | |
|         self::assertCount(4, $spans);
 | |
|         self::assertEquals('H', $spans[0]->textContent);
 | |
|         $subs = $spans[1]->getElementsByTagName('sub');
 | |
|         self::assertCount(1, $subs);
 | |
|         self::assertEquals('2', $subs[0]->textContent);
 | |
|         self::assertEquals('SO', $spans[2]->textContent);
 | |
|         $subs = $spans[3]->getElementsByTagName('sub');
 | |
|         self::assertCount(1, $subs);
 | |
|         self::assertEquals('4', $subs[0]->textContent);
 | |
| 
 | |
|         $tds = $rows[3]->getElementsByTagName('td');
 | |
|         self::assertCount(1, $tds);
 | |
|         $spans = $tds[0]->getElementsByTagName('span');
 | |
|         self::assertCount(0, $spans);
 | |
|         $sups = $tds[0]->getElementsByTagName('sup');
 | |
|         self::assertCount(1, $sups);
 | |
|         self::assertEquals('5', $sups[0]->textContent);
 | |
| 
 | |
|         $tds = $rows[4]->getElementsByTagName('td');
 | |
|         self::assertCount(1, $tds);
 | |
|         $spans = $tds[0]->getElementsByTagName('span');
 | |
|         self::assertCount(0, $spans);
 | |
|         $subs = $tds[0]->getElementsByTagName('sub');
 | |
|         self::assertCount(1, $subs);
 | |
|         self::assertEquals('6', $subs[0]->textContent);
 | |
| 
 | |
|         $this->writeAndReload($spreadsheet, 'Html');
 | |
|     }
 | |
| }
 |