You are here

MediaUiFunctionalTest.php in Drupal 9

Same filename and directory in other branches
  1. 8 core/modules/media/tests/src/Functional/MediaUiFunctionalTest.php

File

core/modules/media/tests/src/Functional/MediaUiFunctionalTest.php
View source
<?php

namespace Drupal\Tests\media\Functional;

use Behat\Mink\Element\NodeElement;
use Drupal\Core\Field\FieldStorageDefinitionInterface;
use Drupal\Core\Url;
use Drupal\field\Entity\FieldConfig;
use Drupal\field\Entity\FieldStorageConfig;

/**
 * Ensures that media UI works correctly.
 *
 * @group media
 */
class MediaUiFunctionalTest extends MediaFunctionalTestBase {

  /**
   * Modules to enable.
   *
   * @var array
   */
  protected static $modules = [
    'block',
    'media_test_source',
  ];

  /**
   * {@inheritdoc}
   */
  protected $defaultTheme = 'stark';

  /**
   * {@inheritdoc}
   */
  protected function setUp() : void {
    parent::setUp();
    $this
      ->drupalPlaceBlock('local_actions_block');
    $this
      ->drupalPlaceBlock('local_tasks_block');
  }

  /**
   * Tests the media actions (add/edit/delete).
   */
  public function testMediaWithOnlyOneMediaType() {
    $session = $this
      ->getSession();
    $page = $session
      ->getPage();
    $assert_session = $this
      ->assertSession();
    $media_type = $this
      ->createMediaType('test', [
      'queue_thumbnail_downloads' => FALSE,
    ]);
    $this
      ->drupalGet('media/add');
    $assert_session
      ->statusCodeEquals(200);
    $assert_session
      ->addressEquals('media/add/' . $media_type
      ->id());
    $assert_session
      ->elementNotExists('css', '#edit-revision');

    // Tests media add form.
    $media_name = $this
      ->randomMachineName();
    $page
      ->fillField('name[0][value]', $media_name);
    $revision_log_message = $this
      ->randomString();
    $page
      ->fillField('revision_log_message[0][value]', $revision_log_message);
    $source_field = $this
      ->randomString();
    $page
      ->fillField('field_media_test[0][value]', $source_field);
    $page
      ->pressButton('Save');
    $media_id = $this->container
      ->get('entity_type.manager')
      ->getStorage('media')
      ->getQuery()
      ->accessCheck(FALSE)
      ->execute();
    $media_id = reset($media_id);

    /** @var \Drupal\media\MediaInterface $media */
    $media = $this->container
      ->get('entity_type.manager')
      ->getStorage('media')
      ->loadUnchanged($media_id);
    $this
      ->assertSame($media
      ->getRevisionLogMessage(), $revision_log_message);
    $this
      ->assertSame($media
      ->getName(), $media_name);

    // Tests media edit form.
    $media_type
      ->setNewRevision(FALSE);
    $media_type
      ->save();
    $media_name2 = $this
      ->randomMachineName();
    $this
      ->drupalGet('media/' . $media_id . '/edit');
    $assert_session
      ->checkboxNotChecked('edit-revision');
    $media_name = $this
      ->randomMachineName();
    $page
      ->fillField('name[0][value]', $media_name2);
    $page
      ->pressButton('Save');

    /** @var \Drupal\media\MediaInterface $media */
    $media = $this->container
      ->get('entity_type.manager')
      ->getStorage('media')
      ->loadUnchanged($media_id);
    $this
      ->assertSame($media
      ->getName(), $media_name2);

    // Test that there is no empty vertical tabs element, if the container is
    // empty (see #2750697).
    // Make the "Publisher ID" and "Created" fields hidden.
    $this
      ->drupalGet('/admin/structure/media/manage/' . $media_type
      ->id() . '/form-display');
    $page
      ->selectFieldOption('fields[created][parent]', 'hidden');
    $page
      ->selectFieldOption('fields[uid][parent]', 'hidden');
    $page
      ->pressButton('Save');

    // Assure we are testing with a user without permission to manage revisions.
    $this
      ->drupalLogin($this->nonAdminUser);

    // Check the container is not present.
    $this
      ->drupalGet('media/' . $media_id . '/edit');
    $assert_session
      ->elementNotExists('css', 'input.vertical-tabs__active-tab');

    // Continue testing as admin.
    $this
      ->drupalLogin($this->adminUser);

    // Enable revisions by default.
    $previous_revision_id = $media
      ->getRevisionId();
    $media_type
      ->setNewRevision(TRUE);
    $media_type
      ->save();
    $this
      ->drupalGet('media/' . $media_id . '/edit');
    $assert_session
      ->checkboxChecked('edit-revision');
    $page
      ->fillField('name[0][value]', $media_name);
    $page
      ->fillField('revision_log_message[0][value]', $revision_log_message);
    $page
      ->pressButton('Save');
    $this
      ->drupalGet('media/' . $media_id);
    $assert_session
      ->statusCodeEquals(404);

    /** @var \Drupal\media\MediaInterface $media */
    $media = $this->container
      ->get('entity_type.manager')
      ->getStorage('media')
      ->loadUnchanged($media_id);
    $this
      ->assertSame($media
      ->getRevisionLogMessage(), $revision_log_message);
    $this
      ->assertNotEquals($previous_revision_id, $media
      ->getRevisionId());

    // Test the status checkbox.
    $this
      ->drupalGet('media/' . $media_id . '/edit');
    $page
      ->uncheckField('status[value]');
    $page
      ->pressButton('Save');

    /** @var \Drupal\media\MediaInterface $media */
    $media = $this->container
      ->get('entity_type.manager')
      ->getStorage('media')
      ->loadUnchanged($media_id);
    $this
      ->assertFalse($media
      ->isPublished());

    // Tests media delete form.
    $this
      ->drupalGet('media/' . $media_id . '/edit');
    $page
      ->clickLink('Delete');
    $assert_session
      ->pageTextContains('This action cannot be undone');
    $page
      ->pressButton('Delete');
    $media_id = \Drupal::entityQuery('media')
      ->accessCheck(FALSE)
      ->execute();
    $this
      ->assertEmpty($media_id);
  }

  /**
   * Tests the "media/add" page.
   *
   * Tests if the "media/add" page gives you a selecting option if there are
   * multiple media types available.
   */
  public function testMediaWithMultipleMediaTypes() {
    $assert_session = $this
      ->assertSession();

    // Tests and creates the first media type.
    $first_media_type = $this
      ->createMediaType('test', [
      'description' => $this
        ->randomMachineName(),
    ]);

    // Test and create a second media type.
    $second_media_type = $this
      ->createMediaType('test', [
      'description' => $this
        ->randomMachineName(),
    ]);

    // Test if media/add displays two media type options.
    $this
      ->drupalGet('media/add');

    // Checks for the first media type.
    $assert_session
      ->pageTextContains($first_media_type
      ->label());
    $assert_session
      ->pageTextContains($first_media_type
      ->getDescription());

    // Checks for the second media type.
    $assert_session
      ->pageTextContains($second_media_type
      ->label());
    $assert_session
      ->pageTextContains($second_media_type
      ->getDescription());
  }

  /**
   * Tests that media in ER fields use the Rendered Entity formatter by default.
   */
  public function testRenderedEntityReferencedMedia() {
    $page = $this
      ->getSession()
      ->getPage();
    $assert_session = $this
      ->assertSession();
    $this
      ->drupalCreateContentType([
      'type' => 'page',
      'name' => 'Page',
    ]);
    $this
      ->drupalGet('/admin/structure/types/manage/page/fields/add-field');
    $page
      ->selectFieldOption('new_storage_type', 'field_ui:entity_reference:media');
    $page
      ->fillField('label', 'Foo field');
    $page
      ->fillField('field_name', 'foo_field');
    $page
      ->pressButton('Save and continue');
    $this
      ->drupalGet('/admin/structure/types/manage/page/display');
    $assert_session
      ->fieldValueEquals('fields[field_foo_field][type]', 'entity_reference_entity_view');
  }

  /**
   * Data provider for testMediaReferenceWidget().
   *
   * @return array[]
   *   Test data. See testMediaReferenceWidget() for the child array structure.
   */
  public function providerTestMediaReferenceWidget() {
    return [
      // Single-value fields with a single media type and the default widget:
      // - The user can create and list the media.
      'single_value:single_type:create_list' => [
        1,
        [
          TRUE,
        ],
        TRUE,
      ],
      // - The user can list but not create the media.
      'single_value:single_type:list' => [
        1,
        [
          FALSE,
        ],
        TRUE,
      ],
      // - The user can create but not list the media.
      'single_value:single_type:create' => [
        1,
        [
          TRUE,
        ],
        FALSE,
      ],
      // - The user can neither create nor list the media.
      'single_value:single_type' => [
        1,
        [
          FALSE,
        ],
        FALSE,
      ],
      // Single-value fields with the tags-style widget:
      // - The user can create and list the media.
      'single_value:single_type:create_list:tags' => [
        1,
        [
          TRUE,
        ],
        TRUE,
        'entity_reference_autocomplete_tags',
      ],
      // - The user can list but not create the media.
      'single_value:single_type:list:tags' => [
        1,
        [
          FALSE,
        ],
        TRUE,
        'entity_reference_autocomplete_tags',
      ],
      // - The user can create but not list the media.
      'single_value:single_type:create:tags' => [
        1,
        [
          TRUE,
        ],
        FALSE,
        'entity_reference_autocomplete_tags',
      ],
      // - The user can neither create nor list the media.
      'single_value:single_type:tags' => [
        1,
        [
          FALSE,
        ],
        FALSE,
        'entity_reference_autocomplete_tags',
      ],
      // Single-value fields with two media types:
      // - The user can create both types.
      'single_value:two_type:create2_list' => [
        1,
        [
          TRUE,
          TRUE,
        ],
        TRUE,
      ],
      // - The user can create only one type.
      'single_value:two_type:create1_list' => [
        1,
        [
          TRUE,
          FALSE,
        ],
        TRUE,
      ],
      // - The user cannot create either type.
      'single_value:two_type:list' => [
        1,
        [
          FALSE,
          FALSE,
        ],
        TRUE,
      ],
      // Multiple-value field with a cardinality of 3, with media the user can
      // create and list.
      'multi_value:single_type:create_list' => [
        3,
        [
          TRUE,
        ],
        TRUE,
      ],
      // The same, with the tags field.
      'multi-value:single_type:create_list:tags' => [
        3,
        [
          TRUE,
        ],
        TRUE,
        'entity_reference_autocomplete_tags',
      ],
      // Unlimited value field.
      'unlimited_value:single_type:create_list' => [
        FieldStorageDefinitionInterface::CARDINALITY_UNLIMITED,
        [
          TRUE,
        ],
        TRUE,
      ],
      // Unlimited value field with the tags widget.
      'unlimited_value:single_type:create_list:tags' => [
        FieldStorageDefinitionInterface::CARDINALITY_UNLIMITED,
        [
          TRUE,
        ],
        TRUE,
        'entity_reference_autocomplete_tags',
      ],
    ];
  }

  /**
   * Tests the default autocomplete widgets for media reference fields.
   *
   * @param int $cardinality
   *   The field cardinality.
   * @param bool[] $media_type_create_access
   *   An array of booleans indicating whether to grant the test user create
   *   access for each media type. A media type is created automatically for
   *   each; for example, an array [TRUE, FALSE] would create two media types,
   *   one that allows the user to create media and a second that does not.
   * @param bool $list_access
   *   Whether to grant the test user access to list media.
   * @param string $widget_id
   *   The widget ID to test.
   *
   * @see media_field_widget_entity_reference_autocomplete_form_alter()
   * @see media_field_widget_multiple_entity_reference_autocomplete_form_alter()
   *
   * @dataProvider providerTestMediaReferenceWidget
   */
  public function testMediaReferenceWidget($cardinality, array $media_type_create_access, $list_access, $widget_id = 'entity_reference_autocomplete') {
    $assert_session = $this
      ->assertSession();

    // Create two content types.
    $non_media_content_type = $this
      ->createContentType();
    $content_type = $this
      ->createContentType();

    // Create some media types.
    $media_types = [];
    $permissions = [];
    $create_media_types = [];
    foreach ($media_type_create_access as $id => $access) {
      if ($access) {
        $create_media_types[] = "media_type_{$id}";
        $permissions[] = "create media_type_{$id} media";
      }
      $this
        ->createMediaType('test', [
        'id' => "media_type_{$id}",
        'label' => "media_type_{$id}",
      ]);
      $media_types["media_type_{$id}"] = "media_type_{$id}";
    }

    // Create a user that can create content of the type, with other
    // permissions as given by the data provider.
    $permissions[] = "create {$content_type->id()} content";
    if ($list_access) {
      $permissions[] = "access media overview";
    }
    $test_user = $this
      ->drupalCreateUser($permissions);

    // Create a non-media entity reference.
    $non_media_storage = FieldStorageConfig::create([
      'field_name' => 'field_not_a_media_field',
      'entity_type' => 'node',
      'type' => 'entity_reference',
      'cardinality' => FieldStorageDefinitionInterface::CARDINALITY_UNLIMITED,
      'settings' => [
        'target_type' => 'node',
      ],
    ]);
    $non_media_storage
      ->save();
    $non_media_field = FieldConfig::create([
      'label' => 'No media here!',
      'field_storage' => $non_media_storage,
      'entity_type' => 'node',
      'bundle' => $non_media_content_type
        ->id(),
      'settings' => [
        'handler' => 'default',
        'handler_settings' => [
          'target_bundles' => [
            $non_media_content_type
              ->id() => $non_media_content_type
              ->id(),
          ],
        ],
      ],
    ]);
    $non_media_field
      ->save();
    \Drupal::entityTypeManager()
      ->getStorage('entity_form_display')
      ->load('node.' . $non_media_content_type
      ->id() . '.default')
      ->setComponent('field_not_a_media_field', [
      'type' => $widget_id,
    ])
      ->save();

    // Create a media field through the user interface to ensure that the
    // help text handling does not break the default value entry on the field
    // settings form.
    // Using submitForm() to avoid dealing with JavaScript on the previous
    // page in the field creation.
    $edit = [
      'new_storage_type' => 'field_ui:entity_reference:media',
      'label' => "Media (cardinality {$cardinality})",
      'field_name' => 'media_reference',
    ];
    $this
      ->drupalGet("admin/structure/types/manage/{$content_type->id()}/fields/add-field");
    $this
      ->submitForm($edit, 'Save and continue');
    $edit = [];
    foreach ($media_types as $type) {
      $edit["settings[handler_settings][target_bundles][{$type}]"] = TRUE;
    }
    $this
      ->drupalGet("admin/structure/types/manage/{$content_type->id()}/fields/node.{$content_type->id()}.field_media_reference");
    $this
      ->submitForm($edit, "Save settings");
    \Drupal::entityTypeManager()
      ->getStorage('entity_form_display')
      ->load('node.' . $content_type
      ->id() . '.default')
      ->setComponent('field_media_reference', [
      'type' => $widget_id,
    ])
      ->save();

    // Some of the expected texts.
    $create_help = 'Create your media on the media add page (opens a new window), then add it by name to the field below.';
    $list_text = 'See the media list (opens a new window) to help locate media.';
    $use_help = 'Type part of the media name.';
    $create_header = "Create new media";
    $use_header = "Use existing media";

    // First check that none of the help texts are on the non-media content.
    $this
      ->drupalGet("/node/add/{$non_media_content_type->id()}");
    $this
      ->assertNoHelpTexts([
      $create_header,
      $create_help,
      $use_header,
      $use_help,
      $list_text,
      'Allowed media types:',
    ]);

    // Now, check that the widget displays the expected help text under the
    // given conditions for the test user.
    $this
      ->drupalLogin($test_user);
    $this
      ->drupalGet("/node/add/{$content_type->id()}");

    // Specific expected help texts for the media field.
    $create_header = "Create new media";
    $use_header = "Use existing media";
    $type_list = 'Allowed media types: ' . implode(", ", array_keys($media_types));
    $fieldset_selector = '#edit-field-media-reference-wrapper fieldset';
    $fieldset = $assert_session
      ->elementExists('css', $fieldset_selector);
    $this
      ->assertSame("Media (cardinality {$cardinality})", $assert_session
      ->elementExists('css', 'legend', $fieldset)
      ->getText());

    // Assert text that should be displayed regardless of other access.
    $this
      ->assertHelpTexts([
      $use_header,
      $use_help,
      $type_list,
    ], $fieldset_selector);

    // The entire section for creating new media should only be displayed if
    // the user can create at least one media of the type.
    if ($create_media_types) {
      if (count($create_media_types) === 1) {
        $url = Url::fromRoute('entity.media.add_form')
          ->setRouteParameter('media_type', $create_media_types[0]);
      }
      else {
        $url = Url::fromRoute('entity.media.add_page');
      }
      $this
        ->assertHelpTexts([
        $create_header,
        $create_help,
      ], $fieldset_selector);
      $this
        ->assertHelpLink($fieldset, 'media add page', [
        'target' => '_blank',
        'href' => $url
          ->toString(),
      ]);
    }
    else {
      $this
        ->assertNoHelpTexts([
        $create_header,
        $create_help,
      ]);
      $this
        ->assertNoHelpLink($fieldset, 'media add page');
    }
    if ($list_access) {
      $this
        ->assertHelpTexts([
        $list_text,
      ], $fieldset_selector);
      $this
        ->assertHelpLink($fieldset, 'media list', [
        'target' => '_blank',
        'href' => Url::fromRoute('entity.media.collection')
          ->toString(),
      ]);
    }
    else {
      $this
        ->assertNoHelpTexts([
        $list_text,
      ]);
      $this
        ->assertNoHelpLink($fieldset, 'media list');
    }
  }

  /**
   * Tests the redirect URL after creating a media item.
   */
  public function testMediaCreateRedirect() {
    $session = $this
      ->getSession();
    $page = $session
      ->getPage();
    $assert_session = $this
      ->assertSession();
    $this
      ->createMediaType('test', [
      'queue_thumbnail_downloads' => FALSE,
    ]);

    // Test a redirect to the media canonical URL for a user without the 'access
    // media overview' permission.
    $this
      ->drupalLogin($this
      ->drupalCreateUser([
      'view media',
      'create media',
    ]));
    $this
      ->drupalGet('media/add');
    $page
      ->fillField('name[0][value]', $this
      ->randomMachineName());
    $page
      ->fillField('field_media_test[0][value]', $this
      ->randomString());
    $page
      ->pressButton('Save');
    $media_id = $this->container
      ->get('entity_type.manager')
      ->getStorage('media')
      ->getQuery()
      ->accessCheck(FALSE)
      ->execute();
    $media_id = reset($media_id);
    $assert_session
      ->addressEquals("media/{$media_id}/edit");

    // Test a redirect to the media overview for a user with the 'access media
    // overview' permission.
    $this
      ->drupalLogin($this
      ->drupalCreateUser([
      'view media',
      'create media',
      'access media overview',
    ]));
    $this
      ->drupalGet('media/add');
    $page
      ->fillField('name[0][value]', $this
      ->randomMachineName());
    $page
      ->fillField('field_media_test[0][value]', $this
      ->randomString());
    $page
      ->pressButton('Save');
    $assert_session
      ->addressEquals('admin/content/media');
  }

  /**
   * Asserts that the given texts are present exactly once.
   *
   * @param string[] $texts
   *   A list of the help texts to check.
   * @param string $selector
   *   (optional) The selector to search.
   */
  public function assertHelpTexts(array $texts, $selector = '') {
    $assert_session = $this
      ->assertSession();
    foreach ($texts as $text) {

      // We only want to escape single quotes, so use str_replace() rather than
      // addslashes().
      $text = str_replace("'", "\\'", $text);
      if ($selector) {
        $assert_session
          ->elementsCount('css', $selector . ":contains('{$text}')", 1);
      }
      else {
        $assert_session
          ->pageTextContains($text);
      }
    }
  }

  /**
   * Asserts that none of the given texts are present.
   *
   * @param string[] $texts
   *   A list of the help texts to check.
   */
  public function assertNoHelpTexts(array $texts) {
    $assert_session = $this
      ->assertSession();
    foreach ($texts as $text) {
      $assert_session
        ->pageTextNotContains($text);
    }
  }

  /**
   * Asserts whether a given link is present.
   *
   * @param \Behat\Mink\Element\NodeElement $element
   *   The element to search.
   * @param string $text
   *   The link text.
   * @param string[] $attributes
   *   An associative array of any expected attributes, keyed by the
   *   attribute name.
   */
  protected function assertHelpLink(NodeElement $element, $text, array $attributes = []) {

    // Find all the links inside the element.
    $link = $element
      ->findLink($text);
    $this
      ->assertNotEmpty($link);
    foreach ($attributes as $attribute => $value) {
      $this
        ->assertSame($link
        ->getAttribute($attribute), $value);
    }
  }

  /**
   * Asserts that a given link is not present.
   *
   * @param \Behat\Mink\Element\NodeElement $element
   *   The element to search.
   * @param string $text
   *   The link text.
   */
  protected function assertNoHelpLink(NodeElement $element, $text) {
    $assert_session = $this
      ->assertSession();

    // Assert that the link and its text are not present anywhere on the page.
    $assert_session
      ->elementNotExists('named', [
      'link',
      $text,
    ], $element);
    $assert_session
      ->pageTextNotContains($text);
  }

  /**
   * Tests the media collection route.
   */
  public function testMediaCollectionRoute() {

    /** @var \Drupal\Core\Entity\EntityStorageInterface $media_storage */
    $media_storage = $this->container
      ->get('entity_type.manager')
      ->getStorage('media');
    $this->container
      ->get('module_installer')
      ->uninstall([
      'views',
    ]);

    // Create a media type and media item.
    $media_type = $this
      ->createMediaType('test');
    $media = $media_storage
      ->create([
      'bundle' => $media_type
        ->id(),
      'name' => 'Unnamed',
    ]);
    $media
      ->save();
    $this
      ->drupalGet($media
      ->toUrl('collection'));
    $assert_session = $this
      ->assertSession();

    // Media list table exists.
    $assert_session
      ->elementExists('css', 'th:contains("Media Name")');
    $assert_session
      ->elementExists('css', 'th:contains("Type")');
    $assert_session
      ->elementExists('css', 'th:contains("Operations")');

    // Media item is present.
    $assert_session
      ->elementExists('css', 'td:contains("Unnamed")');
  }

}

Classes

Namesort descending Description
MediaUiFunctionalTest Ensures that media UI works correctly.