BlockContentTypeTest.php

Same filename in this branch
  1. 11.x core/modules/jsonapi/tests/src/Functional/BlockContentTypeTest.php
Same filename and directory in other branches
  1. 9 core/modules/jsonapi/tests/src/Functional/BlockContentTypeTest.php
  2. 9 core/modules/block_content/tests/src/Functional/BlockContentTypeTest.php
  3. 8.9.x core/modules/jsonapi/tests/src/Functional/BlockContentTypeTest.php
  4. 8.9.x core/modules/block_content/tests/src/Functional/BlockContentTypeTest.php
  5. 10 core/modules/jsonapi/tests/src/Functional/BlockContentTypeTest.php
  6. 10 core/modules/block_content/tests/src/Functional/BlockContentTypeTest.php

Namespace

Drupal\Tests\block_content\Functional

File

core/modules/block_content/tests/src/Functional/BlockContentTypeTest.php

View source
<?php

declare (strict_types=1);
namespace Drupal\Tests\block_content\Functional;

use Drupal\block_content\Entity\BlockContentType;
use Drupal\Component\Utility\Html;
use Drupal\Core\Url;
use Drupal\Tests\system\Functional\Menu\AssertBreadcrumbTrait;

/**
 * Ensures that block type functions work correctly.
 *
 * @group block_content
 */
class BlockContentTypeTest extends BlockContentTestBase {
  use AssertBreadcrumbTrait;
  
  /**
   * {@inheritdoc}
   */
  protected static $modules = [
    'field_ui',
  ];
  
  /**
   * {@inheritdoc}
   */
  protected $defaultTheme = 'stark';
  
  /**
   * Permissions to grant admin user.
   *
   * @var array
   */
  protected $permissions = [
    'administer block content',
    'administer blocks',
    'administer block_content fields',
    'administer block types',
    'administer block content',
    'access block library',
  ];
  
  /**
   * Whether or not to create an initial block type.
   *
   * @var bool
   */
  protected $autoCreateBasicBlockType = FALSE;
  
  /**
   * {@inheritdoc}
   */
  protected function setUp() : void {
    parent::setUp();
    $this->drupalPlaceBlock('page_title_block');
  }
  
  /**
   * Tests the order of the block content types on the add page.
   */
  public function testBlockContentAddPageOrder() : void {
    $this->createBlockContentType([
      'id' => 'bundle_1',
      'label' => 'Bundle 1',
    ]);
    $this->createBlockContentType([
      'id' => 'bundle_2',
      'label' => 'Aaa Bundle 2',
    ]);
    $this->drupalLogin($this->adminUser);
    $this->drupalGet('block/add');
    $this->assertSession()
      ->pageTextMatches('/Aaa Bundle 2(.*)Bundle 1/');
  }
  
  /**
   * Tests creating a block type programmatically and via a form.
   */
  public function testBlockContentTypeCreation() : void {
    // Log in a test user.
    $this->drupalLogin($this->adminUser);
    // Test the page with no block-types.
    $this->drupalGet('block/add');
    $this->assertSession()
      ->statusCodeEquals(200);
    $this->assertSession()
      ->pageTextContains('You have not created any block types yet');
    $this->clickLink('block type creation page');
    // Create a block type via the user interface.
    $edit = [
      'id' => 'foo',
      'label' => 'title for foo',
    ];
    $this->submitForm($edit, 'Save and manage fields');
    // Asserts that form submit redirects to the expected manage fields page.
    $this->assertSession()
      ->addressEquals('admin/structure/block-content/manage/' . $edit['id'] . '/fields');
    $block_type = BlockContentType::load('foo');
    $this->assertInstanceOf(BlockContentType::class, $block_type);
    $field_definitions = \Drupal::service('entity_field.manager')->getFieldDefinitions('block_content', 'foo');
    $this->assertTrue(isset($field_definitions['body']), 'Body field created when using the UI to create block content types.');
    // Check that the block type was created in site default language.
    $default_langcode = \Drupal::languageManager()->getDefaultLanguage()
      ->getId();
    $this->assertEquals($block_type->language()
      ->getId(), $default_langcode);
    // Create block types programmatically.
    $this->createBlockContentType([
      'id' => 'basic',
    ], TRUE);
    $field_definitions = \Drupal::service('entity_field.manager')->getFieldDefinitions('block_content', 'basic');
    $this->assertTrue(isset($field_definitions['body']), "Body field for 'basic' block type created when using the testing API to create block content types.");
    $this->createBlockContentType([
      'id' => 'other',
    ]);
    $field_definitions = \Drupal::service('entity_field.manager')->getFieldDefinitions('block_content', 'other');
    $this->assertFalse(isset($field_definitions['body']), "Body field for 'other' block type not created when using the testing API to create block content types.");
    $block_type = BlockContentType::load('other');
    $this->assertInstanceOf(BlockContentType::class, $block_type);
    $this->drupalGet('block/add/' . $block_type->id());
    $this->assertSession()
      ->statusCodeEquals(200);
  }
  
  /**
   * Tests editing a block type using the UI.
   */
  public function testBlockContentTypeEditing() : void {
    $this->drupalPlaceBlock('system_breadcrumb_block');
    // Now create an initial block-type.
    $this->createBlockContentType([
      'id' => 'basic',
    ], TRUE);
    $this->drupalLogin($this->adminUser);
    // We need two block types to prevent /block/add redirecting.
    $this->createBlockContentType([
      'id' => 'other',
    ]);
    $field_definitions = \Drupal::service('entity_field.manager')->getFieldDefinitions('block_content', 'other');
    $this->assertFalse(isset($field_definitions['body']), 'Body field was not created when using the API to create block content types.');
    // Verify that title and body fields are displayed.
    $this->drupalGet('block/add/basic');
    $this->assertSession()
      ->pageTextContains('Block description');
    $this->assertNotEmpty($this->cssSelect('#edit-body-0-value'), 'Body field was found.');
    // Change the block type name.
    $edit = [
      'label' => 'Bar',
    ];
    $this->drupalGet('admin/structure/block-content/manage/basic');
    $this->assertSession()
      ->titleEquals('Edit basic block type | Drupal');
    $this->submitForm($edit, 'Save');
    $front_page_path = Url::fromRoute('<front>')->toString();
    $this->assertBreadcrumb('admin/structure/block-content/manage/basic/fields', [
      $front_page_path => 'Home',
      'admin/structure/block-content' => 'Block types',
      'admin/structure/block-content/manage/basic' => 'Edit Bar',
    ]);
    \Drupal::service('entity_field.manager')->clearCachedFieldDefinitions();
    $this->drupalGet('block/add');
    $this->assertSession()
      ->pageTextContains('Bar');
    $this->clickLink('Bar');
    // Verify that the original machine name was used in the URL.
    $this->assertSession()
      ->addressEquals(Url::fromRoute('block_content.add_form', [
      'block_content_type' => 'basic',
    ]));
    // Remove the body field.
    $this->drupalGet('admin/structure/block-content/manage/basic/fields/block_content.basic.body/delete');
    $this->submitForm([], 'Delete');
    // Resave the settings for this type.
    $this->drupalGet('admin/structure/block-content/manage/basic');
    $this->submitForm([], 'Save');
    // Check that the body field doesn't exist.
    $this->drupalGet('block/add/basic');
    $this->assertEmpty($this->cssSelect('#edit-body-0-value'), 'Body field was not found.');
  }
  
  /**
   * Tests deleting a block type that still has content.
   */
  public function testBlockContentTypeDeletion() : void {
    // Now create an initial block-type.
    $this->createBlockContentType([
      'id' => 'basic',
    ], TRUE);
    // Create a block type programmatically.
    $type = $this->createBlockContentType([
      'id' => 'foo',
    ]);
    $this->drupalLogin($this->adminUser);
    // Add a new block of this type.
    $block = $this->createBlockContent(FALSE, 'foo');
    // Attempt to delete the block type, which should not be allowed.
    $this->drupalGet('admin/structure/block-content/manage/' . $type->id() . '/delete');
    $this->assertSession()
      ->pageTextContains($type->label() . ' is used by 1 content block on your site. You can not remove this block type until you have removed all of the ' . $type->label() . ' blocks.');
    $this->assertSession()
      ->pageTextNotContains('This action cannot be undone.');
    // Delete the block.
    $block->delete();
    // Attempt to delete the block type, which should now be allowed.
    $this->drupalGet('admin/structure/block-content/manage/' . $type->id() . '/delete');
    $this->assertSession()
      ->pageTextContains('Are you sure you want to delete the block type ' . $type->id() . '?');
    $this->assertSession()
      ->pageTextContains('This action cannot be undone.');
  }
  
  /**
   * Tests that redirects work as expected when multiple block types exist.
   */
  public function testsBlockContentAddTypes() : void {
    // Now create an initial block-type.
    $this->createBlockContentType([
      'id' => 'basic',
    ], TRUE);
    $this->drupalLogin($this->adminUser);
    // Create two block types programmatically.
    $this->createBlockContentType([
      'id' => 'foo',
    ]);
    $this->createBlockContentType([
      'id' => 'bar',
    ]);
    // Get the content block storage.
    $storage = $this->container
      ->get('entity_type.manager')
      ->getStorage('block_content');
    // Install all themes.
    $themes = [
      'olivero',
      'stark',
      'claro',
    ];
    \Drupal::service('theme_installer')->install($themes);
    $theme_settings = $this->config('system.theme');
    foreach ($themes as $default_theme) {
      // Change the default theme.
      $theme_settings->set('default', $default_theme)
        ->save();
      $this->drupalPlaceBlock('local_actions_block');
      // For each installed theme, go to its block page and test the redirects.
      foreach ($themes as $theme) {
        // Test that adding a block from the 'place blocks' form sends you to
        // the block configure form.
        $path = $theme == $default_theme ? 'admin/structure/block' : "admin/structure/block/list/{$theme}";
        $this->drupalGet($path);
        $this->clickLink('Place block');
        $this->clickLink('Add content block');
        $this->clickLink('foo');
        // Create a new block.
        $edit = [
          'info[0][value]' => $this->randomMachineName(8),
        ];
        $this->submitForm($edit, 'Save and configure');
        $blocks = $storage->loadByProperties([
          'info' => $edit['info[0][value]'],
        ]);
        if (!empty($blocks)) {
          $block = reset($blocks);
          $this->assertSession()
            ->addressEquals(Url::fromRoute('block.admin_add', [
            'plugin_id' => 'block_content:' . $block->uuid(),
            'theme' => $theme,
          ]));
          $this->submitForm([
            'region' => 'content',
          ], 'Save block');
          $this->assertSession()
            ->addressEquals(Url::fromRoute('block.admin_display_theme', [
            'theme' => $theme,
          ], [
            'query' => [
              'block-placement' => $theme . '-' . Html::getClass($edit['info[0][value]']),
            ],
          ]));
        }
        else {
          $this->fail('Could not load created block.');
        }
      }
    }
    // Test that adding a block from the 'content blocks list' doesn't send you
    // to the block configure form.
    $this->drupalGet('admin/content/block');
    $this->clickLink('Add content block');
    $this->clickLink('foo');
    $edit = [
      'info[0][value]' => $this->randomMachineName(8),
    ];
    $this->submitForm($edit, 'Save');
    $blocks = $storage->loadByProperties([
      'info' => $edit['info[0][value]'],
    ]);
    if (!empty($blocks)) {
      $this->assertSession()
        ->addressEquals(Url::fromRoute('entity.block_content.collection'));
    }
    else {
      $this->fail('Could not load created block.');
    }
  }

}

Classes

Title Deprecated Summary
BlockContentTypeTest Ensures that block type functions work correctly.

Buggy or inaccurate documentation? Please file an issue. Need support? Need help programming? Connect with the Drupal community.