From 315cc4728dfd79b49fe12bd445e6075eaec87c5f Mon Sep 17 00:00:00 2001 From: "Christoph \"criztovyl\" Schulz" Date: Mon, 1 Jan 2018 12:39:59 +0100 Subject: [PATCH] Support for cell comments in HTML writer and reader The behavior is similar to what is done in LibreOffice. That means if there is a comment it will be shown with a small indicator and the actual comment will be revealed when mouse hover over the indicator. Fixes #308 Closes #310 --- CHANGELOG.md | 2 + docs/references/features-cross-reference.md | 11 ++-- src/PhpSpreadsheet/Reader/Html.php | 12 ++++ src/PhpSpreadsheet/Writer/Html.php | 43 +++++++++++++ .../Functional/HtmlCommentsTest.php | 63 +++++++++++++++++++ 5 files changed, 126 insertions(+), 5 deletions(-) create mode 100644 tests/PhpSpreadsheetTests/Functional/HtmlCommentsTest.php diff --git a/CHANGELOG.md b/CHANGELOG.md index bdc240d9bb..7a9f810e15 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,6 +9,8 @@ and this project adheres to [Semantic Versioning](http://semver.org/). ### Added +- Support cell comments in HTML writer and reader- [#308](https://github.com/PHPOffice/PhpSpreadsheet/issues/308) + ### Fixed - Better auto-detection of CSV separators - [#305](https://github.com/PHPOffice/PhpSpreadsheet/issues/305) diff --git a/docs/references/features-cross-reference.md b/docs/references/features-cross-reference.md index 4bac57a16c..2d28cd4dac 100644 --- a/docs/references/features-cross-reference.md +++ b/docs/references/features-cross-reference.md @@ -1231,14 +1231,14 @@ ● ● N/A - + ● 1 N/A Rich Text - ✖ 1 + ✖ 2 ✔ ✖ ✖ @@ -1256,7 +1256,7 @@ Alignment - ✖ 2 + ✖ 3 ✖ ✖ ✖ @@ -1568,5 +1568,6 @@ -1. Only BIFF8 files support Rich Text. Prior to that, comments could only be plain text -2. Only BIFF8 files support alignment and rotation. Prior to that, comments could only be unformatted text +1. Only text contents +2. Only BIFF8 files support Rich Text. Prior to that, comments could only be plain text +3. Only BIFF8 files support alignment and rotation. Prior to that, comments could only be unformatted text diff --git a/src/PhpSpreadsheet/Reader/Html.php b/src/PhpSpreadsheet/Reader/Html.php index a6720de68f..43d3e66c31 100644 --- a/src/PhpSpreadsheet/Reader/Html.php +++ b/src/PhpSpreadsheet/Reader/Html.php @@ -312,6 +312,14 @@ protected function processDomElement(DOMNode $element, Worksheet $sheet, &$row, case 'em': case 'strong': case 'b': + if (isset($attributeArray['class']) && $attributeArray['class'] === 'comment') { + $sheet->getComment($column . $row) + ->getText() + ->createTextRun($child->textContent); + + break; + } + if ($cellContent > '') { $cellContent .= ' '; } @@ -354,6 +362,10 @@ protected function processDomElement(DOMNode $element, Worksheet $sheet, &$row, } break; + case 'class': + if ($attributeValue === 'comment-indicator') { + break; // Ignore - it's just a red square. + } } } $cellContent .= ' '; diff --git a/src/PhpSpreadsheet/Writer/Html.php b/src/PhpSpreadsheet/Writer/Html.php index 7e1a0048a6..c0e4503e37 100644 --- a/src/PhpSpreadsheet/Writer/Html.php +++ b/src/PhpSpreadsheet/Writer/Html.php @@ -830,6 +830,25 @@ public function buildCSS($generateSurroundingHTML = true) $css['html']['background-color'] = 'white'; } + // CSS for comments as found in LibreOffice + $css['a.comment-indicator:hover + div.comment'] = [ + 'background' => '#ffd', + 'position' => 'absolute', + 'display' => 'block', + 'border' => '1px solid black', + 'padding' => '0.5em', + ]; + + $css['a.comment-indicator'] = [ + 'background' => 'red', + 'display' => 'inline-block', + 'border' => '1px solid black', + 'width' => '0.5em', + 'height' => '0.5em', + ]; + + $css['div.comment']['display'] = 'none'; + // table { } $css['table']['border-collapse'] = 'collapse'; if (!$this->isPdf) { @@ -1385,6 +1404,8 @@ private function generateRow(Worksheet $pSheet, array $pValues, $pRow, $cellType } $html .= '>'; + $html .= $this->writeComment($pSheet, $coordinate); + // Image? $html .= $this->writeImageInCell($pSheet, $coordinate); @@ -1646,4 +1667,26 @@ private function setMargins(Worksheet $pSheet) return "\n"; } + + /** + * Write a comment in the same format as LibreOffice. + * + * @see https://github.com/LibreOffice/core/blob/9fc9bf3240f8c62ad7859947ab8a033ac1fe93fa/sc/source/filter/html/htmlexp.cxx#L1073-L1092 + * + * @param Worksheet $pSheet + * @param string $coordinate + * + * @return string + */ + private function writeComment(Worksheet $pSheet, $coordinate) + { + $result = ''; + if (!$this->isPdf && isset($pSheet->getComments()[$coordinate])) { + $result .= ''; + $result .= '
' . nl2br($pSheet->getComment($coordinate)->getText()->getPlainText()) . '
'; + $result .= PHP_EOL; + } + + return $result; + } } diff --git a/tests/PhpSpreadsheetTests/Functional/HtmlCommentsTest.php b/tests/PhpSpreadsheetTests/Functional/HtmlCommentsTest.php new file mode 100644 index 0000000000..1b497277a6 --- /dev/null +++ b/tests/PhpSpreadsheetTests/Functional/HtmlCommentsTest.php @@ -0,0 +1,63 @@ +createText($valueSingle); + + $plainMulti = new RichText(); + $plainMulti->createText($valueMulti); + + $richSingle = new RichText(); + $richSingle->createTextRun($valueSingle)->getFont()->setBold(true); + + $richMultiSimple = new RichText(); + $richMultiSimple->createTextRun($valueMulti)->getFont()->setBold(true); + + $richMultiMixed = new RichText(); + $richMultiMixed->createText('I am' . PHP_EOL); + $richMultiMixed->createTextRun('multi-line')->getFont()->setBold(true); + $richMultiMixed->createText(PHP_EOL . 'comment!'); + + return [ + 'single line plain text' => [$plainSingle], + 'multi-line plain text' => [$plainMulti], + 'single line simple rich text' => [$richSingle], + 'multi-line simple rich text' => [$richMultiSimple], + 'multi-line mixed rich text' => [$richMultiMixed], + ]; + } + + /** + * @dataProvider providerCommentRichText + * + * @param mixed $richText + */ + public function testComments($richText) + { + $this->spreadsheet = new Spreadsheet(); + + $this->spreadsheet->getActiveSheet()->getCell('A1')->setValue('Comment'); + + $this->spreadsheet->getActiveSheet() + ->getComment('A1') + ->setText($richText); + + $reloadedSpreadsheet = $this->writeAndReload($this->spreadsheet, 'Html'); + + $actual = $reloadedSpreadsheet->getActiveSheet()->getComment('A1')->getText()->getPlainText(); + self::assertSame($richText->getPlainText(), $actual); + } +}