book-2.0.x-dev/tests/src/Functional/BookTestTrait.php
tests/src/Functional/BookTestTrait.php
<?php
declare(strict_types=1);
namespace Drupal\Tests\book\Functional;
use Behat\Mink\Exception\ExpectationException;
use Drupal\Component\Plugin\Exception\InvalidPluginDefinitionException;
use Drupal\Component\Plugin\Exception\PluginNotFoundException;
use Drupal\Core\Entity\EntityInterface;
use Drupal\Core\Entity\EntityMalformedException;
use Drupal\Core\Session\AccountInterface;
use Drupal\Core\Url;
use Drupal\node\NodeInterface;
/**
* Provides common functionality for Book test classes.
*/
trait BookTestTrait {
/**
* A book node.
*
* @var \Drupal\node\NodeInterface
*/
protected NodeInterface $book;
/**
* A user with permission to create and edit books.
*
* @var \Drupal\Core\Session\AccountInterface
*/
protected AccountInterface $bookAuthor;
/**
* Creates a new book with a page hierarchy.
*
* @param array $edit
* (optional) Field data in an associative array. Changes the current input
* fields (where possible) to the values indicated. Defaults to an empty
* array.
*
* @return \Drupal\node\NodeInterface[]
* Array of nodes.
*/
public function createBook(array $edit = []): array {
// Create new book.
$this->drupalLogin($this->bookAuthor);
$this->book = $this->createBookNode('new', NULL, $edit);
$book = $this->book;
/*
* Add page hierarchy to book.
* Book
* |- Node 0
* |- Node 1
* |- Node 2
* |- Node 3
* |- Node 4
*/
$nodes = [];
// Node 0.
$nodes[] = $this->createBookNode($book->id(), NULL, $edit);
// Node 1.
$nodes[] = $this->createBookNode($book->id(), $nodes[0]->book['nid'], $edit);
// Node 2.
$nodes[] = $this->createBookNode($book->id(), $nodes[0]->book['nid'], $edit);
// Node 3.
$nodes[] = $this->createBookNode($book->id(), NULL, $edit);
// Node 4.
$nodes[] = $this->createBookNode($book->id(), NULL, $edit);
$this->drupalLogout();
return $nodes;
}
/**
* Checks the outline of sub-pages; previous, up, and next.
*
* Also checks the printer friendly version of the outline.
*
* @param \Drupal\Core\Entity\EntityInterface $node
* Node to check.
* @param array|null $nodes
* Nodes that should be in outline.
* @param \Drupal\Core\Entity\EntityInterface|bool $previous
* Previous link node.
* @param \Drupal\Core\Entity\EntityInterface|bool $up
* Up link node.
* @param \Drupal\Core\Entity\EntityInterface|bool $next
* Next link node.
* @param array $breadcrumb
* The nodes that should be displayed in the breadcrumb.
*/
public function checkBookNode(EntityInterface $node, array|null $nodes, EntityInterface|bool $previous, EntityInterface|bool $up, EntityInterface|bool $next, array $breadcrumb): void {
try {
$this->drupalGet('node/' . $node->id());
// Check outline structure.
if ($nodes !== NULL) {
$book_navigation = $this->getSession()
->getPage()
->find('css', sprintf('nav[aria-labelledby="book-label-%s"] ul', $this->book->id()));
$this->assertNotNull($book_navigation);
$links = $book_navigation->findAll('css', 'a');
$this->assertCount(count($nodes), $links);
foreach ($nodes as $delta => $node) {
$link = $links[$delta];
$this->assertEquals($node->label(), $link->getText());
$this->assertEquals($node->toUrl()->toString(), $link->getAttribute('href'));
}
}
// Check previous, up, and next links.
if ($previous) {
$previous_element = $this->assertSession()
->elementExists('named_exact', [
'link',
'Go to previous page',
]);
$this->assertEquals($previous->toUrl()->toString(), $previous_element->getAttribute('href'));
}
if ($up) {
$parent_element = $this->assertSession()->elementExists('named_exact', [
'link',
'Go to parent page',
]);
$this->assertEquals($up->toUrl()->toString(), $parent_element->getAttribute('href'));
}
if ($next) {
$next_element = $this->assertSession()->elementExists('named_exact', [
'link',
'Go to next page',
]);
$this->assertEquals($next->toUrl()->toString(), $next_element->getAttribute('href'));
}
// Compute the expected breadcrumb.
$expected_breadcrumb = [];
$expected_breadcrumb[] = Url::fromRoute('<front>')->toString();
foreach ($breadcrumb as $a_node) {
$expected_breadcrumb[] = $a_node->toUrl()->toString();
}
// Fetch links in the current breadcrumb.
$links = $this->xpath('//nav[@aria-labelledby="system-breadcrumb"]/ol/li/a');
$got_breadcrumb = [];
foreach ($links as $link) {
$got_breadcrumb[] = $link->getAttribute('href');
}
// Compare expected and got breadcrumbs.
$this->assertSame($expected_breadcrumb, $got_breadcrumb, 'The breadcrumb is correctly displayed on the page.');
// Check printer friendly version.
$this->drupalGet('book/export/html/' . $node->id());
$this->assertSession()->pageTextContains($node->label());
$this->assertSession()->responseContains($node->body->processed);
}
catch (ExpectationException | EntityMalformedException $e) {
$this->fail($e->getMessage());
}
}
/**
* Creates a book node.
*
* @param int|string $book_nid
* A book node ID or set to 'new' to create a new book.
* @param int|string|null $parent
* (optional) Parent book reference ID. Defaults to NULL.
* @param array $edit
* (optional) Field data in an associative array. Changes the current input
* fields (where possible) to the values indicated. Defaults to an empty
* array.
*
* @return \Drupal\node\NodeInterface
* The created node.
*/
public function createBookNode(int|string $book_nid, mixed $parent = NULL, array $edit = []): NodeInterface {
// $number does not use drupal_static as it should not be reset
// since it uniquely identifies each call to createBookNode().
// Used to ensure that when sorted nodes stay in same order.
static $number = 0;
try {
$edit['title[0][value]'] = str_pad((string) $number, 2, '0', STR_PAD_LEFT) . ' - test node ' . $this->randomMachineName(10);
$edit['body[0][value]'] = 'test body ' . $this->randomMachineName(32) . ' ' . $this->randomMachineName(32);
$edit['book[bid]'] = $book_nid;
$this->drupalGet('node/add/book');
if ($parent !== NULL) {
$this->submitForm($edit, 'Change book (update list of parents)');
$edit['book[pid]'] = $parent;
$this->submitForm($edit, 'Save');
// Make sure the parent was flagged as having children.
$parent_node = $this->container->get('entity_type.manager')
->getStorage('node')
->loadUnchanged($parent);
$this->assertNotEmpty($parent_node->book['has_children'], 'Parent node is marked as having children');
}
else {
$this->submitForm($edit, 'Save');
}
// Check to make sure the book node was created.
$node = $this->drupalGetNodeByTitle($edit['title[0][value]']);
$this->assertNotNull(($node === FALSE ? NULL : $node), 'Book node found in database.');
$number++;
return $node;
}
catch (InvalidPluginDefinitionException | PluginNotFoundException $e) {
$this->fail($e->getMessage());
}
}
/**
* Adds a node to a book.
*
* @param int $book_nid
* The book node ID to add a node to.
* @param int $nid
* The node ID that needs to be added to the book.
* @param array $edit
* (optional) Field data in an associative array. Changes the current input
* fields (where possible) to the values indicated. Defaults to an empty
* array.
*/
public function addNodeToBook(int $book_nid, int $nid, array $edit = []): void {
$this->drupalGet('node/' . $nid . '/edit');
$this->assertSession()->statusCodeEquals(200);
$edit['book[bid]'] = $book_nid;
$this->submitForm($edit, 'Save');
}
}
