diff --git a/css/layout-paragraphs-builder.css b/css/layout-paragraphs-builder.css
index 5f8f271514c9e1b8d149931714fd0b8180d04f5b..99cf51a70a72dfc0e85b20e13feaf9b113d11cd2 100644
--- a/css/layout-paragraphs-builder.css
+++ b/css/layout-paragraphs-builder.css
@@ -42,17 +42,21 @@
   padding: 5px;
   position: relative;
 }
-.dragula-enabled .lpb-component {
+.lpb-component:hover,
+.lpb-component:focus-within {
+  outline: 1px solid blue;
+}
+.lpb-component {
   cursor: grab;
 }
-.lpb-component:hover .lpb-region {
+.lpb-component:hover .lpb-region,
+.lpb-component:focus-within .lpb-region {
   outline: 1px dotted rgba(0, 0, 255, 0.5);
 }
-.lpb-component:hover .lpb-region:hover {
+.lpb-component:hover .lpb-region:hover,
+.lpb-component:focus-within .lpb-region:focus-within {
   outline: 1px solid rgba(0, 0, 255, 0.5);
 }
-.lpb-active-item,
-.lpb-active-item.lpb-region
 .lpb-component:hover .js-lpb-active-item {
   outline: 1px solid blue !important;
 }
@@ -91,7 +95,8 @@
   top: 5px;
   opacity: 0;
 }
-.lpb-component:hover > .lpb-controls {
+.lpb-component:hover > .lpb-controls,
+.lpb-component:focus-within > .lpb-controls {
   opacity: 1;
 }
 .lpb-controls a span {
@@ -114,7 +119,7 @@
   width: 28px;
   margin: 0;
   padding: 0;
-  border: none;
+  border: 1px solid transparent;
   border-radius: 50%;
   opacity: 1;
 }
@@ -126,6 +131,12 @@
   background-color: #eee;
   box-shadow: 0 0 1px rgba(0, 0, 0, 1);
 }
+.lpb-up:focus,
+.lpb-down:focus,
+.lpb-edit:focus,
+.lpb-delete:focus {
+  border-color: blue;
+}
 .lpb-up {
   background: url(../img/icon-up.png) 0 0 no-repeat;
   background-size: cover;
@@ -134,13 +145,13 @@
   background: url(../img/icon-down.png) 0 0 no-repeat;
   background-size: cover;
 }
-.lpb-down[disabled]:hover,
-.lpb-up[disabled]:hover {
+.lpb-down[tabindex="-1"]:hover,
+.lpb-up[tabindex="-1"]:hover {
   background-color: transparent;
   box-shadow: none;
 }
-.lpb-down[disabled],
-.lpb-up[disabled] {
+.lpb-down[tabindex="-1"],
+.lpb-up[tabindex="-1"] {
   opacity: .3;
   cursor: default;
 }
@@ -152,12 +163,6 @@
   background: url(../img/icon-delete.png) 0 0 no-repeat;
   background-size: cover;
 }
-.lpb-region:hover {
-  outline: 1px solid blue;
-}
-.lpb-component:hover {
-  outline: 1px solid blue;
-}
 .lpb-btn {
   position: absolute;
   position: absolute;
@@ -180,13 +185,14 @@
 .lpb-component > .lpb-btn {
   opacity: 0;
 }
-.lpb-component:hover > .lpb-btn {
+.lpb-component:hover > .lpb-btn,
+.lpb-component:focus-within > .lpb-btn {
   opacity: 1;
 }
 .lpb-btn--add {
   position: absolute;
   display: inline-block;
-  border: none;
+  border: 1px solid transparent;
   color: #333;
   font-weight: normal;
   line-height: 1;
@@ -205,6 +211,9 @@
   z-index: 1000;
   opacity: 0;
 }
+.lpb-btn--add:focus {
+  border-color: blue;
+}
 .lpb-btn--add.center {
   top: 50%;
   transform: translate(-50%, -50%);
@@ -216,11 +225,13 @@
 .lpb-btn--add.after {
   bottom: -18px;
 }
-.lpb-component:hover > .lpb-btn--add {
+.lpb-component:hover > .lpb-btn--add,
+.lpb-component:focus-within > .lpb-btn--add {
   visibility: visible;
   opacity: 1;
 }
-.lpb-region:hover > .lpb-btn--add {
+.lpb-region:hover > .lpb-btn--add,
+.lpb-region:focus-within > .lpb-btn--add {
   visibility: visible;
   opacity: 1;
 }
diff --git a/js/layout-paragraphs-builder.js b/js/layout-paragraphs-builder.js
index a7734e6a66d8277266e6fb3e6b472cb078468542..fc90b6ceac9ec652745a2ce185469fac143ccb06 100644
--- a/js/layout-paragraphs-builder.js
+++ b/js/layout-paragraphs-builder.js
@@ -27,19 +27,157 @@
       },
     }).execute();
   });
-
   /**
    * Returns a list of errors for the attempted move, or an empty array if there are no errors.
-   * @param {Element} el The element being moved.
-   * @param {Element} target The distination
    * @param {Element} settings The builder settings.
+   * @param {Element} el The element being moved.
+   * @param {Element} target The destination
+   * @param {Element} source The source
+   * @param {Element} sibling The next sibling element
    * @return {Array} An array of errors.
    */
-  function lpbMoveErrors(el, target, settings) {
+  function moveErrors(settings, el, target, source, sibling) {
     return Drupal._lpbMoveErrors
-      .map(validator => validator.apply(null, [el, target, settings]))
+      .map(validator =>
+        validator.apply(null, [settings, el, target, source, sibling]),
+      )
       .filter(errors => errors !== false && errors !== undefined);
   }
+  function updateMoveButtons($element) {
+    $element.find('.lpb-up, .lpb-down').attr('tabindex', '0');
+    $element
+      .find(
+        '.lpb-component:first-of-type .lpb-up, .lpb-component:last-of-type .lpb-down',
+      )
+      .attr('tabindex', '-1');
+  }
+  function updateUi($element) {
+    reorderComponents($element);
+    updateMoveButtons($element);
+  }
+  /**
+   * Moves a component up or down within a simple list of components.
+   * @param {jQuery} $moveItem The item to move.
+   * @param {int} direction 1 (down) or -1 (up).
+   * @return {void}
+   */
+  function move($moveItem, direction) {
+    const $sibling =
+      direction === 1
+        ? $moveItem.nextAll('.lpb-component').first()
+        : $moveItem.prevAll('.lpb-component').first();
+    const method = direction === 1 ? 'after' : 'before';
+    const { scrollY } = window;
+    const destScroll = scrollY + $sibling.outerHeight() * direction;
+    const distance = Math.abs(destScroll - scrollY);
+
+    if ($sibling.length === 0) {
+      return false;
+    }
+
+    $({ translateY: 0 }).animate(
+      { translateY: 100 * direction },
+      {
+        duration: Math.max(100, Math.min(distance, 500)),
+        easing: 'swing',
+        step() {
+          const a = $sibling.outerHeight() * (this.translateY / 100);
+          const b = -$moveItem.outerHeight() * (this.translateY / 100);
+          $moveItem.css({ transform: `translateY(${a}px)` });
+          $sibling.css({ transform: `translateY(${b}px)` });
+        },
+        complete() {
+          $moveItem.css({ transform: 'none' });
+          $sibling.css({ transform: 'none' });
+          $sibling[method]($moveItem);
+          updateUi($moveItem.closest(`[${idAttr}]`));
+        },
+      },
+    );
+    if (distance > 50) {
+      $('html, body').animate({ scrollTop: destScroll });
+    }
+  }
+  /**
+   * Moves the focused component up or down the DOM to the next valid position
+   * when an arrow key is pressed. Unlike move(), nav()can fully navigate
+   * components to any valid position in an entire layout.
+   * @param {jQuery} $item The jQuery item to move.
+   * @param {int} dir The direction to move (1 == down, -1 == up).
+   * @param {Object} settings The builder ui settings.
+   */
+  function nav($item, dir, settings) {
+    const $element = $item.closest(`[${idAttr}]`);
+    $item.addClass('lpb-active-item');
+    // Add shims as target elements.
+    if (dir === -1) {
+      $(
+        '.lpb-region .lpb-btn--add, .lpb-layout:not(.lpb-active-item)',
+        $element,
+      ).before('<div class="lpb-shim"></div>');
+    } else if (dir === 1) {
+      $('.lpb-region', $element).prepend('<div class="lpb-shim"></div>');
+      $('.lpb-layout:not(.lpb-active-item)', $element).after(
+        '<div class="lpb-shim"></div>',
+      );
+    }
+    // Build a list of possible targets, or move destinatons.
+    const targets = $('.lpb-component, .lpb-shim', $element)
+      .toArray()
+      // Remove child components from possible targets.
+      .filter(i => !$.contains($item[0], i))
+      // Remove layout elements that are not self from possible targets.
+      .filter(i => i.className.indexOf('lpb-layout') === -1 || i === $item[0]);
+    const currentElement = $item[0];
+    let pos = targets.indexOf(currentElement);
+    // Check to see if the next position is allowed by calling the 'accepts' callback.
+    while (
+      targets[pos + dir] !== undefined &&
+      moveErrors(
+        settings,
+        $item[0],
+        targets[pos + dir].parentNode,
+        null,
+        $item.next().length ? $item.next()[0] : null,
+      ).length > 0
+    ) {
+      pos += dir;
+    }
+    if (targets[pos + dir] !== undefined) {
+      // Move after or before the target based on direction.
+      $(targets[pos + dir])[dir === 1 ? 'after' : 'before']($item);
+    }
+    // Remove the shims and save the order.
+    $('.lpb-shim', $element).remove();
+    updateUi($element);
+    $item.removeClass('lpb-active-item');
+  }
+
+  function attachEventListeners($element, settings) {
+    $element.on('click.lp-builder', '.lpb-up', e => {
+      move($(e.target).closest('.lpb-component'), -1);
+      return false;
+    });
+    $element.on('click.lp-builder', '.lpb-down', e => {
+      move($(e.target).closest('.lpb-component'), 1);
+      return false;
+    });
+    $element.on('click.lp-builder', '.lpb-component', e => {
+      $(e.currentTarget).focus();
+      return false;
+    });
+    document.addEventListener('keydown', e => {
+      const $item = $('.lpb-component:focus');
+      if ($item.length) {
+        if (e.code === 'ArrowDown' && $item) {
+          nav($item, 1, settings);
+        }
+        if (e.code === 'ArrowUp' && $item) {
+          nav($item, -1, settings);
+        }
+      }
+    });
+  }
   Drupal._lpbMoveErrors = [];
   /**
    * Registers a move validation function.
@@ -49,13 +187,13 @@
     Drupal._lpbMoveErrors.push(f);
   };
   // Checks nesting depth.
-  Drupal.registerLpbMoveError((el, target, settings) => {
+  Drupal.registerLpbMoveError((settings, el, target) => {
     if (el.className.indexOf('lpb-layout') > -1) {
       return $(target).parents('.lpb-layout').length > settings.nesting_depth;
     }
   });
   // If layout is required, prevents component from being placed outside a layout.
-  Drupal.registerLpbMoveError((el, target, settings) => {
+  Drupal.registerLpbMoveError((settings, el, target) => {
     if (settings.require_layouts) {
       if (
         el.className.indexOf('lpb-component') > -1 &&
@@ -65,9 +203,9 @@
       }
     }
   });
-
   Drupal.behaviors.layoutParagraphsBuilder = {
     attach: function attach(context, settings) {
+      // Run only once - initialize the editor ui.
       $('[data-lp-builder-id]', context)
         .once('lp-builder')
         .each((index, element) => {
@@ -78,10 +216,13 @@
             .find('.lpb-components, .lpb-region')
             .get();
           const drake = dragula(dragContainers, {
-            accepts(el, target) {
+            accepts(el, target, source, sibling) {
               // Returns false if any registered validator returns a value.
               // @see addMoveValidator()
-              return lpbMoveErrors(el, target, lpbSettings).length === 0;
+              return (
+                moveErrors(lpbSettings, el, target, source, sibling).length ===
+                0
+              );
             },
             moves(el, source, handle) {
               const $handle = $(handle);
@@ -95,8 +236,12 @@
               return true;
             },
           });
-          drake.on('drop', () => {
-            reorderComponents($element);
+          drake.on('drop', el => {
+            const $el = $(el);
+            if ($el.prev().is('a')) {
+              $el.insertBefore($el.prev());
+            }
+            updateUi($element);
           });
           drake.on('drag', el => {
             $element.addClass('is-dragging');
@@ -119,7 +264,17 @@
             $(container).removeClass('drag-target');
           });
           $element.data('drake', drake);
+          updateMoveButtons($element);
+          attachEventListeners($element, lpbSettings);
         });
+      // Run every time the behavior is attached.
+      if (context.classList && context.classList.contains('lpb-component')) {
+        $(context)
+          .closest('[data-lp-builder-id]')
+          .each((index, element) => {
+            updateMoveButtons($(element));
+          });
+      }
     },
   };
 })(jQuery, Drupal, Drupal.debounce, drupalSettings, dragula);
diff --git a/src/Controller/LayoutParagraphsBuilderController.php b/src/Controller/LayoutParagraphsBuilderController.php
index 9db9e63f60d74c8e9328cbf78ce66d14e66b01ca..f015cbd71af3ede35d0f1b310a5940b57f8dd61e 100644
--- a/src/Controller/LayoutParagraphsBuilderController.php
+++ b/src/Controller/LayoutParagraphsBuilderController.php
@@ -170,6 +170,7 @@ class LayoutParagraphsBuilderController extends ControllerBase {
       return $type['is_section'] === FALSE;
     });
     $component_menu = [
+      '#title' => $this->t('Choose a component'),
       '#theme' => 'layout_paragraphs_builder_component_menu',
       '#types' => [
         'layout' => $section_components,
diff --git a/src/Element/LayoutParagraphsBuilder.php b/src/Element/LayoutParagraphsBuilder.php
index 1eb10aa90f55327caaf1b05c8a29f43c5b9a8cca..da597656109cc28c42e85927757ed2f2d31340bd 100644
--- a/src/Element/LayoutParagraphsBuilder.php
+++ b/src/Element/LayoutParagraphsBuilder.php
@@ -219,7 +219,7 @@ class LayoutParagraphsBuilder extends RenderElement implements ContainerFactoryP
     $build['#attributes']['data-type'] = $entity->bundle();
     $build['#attributes']['data-id'] = $entity->id();
     $build['#attributes']['class'][] = 'lpb-component';
-    $build['#attributes']['tabindex'] = 0;
+    $build['#attributes']['tabindex'] = '0';
 
     $url_params = [
       'layout_paragraphs_layout' => $layout->id(),
@@ -270,7 +270,6 @@ class LayoutParagraphsBuilder extends RenderElement implements ContainerFactoryP
             'class' => ['lpb-region'],
             'data-region' => $region_name,
             'data-region-uuid' => $entity->uuid() . '-' . $region_name,
-            'tabindex' => 0,
           ],
           'insert_button' => $this->insertComponentButton($url_params, 10000, ['center']),
         ];
diff --git a/src/Event/LayoutParagraphsAllowedTypesEvent.php b/src/Event/LayoutParagraphsAllowedTypesEvent.php
index 7274813156d6b03eb48b24c23db7bb0dd0277f2e..07238ac5191dd4e0e725596214d57892fae3cfb9 100644
--- a/src/Event/LayoutParagraphsAllowedTypesEvent.php
+++ b/src/Event/LayoutParagraphsAllowedTypesEvent.php
@@ -52,7 +52,7 @@ class LayoutParagraphsAllowedTypesEvent extends Event {
    * @param string $region
    *   The region.
    */
-  public function __construct(array $types, LayoutParagraphsLayout $layout, string $parent_uuid, string $region) {
+  public function __construct(array $types, LayoutParagraphsLayout $layout, $parent_uuid = '', $region = '') {
     $this->types = $types;
     $this->layout = $layout;
     $this->parentUuid = $parent_uuid;
diff --git a/src/Form/ComponentFormBase.php b/src/Form/ComponentFormBase.php
index 02dcac5bfcd4da55ac4ebbc906934e1ae121312f..e07a24dfa114f7eff834684ecdc26fd22cfa24b7 100644
--- a/src/Form/ComponentFormBase.php
+++ b/src/Form/ComponentFormBase.php
@@ -126,6 +126,7 @@ abstract class ComponentFormBase extends FormBase {
     $this->paragraphType = $this->paragraph->getParagraphType();
 
     $form += [
+      '#title' => $this->formTitle(),
       '#paragraph' => $this->paragraph,
       '#display' => $display,
       '#tree' => TRUE,
@@ -191,6 +192,16 @@ abstract class ComponentFormBase extends FormBase {
     return $form;
   }
 
+  /**
+   * Create the form title.
+   *
+   * @return \Drupal\Core\StringTranslation\TranslatableMarkup
+   *   The form title.
+   */
+  protected function formTitle() {
+    return $this->t('Component form');
+  }
+
   /**
    * After build callback fixes issues with data-drupal-selector.
    *
diff --git a/src/Form/EditComponentForm.php b/src/Form/EditComponentForm.php
index ad668bf891c338d0b2d4fecf9d07154fd14e0b27..ff0a5401100e5a2db6223a46b24352ccd37f3553 100644
--- a/src/Form/EditComponentForm.php
+++ b/src/Form/EditComponentForm.php
@@ -48,6 +48,16 @@ class EditComponentForm extends ComponentFormBase {
     return $form;
   }
 
+  /**
+   * Create the form title.
+   *
+   * @return \Drupal\Core\StringTranslation\TranslatableMarkup
+   *   The form title.
+   */
+  protected function formTitle() {
+    return $this->t('Edit @type', ['@type' => $this->paragraph->getParagraphType()->label()]);
+  }
+
   /**
    * {@inheritDoc}
    */
diff --git a/src/Form/InsertComponentForm.php b/src/Form/InsertComponentForm.php
index c3c9c95d454f1a493cc05e3776d13fced4da775a..f3e100b363de421039e8daafa69e2bb4f7058426 100644
--- a/src/Form/InsertComponentForm.php
+++ b/src/Form/InsertComponentForm.php
@@ -79,6 +79,16 @@ class InsertComponentForm extends ComponentFormBase {
     return $this->buildComponentForm($form, $form_state);
   }
 
+  /**
+   * Create the form title.
+   *
+   * @return \Drupal\Core\StringTranslation\TranslatableMarkup
+   *   The form title.
+   */
+  protected function formTitle() {
+    return $this->t('Create new @type', ['@type' => $this->paragraph->getParagraphType()->label()]);
+  }
+
   /**
    * {@inheritDoc}
    */
diff --git a/templates/layout-paragraphs-builder-controls.html.twig b/templates/layout-paragraphs-builder-controls.html.twig
index cd2c860f1bd9e6dd6ec592ae0b55b72f76176340..1c3bcf82a35e360780c44e7dd7fd0e32e6edc338 100644
--- a/templates/layout-paragraphs-builder-controls.html.twig
+++ b/templates/layout-paragraphs-builder-controls.html.twig
@@ -1,7 +1,7 @@
   <div class="lpb-controls">
     <span class="lpb-controls-label">{{label}}</span>
-    <a class="lpb-up hidden" href="#move-up"><span>{{ 'Move Up'|t }}</span></a>
-    <a class="lpb-down hidden" href="#move-down"><span>{{ 'Move Down'|t }}</span></a>
+    <a class="lpb-up" href="#move-up"><span>{{ 'Move Up'|t }}</span></a>
+    <a class="lpb-down" href="#move-down"><span>{{ 'Move Down'|t }}</span></a>
     <a class="lpb-edit use-ajax" data-dialog-type="modal" data-dialog-options="{{ dialog_options }}" href="{{edit_url}}"><span>{{ 'Edit'|t }}</span></a>
     <a class="lpb-delete use-ajax" href="{{delete_url}}" data-confirm="{{ 'Really delete? There is no undo.'|t }}"><span>{{ 'Delete'|t }}</span></a>
   </div>