diff --git a/js/builder-form.js b/js/builder-form.js
index fa8ae11a8c54e3992d84cb609cd3a7db9a01f669..f2655e9a60a5793d6fe9a54a3d3f0b3839f70f50 100644
--- a/js/builder-form.js
+++ b/js/builder-form.js
@@ -20,4 +20,4 @@
       });
     }
   };
-})(jQuery, Drupal);
+})(jQuery, Drupal);
\ No newline at end of file
diff --git a/js/builder.es6.js b/js/builder.es6.js
index c59ff4cf39075537e71b81ddd3a410520ed1e51e..1cec0a53bb5877b13e68ad8c3ac0b16772ceacf1 100644
--- a/js/builder.es6.js
+++ b/js/builder.es6.js
@@ -10,7 +10,8 @@
    * @param {Object} settings
    *   The settings object.
    */
-  function attachUiElements($container, id, settings) {
+  function attachUiElements($container, settings) {
+    const id = $container[0].id;
     const lpbBuilderSettings = settings.lpBuilder || {};
     const uiElements = lpbBuilderSettings.uiElements || {};
     const containerUiElements = uiElements[id] || [];
@@ -441,15 +442,11 @@
   Drupal.behaviors.layoutParagraphsBuilder = {
     attach: function attach(context, settings) {
       // Add UI elements to the builder, each component, and each region.
-      [`${idAttr}`, 'data-uuid', 'data-region-uuid'].forEach((attr) => {
-        $(`[${attr}]`)
-          .not('.lpb-formatter')
-          .not('.has-components')
-          .once('lpb-ui-elements')
-          .each((i, el) => {
-            attachUiElements($(el), el.getAttribute(attr), settings);
-          });
-      });
+      $('[data-has-js-ui-element]')
+        .once('lpb-ui-elements')
+        .each((i, el) => {
+          attachUiElements($(el), settings);
+        });
       // Listen to relevant events and update UI.
       const events = [
         'lpb-builder:init.lpb',
diff --git a/js/builder.js b/js/builder.js
index a6ce6bd3e27c43e4bb0f80f5202a3a14bfc9771b..5e45834e783c8037b9f102cbe67bbf803bc47124 100644
--- a/js/builder.js
+++ b/js/builder.js
@@ -20,7 +20,8 @@ function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
 (function ($, Drupal, debounce, dragula) {
   var idAttr = 'data-lpb-id';
 
-  function attachUiElements($container, id, settings) {
+  function attachUiElements($container, settings) {
+    var id = $container[0].id;
     var lpbBuilderSettings = settings.lpBuilder || {};
     var uiElements = lpbBuilderSettings.uiElements || {};
     var containerUiElements = uiElements[id] || [];
@@ -354,10 +355,8 @@ function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
 
   Drupal.behaviors.layoutParagraphsBuilder = {
     attach: function attach(context, settings) {
-      ["".concat(idAttr), 'data-uuid', 'data-region-uuid'].forEach(function (attr) {
-        $("[".concat(attr, "]")).not('.lpb-formatter').not('.has-components').once('lpb-ui-elements').each(function (i, el) {
-          attachUiElements($(el), el.getAttribute(attr), settings);
-        });
+      $('[data-has-js-ui-element]').once('lpb-ui-elements').each(function (i, el) {
+        attachUiElements($(el), settings);
       });
       var events = ['lpb-builder:init.lpb', 'lpb-component:insert.lpb', 'lpb-component:update.lpb', 'lpb-component:move.lpb', 'lpb-component:drop.lpb', 'lpb-component:delete.lpb'].join(' ');
       $('[data-lpb-id]').once('lpb-events').on(events, function (e) {
diff --git a/src/Element/LayoutParagraphsBuilder.php b/src/Element/LayoutParagraphsBuilder.php
index 70f91bf54049831788540e2b3bab1c836c4250cb..632cdd97420349c6dc673688c7bfd722cd57da16 100644
--- a/src/Element/LayoutParagraphsBuilder.php
+++ b/src/Element/LayoutParagraphsBuilder.php
@@ -5,14 +5,15 @@ namespace Drupal\layout_paragraphs\Element;
 use Drupal\Core\Url;
 use Drupal\Core\Render\Markup;
 use Drupal\Core\Render\Renderer;
+use Drupal\Component\Utility\Html;
 use Drupal\Component\Serialization\Json;
 use Drupal\paragraphs\ParagraphInterface;
 use Drupal\Core\Access\AccessResultAllowed;
 use Drupal\Core\Layout\LayoutPluginManager;
 use Drupal\Core\Entity\EntityTypeBundleInfo;
+use Drupal\layout_paragraphs\Utility\Dialog;
 use Drupal\Core\Access\AccessResultForbidden;
 use Drupal\Core\Render\Element\RenderElement;
-use Drupal\layout_paragraphs\Utility\Dialog;
 use Drupal\Core\Entity\EntityRepositoryInterface;
 use Drupal\Core\Entity\EntityTypeManagerInterface;
 use Drupal\layout_paragraphs\LayoutParagraphsSection;
@@ -223,28 +224,13 @@ class LayoutParagraphsBuilder extends RenderElement implements ContainerFactoryP
         'lp-builder',
         'lp-builder-' . $this->layoutParagraphsLayout->id(),
       ],
+      'id' => Html::getUniqueId($this->layoutParagraphsLayout->id()),
       'data-lpb-id' => $this->layoutParagraphsLayout->id(),
     ];
     $element['#attached']['library'] = ['layout_paragraphs/builder'];
     $element['#attached']['drupalSettings']['lpBuilder'][$this->layoutParagraphsLayout->id()] = $this->layoutParagraphsLayout->getSettings();
     $element['#is_empty'] = $this->layoutParagraphsLayout->isEmpty();
     $element['#empty_message'] = $this->layoutParagraphsLayout->getSetting('empty_message', $this->t('Start adding content.'));
-    if ($this->layoutParagraphsLayout->getSetting('require_layouts', FALSE)) {
-      $this->addJsUiElement(
-        $element,
-        $this->layoutParagraphsLayout->id(),
-        $this->doRender($this->insertSectionButton(['layout_paragraphs_layout' => $this->layoutParagraphsLayout->id()], [], 0, ['center'])),
-        'insert'
-      );
-    }
-    else {
-      $this->addJsUiElement(
-        $element,
-        $this->layoutParagraphsLayout->id(),
-        $this->doRender($this->insertComponentButton(['layout_paragraphs_layout' => $this->layoutParagraphsLayout->id()], [], 0, ['center'])),
-        'insert'
-      );
-    }
     $element['#root_components'] = [];
     foreach ($this->layoutParagraphsLayout->getRootComponents() as $component) {
       /** @var \Drupal\layout_paragraphs\LayoutParagraphsComponent $component */
@@ -254,6 +240,22 @@ class LayoutParagraphsBuilder extends RenderElement implements ContainerFactoryP
     if (count($element['#root_components'])) {
       $element['#attributes']['class'][] = 'has-components';
     }
+    else {
+      if ($this->layoutParagraphsLayout->getSetting('require_layouts', FALSE)) {
+        $this->addJsUiElement(
+          $element,
+          $this->doRender($this->insertSectionButton(['layout_paragraphs_layout' => $this->layoutParagraphsLayout->id()], [], 0, ['center'])),
+          'insert'
+        );
+      }
+      else {
+        $this->addJsUiElement(
+          $element,
+          $this->doRender($this->insertComponentButton(['layout_paragraphs_layout' => $this->layoutParagraphsLayout->id()], [], 0, ['center'])),
+          'insert'
+        );
+      }
+    }
     return $element;
   }
 
@@ -279,6 +281,7 @@ class LayoutParagraphsBuilder extends RenderElement implements ContainerFactoryP
     $build['#attributes']['data-type'] = $entity->bundle();
     $build['#attributes']['data-id'] = $entity->id();
     $build['#attributes']['class'][] = 'js-lpb-component';
+    $build['#attributes']['id'] = Html::getUniqueId($entity->id());
     $build['#layout_paragraphs_component'] = TRUE;
     if ($entity->isNew()) {
       $build['#attributes']['class'][] = 'is_new';
@@ -307,23 +310,22 @@ class LayoutParagraphsBuilder extends RenderElement implements ContainerFactoryP
       '#uuid' => $entity->uuid(),
       '#layout_paragraphs_layout' => $this->layoutParagraphsLayout,
       '#edit_access' => $this->editAccess($entity),
-      '#duplicate_access' => $this->createAccess(),
+      '#duplicate_access' => $this->createAccess() && $this->checkCardinality(),
       '#delete_access' => $this->deleteAccess($entity),
     ];
-    $this->addJsUiElement($build, $entity->uuid(), $this->doRender($controls), 'controls', 'prepend');
+    $build['#attached']['drupalSettings']['lpBuilder']['uiElements'][$entity->uuid()] = [];
+    $this->addJsUiElement($build, $this->doRender($controls), 'controls', 'prepend');
 
-    if ($this->createAccess()) {
+    if ($this->createAccess() && $this->checkCardinality()) {
       if (!$component->getParentUuid() && $this->layoutParagraphsLayout->getSetting('require_layouts')) {
         $this->addJsUiElement(
           $build,
-          $entity->uuid(),
           $this->doRender($this->insertSectionButton($url_params, $query_params + ['placement' => 'before'], -10000, ['before'])),
           'insert_before',
           'prepend'
         );
         $this->addJsUiElement(
           $build,
-          $entity->uuid(),
           $this->doRender($this->insertSectionButton($url_params, $query_params + ['placement' => 'after'], 10000, ['after'])),
           'insert_after',
           'append'
@@ -332,14 +334,12 @@ class LayoutParagraphsBuilder extends RenderElement implements ContainerFactoryP
       else {
         $this->addJsUiElement(
           $build,
-          $entity->uuid(),
           $this->doRender($this->insertComponentButton($url_params, $query_params + ['placement' => 'before'], -10000, ['before'])),
           'insert_before',
           'prepend'
         );
         $this->addJsUiElement(
           $build,
-          $entity->uuid(),
           $this->doRender($this->insertComponentButton($url_params, $query_params + ['placement' => 'after'], -10000, ['after'])),
           'insert_after',
           'append'
@@ -371,12 +371,12 @@ class LayoutParagraphsBuilder extends RenderElement implements ContainerFactoryP
             ],
             'data-region' => $region_name,
             'data-region-uuid' => $entity->uuid() . '-' . $region_name,
+            'id' => Html::getUniqueId($entity->uuid() . '-' . $region_name),
           ],
         ];
-        if ($this->createAccess()) {
+        if ($this->createAccess() && $this->checkCardinality()) {
           $this->addJsUiElement(
-            $build,
-            $entity->uuid() . '-' . $region_name,
+            $build['regions'][$region_name],
             $this->doRender($this->insertComponentButton($url_params, $query_params, 10000, ['center'])),
             'insert'
           );
@@ -681,8 +681,6 @@ class LayoutParagraphsBuilder extends RenderElement implements ContainerFactoryP
    *
    * @param array $build
    *   The build array to attach JS settings to.
-   * @param string $id
-   *   The element container's id.
    * @param \Drupal\Core\Render\Markup $element
    *   The UI element.
    * @param string $key
@@ -690,7 +688,9 @@ class LayoutParagraphsBuilder extends RenderElement implements ContainerFactoryP
    * @param string $method
    *   The javascript method to use to attach $element to its container.
    */
-  public function addJsUiElement(array &$build, string $id, Markup $element, string $key, string $method = 'append') {
+  public function addJsUiElement(array &$build, Markup $element, string $key, string $method = 'append') {
+    $id = $build['#attributes']['id'];
+    $build['#attributes']['data-has-js-ui-element'] = TRUE;
     $build['#attached']['drupalSettings']['lpBuilder']['uiElements'][$id][$key] = [
       'element' => $element,
       'method' => $method,
@@ -710,4 +710,32 @@ class LayoutParagraphsBuilder extends RenderElement implements ContainerFactoryP
     return $this->renderer->render($render_array);
   }
 
+  /**
+   * Checks if adding a component would exceed the field's cardinality limit.
+   *
+   * @return bool
+   *   True if a compoment can be added without exceeding cardinality.
+   */
+  protected function checkCardinality() {
+    $cardinality = $this->getCardinality();
+    if ($cardinality > 0) {
+      $count = $this->layoutParagraphsLayout->getParagraphsReferenceField()->count();
+      return $cardinality > $count;
+    }
+    return TRUE;
+  }
+
+  /**
+   * Gets the cardinality field setting for a Layout Paragraphs reference field.
+   *
+   * @return int
+   *   The cardinality setting.
+   */
+  protected function getCardinality() {
+    $field_name = $this->layoutParagraphsLayout->getFieldName();
+    $field_config = $this->layoutParagraphsLayout->getEntity()->{$field_name}->getFieldDefinition();
+    $field_definition = $field_config->getFieldStorageDefinition();
+    return $field_definition->getCardinality();
+  }
+
 }
diff --git a/src/EventSubscriber/LayoutParagraphsUpdateLayoutSubscriber.php b/src/EventSubscriber/LayoutParagraphsUpdateLayoutSubscriber.php
index 80da33aa1d6e5ea998a3f805405c56edf9a0861d..d193ed4381f4553f38981f51cdfab784fe2452c4 100644
--- a/src/EventSubscriber/LayoutParagraphsUpdateLayoutSubscriber.php
+++ b/src/EventSubscriber/LayoutParagraphsUpdateLayoutSubscriber.php
@@ -15,22 +15,64 @@ class LayoutParagraphsUpdateLayoutSubscriber implements EventSubscriberInterface
    */
   public static function getSubscribedEvents() {
     return [
-      LayoutParagraphsUpdateLayoutEvent::EVENT_NAME => 'compareLayouts',
+      LayoutParagraphsUpdateLayoutEvent::EVENT_NAME => 'layoutUpdated',
     ];
   }
 
   /**
-   * Restricts available types based on settings in layout.
+   * Determines if a Layout Paragraphs Builder UI needs to be refreshed.
+   *
+   * Some interactions will create conditions where the entire builder UI needs
+   * to be refreshed, rather than simply returning a single new or edited
+   * component. For example: when removing the only component from a layout,
+   * the layout should be refreshed to show the correct ui/messaging for an
+   * empty container.
    *
    * @param \Drupal\layout_paragraphs\Event\LayoutParagraphsUpdateLayoutEvent $event
-   *   The allowed types event.
+   *   The event.
    */
-  public function compareLayouts(LayoutParagraphsUpdateLayoutEvent $event) {
+  public function layoutUpdated(LayoutParagraphsUpdateLayoutEvent $event) {
+    $event->needsRefresh = $this->compareEmptyState($event) || $this->compareMaximumReached($event);
+  }
 
+  /**
+   * Compares the empty state of the original and updated layout.
+   *
+   * If the original layout was empty and the updated layout is not, or visa
+   * versa, the entire layout builder ui needs to be refreshed.
+   *
+   * @param \Drupal\layout_paragraphs\Event\LayoutParagraphsUpdateLayoutEvent $event
+   *   The event.
+   */
+  public function compareEmptyState(LayoutParagraphsUpdateLayoutEvent $event) {
     $original = $event->getOriginalLayout()->getParagraphsReferenceField();
     $layout = $event->getUpdatedLayout()->getParagraphsReferenceField();
+    return $original->isEmpty() != $layout->isEmpty();
+  }
+
+  /**
+   * Compares count == cardinality of the original and updated layouts.
+   *
+   * @param \Drupal\layout_paragraphs\Event\LayoutParagraphsUpdateLayoutEvent $event
+   *   The event.
+   *
+   * @return bool
+   *   True if the count == cardinality limit has changed.
+   */
+  public function compareMaximumReached(LayoutParagraphsUpdateLayoutEvent $event) {
+
+    $original_count = $event->getOriginalLayout()->getParagraphsReferenceField()->count();
+    $updated_count = $event->getUpdatedLayout()->getParagraphsReferenceField()->count();
+    $cardinality = $event->getUpdatedLayout()
+      ->getParagraphsReferenceField()
+      ->getFieldDefinition()
+      ->getFieldStorageDefinition()
+      ->getCardinality();
+
+    $original_is_max = $cardinality > 0 && $cardinality <= $original_count;
+    $updated_is_max = $cardinality > 0 && $cardinality <= $updated_count;
 
-    $event->needsRefresh = ($original->isEmpty() != $layout->isEmpty());
+    return $original_is_max != $updated_is_max;
   }
 
 }
diff --git a/tests/src/FunctionalJavascript/CardinalityTest.php b/tests/src/FunctionalJavascript/CardinalityTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..e679161112e14a655c597f62a604538209da2f79
--- /dev/null
+++ b/tests/src/FunctionalJavascript/CardinalityTest.php
@@ -0,0 +1,70 @@
+<?php
+
+namespace Drupal\Tests\layout_paragraphs\FunctionalJavascript;
+
+/**
+ * Tests cardinality settings for a Layout Paragraphs field widget.
+ *
+ * @group layout_paragraphs
+ */
+class CardinalityTest extends BuilderTestBase {
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function setUp(): void {
+    parent::setUp();
+    $this->loginWithPermissions([
+      'administer site configuration',
+      'administer node fields',
+      'administer node display',
+      'administer paragraphs types',
+    ]);
+    $this->drupalGet('admin/structure/types/manage/page/fields/node.page.field_content/storage');
+    $this->submitForm([
+      'cardinality' => 'number',
+      'cardinality_number' => 2,
+    ], 'Save field settings');
+  }
+
+  /**
+   * Tests cardinality settings for a Layout Paragraphs field widget.
+   */
+  public function testCardinality() {
+
+    $this->loginWithPermissions([
+      'create page content',
+      'edit own page content',
+    ]);
+
+    $this->drupalGet('node/add/page');
+    $page = $this->getSession()->getPage();
+
+    // Add a three-column section.
+    $this->addSectionComponent(2, '.lpb-btn--add');
+
+    // Cardinality is set to 2. We should still have (+) buttons.
+    $this->assertSession()->elementExists('css', '.layout__region--first .lpb-btn--add');
+    $this->htmlOutput($this->getSession()->getPage()->getHtml());
+
+    // Add a text component.
+    $this->addTextComponent('Some arbitrary text', '.layout__region--first .lpb-btn--add');
+
+    // Maximum number has been reached. There should be no more (+) buttons.
+    $this->assertSession()->elementNotExists('css', '.layout__region--first .lpb-btn--add');
+    $this->htmlOutput($this->getSession()->getPage()->getHtml());
+
+    // Remove a component.
+    $button = $page->find('css', '.layout__region--first a.lpb-delete');
+    $button->click();
+    $this->assertSession()->assertWaitOnAjaxRequest();
+    $button = $page->find('css', 'button.lpb-btn--confirm-delete');
+    $button->click();
+    $this->assertSession()->assertWaitOnAjaxRequest();
+
+    // We no longer have the maximum allowed items, and should have (+) buttons.
+    $this->assertSession()->elementExists('css', '.layout__region--first .lpb-btn--add');
+    $this->htmlOutput($this->getSession()->getPage()->getHtml());
+  }
+
+}