StyleTableTest.php

Same filename in this branch
  1. 10 core/modules/views_ui/tests/src/Functional/StyleTableTest.php
Same filename in other branches
  1. 9 core/modules/views/tests/src/Functional/Plugin/StyleTableTest.php
  2. 9 core/modules/views_ui/tests/src/Functional/StyleTableTest.php
  3. 8.9.x core/modules/views/tests/src/Functional/Plugin/StyleTableTest.php
  4. 8.9.x core/modules/views_ui/tests/src/Functional/StyleTableTest.php
  5. 11.x core/modules/views/tests/src/Functional/Plugin/StyleTableTest.php
  6. 11.x core/modules/views_ui/tests/src/Functional/StyleTableTest.php

Namespace

Drupal\Tests\views\Functional\Plugin

File

core/modules/views/tests/src/Functional/Plugin/StyleTableTest.php

View source
<?php

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

use Drupal\Core\Database\Database;
use Drupal\dynamic_page_cache\EventSubscriber\DynamicPageCacheSubscriber;
use Drupal\Tests\views\Functional\ViewTestBase;
use Drupal\views\Entity\View;

/**
 * Tests the table style views plugin.
 *
 * @group views
 */
class StyleTableTest extends ViewTestBase {
    
    /**
     * Views used by this test.
     *
     * @var array
     */
    public static $testViews = [
        'test_table',
    ];
    
    /**
     * {@inheritdoc}
     */
    protected $defaultTheme = 'stark';
    
    /**
     * {@inheritdoc}
     */
    protected function setUp($import_test_views = TRUE, $modules = [
        'views_test_config',
    ]) : void {
        parent::setUp($import_test_views, $modules);
        $this->enableViewsTestModule();
    }
    
    /**
     * Tests table caption/summary/description.
     */
    public function testAccessibilitySettings() : void {
        $this->drupalGet('test-table');
        $this->assertSession()
            ->elementExists('xpath', '//caption/child::text()');
        $this->assertSession()
            ->elementTextEquals('xpath', '//caption/child::text()', 'caption-text');
        $this->assertSession()
            ->elementExists('xpath', '//summary/child::text()');
        $this->assertSession()
            ->elementTextEquals('xpath', '//summary/child::text()', 'summary-text');
        // Check that the summary has the right accessibility settings.
        $this->assertSession()
            ->elementAttributeExists('xpath', '//summary', 'role');
        $this->assertSession()
            ->elementAttributeExists('xpath', '//summary', 'aria-expanded');
        $this->assertSession()
            ->elementExists('xpath', '//caption/details/child::text()[normalize-space()]');
        $this->assertSession()
            ->elementTextEquals('xpath', '//caption/details/child::text()[normalize-space()]', 'description-text');
        // Remove the caption and ensure the caption is not displayed anymore.
        $view = View::load('test_table');
        $display =& $view->getDisplay('default');
        $display['display_options']['style']['options']['caption'] = '';
        $view->save();
        $this->drupalGet('test-table');
        $this->assertSession()
            ->elementTextEquals('xpath', '//caption/child::text()', '');
        // Remove the table summary.
        $display =& $view->getDisplay('default');
        $display['display_options']['style']['options']['summary'] = '';
        $view->save();
        $this->drupalGet('test-table');
        $this->assertSession()
            ->elementNotExists('xpath', '//summary/child::text()');
        // Remove the table description.
        $display =& $view->getDisplay('default');
        $display['display_options']['style']['options']['description'] = '';
        $view->save();
        $this->drupalGet('test-table');
        $this->assertSession()
            ->elementNotExists('xpath', '//caption/details/child::text()[normalize-space()]');
    }
    
    /**
     * Tests table fields in columns.
     */
    public function testFieldInColumns() : void {
        $this->drupalGet('test-table');
        // Ensure that both columns are in separate tds.
        // Check for class " views-field-job ", because just "views-field-job" won't
        // do: "views-field-job-1" would also contain "views-field-job".
        // @see Drupal\system\Tests\Form\ElementTest::testButtonClasses().
        $this->assertSession()
            ->elementExists('xpath', '//tbody/tr/td[contains(concat(" ", @class, " "), " views-field-job ")]');
        $this->assertSession()
            ->elementExists('xpath', '//tbody/tr/td[contains(concat(" ", @class, " "), " views-field-job-1 ")]');
        // Combine the second job-column with the first one, with ', ' as separator.
        $view = View::load('test_table');
        $display =& $view->getDisplay('default');
        $display['display_options']['style']['options']['columns']['job_1'] = 'job';
        $display['display_options']['style']['options']['info']['job']['separator'] = ', ';
        $view->save();
        // Ensure that both columns are properly combined.
        $this->drupalGet('test-table');
        $this->assertSession()
            ->elementExists('xpath', '//tbody/tr/td[contains(concat(" ", @class, " "), " views-field-job views-field-job-1 ")]');
        $this->assertSession()
            ->elementExists('xpath', '//tbody/tr/td[contains(., "Drummer, Drummer")]');
    }
    
    /**
     * Tests that a number with the value of "0" is displayed in the table.
     */
    public function testNumericFieldVisible() : void {
        // Adds a new data point in the views_test_data table to have a person with
        // an age of zero.
        $data_set = $this->dataSet();
        $query = Database::getConnection()->insert('views_test_data')
            ->fields(array_keys($data_set[0]));
        $query->values([
            'name' => 'James McCartney',
            'age' => 0,
            'job' => 'Baby',
            'created' => gmmktime(6, 30, 10, 1, 1, 2000),
            'status' => 1,
        ]);
        $query->execute();
        $this->drupalGet('test-table');
        $this->assertSession()
            ->elementExists('xpath', '//tbody/tr/td[contains(., "Baby")]');
        $this->assertSession()
            ->elementExists('xpath', '//tbody/tr/td[text()=0]');
    }
    
    /**
     * Tests that empty columns are hidden when empty_column is set.
     */
    public function testEmptyColumn() : void {
        // Empty the 'job' data.
        \Drupal::database()->update('views_test_data')
            ->fields([
            'job' => '',
        ])
            ->execute();
        $this->drupalGet('test-table');
        // Test that only one of the job columns still shows.
        // Ensure that empty column header is hidden.
        $this->assertSession()
            ->elementsCount('xpath', '//thead/tr/th/a[text()="Job"]', 1);
        $this->assertSession()
            ->elementNotExists('xpath', '//tbody/tr/td[contains(concat(" ", @class, " "), " views-field-job-1 ")]');
    }
    
    /**
     * Tests grouping by a field.
     */
    public function testGrouping() : void {
        
        /** @var \Drupal\views\ViewEntityInterface $view */
        $view = \Drupal::entityTypeManager()->getStorage('view')
            ->load('test_table');
        // Get a reference to the display configuration so we can alter some
        // specific style options.
        $display =& $view->getDisplay('default');
        // Set job as the grouping field.
        $display['display_options']['style']['options']['grouping'][0] = [
            'field' => 'job',
            'rendered' => TRUE,
            'rendered_strip' => FALSE,
        ];
        // Clear the caption text, the rendered job field will be used as a caption.
        $display['display_options']['style']['options']['caption'] = '';
        $display['display_options']['style']['options']['summary'] = '';
        $display['display_options']['style']['options']['description'] = '';
        $view->save();
        // Add a record containing unsafe markup to be sure it's filtered out.
        $unsafe_markup = '<script>alert("Rapper");</script>';
        $unsafe_markup_data = [
            'name' => 'Marshall',
            'age' => 42,
            'job' => $unsafe_markup,
            'created' => gmmktime(0, 0, 0, 2, 15, 2001),
            'status' => 1,
        ];
        $database = $this->container
            ->get('database');
        $database->insert('views_test_data')
            ->fields(array_keys($unsafe_markup_data))
            ->values($unsafe_markup_data)
            ->execute();
        $this->drupalGet('test-table');
        $expected_captions = [
            'Job: Speaker',
            'Job: Songwriter',
            'Job: Drummer',
            'Job: Singer',
            'Job: ' . $unsafe_markup,
        ];
        // Ensure that we don't find the caption containing unsafe markup.
        $this->assertSession()
            ->responseNotContains($unsafe_markup);
        // Ensure that the summary isn't shown.
        $this->assertSession()
            ->elementNotExists('xpath', '//caption/details');
        // Ensure that all expected captions are found.
        foreach ($expected_captions as $raw_caption) {
            $this->assertSession()
                ->assertEscaped($raw_caption);
        }
        $display =& $view->getDisplay('default');
        // Remove the label from the grouping field.
        $display['display_options']['fields']['job']['label'] = '';
        $view->save();
        $this->drupalGet('test-table');
        $expected_captions = [
            'Speaker',
            'Songwriter',
            'Drummer',
            'Singer',
            $unsafe_markup,
        ];
        // Ensure that we don't find the caption containing unsafe markup.
        $this->assertSession()
            ->responseNotContains($unsafe_markup);
        // Ensure that all expected captions are found.
        foreach ($expected_captions as $raw_caption) {
            $this->assertSession()
                ->assertEscaped($raw_caption);
        }
    }
    
    /**
     * Tests responsive classes and column assigning.
     */
    public function testResponsiveMergedColumns() : void {
        
        /** @var \Drupal\views\ViewEntityInterface $view */
        $view = \Drupal::entityTypeManager()->getStorage('view')
            ->load('test_table');
        // Merge the two job columns together and set the responsive priority on
        // the column that is merged to.
        $display =& $view->getDisplay('default');
        $display['display_options']['style']['options']['columns']['job'] = 'job_1';
        $display['display_options']['style']['options']['info']['job_1']['separator'] = ', ';
        $display['display_options']['style']['options']['info']['job_1']['responsive'] = 'priority-low';
        $view->save();
        // Ensure that both columns are properly combined.
        $this->drupalGet('test-table');
        $this->assertSession()
            ->elementExists('xpath', '//tbody/tr/td[contains(concat(" ", @class, " "), " priority-low views-field views-field-job views-field-job-1 ")]');
        $this->assertSession()
            ->elementExists('xpath', '//tbody/tr/td[contains(., "Drummer, Drummer")]');
    }
    
    /**
     * Tests the cacheability of the table display.
     */
    public function testTableCacheability() : void {
        \Drupal::service('module_installer')->uninstall([
            'page_cache',
        ]);
        $url = 'test-table';
        $this->drupalGet($url);
        $this->assertSession()
            ->statusCodeEquals(200);
        $this->assertSession()
            ->responseHeaderEquals(DynamicPageCacheSubscriber::HEADER, 'MISS');
        $this->drupalGet($url);
        $this->assertSession()
            ->responseHeaderEquals(DynamicPageCacheSubscriber::HEADER, 'HIT');
    }

}

Classes

Title Deprecated Summary
StyleTableTest Tests the table style views plugin.

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