From d8b8a0aa2602b12e397bb6d371049de16210ed58 Mon Sep 17 00:00:00 2001
From: xjm <xjm@65776.no-reply.drupal.org>
Date: Sat, 23 Feb 2019 15:15:16 -0600
Subject: [PATCH] Issue #2942661 by tedbow, tim.plunkett, bkosborne,
 johndevman, xjm, phenaproxima, Kristen Pol: Sections should have third-party
 settings

---
 .../config/schema/layout_builder.schema.yml   |   5 +
 .../layout_builder.post_update.php            |   7 +
 core/modules/layout_builder/src/Section.php   |  71 ++++++-
 .../fixtures/update/layout-builder-enable.php |   1 +
 .../layout_builder_defaults_test.schema.yml   |   6 +
 .../LayoutBuilderEnableUpdatePathTest.php     |   1 +
 .../src/Kernel/DefaultsSectionStorageTest.php |  11 +-
 .../tests/src/Unit/SectionTest.php            | 189 +++++++++++++++++-
 8 files changed, 280 insertions(+), 11 deletions(-)
 create mode 100644 core/modules/layout_builder/tests/modules/layout_builder_defaults_test/config/schema/layout_builder_defaults_test.schema.yml

diff --git a/core/modules/layout_builder/config/schema/layout_builder.schema.yml b/core/modules/layout_builder/config/schema/layout_builder.schema.yml
index d92cc4286e9f..f9f2ee8e9e2e 100644
--- a/core/modules/layout_builder/config/schema/layout_builder.schema.yml
+++ b/core/modules/layout_builder/config/schema/layout_builder.schema.yml
@@ -28,6 +28,11 @@ layout_builder.section:
       label: 'Components'
       sequence:
         type: layout_builder.component
+    third_party_settings:
+      type: sequence
+      label: 'Third party settings'
+      sequence:
+        type: '[%parent.%parent.%type].third_party.[%key]'
 
 layout_builder.component:
   type: mapping
diff --git a/core/modules/layout_builder/layout_builder.post_update.php b/core/modules/layout_builder/layout_builder.post_update.php
index 8b69590d54bc..ff4ef098bd73 100644
--- a/core/modules/layout_builder/layout_builder.post_update.php
+++ b/core/modules/layout_builder/layout_builder.post_update.php
@@ -146,3 +146,10 @@ function layout_builder_post_update_fix_tempstore_keys() {
     }
   }
 }
+
+/**
+ * Clear caches due to config schema additions.
+ */
+function layout_builder_post_update_section_third_party_settings_schema() {
+  // Empty post-update hook.
+}
diff --git a/core/modules/layout_builder/src/Section.php b/core/modules/layout_builder/src/Section.php
index 1ec0b8af10ae..26f597b3fd5c 100644
--- a/core/modules/layout_builder/src/Section.php
+++ b/core/modules/layout_builder/src/Section.php
@@ -2,6 +2,8 @@
 
 namespace Drupal\layout_builder;
 
+use Drupal\Core\Config\Entity\ThirdPartySettingsInterface;
+
 /**
  * Provides a domain object for layout sections.
  *
@@ -22,7 +24,7 @@
  * @todo Determine whether an interface will be provided for this in
  *   https://www.drupal.org/project/drupal/issues/2930334.
  */
-class Section {
+class Section implements ThirdPartySettingsInterface {
 
   /**
    * The layout plugin ID.
@@ -45,6 +47,15 @@ class Section {
    */
   protected $components = [];
 
+  /**
+   * Third party settings.
+   *
+   * An array of key/value pairs keyed by provider.
+   *
+   * @var array[]
+   */
+  protected $thirdPartySettings = [];
+
   /**
    * Constructs a new Section.
    *
@@ -54,13 +65,16 @@ class Section {
    *   (optional) The layout plugin settings.
    * @param \Drupal\layout_builder\SectionComponent[] $components
    *   (optional) The components.
+   * @param array[] $third_party_settings
+   *   (optional) Any third party settings.
    */
-  public function __construct($layout_id, array $layout_settings = [], array $components = []) {
+  public function __construct($layout_id, array $layout_settings = [], array $components = [], array $third_party_settings = []) {
     $this->layoutId = $layout_id;
     $this->layoutSettings = $layout_settings;
     foreach ($components as $component) {
       $this->setComponent($component);
     }
+    $this->thirdPartySettings = $third_party_settings;
   }
 
   /**
@@ -334,6 +348,7 @@ public function toArray() {
       'components' => array_map(function (SectionComponent $component) {
         return $component->toArray();
       }, $this->getComponents()),
+      'third_party_settings' => $this->thirdPartySettings,
     ];
   }
 
@@ -349,10 +364,18 @@ public function toArray() {
    *   The section object.
    */
   public static function fromArray(array $section) {
+    // Ensure expected array keys are present.
+    $section += [
+      'layout_id' => '',
+      'layout_settings' => [],
+      'components' => [],
+      'third_party_settings' => [],
+    ];
     return new static(
       $section['layout_id'],
       $section['layout_settings'],
-      array_map([SectionComponent::class, 'fromArray'], $section['components'])
+      array_map([SectionComponent::class, 'fromArray'], $section['components']),
+      $section['third_party_settings']
     );
   }
 
@@ -365,4 +388,46 @@ public function __clone() {
     }
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function getThirdPartySetting($provider, $key, $default = NULL) {
+    return isset($this->thirdPartySettings[$provider][$key]) ? $this->thirdPartySettings[$provider][$key] : $default;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getThirdPartySettings($provider) {
+    return isset($this->thirdPartySettings[$provider]) ? $this->thirdPartySettings[$provider] : [];
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function setThirdPartySetting($provider, $key, $value) {
+    $this->thirdPartySettings[$provider][$key] = $value;
+    return $this;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function unsetThirdPartySetting($provider, $key) {
+    unset($this->thirdPartySettings[$provider][$key]);
+    // If the third party is no longer storing any information, completely
+    // remove the array holding the settings for this provider.
+    if (empty($this->thirdPartySettings[$provider])) {
+      unset($this->thirdPartySettings[$provider]);
+    }
+    return $this;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getThirdPartyProviders() {
+    return array_keys($this->thirdPartySettings);
+  }
+
 }
diff --git a/core/modules/layout_builder/tests/fixtures/update/layout-builder-enable.php b/core/modules/layout_builder/tests/fixtures/update/layout-builder-enable.php
index fc69baae3375..980a87064ef7 100644
--- a/core/modules/layout_builder/tests/fixtures/update/layout-builder-enable.php
+++ b/core/modules/layout_builder/tests/fixtures/update/layout-builder-enable.php
@@ -32,6 +32,7 @@
       'weight' => 0,
     ],
   ],
+  'third_party_settings' => [],
 ];
 $connection->update('config')
   ->fields([
diff --git a/core/modules/layout_builder/tests/modules/layout_builder_defaults_test/config/schema/layout_builder_defaults_test.schema.yml b/core/modules/layout_builder/tests/modules/layout_builder_defaults_test/config/schema/layout_builder_defaults_test.schema.yml
new file mode 100644
index 000000000000..cd7d15152603
--- /dev/null
+++ b/core/modules/layout_builder/tests/modules/layout_builder_defaults_test/config/schema/layout_builder_defaults_test.schema.yml
@@ -0,0 +1,6 @@
+layout_builder.section.third_party.layout_builder_defaults_test:
+  type: mapping
+  mapping:
+    which_party:
+      label: 'Which party?'
+      type: string
diff --git a/core/modules/layout_builder/tests/src/Functional/Update/LayoutBuilderEnableUpdatePathTest.php b/core/modules/layout_builder/tests/src/Functional/Update/LayoutBuilderEnableUpdatePathTest.php
index 5577539359dd..413d0eae0ab3 100644
--- a/core/modules/layout_builder/tests/src/Functional/Update/LayoutBuilderEnableUpdatePathTest.php
+++ b/core/modules/layout_builder/tests/src/Functional/Update/LayoutBuilderEnableUpdatePathTest.php
@@ -47,6 +47,7 @@ public function testRunUpdates() {
               'weight' => 0,
             ],
           ],
+          'third_party_settings' => [],
         ],
       ],
     ];
diff --git a/core/modules/layout_builder/tests/src/Kernel/DefaultsSectionStorageTest.php b/core/modules/layout_builder/tests/src/Kernel/DefaultsSectionStorageTest.php
index 2f355e221a17..637da2994751 100644
--- a/core/modules/layout_builder/tests/src/Kernel/DefaultsSectionStorageTest.php
+++ b/core/modules/layout_builder/tests/src/Kernel/DefaultsSectionStorageTest.php
@@ -103,9 +103,14 @@ public function testAccess($expected, $operation, $is_enabled, array $section_da
    */
   public function providerTestAccess() {
     $section_data = [
-      new Section('layout_onecol', [], [
-        'first-uuid' => new SectionComponent('first-uuid', 'content', ['id' => 'foo']),
-      ]),
+      new Section(
+        'layout_onecol',
+        [],
+        [
+          'first-uuid' => new SectionComponent('first-uuid', 'content', ['id' => 'foo'], ['harold' => 'maude']),
+        ],
+        ['layout_builder_defaults_test' => ['which_party' => 'third']]
+      ),
     ];
 
     // Data provider values are:
diff --git a/core/modules/layout_builder/tests/src/Unit/SectionTest.php b/core/modules/layout_builder/tests/src/Unit/SectionTest.php
index eff239507d2b..232e1719c525 100644
--- a/core/modules/layout_builder/tests/src/Unit/SectionTest.php
+++ b/core/modules/layout_builder/tests/src/Unit/SectionTest.php
@@ -25,11 +25,19 @@ class SectionTest extends UnitTestCase {
   protected function setUp() {
     parent::setUp();
 
-    $this->section = new Section('layout_onecol', [], [
-      new SectionComponent('existing-uuid', 'some-region', ['id' => 'existing-block-id']),
-      (new SectionComponent('second-uuid', 'ordered-region', ['id' => 'second-block-id']))->setWeight(3),
-      (new SectionComponent('first-uuid', 'ordered-region', ['id' => 'first-block-id']))->setWeight(2),
-    ]);
+    $this->section = new Section(
+      'layout_onecol',
+      [],
+      [
+        new SectionComponent('existing-uuid', 'some-region', ['id' => 'existing-block-id']),
+        (new SectionComponent('second-uuid', 'ordered-region', ['id' => 'second-block-id']))->setWeight(3),
+        (new SectionComponent('first-uuid', 'ordered-region', ['id' => 'first-block-id']))->setWeight(2),
+      ],
+      [
+        'bad_judgement' => ['blink_speed' => 'fast', 'spin_direction' => 'clockwise'],
+        'hunt_and_peck' => ['delay' => '300ms'],
+      ]
+    );
   }
 
   /**
@@ -179,4 +187,175 @@ protected function assertComponents(array $expected, Section $section) {
     $this->assertSame(array_keys($expected), array_keys($result));
   }
 
+  /**
+   * @covers ::getThirdPartySettings
+   * @dataProvider providerTestGetThirdPartySettings
+   */
+  public function testGetThirdPartySettings($provider, $expected) {
+    $this->assertSame($expected, $this->section->getThirdPartySettings($provider));
+  }
+
+  /**
+   * Provides test data for ::testGetThirdPartySettings().
+   */
+  public function providerTestGetThirdPartySettings() {
+    $data = [];
+    $data[] = [
+      'bad_judgement',
+      ['blink_speed' => 'fast', 'spin_direction' => 'clockwise'],
+    ];
+    $data[] = [
+      'hunt_and_peck',
+      ['delay' => '300ms'],
+    ];
+    $data[] = [
+      'non_existing_provider',
+      [],
+    ];
+    return $data;
+  }
+
+  /**
+   * @covers ::getThirdPartySetting
+   * @dataProvider providerTestGetThirdPartySetting
+   */
+  public function testGetThirdPartySetting($provider, $key, $expected, $default = FALSE) {
+    if ($default) {
+      $this->assertSame($expected, $this->section->getThirdPartySetting($provider, $key, $default));
+    }
+    else {
+      $this->assertSame($expected, $this->section->getThirdPartySetting($provider, $key));
+    }
+  }
+
+  /**
+   * Provides test data for ::testGetThirdPartySetting().
+   */
+  public function providerTestGetThirdPartySetting() {
+    $data = [];
+    $data[] = [
+      'bad_judgement',
+      'blink_speed',
+      'fast',
+    ];
+    $data[] = [
+      'hunt_and_peck',
+      'delay',
+      '300ms',
+    ];
+    $data[] = [
+      'hunt_and_peck',
+      'non_existing_key',
+      NULL,
+    ];
+    $data[] = [
+      'non_existing_provider',
+      'non_existing_key',
+      NULL,
+    ];
+    $data[] = [
+      'non_existing_provider',
+      'non_existing_key',
+      'default value',
+      'default value',
+    ];
+    return $data;
+  }
+
+  /**
+   * @covers ::setThirdPartySetting
+   * @dataProvider providerTestSetThirdPartySetting
+   */
+  public function testSetThirdPartySetting($provider, $key, $value, $expected) {
+    $this->section->setThirdPartySetting($provider, $key, $value);
+    $this->assertSame($expected, $this->section->getThirdPartySettings($provider));
+  }
+
+  /**
+   * Provides test data for ::testSetThirdPartySettings().
+   */
+  public function providerTestSetThirdPartySetting() {
+    $data = [];
+    $data[] = [
+      'bad_judgement',
+      'blink_speed',
+      'super fast',
+      [
+        'blink_speed' => 'super fast',
+        'spin_direction' => 'clockwise',
+      ],
+    ];
+    $data[] = [
+      'bad_judgement',
+      'new_setting',
+      'new_value',
+      [
+        'blink_speed' => 'fast',
+        'spin_direction' => 'clockwise',
+        'new_setting' => 'new_value',
+      ],
+    ];
+    $data[] = [
+      'new_provider',
+      'new_setting',
+      'new_value',
+      [
+        'new_setting' => 'new_value',
+      ],
+    ];
+    return $data;
+  }
+
+  /**
+   * @covers ::unsetThirdPartySetting
+   * @dataProvider providerTestUnsetThirdPartySetting
+   */
+  public function testUnsetThirdPartySetting() {
+    $this->section->unsetThirdPartySetting('bad_judgement', 'blink_speed');
+    $this->assertSame(['spin_direction' => 'clockwise'], $this->section->getThirdPartySettings('bad_judgement'));
+    $this->section->unsetThirdPartySetting('hunt_and_peck', 'delay');
+    $this->assertSame([], $this->section->getThirdPartySettings('hunt_and_peck'));
+    $this->section->unsetThirdPartySetting('bad_judgement', 'non_existing_key');
+    $this->section->unsetThirdPartySetting('non_existing_provider', 'non_existing_key');
+  }
+
+  /**
+   * Provides test data for ::testUnsetThirdPartySettings().
+   */
+  public function providerTestUnsetThirdPartySetting() {
+    $data = [];
+    $data[] = [
+      'bad_judgement',
+      'blink_speed',
+      [
+        'spin_direction' => 'clockwise',
+      ],
+    ];
+    $data[] = [
+      'hunt_and_peck',
+      'delay',
+      [],
+    ];
+    $data[] = [
+      'bad_judgement',
+      'non_existing_key',
+      [],
+    ];
+    $data[] = [
+      'non_existing_provider',
+      'non_existing_key',
+      [],
+    ];
+    return $data;
+  }
+
+  /**
+   * @covers ::getThirdPartyProviders
+   */
+  public function testGetThirdPartyProviders() {
+    $this->assertSame(['bad_judgement', 'hunt_and_peck'], $this->section->getThirdPartyProviders());
+    $this->section->unsetThirdPartySetting('hunt_and_peck', 'delay');
+    $this->assertSame(['bad_judgement'], $this->section->getThirdPartyProviders());
+  }
+
 }
-- 
GitLab