From 2760e5abb74eafbcd7149bd3035ff8820562fc29 Mon Sep 17 00:00:00 2001 From: oleibman <10341515+oleibman@users.noreply.github.com> Date: Thu, 30 May 2024 00:19:11 -0700 Subject: [PATCH 1/3] Conditional Color Scale Improvements Fix #4049. Some possible options were not included for read or write. In addition, although it isn't well documented, it appears that 2-color scale always has 2 cvfo entries in Xml in order minimum/maximum, and 3-color scale always has 3 entries in order minimum/midpoint/maximum. --- .../Reader/Xlsx/ConditionalStyles.php | 29 +++++---- src/PhpSpreadsheet/Writer/Xlsx/Worksheet.php | 57 ++++++++++++++++-- .../Reader/Xlsx/Issue4049Test.php | 38 ++++++++++++ tests/data/Reader/XLSX/issue.4049.xlsx | Bin 0 -> 4774 bytes 4 files changed, 106 insertions(+), 18 deletions(-) create mode 100644 tests/PhpSpreadsheetTests/Reader/Xlsx/Issue4049Test.php create mode 100644 tests/data/Reader/XLSX/issue.4049.xlsx diff --git a/src/PhpSpreadsheet/Reader/Xlsx/ConditionalStyles.php b/src/PhpSpreadsheet/Reader/Xlsx/ConditionalStyles.php index a26b9f6334..e353227e74 100644 --- a/src/PhpSpreadsheet/Reader/Xlsx/ConditionalStyles.php +++ b/src/PhpSpreadsheet/Reader/Xlsx/ConditionalStyles.php @@ -285,29 +285,34 @@ private function readDataBarOfConditionalRule(SimpleXMLElement $cfRule, array $c private function readColorScale(SimpleXMLElement|stdClass $cfRule): ConditionalColorScale { $colorScale = new ConditionalColorScale(); - $types = []; + $count = count($cfRule->colorScale->cfvo); + $idx = 0; foreach ($cfRule->colorScale->cfvo as $cfvoXml) { $attr = $cfvoXml->attributes() ?? []; $type = (string) ($attr['type'] ?? ''); - $types[] = $type; $val = $attr['val'] ?? null; - if ($type === 'min') { - $colorScale->setMinimumConditionalFormatValueObject(new ConditionalFormatValueObject($type, $val)); - } elseif ($type === 'percentile') { - $colorScale->setMidpointConditionalFormatValueObject(new ConditionalFormatValueObject($type, $val)); - } elseif ($type === 'max') { - $colorScale->setMaximumConditionalFormatValueObject(new ConditionalFormatValueObject($type, $val)); + if ($idx === 0) { + $method = 'setMinimumConditionalFormatValueObject'; + } elseif ($idx === 1 && $count === 3) { + $method = 'setMidpointConditionalFormatValueObject'; + } else { + $method = 'setMaximumConditionalFormatValueObject'; + } + if ($type !== 'formula') { + $colorScale->$method(new ConditionalFormatValueObject($type, $val)); + } else { + $colorScale->$method(new ConditionalFormatValueObject($type, null, $val)); } + ++$idx; } $idx = 0; foreach ($cfRule->colorScale->color as $color) { - $type = $types[$idx]; $rgb = $this->styleReader->readColor($color); - if ($type === 'min') { + if ($idx === 0) { $colorScale->setMinimumColor(new Color($rgb)); - } elseif ($type === 'percentile') { + } elseif ($idx === 1 && $count === 3) { $colorScale->setMidpointColor(new Color($rgb)); - } elseif ($type === 'max') { + } else { $colorScale->setMaximumColor(new Color($rgb)); } ++$idx; diff --git a/src/PhpSpreadsheet/Writer/Xlsx/Worksheet.php b/src/PhpSpreadsheet/Writer/Xlsx/Worksheet.php index 804a2eeb8b..4a981aa93c 100644 --- a/src/PhpSpreadsheet/Writer/Xlsx/Worksheet.php +++ b/src/PhpSpreadsheet/Writer/Xlsx/Worksheet.php @@ -763,8 +763,23 @@ private static function writeColorScaleElements(XMLWriter $objWriter, ?Condition $useMin = $minCfvo !== null || $minArgb !== null; if ($useMin) { $objWriter->startElement('cfvo'); - $objWriter->writeAttribute('type', $minCfvo?->getType() ?? 'min'); - self::writeAttributeIf($objWriter, $minCfvo?->getValue() !== null, 'val', (string) $minCfvo?->getValue()); + $type = 'min'; + $value = null; + if ($minCfvo !== null) { + $typex = $minCfvo->getType(); + if ($typex === 'formula') { + $value = $minCfvo->getCellFormula(); + if ($value !== null) { + $type = $typex; + } + } else { + $type = $typex; + $defaults = ['number' => '0', 'percent' => '0', 'percentile' => '10']; + $value = $minCfvo->getValue() ?? $defaults[$type] ?? null; + } + } + $objWriter->writeAttribute('type', $type); + self::writeAttributeIf($objWriter, $value !== null, 'val', (string) $value); $objWriter->endElement(); } $midCfvo = $colorScale->getMidpointConditionalFormatValueObject(); @@ -772,8 +787,23 @@ private static function writeColorScaleElements(XMLWriter $objWriter, ?Condition $useMid = $midCfvo !== null || $midArgb !== null; if ($useMid) { $objWriter->startElement('cfvo'); - $objWriter->writeAttribute('type', $midCfvo?->getType() ?? 'percentile'); - $objWriter->writeAttribute('val', (string) (($midCfvo?->getValue()) ?? '50')); + $type = 'percentile'; + $value = '50'; + if ($midCfvo !== null) { + $type = $midCfvo->getType(); + if ($type === 'formula') { + $value = $midCfvo->getCellFormula(); + if ($value === null) { + $type = 'percentile'; + $value = '50'; + } + } else { + $defaults = ['number' => '0', 'percent' => '50', 'percentile' => '50']; + $value = $midCfvo->getValue() ?? $defaults[$type] ?? null; + } + } + $objWriter->writeAttribute('type', $type); + self::writeAttributeIf($objWriter, $value !== null, 'val', (string) $value); $objWriter->endElement(); } $maxCfvo = $colorScale->getMaximumConditionalFormatValueObject(); @@ -781,8 +811,23 @@ private static function writeColorScaleElements(XMLWriter $objWriter, ?Condition $useMax = $maxCfvo !== null || $maxArgb !== null; if ($useMax) { $objWriter->startElement('cfvo'); - $objWriter->writeAttribute('type', $maxCfvo?->getType() ?? 'max'); - self::writeAttributeIf($objWriter, $maxCfvo?->getValue() !== null, 'val', (string) $maxCfvo?->getValue()); + $type = 'max'; + $value = null; + if ($maxCfvo !== null) { + $typex = $maxCfvo->getType(); + if ($typex === 'formula') { + $value = $maxCfvo->getCellFormula(); + if ($value !== null) { + $type = $typex; + } + } else { + $type = $typex; + $defaults = ['number' => '0', 'percent' => '100', 'percentile' => '90']; + $value = $maxCfvo->getValue() ?? $defaults[$type] ?? null; + } + } + $objWriter->writeAttribute('type', $type); + self::writeAttributeIf($objWriter, $value !== null, 'val', (string) $value); $objWriter->endElement(); } if ($useMin) { diff --git a/tests/PhpSpreadsheetTests/Reader/Xlsx/Issue4049Test.php b/tests/PhpSpreadsheetTests/Reader/Xlsx/Issue4049Test.php new file mode 100644 index 0000000000..056192d168 --- /dev/null +++ b/tests/PhpSpreadsheetTests/Reader/Xlsx/Issue4049Test.php @@ -0,0 +1,38 @@ +load($xlsxFile); + $sheet = $spreadsheet->getActiveSheet(); + $conditionals = $sheet->getConditionalStylesCollection(); + self::assertCount(1, $conditionals); + self::assertSame('E9:E14', array_keys($conditionals)[0]); + $cond1 = $conditionals['E9:E14']; + self::assertCount(1, $cond1); + self::assertSame('colorScale', $cond1[0]->getConditionType()); + $colorScale = $cond1[0]->getColorScale(); + self::assertNotNull($colorScale); + $min = $colorScale->getMinimumConditionalFormatValueObject(); + self::assertSame('formula', $min->getType()); + self::assertSame('25', $min->getCellFormula()); + + self::assertNull($colorScale->getMidpointConditionalFormatValueObject()); + + $max = $colorScale->getMaximumConditionalFormatValueObject(); + self::assertSame('max', $max->getType()); + self::assertNull($max->getValue()); + + $spreadsheet->disconnectWorksheets(); + } +} diff --git a/tests/data/Reader/XLSX/issue.4049.xlsx b/tests/data/Reader/XLSX/issue.4049.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..ef8d47351cd63c5a70a1d38ca851fef1e7fca131 GIT binary patch literal 4774 zcmai22RzjO|Hs+ljziA&wL&<1WMn2Ht8f_^XRlCZ)+wAw_ADbalslBYvSn3t=4EFj zgvj_GeShE7|JV2Ty7l=y9`}CU`}KOiUeDL_b-G#rLRvghQc}G067|M-r;Gu2?c*$L z<6-IT;A-djp9>Ko9~bA$0W;SY=oykP;1#NClMXNjunMjDfDV)j%k9ea=wUZi?s(v! zrlf;EKC4TQ;_liTSFfsXs#f3t&LtELksn>fx)~?cnj1RtoOI~OO+<&fX>6P8_X>za zX@6b8U#tG+1t{vf4_aZG*X3ogqGuN0R#48e=f}wwk@a1^bG`4`lG?^a(lr0qub%Ks zFEsXC(%p2n2>fWD2UfKg#LV7x1Nq?J} zDCh5ztUM>+Ty8#D+Kc|{rG>5*2`Pz{1A+)2503+x46j^nR_&D+sd6T!@WwiQa&MStikILxUe1&P4?=>7n5Zcl)la|+nb3}zw@9|&e zw1*Q%lICaosPcBX zbw;Yen$znFK4(L^ClK`D{kN|rgT~1ZtE(yIcXk_Odbwfd)oY=PM;ajyL$Si|n5Q1M z9aKRfm!}naeK;5ByW&`P%TTJ4_tD4(fh;CV;88`6DXW1!(+=|o)LjQtJUAKS?E@@t z?j?B$V5I#5AJBCp$s#h}U+UoASjN*qb)=h@2KB|*w**q-bsC3JbNLnJ1O5RFGdFyh zr7lex%zlwyr(??C)?^J?{m%Y^jz$VL#xW3+!RI_6E1kf=F{XNO=FWkFMh<9BT2to~h4FCSs0l|JT>6=ilk z<7uK@h(9xB2dE;Y0J@^}_&mk<7Bw%iW24qbTl9FH>T;X5v>&XL-r_q`>bx?dQDN_) z2a3z++v1YG2->K4awE&`L1W1Z@HoNP5KQz;1bBGcg#VUcoIh&yRDxM}z@1O0nb7fl zbiWpP3lv0*1^a|U{o{x4MToDehq>ml-qgqkIYFV+#a?v}ch5nux;*K0|Mo1&n8uR+Y| z{n%N@@9dd;x%)~!hG&^IoRL-nEd}^n$z3o(j*1$>VZq0T&LnWTACA_y#@W%UnngZGZh?ISE&JH>VJp@eGH;>fw)#{cL~tSvE_tY5X^{$ zi-|SG_B?BZ5COxY#O<8yEsQ}R`dQ2FX^H(U&m&*_kl2)aDf16XK7y&gIBtHTd2XXD zN|0~?EKqkpGNFx}gZi8p{aIh7ls)=6xmJB`-v`PuY;-Q_kE~6^5^wycta&C5%}U`z1bnPT*H_? z6)V(CMeB%t)HH|q;vDGZNF$>Wpv8ZITcKc@Y1%JSvhebBUQEA zbNP|23qFV`7RTyp49qO>Wt%^sZrMyY;3Z-QR;Ng_>}-c&shgU%YxS{{s$DHXjAVu7 zuLK5HxlB#DI(Y#iJ@)3{K2gf;eMS4J8KajVhw+>B1dG$ft6jkX{2F0^d$sd?`P}m6 z)N-!C*C{Mp5$J^axBP-4qJa07b`yn-%bZ2IG5VUojmWs?%33rs+O74XOObaR<4n06 zjfF>=+nAR{u+{sS-LtV+WkKs@Ix9Vc4g%V0-EcA;kL9P#Xg6OU{^d{Od?e>@)XquW zNmMp5vD@5^mHg^QPhk-J{IZ2`i*boS3k$V~w*_xYRbQT!OpcYMVgUibs!P@>#r9ZFbvLrTC0ic(t5_r@(B8A1<= zkcwK|AA|E#N0)6O=bS1redBwtrjv#{dYG5arm9kt&4%S=7Q4}PEsjWq_FN7$%mH5h zM*4a2(NPn>-r=|qm?2+T5_87D)%mkzRQ+zojU3NE%(cHty+dcQT0+N0FRiIA=+w-bthVuP7 zAZ&!~9_YwAnfUhs)>4aWFq=Aot&*LqV_m3SlH)>y=Z09G!*#M!s(&jq+8>4HX>aKP zw=wYYI93P8qJXkTiK>Miba4>uLn7)=Q3GvHxHAG?#OHyZ=YoQ4t-fxf_I-CZ_XIHk zg<8ccA4O%Lul*?_fxzbrcN!4XYE&r#@!@P_B|Q%GdSchQuxrD{;?in>%I$xshH;h z8BNt*riapeBJ>(%;oq~K8V|_kQvEFJ+SG;A5giEMGk{Fn%z6sLxjM1RYTp^c{o6wq z^X4o82(Lvwtl&fYJpctFxsJjiUc%Xk_DSm8=wPqAqN`i=tLbu>4{6r4dN4dIfqoZy?uZJ&!P@&r z6ZK%ImY5P%vhy|=M#YrwF#jq@mFe!{O#H%IOwul*TkspzC5QKuLv)(iQDBP1cDaJ} z-6A37OKKjyBV{?#;gMcuR`8x778h$H1Q3zRD#C3C&+pO&U^HtMJWTelc<+cz=+%pb z)XYAMfBLA+X-od{n{<`6N`H(>?u_${!~@vV;VKZPv1XvJ!pfX!pHa=&(WB4B z{Yjj1PP1xEM>)J0@lLvfq|cADpUOLeD|;HEs5tDT9;e0E7bt2UoMbUd7FG8i*DA{X zwPgZ-6PlN_l<( zYHGrd|6IrP%m*KY&wyOc-9kea^)v@IcXhJ0P=OhJX|dRWi@NqiUzO7@aB$8uKtA^; z`-OBe3dU$Y^I5Iuxe^~A^%B!$e1*|W`~CsCeR18RXLz%Yjnz}@rT4a={~=psUP8HI zp+rzX)Go~Q3YQpA?4T3$$O<_YUeH#y5N^3L1QAJzCSNKH;YwmM4ZLG_20a!{5;@2d z*-xvpuT(~+zZmgIm!D#TWtreJMVZ>v{TS`26J4y`ol|-VcLqHFs*4$aoWT#Ra{44r z)Jkcbl4C0rq=q|=2+Rm7USBgZL;IY~SUw48=!0lEH;4A<-gc%WRu-gu);#j4ZU1P% z9(};*w;sah-7HXuq#^VJJ1^WIS_@D9&I?Mdcv%BvrF)QKd1K1OynN)f80NMJ1=YGq z2q&^%-*(1*pqi9geoCB!GR-*uy1GkDjaoH-&!yCPh^~bVvi@4^b``-b2F^p?3*y># z?q_=)E?za_=UPASk)C2WQY`E8w1Y(_zVV-`z-G#f18;!%wgdw*sz3fwn-gzIlX zr$iMB1rZJ41fA`+zPsgv>=zP$czgn^j zWOaJnw5R*j%!og-T_=x6p$8M>MV-6ZhIhM?PguGu#Z_7HRTGUBB3MD)dseQH)ea)+ zVgELtQ?VyS-R)JpV*>H6Q2)+&Ba`mkl#)S$fLw~3t4m+l(ra9Geg3-rWU!r$&LWaa z%MD6y%yY=4I2qA?KD+5zT#%e`J?oWg=l-XjyP@Ld>IHZ8vM}^@hkKeG z7ipzIj}9&?_6ZI-Hs~Z=W!S^gwwq>1H4=cvE85oYg?%SRrqGvRT*d4m@679}IxdJU zR~Bq$qT7hUsaOFKzK)!W&qE~xO(Gw*Z7%B2CNw*}Xtw6w*055jO%vLtF2dSeO|69K79@v;yM)=x3 zI<&WeD?W1Cq2s0-+N?)$)}>BCL`;@}4?Oz$j0DfnRuNwAJN~+jH(a^v7Btyq(aJ3E zMsI0e7U8gW1fkJ$l25Nmtw~I=&z*I@!tZ@O5W!UofIA-mM`*{;9g zB;z0E!^+oP)^M!KL_C7 zQBJG<MI` s_LzcxS|ZM!Q-t*U!qZ9jYwUw@8uP!Dx2_fu@i96%?vZvp_n!Ftf6@x1&Hw-a literal 0 HcmV?d00001 From fcc5cf13698868de8c48b5662bc68cada2723c5a Mon Sep 17 00:00:00 2001 From: oleibman <10341515+oleibman@users.noreply.github.com> Date: Thu, 30 May 2024 09:35:40 -0700 Subject: [PATCH 2/3] Minor Test Improvement --- .../PhpSpreadsheetTests/Reader/Xlsx/Issue4049Test.php | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/tests/PhpSpreadsheetTests/Reader/Xlsx/Issue4049Test.php b/tests/PhpSpreadsheetTests/Reader/Xlsx/Issue4049Test.php index 056192d168..d1d6cde386 100644 --- a/tests/PhpSpreadsheetTests/Reader/Xlsx/Issue4049Test.php +++ b/tests/PhpSpreadsheetTests/Reader/Xlsx/Issue4049Test.php @@ -5,15 +5,17 @@ namespace PhpOffice\PhpSpreadsheetTests\Reader\Xlsx; use PhpOffice\PhpSpreadsheet\Reader\Xlsx; -use PHPUnit\Framework\TestCase; +use PhpOffice\PhpSpreadsheetTests\Functional\AbstractFunctional; -class Issue4049Test extends TestCase +class Issue4049Test extends AbstractFunctional { - public static function testPr1869(): void + public function testColorScale(): void { $xlsxFile = 'tests/data/Reader/XLSX/issue.4049.xlsx'; $reader = new Xlsx(); - $spreadsheet = $reader->load($xlsxFile); + $oldSpreadsheet = $reader->load($xlsxFile); + $spreadsheet = $this->writeAndReload($oldSpreadsheet, 'Xlsx'); + $oldSpreadsheet->disconnectWorksheets(); $sheet = $spreadsheet->getActiveSheet(); $conditionals = $sheet->getConditionalStylesCollection(); self::assertCount(1, $conditionals); From 24e6f4d8378a12474d3fc4b7d939ef114a6aff43 Mon Sep 17 00:00:00 2001 From: oleibman <10341515+oleibman@users.noreply.github.com> Date: Fri, 31 May 2024 08:48:34 -0700 Subject: [PATCH 3/3] Update CHANGELOG.md --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index d89d59c7c4..daf78caf04 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -29,6 +29,7 @@ and this project adheres to [Semantic Versioning](https://semver.org). - POWER Null/Bool Args. [PR #4031](https://github.com/PHPOffice/PhpSpreadsheet/pull/4031) - Do Not Output Alignment and Protection for Conditional Format. [Issue #4025](https://github.com/PHPOffice/PhpSpreadsheet/issues/4025) [PR #4027](https://github.com/PHPOffice/PhpSpreadsheet/pull/4027) - Xls Conditional Format Improvements. [PR #4030](https://github.com/PHPOffice/PhpSpreadsheet/pull/4030) +- Conditional Color Scale Improvements. [Issue #4049](https://github.com/PHPOffice/PhpSpreadsheet/issues/4049) [PR #4050](https://github.com/PHPOffice/PhpSpreadsheet/pull/4050) ## 2024-05-11 - 2.1.0