You are here

public function ImageTest::testLinkability in Drupal 10

Tests linkability of the image CKEditor widget.

Due to the complex overrides that `drupalImage.DrupalImage` is making, this is explicitly testing the "editingDowncast" and "dataDowncast" results. These are CKEditor 5 concepts.

@dataProvider providerLinkability

See also

https://ckeditor.com/docs/ckeditor5/latest/framework/guides/architecture...

File

core/modules/ckeditor5/tests/src/FunctionalJavascript/ImageTest.php, line 256

Class

ImageTest
@coversDefaultClass \Drupal\ckeditor5\Plugin\CKEditor5Plugin\ImageUpload @group ckeditor5 @internal

Namespace

Drupal\Tests\ckeditor5\FunctionalJavascript

Code

public function testLinkability(string $image_type, bool $unrestricted) {
  assert($image_type === 'inline' || $image_type === 'block');

  // Disable filter_html.
  if ($unrestricted) {
    FilterFormat::load('test_format')
      ->setFilterConfig('filter_html', [
      'status' => FALSE,
    ])
      ->save();
  }

  // Make the test content have either a block image or an inline image.
  $img_tag = '<img alt="drupalimage test image" data-entity-type="file" data-entity-uuid="' . $this->file
    ->uuid() . '" src="' . $this->file
    ->createFileUrl() . '" />';
  $this->host->body->value .= $image_type === 'block' ? $img_tag : "<p>{$img_tag}</p>";
  $this->host
    ->save();

  // Adjust the expectations accordingly.
  $expected_widget_class = $image_type === 'block' ? 'image' : 'image-inline';
  $page = $this
    ->getSession()
    ->getPage();
  $this
    ->drupalGet($this->host
    ->toUrl('edit-form'));
  $this
    ->waitForEditor();
  $assert_session = $this
    ->assertSession();

  // Initial state: the image CKEditor Widget is not selected.
  $drupalimage = $assert_session
    ->waitForElementVisible('css', ".ck-content .ck-widget.{$expected_widget_class}");
  $this
    ->assertNotEmpty($drupalimage);
  $this
    ->assertFalse($drupalimage
    ->hasClass('.ck-widget_selected'));

  // Assert the "editingDowncast" HTML before making changes.
  $assert_session
    ->elementExists('css', '.ck-content .ck-widget.' . $expected_widget_class . ' > img[src*="image-test.png"][alt="drupalimage test image"]');

  // Assert the "dataDowncast" HTML before making changes.
  $xpath = new \DOMXPath($this
    ->getEditorDataAsDom());
  $this
    ->assertNotEmpty($xpath
    ->query('//img[@alt="drupalimage test image"]'));
  $this
    ->assertEmpty($xpath
    ->query('//a'));

  // Assert the link button is present and not pressed.
  $link_button = $this
    ->getEditorButton('Link');
  $this
    ->assertSame('false', $link_button
    ->getAttribute('aria-pressed'));

  // Tests linking images.
  $drupalimage
    ->click();
  $this
    ->assertTrue($drupalimage
    ->hasClass('ck-widget_selected'));
  $this
    ->assertEditorButtonEnabled('Link');

  // Assert structure of image toolbar balloon.
  $this
    ->assertVisibleBalloon('.ck-toolbar[aria-label="Image toolbar"]');
  $link_image_button = $this
    ->getBalloonButton('Link image');

  // Click the "Link image" button.
  $this
    ->assertSame('false', $link_image_button
    ->getAttribute('aria-pressed'));
  $link_image_button
    ->press();

  // Assert structure of link form balloon.
  $balloon = $this
    ->assertVisibleBalloon('.ck-link-form');
  $url_input = $balloon
    ->find('css', '.ck-labeled-field-view__input-wrapper .ck-input-text');

  // Fill in link form balloon's <input> and hit "Save".
  $url_input
    ->setValue('http://www.drupal.org/association');
  $balloon
    ->pressButton('Save');

  // Assert the "editingDowncast" HTML after making changes. First assert the
  // link exists, then assert the expected DOM structure in detail.
  $assert_session
    ->elementExists('css', '.ck-content a[href*="//www.drupal.org/association"]');

  // For inline images, the link is wrapping the widget; for block images the
  // link lives inside the widget. (This is how it is implemented upstream, it
  // could be implemented differently, we just want to ensure we do not break
  // it. Drupal only cares about having its own "dataDowncast", the
  // "editingDowncast" is considered an implementation detail.)
  $assert_session
    ->elementExists('css', $image_type === 'inline' ? '.ck-content a[href*="//www.drupal.org/association"] .ck-widget.' . $expected_widget_class . ' > img[src*="image-test.png"][alt="drupalimage test image"]' : '.ck-content .ck-widget.' . $expected_widget_class . ' a[href*="//www.drupal.org/association"] > img[src*="image-test.png"][alt="drupalimage test image"]');

  // Assert the "dataDowncast" HTML after making changes.
  $xpath = new \DOMXPath($this
    ->getEditorDataAsDom());
  $this
    ->assertCount(1, $xpath
    ->query('//a[@href="http://www.drupal.org/association"]/img[@alt="drupalimage test image"]'));
  $this
    ->assertEmpty($xpath
    ->query('//a[@href="http://www.drupal.org/association" and @class="trusted"]'));

  // Add `class="trusted"` to the link.
  $xpath = new \DOMXPath($this
    ->getEditorDataAsDom());
  $this
    ->assertEmpty($xpath
    ->query('//a[@href="http://www.drupal.org/association" and @class="trusted"]'));
  $this
    ->pressEditorButton('Source');
  $source_text_area = $assert_session
    ->waitForElement('css', '.ck-source-editing-area textarea');
  $this
    ->assertNotEmpty($source_text_area);
  $new_value = str_replace('<a ', '<a class="trusted" ', $source_text_area
    ->getValue());
  $source_text_area
    ->setValue('<p>temp</p>');
  $source_text_area
    ->setValue($new_value);
  $this
    ->pressEditorButton('Source');

  // When unrestricted, additional attributes on links should be retained.
  $xpath = new \DOMXPath($this
    ->getEditorDataAsDom());
  $this
    ->assertCount($unrestricted ? 1 : 0, $xpath
    ->query('//a[@href="http://www.drupal.org/association" and @class="trusted"]'));

  // Save the entity whose text field is being edited.
  $page
    ->pressButton('Save');

  // Assert the HTML the end user sees.
  $assert_session
    ->elementExists('css', $unrestricted ? 'a[href="http://www.drupal.org/association"].trusted img[src*="image-test.png"]' : 'a[href="http://www.drupal.org/association"] img[src*="image-test.png"]');

  // Go back to edit the now *linked* <drupal-media>. Everything from this
  // point onwards is effectively testing "upcasting" and proving there is no
  // data loss.
  $this
    ->drupalGet($this->host
    ->toUrl('edit-form'));
  $this
    ->waitForEditor();

  // Assert the "dataDowncast" HTML before making changes.
  $xpath = new \DOMXPath($this
    ->getEditorDataAsDom());
  $this
    ->assertNotEmpty($xpath
    ->query('//img[@alt="drupalimage test image"]'));
  $this
    ->assertNotEmpty($xpath
    ->query('//a[@href="http://www.drupal.org/association"]'));
  $this
    ->assertNotEmpty($xpath
    ->query('//a[@href="http://www.drupal.org/association"]/img[@alt="drupalimage test image"]'));
  $this
    ->assertCount($unrestricted ? 1 : 0, $xpath
    ->query('//a[@href="http://www.drupal.org/association" and @class="trusted"]'));

  // Tests unlinking images.
  $drupalimage
    ->click();
  $this
    ->assertEditorButtonEnabled('Link');
  $this
    ->assertSame('true', $this
    ->getEditorButton('Link')
    ->getAttribute('aria-pressed'));

  // Assert structure of image toolbar balloon.
  $this
    ->assertVisibleBalloon('.ck-toolbar[aria-label="Image toolbar"]');
  $link_image_button = $this
    ->getBalloonButton('Link image');
  $this
    ->assertSame('true', $link_image_button
    ->getAttribute('aria-pressed'));
  $link_image_button
    ->click();

  // Assert structure of link actions balloon.
  $this
    ->getBalloonButton('Edit link');
  $unlink_image_button = $this
    ->getBalloonButton('Unlink');

  // Click the "Unlink" button.
  $unlink_image_button
    ->click();
  $this
    ->assertSame('false', $this
    ->getEditorButton('Link')
    ->getAttribute('aria-pressed'));

  // Assert the "editingDowncast" HTML after making changes. Assert the
  // widget exists but not the link, or *any* link for that matter. Then
  // assert the expected DOM structure in detail.
  $assert_session
    ->elementExists('css', '.ck-content .ck-widget.' . $expected_widget_class);
  $assert_session
    ->elementNotExists('css', '.ck-content a');
  $assert_session
    ->elementExists('css', '.ck-content .ck-widget.' . $expected_widget_class . ' > img[src*="image-test.png"][alt="drupalimage test image"]');

  // Assert the "dataDowncast" HTML after making changes.
  $xpath = new \DOMXPath($this
    ->getEditorDataAsDom());
  $this
    ->assertCount(0, $xpath
    ->query('//a[@href="http://www.drupal.org/association"]/img[@alt="drupalimage test image"]'));
  $this
    ->assertCount(1, $xpath
    ->query('//img[@alt="drupalimage test image"]'));
  $this
    ->assertCount(0, $xpath
    ->query('//a'));
}