diff --git a/core/modules/telephone/src/Plugin/Field/FieldFormatter/TelephoneLinkFormatter.php b/core/modules/telephone/src/Plugin/Field/FieldFormatter/TelephoneLinkFormatter.php
index afdf54146d660629e51edff9b41f8fa5d9def47a..60ee25cbc3d91d4cdf028324d364772f7be24db1 100644
--- a/core/modules/telephone/src/Plugin/Field/FieldFormatter/TelephoneLinkFormatter.php
+++ b/core/modules/telephone/src/Plugin/Field/FieldFormatter/TelephoneLinkFormatter.php
@@ -67,6 +67,21 @@ public function viewElements(FieldItemListInterface $items, $langcode) {
$title_setting = $this->getSetting('title');
foreach ($items as $delta => $item) {
+ // If the telephone number is 5 or less digits, parse_url() will think
+ // it's a port number rather than a phone number which causes the link
+ // formatter to throw an InvalidArgumentException. Avoid this by inserting
+ // a dash (-) after the first digit - RFC 3966 defines the dash as a
+ // visual separator character and so will be removed before the phone
+ // number is used. See https://bugs.php.net/bug.php?id=70588 for more.
+ // While the bug states this only applies to numbers <= 65535, a 5 digit
+ // number greater than 65535 will cause parse_url() to return FALSE so
+ // we need the work around on any 5 digit (or less) number.
+ // First we strip whitespace so we're counting actual digits.
+ $phone_number = preg_replace('/\s+/', '', $item->value);
+ if (strlen($phone_number) <= 5) {
+ $phone_number = substr_replace($phone_number, '-', 1, 0);
+ }
+
// Render each element as link.
$element[$delta] = [
'#type' => 'link',
@@ -74,7 +89,7 @@ public function viewElements(FieldItemListInterface $items, $langcode) {
// itself as title.
'#title' => $title_setting ?: $item->value,
// Prepend 'tel:' to the telephone number.
- '#url' => Url::fromUri('tel:' . rawurlencode(preg_replace('/\s+/', '', $item->value))),
+ '#url' => Url::fromUri('tel:' . rawurlencode($phone_number)),
'#options' => ['external' => TRUE],
];
diff --git a/core/modules/telephone/tests/src/Functional/TelephoneFieldTest.php b/core/modules/telephone/tests/src/Functional/TelephoneFieldTest.php
index 75e626f66dd5c3b6f592b4705fc70cc7bea9f7c1..8d351819397b255ecd8a57546622e3876f266b36 100644
--- a/core/modules/telephone/tests/src/Functional/TelephoneFieldTest.php
+++ b/core/modules/telephone/tests/src/Functional/TelephoneFieldTest.php
@@ -31,20 +31,15 @@ class TelephoneFieldTest extends BrowserTestBase {
*/
protected $webUser;
+ /**
+ * {@inheritdoc}
+ */
protected function setUp() {
parent::setUp();
$this->drupalCreateContentType(['type' => 'article']);
$this->webUser = $this->drupalCreateUser(['create article content', 'edit own article content']);
$this->drupalLogin($this->webUser);
- }
-
- // Test fields.
-
- /**
- * Helper function for testTelephoneField().
- */
- public function testTelephoneField() {
// Add the telephone field to the article content type.
FieldStorageConfig::create([
@@ -74,29 +69,54 @@ public function testTelephoneField() {
'weight' => 1,
])
->save();
+ }
- // Display creation form.
+ /**
+ * Test to confirm the widget is setup.
+ *
+ * @covers \Drupal\telephone\Plugin\Field\FieldWidget\TelephoneDefaultWidget::formElement
+ */
+ public function testTelephoneWidget() {
$this->drupalGet('node/add/article');
$this->assertFieldByName("field_telephone[0][value]", '', 'Widget found.');
$this->assertRaw('placeholder="123-456-7890"');
+ }
+ /**
+ * Test the telephone formatter.
+ *
+ * @covers \Drupal\telephone\Plugin\Field\FieldFormatter\TelephoneLinkFormatter::viewElements
+ *
+ * @dataProvider providerPhoneNumbers
+ */
+ public function testTelephoneFormatter($input, $expected) {
// Test basic entry of telephone field.
$edit = [
'title[0][value]' => $this->randomMachineName(),
- 'field_telephone[0][value]' => "123456789",
+ 'field_telephone[0][value]' => $input,
];
- $this->drupalPostForm(NULL, $edit, t('Save'));
- $this->assertRaw('', 'A telephone link is provided on the article node page.');
+ $this->drupalPostForm('node/add/article', $edit, t('Save'));
+ $this->assertRaw('');
+ }
- // Add number with a space in it. Need to ensure it is stripped on output.
- $edit = [
- 'title[0][value]' => $this->randomMachineName(),
- 'field_telephone[0][value]' => "1234 56789",
+ /**
+ * Provides the phone numbers to check and expected results.
+ */
+ public function providerPhoneNumbers() {
+ return [
+ 'standard phone number' => ['123456789', '123456789'],
+ 'whitespace is removed' => ['1234 56789', '123456789'],
+ 'parse_url(0) return FALSE workaround' => ['0', '0-'],
+ 'php bug 70588 workaround - lower edge check' => ['1', '1-'],
+ 'php bug 70588 workaround' => ['123', '1-23'],
+ 'php bug 70588 workaround - with whitespace removal' => ['1 2 3 4 5', '1-2345'],
+ 'php bug 70588 workaround - upper edge check' => ['65534', '6-5534'],
+ 'php bug 70588 workaround - edge check' => ['65535', '6-5535'],
+ 'php bug 70588 workaround - invalid port number - lower edge check' => ['65536', '6-5536'],
+ 'php bug 70588 workaround - invalid port number - upper edge check' => ['99999', '9-9999'],
+ 'lowest number not affected by php bug 70588' => ['100000', '100000'],
];
-
- $this->drupalPostForm('node/add/article', $edit, t('Save'));
- $this->assertRaw('', 'Telephone link is output with whitespace removed.');
}
}