previousLocaleSetting = null; } protected function tearDown(): void { parent::tearDown(); // If a test sets the locale, reset it on teardown if ($this->previousLocaleSetting) { setlocale(LC_CTYPE, $this->previousLocaleSetting); } } /** * Test {@link Text->LimitCharacters()} */ public static function providerLimitCharacters() { // Plain text values always encoded safely // HTML stored in non-html fields is treated literally. return [ ['The little brown fox jumped over the lazy cow.', 'The little brown fox…'], ['
Short & Sweet
', '<p>Short & Sweet</p>'], ['This text contains & in it', 'This text contains &…'], ['Is an umault in schön?', 'Is an umault in schö…'], ]; } /** * Test {@link Text->LimitCharacters()} * * @param string $originalValue * @param string $expectedValue */ #[DataProvider('providerLimitCharacters')] public function testLimitCharacters($originalValue, $expectedValue) { $textObj = DBField::create_field('Text', $originalValue); $result = $textObj->obj('LimitCharacters')->forTemplate(); $this->assertEquals($expectedValue, $result); } /** * @return array */ public static function providerLimitCharactersToClosestWord() { return [ // Standard words limited, ellipsis added if truncated ['Lorem ipsum dolor sit amet', 24, 'Lorem ipsum dolor sit…'], // Complete words less than the character limit don't get truncated, ellipsis not added ['Lorem ipsum', 24, 'Lorem ipsum'], ['Lorem', 24, 'Lorem'], ['', 24, ''], // No words produces nothing! // Special characters are encoded safely ['Nice & Easy', 24, 'Nice & Easy'], // HTML stored in non-html fields is treated literally. // If storing HTML you should use DBHTMLText instead ['Lorem ipsum dolor sit amet
', 24, '<p>Lorem ipsum dolor…'], ['Lorem ipsum dolor sit amet
', 24, '<p><span>Lorem ipsum…'], ['Lorem ipsum
', 24, '<p>Lorem ipsum</p>'], ['Lorem & ipsum dolor sit amet', 24, 'Lorem & ipsum dolor…'], ['Is an umault in schön or not?', 22, 'Is an umault in schön…'], ]; } /** * Test {@link Text->LimitCharactersToClosestWord()} * * * @param string $originalValue Raw string input * @param int $limit * @param string $expectedValue Expected template value */ #[DataProvider('providerLimitCharactersToClosestWord')] public function testLimitCharactersToClosestWord($originalValue, $limit, $expectedValue) { $textObj = DBField::create_field('Text', $originalValue); $result = $textObj->obj('LimitCharactersToClosestWord', [$limit])->forTemplate(); $this->assertEquals($expectedValue, $result); } /** * Test {@link Text->LimitWordCount()} */ public static function providerLimitWordCount() { return [ // Standard words limited, ellipsis added if truncated ['The little brown fox jumped over the lazy cow.', 3, 'The little brown…'], [' This text has white space around the ends ', 3, 'This text has…'], // Words less than the limit word count don't get truncated, ellipsis not added ['Two words', 3, 'Two words'], // Two words shouldn't have an ellipsis ['These three words', 3, 'These three words'], // Three words shouldn't have an ellipsis ['One', 3, 'One'], // Neither should one word ['', 3, ''], // No words produces nothing! // Text with special characters ['Nice & Easy', 3, 'Nice & Easy'], ['One & Two & Three', 3, 'One & Two…'], // HTML stored in non-html fields is treated literally. // If storing HTML you should use DBHTMLText instead ['Text inside a paragraph tag should also work
', 3, '<p>Text inside a…'], ['Two words
', 3, '<p>Two words</p>'], // Check UTF8 ['Is an umault in schön or not?', 5, 'Is an umault in schön…'], ]; } /** * Test {@link DBText->LimitWordCount()} * * * @param string $originalValue Raw string input * @param int $limit Number of words * @param string $expectedValue Expected template value */ #[DataProvider('providerLimitWordCount')] public function testLimitWordCount($originalValue, $limit, $expectedValue) { $textObj = DBField::create_field('Text', $originalValue); $result = $textObj->obj('LimitWordCount', [$limit])->forTemplate(); $this->assertEquals($expectedValue, $result); } public static function providerLimitSentences() { return [ ['', 2, ''], ['First sentence.', 2, 'First sentence.'], ['First sentence. Second sentence.', 2, 'First sentence. Second sentence.'], // HTML stored in non-html fields is treated literally. // If storing HTML you should use DBHTMLText instead ['First sentence.
', 2, '<p>First sentence.</p>'], ['First sentence. Second sentence. Third sentence
', 2, '<p>First sentence. Second sentence.'], // Check UTF8 ['Is schön. Isn\'t schön.', 1, 'Is schön.'], ]; } /** * Test {@link DBText->LimitSentences()} * * @param string $originalValue * @param int $limit Number of sentences * @param string $expectedValue Expected template value */ #[DataProvider('providerLimitSentences')] public function testLimitSentences($originalValue, $limit, $expectedValue) { $textObj = DBField::create_field('Text', $originalValue); $result = $textObj->obj('LimitSentences', [$limit])->forTemplate(); $this->assertEquals($expectedValue, $result); } public static function providerFirstSentence() { return [ ['', ''], ['First sentence.', 'First sentence.'], ['First sentence. Second sentence', 'First sentence.'], ['First sentence? Second sentence', 'First sentence?'], ['First sentence! Second sentence', 'First sentence!'], // HTML stored in non-html fields is treated literally. // If storing HTML you should use DBHTMLText instead ['First sentence. Second sentence. Third sentence
', '<p>First sentence.'], // Check UTF8 ['Is schön. Isn\'t schön.', 'Is schön.'], ]; } /** * @param string $originalValue * @param string $expectedValue */ #[DataProvider('providerFirstSentence')] public function testFirstSentence($originalValue, $expectedValue) { $textObj = DBField::create_field('Text', $originalValue); $result = $textObj->obj('FirstSentence')->forTemplate(); $this->assertEquals($expectedValue, $result); } /** * each test is in the format input, character limit, highlight, expected output * * @return array */ public static function providerContextSummary() { return [ [ 'This is some text. It is a test', 20, 'test', '… text. It is a test' ], [ // Retains case of original string 'This is some test text. Test test what if you have multiple keywords.', 50, 'some test', 'This is some test text.' . ' Test test what if you have…' ], [ 'Here is some text & HTML included', 20, 'html', '… text & HTML inc…' ], [ 'A dog ate a cat while looking at a Foobar', 100, 'a', // test that it does not highlight too much (eg every a) 'A dog ate a cat while looking at a Foobar', ], [ 'A dog ate a cat while looking at a Foobar', 100, 'ate', // it should highlight 3 letters or more. 'A dog ate a cat while looking at a Foobar', ], [ 'both schön and können have umlauts', 21, 'schön', // check UTF8 support 'both schön and können…', ], [ 'both schön and können have umlauts', 21, '', // check non-existent search term 'both schön and können…', ] ]; } /** * each test is in the format input, word limit, add ellipsis (false or string), expected output * * @return array */ public static function providerSummary() { return [ 'simple test' => [ 'This is some text. It is a test', 3, false, 'This is some…', ], 'custom ellipses' => [ // check custom ellipsis 'This is a test text in a longer sentence and a custom ellipsis.', 8, '...', // regular dots instead of the ellipsis character 'This is a test text in a longer...', ], 'umlauts' => [ 'both schön and können have umlauts', 5, false, 'both schön and können have…', ], 'invalid UTF' => [ // check invalid UTF8 handling — input is an invalid UTF sequence, output should be empty string "\xf0\x28\x8c\xbc", 50, false, '', ], 'treats period as sentence boundary' => [ 'This is some text. It is a test. There are three sentences.', 10, false, 'This is some text. It is a test.', ], 'treats exclamation mark as sentence boundary' => [ 'This is some text! It is a test! There are three sentences.', 10, false, 'This is some text! It is a test!', ], 'treats question mark as sentence boundary' => [ 'This is some text? It is a test? There are three sentences.', 10, false, 'This is some text? It is a test?', ], 'does not treat colon as sentence boundary' => [ 'This is some text: It is a test: There are three sentences.', 10, false, 'This is some text: It is a test: There are…', ], ]; } /** * @param string $originalValue Input * @param int $limit Number of characters * @param string $keywords Keywords to highlight * @param string $expectedValue Expected output (XML encoded safely) */ #[DataProvider('providerContextSummary')] public function testContextSummary($originalValue, $limit, $keywords, $expectedValue) { $text = DBField::create_field('Text', $originalValue); $result = $text->obj('ContextSummary', [$limit, $keywords])->forTemplate(); // it should highlight 3 letters or more. $this->assertEquals($expectedValue, $result); } public function testRAW() { $data = DBField::create_field('Text', 'This & This'); $this->assertEquals($data->RAW(), 'This & This'); } public function testXML() { $data = DBField::create_field('Text', 'This & This'); $this->assertEquals($data->XML(), 'This & This'); } public function testHTML() { $data = DBField::create_field('Text', 'This & This'); $this->assertEquals($data->HTML(), 'This & This'); } public function testJS() { $data = DBField::create_field('Text', '"this is a test"'); $this->assertEquals($data->JS(), '\"this is a test\"'); } public function testATT() { $data = DBField::create_field('Text', '"this is a test"'); $this->assertEquals($data->ATT(), '"this is a test"'); } public function testValidUtf8() { // Install a UTF-8 locale $this->previousLocaleSetting = setlocale(LC_CTYPE, 0); $locales = ['en_US.UTF-8', 'en_NZ.UTF-8', 'de_DE.UTF-8']; $localeInstalled = false; foreach ($locales as $locale) { if ($localeInstalled = setlocale(LC_CTYPE, $locale)) { break; } } // If the system doesn't have any of the UTF-8 locales, exit early if ($localeInstalled === false) { $this->markTestIncomplete('Unable to run this test because of missing locale!'); return; } $problematicText = html_entity_decode('This is a Test with non-breaking space!', ENT_COMPAT, 'UTF-8'); $textObj = new DBText('Test'); $textObj->setValue($problematicText); $this->assertTrue(mb_check_encoding($textObj->FirstSentence(), 'UTF-8')); } public function testDefaultEllipsis() { $textObj = new DBText('Test'); $this->assertEquals('…', $textObj->defaultEllipsis()); } /** * @param string $originalValue Input * @param int $words Number of words * @param mixed $add Ellipsis (false for default or string for custom text) * @param string $expectedValue Expected output (XML encoded safely) */ #[DataProvider('providerSummary')] public function testSummary($originalValue, $words, $add, $expectedValue) { $text = DBField::create_field(DBText::class, $originalValue); $result = $text->obj('Summary', [$words, $add])->forTemplate(); $this->assertEquals($expectedValue, $result); } public function testSummaryConfiguration() { $text = DBField::create_field(DBText::class, 'This is some text: It is a test: There are three sentences.'); // Doesn't treat colon as a boundary by default $this->assertSame('This is some text: It is a test: There are…', $text->Summary(10)); DBText::config()->merge('summary_sentence_separators', [':']); // Does treat colon as a boundary if configured to do so $this->assertSame('This is some text: It is a test:', $text->Summary(10)); } }