From 7ae90387260e16bc3dd9a1d2bfd92dd9d699c811 Mon Sep 17 00:00:00 2001 From: Steve Boyd Date: Wed, 31 May 2023 15:55:23 +1200 Subject: [PATCH 1/3] DEP Explicitly require psr/http-message ^1 --- composer.json | 1 + 1 file changed, 1 insertion(+) diff --git a/composer.json b/composer.json index e71bcc6ccf8..368af968ccd 100644 --- a/composer.json +++ b/composer.json @@ -33,6 +33,7 @@ "monolog/monolog": "^1.16", "nikic/php-parser": "^4.10.5", "psr/container": "^1", + "psr/http-message": "^1", "silverstripe/config": "^1@dev", "silverstripe/assets": "^1@dev", "silverstripe/vendor-plugin": "^1.6", From 51fd1d6b7e45d7a7c0c26cae3cab7811b871362e Mon Sep 17 00:00:00 2001 From: Steve Boyd Date: Tue, 13 Jun 2023 10:11:12 +1200 Subject: [PATCH 2/3] FIX Handle __TRAIT__ in i18nTextCollector --- src/i18n/TextCollection/i18nTextCollector.php | 10 +++--- tests/php/i18n/i18nTextCollectorTest.php | 33 +++++++++++++++++++ 2 files changed, 38 insertions(+), 5 deletions(-) diff --git a/src/i18n/TextCollection/i18nTextCollector.php b/src/i18n/TextCollection/i18nTextCollector.php index 558dadf63dc..212655e87a1 100644 --- a/src/i18n/TextCollection/i18nTextCollector.php +++ b/src/i18n/TextCollection/i18nTextCollector.php @@ -625,14 +625,14 @@ public function collectFromCode($content, $fileName, Module $module) } } - // Check class - if ($id === T_CLASS) { + // Check class and trait + if ($id === T_CLASS || $id === T_TRAIT) { // Skip if previous token was '::'. E.g. 'Object::class' if (is_array($previousToken) && $previousToken[0] === T_DOUBLE_COLON) { if ($inSelf) { // Handle self::class by allowing logic further down - // for __CLASS__ to handle an array of class parts - $id = T_CLASS_C; + // for __CLASS__/__TRAIT__ to handle an array of class parts + $id = $id === T_TRAIT ? T_TRAIT_C : T_CLASS_C; $inSelf = false; } elseif ($potentialClassName) { $id = T_CONSTANT_ENCAPSED_STRING; @@ -722,7 +722,7 @@ function ($input) { } else { throw new LogicException("Invalid string escape: " . $text); } - } elseif ($id === T_CLASS_C) { + } elseif ($id === T_CLASS_C || $id === T_TRAIT_C) { // Evaluate __CLASS__ . '.KEY' and self::class concatenation $text = implode('\\', $currentClass); } else { diff --git a/tests/php/i18n/i18nTextCollectorTest.php b/tests/php/i18n/i18nTextCollectorTest.php index df33ced4706..f79c534c360 100644 --- a/tests/php/i18n/i18nTextCollectorTest.php +++ b/tests/php/i18n/i18nTextCollectorTest.php @@ -407,6 +407,39 @@ public function pointlessFunction4() ); } + public function testCollectFromTrait() + { + $c = i18nTextCollector::create(); + $mymodule = ModuleLoader::inst()->getManifest()->getModule('i18ntestmodule'); + $php = <<assertEquals( + [ + 'SilverStripe\\Framework\\Core\\MyTrait.NEWLINES' => "New Lines", + ], + $c->collectFromCode($php, null, $mymodule) + ); + } + public function testNewlinesInEntityValues() { $c = i18nTextCollector::create(); From fcf5e324dd00c8a8cd42b014f92858734077b7f2 Mon Sep 17 00:00:00 2001 From: Thomas Portelange Date: Fri, 1 Mar 2024 15:10:50 +0100 Subject: [PATCH 3/3] FIX Handle non-breakable spaces Fixes issue https://github.com/silverstripe/silverstripe-framework/issues/11162 --- src/Forms/NumericField.php | 12 +++++++- tests/php/Forms/NumericFieldTest.php | 43 +++++++--------------------- 2 files changed, 22 insertions(+), 33 deletions(-) diff --git a/src/Forms/NumericField.php b/src/Forms/NumericField.php index 4c05caa463e..baf568a626c 100644 --- a/src/Forms/NumericField.php +++ b/src/Forms/NumericField.php @@ -101,10 +101,20 @@ protected function getNumberType() return NumberFormatter::TYPE_DOUBLE; } + /** + * In some cases and locales, validation expects non-breaking spaces. + * This homogenises regular, narrow and thin non-breaking spaces to a regular space character. + * + */ + private function clean(?string $value): string + { + return trim(str_replace(["\u{00A0}", "\u{202F}", "\u{2009}"], ' ', $value ?? '')); + } + public function setSubmittedValue($value, $data = null) { // Save original value in case parse fails - $value = trim($value ?? ''); + $value = $this->clean($value); $this->originalValue = $value; // Empty string is no-number (not 0) diff --git a/tests/php/Forms/NumericFieldTest.php b/tests/php/Forms/NumericFieldTest.php index f07a6dfc51c..45e75a93d70 100644 --- a/tests/php/Forms/NumericFieldTest.php +++ b/tests/php/Forms/NumericFieldTest.php @@ -11,27 +11,6 @@ class NumericFieldTest extends SapphireTest { protected $usesDatabase = false; - /** - * In some cases and locales, validation expects non-breaking spaces. - * This homogenises narrow and regular NBSPs to a regular space character - * - * Duplicates non-public NumericField::clean method - * - * @param string $input - * @return string The input value, with all non-breaking spaces replaced with spaces - */ - protected function clean($input) - { - return str_replace( - [ - html_entity_decode(' ', 0, 'UTF-8'), - html_entity_decode(' ', 0, 'UTF-8'), // narrow non-breaking space - ], - ' ', - trim($input ?? '') - ); - } - /** * Test that data loaded in via Form::loadDataFrom(DataObject) will populate the field correctly, * and can format the database value appropriately for the frontend @@ -64,8 +43,8 @@ public function testSetValue($locale, $scale, $input, $output) // Test expected formatted value $this->assertEquals( - $this->clean($output), - $this->clean($field->Value()), + (string) $output, + $field->Value(), "Expected $input to be formatted as $output in locale $locale" ); @@ -92,11 +71,11 @@ public function dataForTestSetValue() ['nl_NL', null, '12.1', '12,1'], ['nl_NL', 1, '14000.5', "14.000,5"], // fr - ['fr_FR', 0, '13000', "13 000"], + ['fr_FR', 0, '13000', "13 000"], // With a narrow non breaking space ['fr_FR', 0, '15', '15'], ['fr_FR', null, '12.0', '12,0'], ['fr_FR', null, '12.1', '12,1'], - ['fr_FR', 1, '14000.5', "14 000,5"], + ['fr_FR', 1, '14000.5', "14 000,5"], // With a narrow non breaking space // us ['en_US', 0, '13000', "13,000"], ['en_US', 0, '15', '15'], @@ -175,15 +154,15 @@ public function dataForTestSubmittedValue() ['nl_NL', 1, '15,000.5', false], // fr - ['fr_FR', 0, '13000', 13000, '13 000'], + ['fr_FR', 0, '13000', 13000, '13 000'], // With a narrow non breaking space ['fr_FR', 2, '12,00', 12.0], ['fr_FR', 2, '12.00', false], - ['fr_FR', 1, '11 000', 11000, '11 000,0'], - ['fr_FR', 0, '11.000', 11000, '11 000'], + ['fr_FR', 1, '11 000', 11000, '11 000,0'], // With a narrow non breaking space + ['fr_FR', 0, '11.000', 11000, '11 000'], // With a narrow non breaking space ['fr_FR', null, '11,000', 11.000, '11,0'], - ['fr_FR', 1, '15 000,5', 15000.5], + ['fr_FR', 1, '15 000,5', 15000.5, '15 000,5'], // With a narrow non breaking space ['fr_FR', 1, '15 000.5', false], - ['fr_FR', 1, '15.000,5', 15000.5, '15 000,5'], + ['fr_FR', 1, '15.000,5', 15000.5, '15 000,5'], // With a narrow non breaking space ['fr_FR', 1, '15,000.5', false], // us ['en_US', 0, '13000', 13000, '13,000'], @@ -260,8 +239,8 @@ public function testSetSubmittedValue($locale, $scale, $submittedValue, $dataVal $cleanedInput = $submittedValue; } $this->assertEquals( - $this->clean($cleanedInput), - $this->clean($field->Value()), + $cleanedInput, + $field->Value(), "Expected input $submittedValue to be cleaned up as $cleanedInput in locale $locale" ); }