diff --git a/core/modules/forum/forum.admin.inc b/core/modules/forum/forum.admin.inc
index 451f7d1e476ba90265ae371a67ff4efca8af64bf..332f11ca5f41b02ced0f02cf47337db6de15a9a4 100644
--- a/core/modules/forum/forum.admin.inc
+++ b/core/modules/forum/forum.admin.inc
@@ -5,6 +5,7 @@
  * Administrative page callbacks for the Forum module.
  */
 
+use Drupal\taxonomy\Form\OverviewTerms;
 use Drupal\taxonomy\Entity\Term;
 
 /**
@@ -22,7 +23,9 @@ function forum_overview($form, &$form_state) {
 
   $vid = $config->get('vocabulary');
   $vocabulary = entity_load('taxonomy_vocabulary', $vid);
-  $form = taxonomy_overview_terms($form, $form_state, $vocabulary);
+  // @todo temporary, will be fixed in http://drupal.org/node/1974210.
+  $overview = OverviewTerms::create(Drupal::getContainer());
+  $form = $overview->buildForm($form, $form_state, $vocabulary);
 
   foreach (element_children($form['terms']) as $key) {
     if (isset($form['terms'][$key]['#term'])) {
@@ -50,7 +53,8 @@ function forum_overview($form, &$form_state) {
   unset($form['actions']['reset_alphabetical']);
 
   // The form needs to have submit and validate handlers set explicitly.
-  $form['#submit'] = array('taxonomy_overview_terms_submit'); // Use the existing taxonomy overview submit handler.
+  // Use the existing taxonomy overview submit handler.
+  $form['#submit'] = array(array($overview, 'submitForm'));
   $form['terms']['#empty'] = t('No containers or forums available. <a href="@container">Add container</a> or <a href="@forum">Add forum</a>.', array('@container' => url('admin/structure/forum/add/container'), '@forum' => url('admin/structure/forum/add/forum')));
   return $form;
 }
diff --git a/core/modules/taxonomy/lib/Drupal/taxonomy/Form/OverviewTerms.php b/core/modules/taxonomy/lib/Drupal/taxonomy/Form/OverviewTerms.php
new file mode 100644
index 0000000000000000000000000000000000000000..d79af40fa02ec32228d5efa076967c4f1b559d11
--- /dev/null
+++ b/core/modules/taxonomy/lib/Drupal/taxonomy/Form/OverviewTerms.php
@@ -0,0 +1,470 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\taxonomy\Form\OverviewTerms
+ */
+
+namespace Drupal\taxonomy\Form;
+
+use Drupal\Core\Form\FormBase;
+use Drupal\Core\Config\ConfigFactory;
+use Drupal\Core\Extension\ModuleHandlerInterface;
+use Drupal\taxonomy\VocabularyInterface;
+use Symfony\Component\DependencyInjection\ContainerInterface;
+
+/*
+ * Provides terms overview form for a taxonomy vocabulary.
+ */
+class OverviewTerms extends FormBase {
+
+  /**
+   * Taxonomy config.
+   *
+   * @var \Drupal\Core\Config\Config
+   */
+  protected $config;
+
+  /**
+   * The module handler service.
+   *
+   * @var \Drupal\Core\Extension\ModuleHandlerInterface
+   */
+  protected $moduleHandler;
+
+  /**
+   * Constructs an OverviewTerms object.
+   *
+   * @param \Drupal\Core\Config\ConfigFactory $config_factory
+   *   The config factory service.
+   * @param \Drupal\Core\Extension\ModuleHandlerInterface $module_handler
+   *   The module handler service.
+   */
+  public function __construct(ConfigFactory $config_factory, ModuleHandlerInterface $module_handler) {
+    $this->config = $config_factory->get('taxonomy.settings');
+    $this->moduleHandler = $module_handler;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function create(ContainerInterface $container) {
+    return new static(
+      $container->get('config.factory'),
+      $container->get('module_handler')
+    );
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getFormId() {
+    return 'taxonomy_overview_terms';
+  }
+
+  /**
+   * Form constructor.
+   *
+   * Display a tree of all the terms in a vocabulary, with options to edit
+   * each one. The form is made drag and drop by the theme function.
+   *
+   * @param array $form
+   *   An associative array containing the structure of the form.
+   * @param array $form_state
+   *   An associative array containing the current state of the form.
+   * @param \Drupal\taxonomy\VocabularyInterface $taxonomy_vocabulary
+   *   The vocabulary to display the overview form for.
+   *
+   * @return array
+   *   The form structure.
+   */
+  public function buildForm(array $form, array &$form_state, VocabularyInterface $taxonomy_vocabulary = NULL) {
+    // @todo Remove global variables when http://drupal.org/node/2044435 is in.
+    global $pager_page_array, $pager_total, $pager_total_items;
+
+    $form_state['taxonomy']['vocabulary'] = $taxonomy_vocabulary;
+    $parent_fields = FALSE;
+
+    $page = $this->getRequest()->query->get('page') ?: 0;
+    // Number of terms per page.
+    $page_increment = $this->config->get('terms_per_page_admin');
+    // Elements shown on this page.
+    $page_entries = 0;
+    // Elements at the root level before this page.
+    $before_entries = 0;
+    // Elements at the root level after this page.
+    $after_entries = 0;
+    // Elements at the root level on this page.
+    $root_entries = 0;
+
+    // Terms from previous and next pages are shown if the term tree would have
+    // been cut in the middle. Keep track of how many extra terms we show on
+    // each page of terms.
+    $back_step = NULL;
+    $forward_step = 0;
+
+    // An array of the terms to be displayed on this page.
+    $current_page = array();
+
+    $delta = 0;
+    $term_deltas = array();
+    // @todo taxonomy_get_tree needs to be converted to a service and injected.
+    //   Will be fixed in http://drupal.org/node/1976298.
+    $tree = taxonomy_get_tree($taxonomy_vocabulary->id(), 0, NULL, TRUE);
+    $term = current($tree);
+    do {
+      // In case this tree is completely empty.
+      if (empty($term)) {
+        break;
+      }
+      $delta++;
+      // Count entries before the current page.
+      if ($page && ($page * $page_increment) > $before_entries && !isset($back_step)) {
+        $before_entries++;
+        continue;
+      }
+      // Count entries after the current page.
+      elseif ($page_entries > $page_increment && isset($complete_tree)) {
+        $after_entries++;
+        continue;
+      }
+
+      // Do not let a term start the page that is not at the root.
+      if (isset($term->depth) && ($term->depth > 0) && !isset($back_step)) {
+        $back_step = 0;
+        while ($pterm = prev($tree)) {
+          $before_entries--;
+          $back_step++;
+          if ($pterm->depth == 0) {
+            prev($tree);
+            // Jump back to the start of the root level parent.
+            continue 2;
+          }
+        }
+      }
+      $back_step = isset($back_step) ? $back_step : 0;
+
+      // Continue rendering the tree until we reach the a new root item.
+      if ($page_entries >= $page_increment + $back_step + 1 && $term->depth == 0 && $root_entries > 1) {
+        $complete_tree = TRUE;
+        // This new item at the root level is the first item on the next page.
+        $after_entries++;
+        continue;
+      }
+      if ($page_entries >= $page_increment + $back_step) {
+        $forward_step++;
+      }
+
+      // Finally, if we've gotten down this far, we're rendering a term on this
+      // page.
+      $page_entries++;
+      $term_deltas[$term->id()] = isset($term_deltas[$term->id()]) ? $term_deltas[$term->id()] + 1 : 0;
+      $key = 'tid:' . $term->id() . ':' . $term_deltas[$term->id()];
+
+      // Keep track of the first term displayed on this page.
+      if ($page_entries == 1) {
+        $form['#first_tid'] = $term->id();
+      }
+      // Keep a variable to make sure at least 2 root elements are displayed.
+      if ($term->parents[0] == 0) {
+        $root_entries++;
+      }
+      $current_page[$key] = $term;
+    } while ($term = next($tree));
+
+    // Because we didn't use a pager query, set the necessary pager variables.
+    $total_entries = $before_entries + $page_entries + $after_entries;
+    $pager_total_items[0] = $total_entries;
+    $pager_page_array[0] = $page;
+    $pager_total[0] = ceil($total_entries / $page_increment);
+
+    // If this form was already submitted once, it's probably hit a validation
+    // error. Ensure the form is rebuilt in the same order as the user
+    // submitted.
+    if (!empty($form_state['input'])) {
+      // Get the $_POST order.
+      $order = array_flip(array_keys($form_state['input']['terms']));
+      // Update our form with the new order.
+      $current_page = array_merge($order, $current_page);
+      foreach ($current_page as $key => $term) {
+        // Verify this is a term for the current page and set at the current
+        // depth.
+        if (is_array($form_state['input']['terms'][$key]) && is_numeric($form_state['input']['terms'][$key]['term']['tid'])) {
+          $current_page[$key]->depth = $form_state['input']['terms'][$key]['term']['depth'];
+        }
+        else {
+          unset($current_page[$key]);
+        }
+      }
+    }
+
+    $errors = form_get_errors() != FALSE ? form_get_errors() : array();
+    $destination = drupal_get_destination();
+    $row_position = 0;
+    // Build the actual form.
+    $form['terms'] = array(
+      '#type' => 'table',
+      '#header' => array($this->t('Name'), $this->t('Weight'), $this->t('Operations')),
+      '#empty' => $this->t('No terms available. <a href="@link">Add term</a>.', array('@link' => url('admin/structure/taxonomy/manage/' . $taxonomy_vocabulary->id() . '/add'))),
+      '#attributes' => array(
+        'id' => 'taxonomy',
+      ),
+    );
+    foreach ($current_page as $key => $term) {
+      $uri = $term->uri();
+      $edit_uri = $term->uri('edit-form');
+      $form['terms'][$key]['#term'] = $term;
+      $indentation = array();
+      if (isset($term->depth) && $term->depth > 0) {
+        $indentation = array(
+          '#theme' => 'indentation',
+          '#size' => $term->depth,
+        );
+      }
+      $form['terms'][$key]['term'] = array(
+        '#prefix' => !empty($indentation) ? drupal_render($indentation) : '',
+        '#type' => 'link',
+        '#title' => $term->label(),
+        '#href' => $uri['path'],
+      );
+      if ($taxonomy_vocabulary->hierarchy != TAXONOMY_HIERARCHY_MULTIPLE && count($tree) > 1) {
+        $parent_fields = TRUE;
+        $form['terms'][$key]['term']['tid'] = array(
+          '#type' => 'hidden',
+          '#value' => $term->id(),
+          '#attributes' => array(
+            'class' => array('term-id'),
+          ),
+        );
+        $form['terms'][$key]['term']['parent'] = array(
+          '#type' => 'hidden',
+          // Yes, default_value on a hidden. It needs to be changeable by the
+          // javascript.
+          '#default_value' => $term->parent->value,
+          '#attributes' => array(
+            'class' => array('term-parent'),
+          ),
+        );
+        $form['terms'][$key]['term']['depth'] = array(
+          '#type' => 'hidden',
+          // Same as above, the depth is modified by javascript, so it's a
+          // default_value.
+          '#default_value' => $term->depth,
+          '#attributes' => array(
+            'class' => array('term-depth'),
+          ),
+        );
+      }
+      $form['terms'][$key]['weight'] = array(
+        '#type' => 'weight',
+        '#delta' => $delta,
+        '#title_display' => 'invisible',
+        '#title' => $this->t('Weight for added term'),
+        '#default_value' => $term->weight->value,
+        '#attributes' => array(
+          'class' => array('term-weight'),
+        ),
+      );
+      $operations = array(
+        'edit' => array(
+          'title' => $this->t('edit'),
+          'href' => $edit_uri['path'],
+          'query' => $destination,
+        ),
+        'delete' => array(
+          'title' => $this->t('delete'),
+          'href' => $uri['path'] . '/delete',
+          'query' => $destination,
+        ),
+      );
+      if ($this->moduleHandler->moduleExists('content_translation') && content_translation_translate_access($term)) {
+        $operations['translate'] = array(
+          'title' => $this->t('translate'),
+          'href' => $uri['path'] . '/translations',
+          'query' => $destination,
+        );
+      }
+      $form['terms'][$key]['operations'] = array(
+        '#type' => 'operations',
+        '#links' => $operations,
+      );
+
+      $form['terms'][$key]['#attributes']['class'] = array();
+      if ($parent_fields) {
+        $form['terms'][$key]['#attributes']['class'][] = 'draggable';
+      }
+
+      // Add classes that mark which terms belong to previous and next pages.
+      if ($row_position < $back_step || $row_position >= $page_entries - $forward_step) {
+        $form['terms'][$key]['#attributes']['class'][] = 'taxonomy-term-preview';
+      }
+
+      if ($row_position !== 0 && $row_position !== count($tree) - 1) {
+        if ($row_position == $back_step - 1 || $row_position == $page_entries - $forward_step - 1) {
+          $form['terms'][$key]['#attributes']['class'][] = 'taxonomy-term-divider-top';
+        }
+        elseif ($row_position == $back_step || $row_position == $page_entries - $forward_step) {
+          $form['terms'][$key]['#attributes']['class'][] = 'taxonomy-term-divider-bottom';
+        }
+      }
+
+      // Add an error class if this row contains a form error.
+      foreach ($errors as $error_key => $error) {
+        if (strpos($error_key, $key) === 0) {
+          $form['terms'][$key]['#attributes']['class'][] = 'error';
+        }
+      }
+      $row_position++;
+    }
+
+    if ($parent_fields) {
+      $form['terms']['#tabledrag'][] = array(
+        'match',
+        'parent',
+        'term-parent',
+        'term-parent',
+        'term-id',
+        FALSE,
+      );
+      $form['terms']['#tabledrag'][] = array(
+        'depth',
+        'group',
+        'term-depth',
+        NULL,
+        NULL,
+        FALSE
+      );
+      $form['terms']['#attached']['library'][] = array('taxonomy', 'drupal.taxonomy');
+      $form['terms']['#attached']['js'][] = array(
+        'data' => array('taxonomy' => array('backStep' => $back_step, 'forwardStep' => $forward_step)),
+        'type' => 'setting',
+      );
+    }
+    $form['terms']['#tabledrag'][] = array('order', 'sibling', 'term-weight');
+
+    if ($taxonomy_vocabulary->hierarchy != TAXONOMY_HIERARCHY_MULTIPLE && count($tree) > 1) {
+      $form['actions'] = array('#type' => 'actions', '#tree' => FALSE);
+      $form['actions']['submit'] = array(
+        '#type' => 'submit',
+        '#value' => $this->t('Save'),
+        '#button_type' => 'primary',
+      );
+      $form['actions']['reset_alphabetical'] = array(
+        '#type' => 'submit',
+        '#submit' => array(array($this, 'submitReset')),
+        '#value' => $this->t('Reset to alphabetical'),
+      );
+      $form_state['redirect'] = array(current_path(), ($page ? array('query' => array('page' => $page)) : array()));
+    }
+
+    return $form;
+  }
+
+  /**
+   * Form submission handler.
+   *
+   * Rather than using a textfield or weight field, this form depends entirely
+   * upon the order of form elements on the page to determine new weights.
+   *
+   * Because there might be hundreds or thousands of taxonomy terms that need to
+   * be ordered, terms are weighted from 0 to the number of terms in the
+   * vocabulary, rather than the standard -10 to 10 scale. Numbers are sorted
+   * lowest to highest, but are not necessarily sequential. Numbers may be
+   * skipped when a term has children so that reordering is minimal when a child
+   * is added or removed from a term.
+   *
+   * @param array $form
+   *   An associative array containing the structure of the form.
+   * @param array $form_state
+   *   An associative array containing the current state of the form.
+   */
+  public function submitForm(array &$form, array &$form_state) {
+    // Sort term order based on weight.
+    uasort($form_state['values']['terms'], 'drupal_sort_weight');
+
+    $vocabulary = $form_state['taxonomy']['vocabulary'];
+    // Update the current hierarchy type as we go.
+    $hierarchy = TAXONOMY_HIERARCHY_DISABLED;
+
+    $changed_terms = array();
+    // @todo taxonomy_get_tree needs to be converted to a service and injected.
+    //   Will be fixed in http://drupal.org/node/1976298.
+    $tree = taxonomy_get_tree($vocabulary->id(), 0, NULL, TRUE);
+
+    if (empty($tree)) {
+      return;
+    }
+
+    // Build a list of all terms that need to be updated on previous pages.
+    $weight = 0;
+    $term = $tree[0];
+    while ($term->id() != $form['#first_tid']) {
+      if ($term->parent->value == 0 && $term->weight->value != $weight) {
+        $term->weight->value = $weight;
+        $changed_terms[$term->id()] = $term;
+      }
+      $weight++;
+      $hierarchy = $term->parent->value != 0 ? TAXONOMY_HIERARCHY_SINGLE : $hierarchy;
+      $term = $tree[$weight];
+    }
+
+    // Renumber the current page weights and assign any new parents.
+    $level_weights = array();
+    foreach ($form_state['values']['terms'] as $tid => $values) {
+      if (isset($form['terms'][$tid]['#term'])) {
+        $term = $form['terms'][$tid]['#term'];
+        // Give terms at the root level a weight in sequence with terms on previous pages.
+        if ($values['term']['parent'] == 0 && $term->weight->value != $weight) {
+          $term->weight->value = $weight;
+          $changed_terms[$term->id()] = $term;
+        }
+        // Terms not at the root level can safely start from 0 because they're all on this page.
+        elseif ($values['term']['parent'] > 0) {
+          $level_weights[$values['term']['parent']] = isset($level_weights[$values['term']['parent']]) ? $level_weights[$values['term']->parent->value] + 1 : 0;
+          if ($level_weights[$values['term']['parent']] != $term->weight->value) {
+            $term->weight->value = $level_weights[$values['term']['parent']];
+            $changed_terms[$term->id()] = $term;
+          }
+        }
+        // Update any changed parents.
+        if ($values['term']['parent'] != $term->parent->value) {
+          $term->parent->value = $values['term']['parent'];
+          $changed_terms[$term->id()] = $term;
+        }
+        $hierarchy = $term->parent->value != 0 ? TAXONOMY_HIERARCHY_SINGLE : $hierarchy;
+        $weight++;
+      }
+    }
+
+    // Build a list of all terms that need to be updated on following pages.
+    for ($weight; $weight < count($tree); $weight++) {
+      $term = $tree[$weight];
+      if ($term->parent->value == 0 && $term->weight->value != $weight) {
+        $term->parent->value = $term->parent->value;
+        $term->weight->value = $weight;
+        $changed_terms[$term->id()] = $term;
+      }
+      $hierarchy = $term->parent->value != 0 ? TAXONOMY_HIERARCHY_SINGLE : $hierarchy;
+    }
+
+    // Save all updated terms.
+    foreach ($changed_terms as $term) {
+      $term->save();
+    }
+
+    // Update the vocabulary hierarchy to flat or single hierarchy.
+    if ($vocabulary->hierarchy != $hierarchy) {
+      $vocabulary->hierarchy = $hierarchy;
+      $vocabulary->save();
+    }
+    drupal_set_message($this->t('The configuration options have been saved.'));
+  }
+
+  /**
+   * Redirects to confirmation form for the reset action.
+   */
+  public function submitReset(array &$form, array &$form_state) {
+    $form_state['redirect'] = 'admin/structure/taxonomy/manage/' . $form_state['taxonomy']['vocabulary']->id() . '/reset';
+  }
+
+}
diff --git a/core/modules/taxonomy/taxonomy.admin.inc b/core/modules/taxonomy/taxonomy.admin.inc
deleted file mode 100644
index bd8a2b5d2c8d665307d7fee481adf6d83dad5f6b..0000000000000000000000000000000000000000
--- a/core/modules/taxonomy/taxonomy.admin.inc
+++ /dev/null
@@ -1,378 +0,0 @@
-<?php
-
-/**
- * @file
- * Administrative page callbacks for the taxonomy module.
- */
-
-use Drupal\taxonomy\Entity\Vocabulary;
-
-/**
- * Form builder for the taxonomy terms overview.
- *
- * Display a tree of all the terms in a vocabulary, with options to edit
- * each one. The form is made drag and drop by the theme function.
- *
- * @param Drupal\taxonomy\Entity\Vocabulary $vocabulary
- *   The taxonomy vocabulary entity to list terms for.
- *
- * @ingroup forms
- * @see taxonomy_overview_terms_submit()
- * @see theme_taxonomy_overview_terms()
- */
-function taxonomy_overview_terms($form, &$form_state, Vocabulary $vocabulary) {
-  global $pager_page_array, $pager_total, $pager_total_items;
-
-  $form_state['taxonomy']['vocabulary'] = $vocabulary;
-  $parent_fields = FALSE;
-
-  $page            = isset($_GET['page']) ? $_GET['page'] : 0;
-  $page_increment  = Drupal::config('taxonomy.settings')->get('terms_per_page_admin');
-  $page_entries    = 0;
-  $before_entries  = 0;
-  $after_entries   = 0;
-  $root_entries    = 0;
-
-  // Terms from previous and next pages are shown if the term tree would have
-  // been cut in the middle. Keep track of how many extra terms we show on each
-  // page of terms.
-  $back_step    = NULL;
-  $forward_step = 0;
-
-  // An array of the terms to be displayed on this page.
-  $current_page = array();
-
-  $delta = 0;
-  $term_deltas = array();
-  $tree = taxonomy_get_tree($vocabulary->id(), 0, NULL, TRUE);
-  $term = current($tree);
-  do {
-    // In case this tree is completely empty.
-    if (empty($term)) {
-      break;
-    }
-    $delta++;
-    // Count entries before the current page.
-    if ($page && ($page * $page_increment) > $before_entries && !isset($back_step)) {
-      $before_entries++;
-      continue;
-    }
-    // Count entries after the current page.
-    elseif ($page_entries > $page_increment && isset($complete_tree)) {
-      $after_entries++;
-      continue;
-    }
-
-    // Do not let a term start the page that is not at the root.
-    if (isset($term->depth) && ($term->depth > 0) && !isset($back_step)) {
-      $back_step = 0;
-      while ($pterm = prev($tree)) {
-        $before_entries--;
-        $back_step++;
-        if ($pterm->depth == 0) {
-          prev($tree);
-          // Jump back to the start of the root level parent.
-          continue 2;
-       }
-      }
-    }
-    $back_step = isset($back_step) ? $back_step : 0;
-
-    // Continue rendering the tree until we reach the a new root item.
-    if ($page_entries >= $page_increment + $back_step + 1 && $term->depth == 0 && $root_entries > 1) {
-      $complete_tree = TRUE;
-      // This new item at the root level is the first item on the next page.
-      $after_entries++;
-      continue;
-    }
-    if ($page_entries >= $page_increment + $back_step) {
-      $forward_step++;
-    }
-
-    // Finally, if we've gotten down this far, we're rendering a term on this page.
-    $page_entries++;
-    $term_deltas[$term->id()] = isset($term_deltas[$term->id()]) ? $term_deltas[$term->id()] + 1 : 0;
-    $key = 'tid:' . $term->id() . ':' . $term_deltas[$term->id()];
-
-    // Keep track of the first term displayed on this page.
-    if ($page_entries == 1) {
-      $form['#first_tid'] = $term->id();
-    }
-    // Keep a variable to make sure at least 2 root elements are displayed.
-    if ($term->parents[0] == 0) {
-      $root_entries++;
-    }
-    $current_page[$key] = $term;
-  } while ($term = next($tree));
-
-  // Because we didn't use a pager query, set the necessary pager variables.
-  $total_entries = $before_entries + $page_entries + $after_entries;
-  $pager_total_items[0] = $total_entries;
-  $pager_page_array[0] = $page;
-  $pager_total[0] = ceil($total_entries / $page_increment);
-
-  // If this form was already submitted once, it's probably hit a validation
-  // error. Ensure the form is rebuilt in the same order as the user submitted.
-  if (!empty($form_state['input'])) {
-    // Get the $_POST order.
-    $order = array_flip(array_keys($form_state['input']['terms']));
-    // Update our form with the new order.
-    $current_page = array_merge($order, $current_page);
-    foreach ($current_page as $key => $term) {
-      // Verify this is a term for the current page and set at the current
-      // depth.
-      if (is_array($form_state['input']['terms'][$key]) && is_numeric($form_state['input']['terms'][$key]['term']['tid'])) {
-        $current_page[$key]->depth = $form_state['input']['terms'][$key]['term']['depth'];
-      }
-      else {
-        unset($current_page[$key]);
-      }
-    }
-  }
-
-  $errors = form_get_errors() != FALSE ? form_get_errors() : array();
-  $destination = drupal_get_destination();
-  $row_position = 0;
-  // Build the actual form.
-  $form['terms'] = array(
-    '#type' => 'table',
-    '#header' => array(t('Name'), t('Weight'), t('Operations')),
-    '#empty' => t('No terms available. <a href="@link">Add term</a>.', array('@link' => url('admin/structure/taxonomy/manage/' . $vocabulary->id() . '/add'))),
-    '#attributes' => array(
-      'id' => 'taxonomy',
-    ),
-  );
-  foreach ($current_page as $key => $term) {
-    $form['terms'][$key]['#term'] = $term;
-    $indentation = array();
-    if (isset($term->depth) && $term->depth > 0) {
-      $indentation = array(
-        '#theme' => 'indentation',
-        '#size' => $term->depth,
-      );
-    }
-    $form['terms'][$key]['term'] = array(
-      '#prefix' => !empty($indentation) ? drupal_render($indentation) : '',
-      '#type' => 'link',
-      '#title' => $term->label(),
-      '#href' => "taxonomy/term/" . $term->id(),
-    );
-    if ($vocabulary->hierarchy != TAXONOMY_HIERARCHY_MULTIPLE && count($tree) > 1) {
-      $parent_fields = TRUE;
-      $form['terms'][$key]['term']['tid'] = array(
-        '#type' => 'hidden',
-        '#value' => $term->id(),
-        '#attributes' => array(
-          'class' => array('term-id'),
-        ),
-      );
-      $form['terms'][$key]['term']['parent'] = array(
-        '#type' => 'hidden',
-        // Yes, default_value on a hidden. It needs to be changeable by the
-        // javascript.
-        '#default_value' => $term->parent->value,
-        '#attributes' => array(
-          'class' => array('term-parent'),
-        ),
-      );
-      $form['terms'][$key]['term']['depth'] = array(
-        '#type' => 'hidden',
-        // Same as above, the depth is modified by javascript, so it's a
-        // default_value.
-        '#default_value' => $term->depth,
-        '#attributes' => array(
-          'class' => array('term-depth'),
-        ),
-      );
-    }
-    $form['terms'][$key]['weight'] = array(
-      '#type' => 'weight',
-      '#delta' => $delta,
-      '#title_display' => 'invisible',
-      '#title' => t('Weight for added term'),
-      '#default_value' => $term->weight->value,
-      '#attributes' => array(
-        'class' => array('term-weight'),
-      ),
-    );
-    $operations = array(
-      'edit' => array(
-        'title' => t('edit'),
-        'href' => 'taxonomy/term/' . $term->id() . '/edit',
-        'query' => $destination,
-      ),
-      'delete' => array(
-        'title' => t('delete'),
-        'href' => 'taxonomy/term/' . $term->id() . '/delete',
-        'query' => $destination,
-      ),
-    );
-    if (module_invoke('content_translation', 'translate_access', $term)) {
-      $operations['translate'] = array(
-        'title' => t('translate'),
-        'href' => 'taxonomy/term/' . $term->id() . '/translations',
-        'query' => $destination,
-      );
-    }
-    $form['terms'][$key]['operations'] = array(
-      '#type' => 'operations',
-      '#links' => $operations,
-    );
-
-    $form['terms'][$key]['#attributes']['class'] = array();
-    if ($parent_fields) {
-      $form['terms'][$key]['#attributes']['class'][] = 'draggable';
-    }
-
-    // Add classes that mark which terms belong to previous and next pages.
-    if ($row_position < $back_step || $row_position >= $page_entries - $forward_step) {
-      $form['terms'][$key]['#attributes']['class'][] = 'taxonomy-term-preview';
-    }
-
-    if ($row_position !== 0 && $row_position !== count($tree) - 1) {
-      if ($row_position == $back_step - 1 || $row_position == $page_entries - $forward_step - 1) {
-        $form['terms'][$key]['#attributes']['class'][] = 'taxonomy-term-divider-top';
-      }
-      elseif ($row_position == $back_step || $row_position == $page_entries - $forward_step) {
-        $form['terms'][$key]['#attributes']['class'][] = 'taxonomy-term-divider-bottom';
-      }
-    }
-
-    // Add an error class if this row contains a form error.
-    foreach ($errors as $error_key => $error) {
-      if (strpos($error_key, $key) === 0) {
-        $form['terms'][$key]['#attributes']['class'][] = 'error';
-      }
-    }
-    $row_position++;
-  }
-
-
-  if ($parent_fields) {
-    $form['terms']['#tabledrag'][] = array('match', 'parent', 'term-parent', 'term-parent', 'term-id', FALSE);
-    $form['terms']['#tabledrag'][] = array('depth', 'group', 'term-depth', NULL, NULL, FALSE);
-    $form['terms']['#attached']['library'][] = array('taxonomy', 'drupal.taxonomy');
-    $form['terms']['#attached']['js'][] = array(
-      'data' => array('taxonomy' => array('backStep' => $back_step, 'forwardStep' => $forward_step)),
-      'type' => 'setting',
-    );
-  }
-  $form['terms']['#tabledrag'][] = array('order', 'sibling', 'term-weight');
-
-  if ($vocabulary->hierarchy != TAXONOMY_HIERARCHY_MULTIPLE && count($tree) > 1) {
-    $form['actions'] = array('#type' => 'actions', '#tree' => FALSE);
-    $form['actions']['submit'] = array(
-      '#type' => 'submit',
-      '#value' => t('Save'),
-      '#button_type' => 'primary',
-    );
-    $form['actions']['reset_alphabetical'] = array(
-      '#type' => 'submit',
-      '#value' => t('Reset to alphabetical')
-    );
-    $form_state['redirect'] = array(current_path(), (isset($_GET['page']) ? array('query' => array('page' => $_GET['page'])) : array()));
-  }
-
-  return $form;
-}
-
-/**
- * Submit handler for terms overview form.
- *
- * Rather than using a textfield or weight field, this form depends entirely
- * upon the order of form elements on the page to determine new weights.
- *
- * Because there might be hundreds or thousands of taxonomy terms that need to
- * be ordered, terms are weighted from 0 to the number of terms in the
- * vocabulary, rather than the standard -10 to 10 scale. Numbers are sorted
- * lowest to highest, but are not necessarily sequential. Numbers may be skipped
- * when a term has children so that reordering is minimal when a child is
- * added or removed from a term.
- *
- * @see taxonomy_overview_terms()
- */
-function taxonomy_overview_terms_submit($form, &$form_state) {
-  if ($form_state['triggering_element']['#value'] == t('Reset to alphabetical')) {
-    // Redirect to confirmation form for the reset action.
-    $form_state['redirect'] = 'admin/structure/taxonomy/manage/' . $form_state['taxonomy']['vocabulary']->id() . '/reset';
-    return;
-  }
-
-  // Sort term order based on weight.
-  uasort($form_state['values']['terms'], 'drupal_sort_weight');
-
-  $vocabulary = $form_state['taxonomy']['vocabulary'];
-  // Update the current hierarchy type as we go.
-  $hierarchy = TAXONOMY_HIERARCHY_DISABLED;
-
-  $changed_terms = array();
-  $tree = taxonomy_get_tree($vocabulary->id(), 0, NULL, TRUE);
-
-  if (empty($tree)) {
-    return;
-  }
-
-  // Build a list of all terms that need to be updated on previous pages.
-  $weight = 0;
-  $term = $tree[0];
-  while ($term->id() != $form['#first_tid']) {
-    if ($term->parent->value == 0 && $term->weight->value != $weight) {
-      $term->weight->value = $weight;
-      $changed_terms[$term->id()] = $term;
-    }
-    $weight++;
-    $hierarchy = $term->parent->value != 0 ? TAXONOMY_HIERARCHY_SINGLE : $hierarchy;
-    $term = $tree[$weight];
-  }
-
-  // Renumber the current page weights and assign any new parents.
-  $level_weights = array();
-  foreach ($form_state['values']['terms'] as $tid => $values) {
-    if (isset($form['terms'][$tid]['#term'])) {
-      $term = $form['terms'][$tid]['#term'];
-      // Give terms at the root level a weight in sequence with terms on previous pages.
-      if ($values['term']['parent'] == 0 && $term->weight->value != $weight) {
-        $term->weight->value = $weight;
-        $changed_terms[$term->id()] = $term;
-      }
-      // Terms not at the root level can safely start from 0 because they're all on this page.
-      elseif ($values['term']['parent'] > 0) {
-        $level_weights[$values['term']['parent']] = isset($level_weights[$values['term']['parent']]) ? $level_weights[$values['term']->parent->value] + 1 : 0;
-        if ($level_weights[$values['term']['parent']] != $term->weight->value) {
-          $term->weight->value = $level_weights[$values['term']['parent']];
-          $changed_terms[$term->id()] = $term;
-        }
-      }
-      // Update any changed parents.
-      if ($values['term']['parent'] != $term->parent->value) {
-        $term->parent->value = $values['term']['parent'];
-        $changed_terms[$term->id()] = $term;
-      }
-      $hierarchy = $term->parent->value != 0 ? TAXONOMY_HIERARCHY_SINGLE : $hierarchy;
-      $weight++;
-    }
-  }
-
-  // Build a list of all terms that need to be updated on following pages.
-  for ($weight; $weight < count($tree); $weight++) {
-    $term = $tree[$weight];
-    if ($term->parent->value == 0 && $term->weight->value != $weight) {
-      $term->parent->value = $term->parent->value;
-      $term->weight->value = $weight;
-      $changed_terms[$term->id()] = $term;
-    }
-    $hierarchy = $term->parent->value != 0 ? TAXONOMY_HIERARCHY_SINGLE : $hierarchy;
-  }
-
-  // Save all updated terms.
-  foreach ($changed_terms as $term) {
-    $term->save();
-  }
-
-  // Update the vocabulary hierarchy to flat or single hierarchy.
-  if ($vocabulary->hierarchy != $hierarchy) {
-    $vocabulary->hierarchy = $hierarchy;
-    $vocabulary->save();
-  }
-  drupal_set_message(t('The configuration options have been saved.'));
-}
diff --git a/core/modules/taxonomy/taxonomy.module b/core/modules/taxonomy/taxonomy.module
index 306d6d0ec4ee4106b8c53545757f041041084e4f..4a78f37e58caed884ecee3ecd575ace5e58b7d28 100644
--- a/core/modules/taxonomy/taxonomy.module
+++ b/core/modules/taxonomy/taxonomy.module
@@ -287,13 +287,9 @@ function taxonomy_menu() {
   );
 
   $items['admin/structure/taxonomy/manage/%taxonomy_vocabulary'] = array(
+    'route_name' => 'taxonomy_overview_terms',
     'title callback' => 'entity_page_label',
     'title arguments' => array(4),
-    'page callback' => 'drupal_get_form',
-    'page arguments' => array('taxonomy_overview_terms', 4),
-    'access callback' => 'entity_page_access',
-    'access arguments' => array(4, 'view'),
-    'file' => 'taxonomy.admin.inc',
   );
   $items['admin/structure/taxonomy/manage/%taxonomy_vocabulary/list'] = array(
     'title' => 'List',
@@ -306,7 +302,6 @@ function taxonomy_menu() {
     'access callback' => 'entity_page_access',
     'access arguments' => array(4, 'update'),
     'type' => MENU_LOCAL_TASK,
-    'file' => 'taxonomy.admin.inc',
   );
   $items['admin/structure/taxonomy/%taxonomy_vocabulary/delete'] = array(
     'title' => 'Delete',
diff --git a/core/modules/taxonomy/taxonomy.routing.yml b/core/modules/taxonomy/taxonomy.routing.yml
index 0245c19d77946b0d61bed72bb0318ec789641467..6a05d26c59668134228fbd8aeb19c8a010dbd3b5 100644
--- a/core/modules/taxonomy/taxonomy.routing.yml
+++ b/core/modules/taxonomy/taxonomy.routing.yml
@@ -53,3 +53,11 @@ taxonomy_autocomplete:
     _controller: '\Drupal\taxonomy\Controller\TermAutocompleteController::autocomplete'
   requirements:
     _permission: 'access content'
+
+taxonomy_overview_terms:
+  pattern: 'admin/structure/taxonomy/manage/{taxonomy_vocabulary}'
+  defaults:
+    _form: 'Drupal\taxonomy\Form\OverviewTerms'
+  requirements:
+    _entity_access: 'taxonomy_vocabulary.view'
+