From 695846b55a61d712e2ff414504afcdea98172b6f Mon Sep 17 00:00:00 2001
From: Lee Rowlands <lee.rowlands@previousnext.com.au>
Date: Tue, 23 Jun 2020 16:39:35 +1000
Subject: [PATCH] Issue #3044292 by johnwebdev, Sam152, Berdir: Inline content
 blocks created by layout builder should not directly be impacted by content
 moderation

---
 .../Handler/BlockContentModerationHandler.php | 11 ++++
 .../src/Entity/Handler/ModerationHandler.php  |  8 +++
 .../Handler/ModerationHandlerInterface.php    | 23 ++++++++
 .../src/ModerationInformation.php             |  6 +-
 ...uilderContentModerationIntegrationTest.php | 56 ++++++++++++++++++-
 .../src/Unit/ModerationInformationTest.php    |  3 +
 6 files changed, 104 insertions(+), 3 deletions(-)

diff --git a/core/modules/content_moderation/src/Entity/Handler/BlockContentModerationHandler.php b/core/modules/content_moderation/src/Entity/Handler/BlockContentModerationHandler.php
index a618b2246e8b..2b9a8b2ba99f 100644
--- a/core/modules/content_moderation/src/Entity/Handler/BlockContentModerationHandler.php
+++ b/core/modules/content_moderation/src/Entity/Handler/BlockContentModerationHandler.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\content_moderation\Entity\Handler;
 
+use Drupal\Core\Entity\ContentEntityInterface;
 use Drupal\Core\Form\FormStateInterface;
 
 /**
@@ -29,4 +30,14 @@ public function enforceRevisionsBundleFormAlter(array &$form, FormStateInterface
     $form['revision']['#description'] = $this->t('Revisions must be required when moderation is enabled.');
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function isModeratedEntity(ContentEntityInterface $entity) {
+    // Only reusable blocks can be moderated individually. Non-reusable or
+    // inline blocks are moderated as part of the entity they are a composite
+    // of.
+    return $entity->isReusable();
+  }
+
 }
diff --git a/core/modules/content_moderation/src/Entity/Handler/ModerationHandler.php b/core/modules/content_moderation/src/Entity/Handler/ModerationHandler.php
index 243208a000f8..446f073312b1 100644
--- a/core/modules/content_moderation/src/Entity/Handler/ModerationHandler.php
+++ b/core/modules/content_moderation/src/Entity/Handler/ModerationHandler.php
@@ -28,6 +28,14 @@ public static function createInstance(ContainerInterface $container, EntityTypeI
     return new static();
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function isModeratedEntity(ContentEntityInterface $entity) {
+    // Moderate all entities included in the moderation workflow by default.
+    return TRUE;
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/modules/content_moderation/src/Entity/Handler/ModerationHandlerInterface.php b/core/modules/content_moderation/src/Entity/Handler/ModerationHandlerInterface.php
index 7e6f109fbd12..f585517ccaeb 100644
--- a/core/modules/content_moderation/src/Entity/Handler/ModerationHandlerInterface.php
+++ b/core/modules/content_moderation/src/Entity/Handler/ModerationHandlerInterface.php
@@ -16,6 +16,29 @@
  */
 interface ModerationHandlerInterface {
 
+  /**
+   * Determines if an entity should be moderated.
+   *
+   * At the workflow level, moderation is enabled or disabled for entire entity
+   * types or bundles. After a bundle has been enabled, there maybe be further
+   * decisions each entity type may make to evaluate if a given entity is
+   * appropriate to be included in a moderation workflow. The handler is only
+   * consulted after the user has configured the associated entity type and
+   * bundle to be included in a moderation workflow.
+   *
+   * Returning FALSE will remove the moderation state field widget from the
+   * associated entity form and opt out of all moderation related entity
+   * semantics, such as creating new revisions and changing the publishing
+   * status of a revision.
+   *
+   * @param \Drupal\Core\Entity\ContentEntityInterface $entity
+   *   The entity we may be moderating.
+   *
+   * @return bool
+   *   TRUE if this entity should be moderated, FALSE otherwise.
+   */
+  public function isModeratedEntity(ContentEntityInterface $entity);
+
   /**
    * Operates on moderated content entities preSave().
    *
diff --git a/core/modules/content_moderation/src/ModerationInformation.php b/core/modules/content_moderation/src/ModerationInformation.php
index c5510aa05ce7..a41b154f279d 100644
--- a/core/modules/content_moderation/src/ModerationInformation.php
+++ b/core/modules/content_moderation/src/ModerationInformation.php
@@ -52,8 +52,10 @@ public function isModeratedEntity(EntityInterface $entity) {
     if (!$entity instanceof ContentEntityInterface) {
       return FALSE;
     }
-
-    return $this->shouldModerateEntitiesOfBundle($entity->getEntityType(), $entity->bundle());
+    if (!$this->shouldModerateEntitiesOfBundle($entity->getEntityType(), $entity->bundle())) {
+      return FALSE;
+    }
+    return $this->entityTypeManager->getHandler($entity->getEntityTypeId(), 'moderation')->isModeratedEntity($entity);
   }
 
   /**
diff --git a/core/modules/content_moderation/tests/src/Functional/LayoutBuilderContentModerationIntegrationTest.php b/core/modules/content_moderation/tests/src/Functional/LayoutBuilderContentModerationIntegrationTest.php
index 2f3d3e2bb7e8..ebc3fca91891 100644
--- a/core/modules/content_moderation/tests/src/Functional/LayoutBuilderContentModerationIntegrationTest.php
+++ b/core/modules/content_moderation/tests/src/Functional/LayoutBuilderContentModerationIntegrationTest.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\Tests\content_moderation\Functional;
 
+use Drupal\block_content\Entity\BlockContentType;
 use Drupal\layout_builder\Entity\LayoutBuilderEntityViewDisplay;
 use Drupal\Tests\BrowserTestBase;
 use Drupal\Tests\content_moderation\Traits\ContentModerationTestTrait;
@@ -24,6 +25,7 @@ class LayoutBuilderContentModerationIntegrationTest extends BrowserTestBase {
     'node',
     'content_moderation',
     'menu_ui',
+    'block_content',
   ];
 
   /**
@@ -41,10 +43,21 @@ protected function setUp(): void {
     //   https://www.drupal.org/project/drupal/issues/2917777.
     $this->drupalPlaceBlock('local_tasks_block');
 
+    $workflow = $this->createEditorialWorkflow();
+
     // Add a new bundle and add an editorial workflow.
     $this->createContentType(['type' => 'bundle_with_section_field']);
-    $workflow = $this->createEditorialWorkflow();
     $workflow->getTypePlugin()->addEntityTypeAndBundle('node', 'bundle_with_section_field');
+
+    // Add a new block content bundle to the editorial workflow.
+    BlockContentType::create([
+      'id' => 'basic',
+      'label' => 'Basic',
+      'revision' => 1,
+    ])->save();
+    block_content_add_body_field('basic');
+
+    $workflow->getTypePlugin()->addEntityTypeAndBundle('block_content', 'basic');
     $workflow->save();
 
     // Enable layout overrides.
@@ -62,6 +75,7 @@ protected function setUp(): void {
       'view latest version',
       'use editorial transition create_new_draft',
       'use editorial transition publish',
+      'create and edit custom blocks',
     ]));
   }
 
@@ -138,4 +152,44 @@ public function testLayoutModeration() {
     $assert_session->pageTextNotContains('Powered by Drupal');
   }
 
+  /**
+   * Test placing inline blocks that belong to a moderated custom block bundle.
+   */
+  public function testModeratedInlineBlockBundles() {
+    $page = $this->getSession()->getPage();
+    $assert_session = $this->assertSession();
+
+    $node = $this->createNode([
+      'type' => 'bundle_with_section_field',
+      'title' => 'The first node title',
+      'moderation_state' => 'published',
+    ]);
+    $this->drupalGet("node/{$node->id()}/layout");
+    $page->clickLink('Add block');
+    $this->clickLink('Create custom block');
+
+    $assert_session->fieldNotExists('settings[block_form][moderation_state][0][state]');
+    $this->submitForm([
+      'settings[label]' => 'Test inline block',
+      'settings[block_form][body][0][value]' => 'Example block body',
+    ], 'Add block');
+
+    // Save a draft of the page with the inline block and ensure the drafted
+    // content appears on the latest version page.
+    $this->assertSession()->pageTextContains('Example block body');
+    $this->submitForm([
+      'moderation_state[0][state]' => 'draft',
+    ], 'Save layout');
+    $assert_session->pageTextContains('The layout override has been saved.');
+    $assert_session->pageTextContains('Example block body');
+
+    // Publish the draft of the page ensure the draft inline block content
+    // apears on the published page.
+    $this->submitForm([
+      'new_state' => 'published',
+    ], 'Apply');
+    $assert_session->pageTextContains('The moderation state has been updated.');
+    $assert_session->pageTextContains('Example block body');
+  }
+
 }
diff --git a/core/modules/content_moderation/tests/src/Unit/ModerationInformationTest.php b/core/modules/content_moderation/tests/src/Unit/ModerationInformationTest.php
index 329a291e40ab..eb88edd31a1d 100644
--- a/core/modules/content_moderation/tests/src/Unit/ModerationInformationTest.php
+++ b/core/modules/content_moderation/tests/src/Unit/ModerationInformationTest.php
@@ -13,6 +13,7 @@
 use Drupal\content_moderation\ModerationInformation;
 use Drupal\Tests\UnitTestCase;
 use Drupal\workflows\WorkflowInterface;
+use Prophecy\Argument;
 
 /**
  * @coversDefaultClass \Drupal\content_moderation\ModerationInformation
@@ -38,6 +39,7 @@ protected function getUser() {
    */
   protected function getEntityTypeManager() {
     $entity_type_manager = $this->prophesize(EntityTypeManagerInterface::class);
+    $entity_type_manager->getHandler(Argument::any(), 'moderation')->willReturn(new ModerationHandler());
     return $entity_type_manager->reveal();
   }
 
@@ -94,6 +96,7 @@ public function testIsModeratedEntity($workflow, $expected) {
     ]);
     $entity = $this->prophesize(ContentEntityInterface::class);
     $entity->getEntityType()->willReturn($entity_type);
+    $entity->getEntityTypeId()->willReturn($entity_type->id());
     $entity->bundle()->willReturn('test_bundle');
 
     $this->assertEquals($expected, $moderation_information->isModeratedEntity($entity->reveal()));
-- 
GitLab