diff --git a/typo3/sysext/core/Classes/DataHandling/DataHandler.php b/typo3/sysext/core/Classes/DataHandling/DataHandler.php
index 1f88baa951fd0aefa3cef1d6cfc04024b43b8e4c..7e8844cdfe437466337bd13c38610e5ea1e7fde9 100644
--- a/typo3/sysext/core/Classes/DataHandling/DataHandler.php
+++ b/typo3/sysext/core/Classes/DataHandling/DataHandler.php
@@ -932,7 +932,7 @@ class DataHandler implements LoggerAwareInterface
      * Processing the data-array
      * Call this function to process the data-array set by start()
      *
-     * @return void|FALSE
+     * @return bool|void
      */
     public function process_datamap()
     {
@@ -1013,12 +1013,11 @@ class DataHandler implements LoggerAwareInterface
                     $fieldArray = $this->newFieldArray($table);
                     // A pid must be set for new records.
                     if (isset($incomingFieldArray['pid'])) {
-                        // $value = the pid
                         $pid_value = $incomingFieldArray['pid'];
                         // Checking and finding numerical pid, it may be a string-reference to another value
-                        $OK = 1;
+                        $canProceed = true;
                         // If a NEW... id
-                        if (strstr($pid_value, 'NEW')) {
+                        if (strpos($pid_value, 'NEW') !== false) {
                             if ($pid_value[0] === '-') {
                                 $negFlag = -1;
                                 $pid_value = substr($pid_value, 1);
@@ -1032,35 +1031,12 @@ class DataHandler implements LoggerAwareInterface
                                 }
                                 $pid_value = (int)($negFlag * $this->substNEWwithIDs[$pid_value]);
                             } else {
-                                $OK = 0;
+                                $canProceed = false;
                             }
                         }
                         $pid_value = (int)$pid_value;
-                        // The $pid_value is now the numerical pid at this point
-                        if ($OK) {
-                            $sortRow = $GLOBALS['TCA'][$table]['ctrl']['sortby'];
-                            // Points to a page on which to insert the element, possibly in the top of the page
-                            if ($pid_value >= 0) {
-                                // If this table is sorted we better find the top sorting number
-                                if ($sortRow) {
-                                    $fieldArray[$sortRow] = $this->getSortNumber($table, 0, $pid_value);
-                                }
-                                // The numerical pid is inserted in the data array
-                                $fieldArray['pid'] = $pid_value;
-                            } else {
-                                // points to another record before ifself
-                                // If this table is sorted we better find the top sorting number
-                                if ($sortRow) {
-                                    // Because $pid_value is < 0, getSortNumber returns an array
-                                    $tempArray = $this->getSortNumber($table, 0, $pid_value);
-                                    $fieldArray['pid'] = $tempArray['pid'];
-                                    $fieldArray[$sortRow] = $tempArray['sortNumber'];
-                                } else {
-                                    // Here we fetch the PID of the record that we point to...
-                                    $tempdata = $this->recordInfo($table, abs($pid_value), 'pid');
-                                    $fieldArray['pid'] = $tempdata['pid'];
-                                }
-                            }
+                        if ($canProceed) {
+                            $fieldArray = $this->resolveSortingAndPidForNewRecord($table, $pid_value, $fieldArray);
                         }
                     }
                     $theRealPid = $fieldArray['pid'];
@@ -1319,6 +1295,43 @@ class DataHandler implements LoggerAwareInterface
         }
     }
 
+    /**
+     * Sets the "sorting" DB field and the "pid" field of an incoming record that should be added (NEW1234)
+     * depending on the record that should be added or where it should be added.
+     *
+     * This method is called from process_datamap()
+     *
+     * @param string $table the table name of the record to insert
+     * @param int $pid the real PID (numeric) where the record should be
+     * @param array $fieldArray field+value pairs to add
+     * @return array the modified field array
+     */
+    protected function resolveSortingAndPidForNewRecord(string $table, int $pid, array $fieldArray): array
+    {
+        $sortRow = $GLOBALS['TCA'][$table]['ctrl']['sortby'];
+        // Points to a page on which to insert the element, possibly in the top of the page
+        if ($pid >= 0) {
+            // The numerical pid is inserted in the data array
+            $fieldArray['pid'] = $pid;
+            // If this table is sorted we better find the top sorting number
+            if ($sortRow) {
+                $fieldArray[$sortRow] = $this->getSortNumber($table, 0, $pid);
+            }
+        } elseif ($sortRow) {
+            // Points to another record before itself
+            // If this table is sorted we better find the top sorting number
+            // Because $pid is < 0, getSortNumber() returns an array
+            $sortingInfo = $this->getSortNumber($table, 0, $pid);
+            $fieldArray['pid'] = $sortingInfo['pid'];
+            $fieldArray[$sortRow] = $sortingInfo['sortNumber'];
+        } else {
+            // Here we fetch the PID of the record that we point to
+            $record = $this->recordInfo($table, abs($pid), 'pid');
+            $fieldArray['pid'] = $record['pid'];
+        }
+        return $fieldArray;
+    }
+
     /**
      * Fix shadowing of data in case we are editing an offline version of a live "New" placeholder record:
      *
@@ -3300,7 +3313,7 @@ class DataHandler implements LoggerAwareInterface
         }
         /** @var $copyTCE DataHandler */
         $copyTCE = $this->getLocalTCE();
-        $copyTCE->start($pasteDatamap, '', $this->BE_USER);
+        $copyTCE->start($pasteDatamap, [], $this->BE_USER);
         $copyTCE->process_datamap();
         $this->errorLog = array_merge($this->errorLog, $copyTCE->errorLog);
         unset($copyTCE);
@@ -3477,25 +3490,14 @@ class DataHandler implements LoggerAwareInterface
         // Initialize:
         $uid = (int)$uid;
         $destPid = (int)$destPid;
-        // Finding list of tables to copy.
-        // These are the tables, the user may modify
-        $copyTablesArray = $this->admin ? $this->compileAdminTables() : explode(',', $this->BE_USER->groupData['tables_modify']);
-        // If not all tables are allowed then make a list of allowed tables: That is the tables that figure in both allowed tables AND the copyTable-list
-        if (!strstr($this->copyWhichTables, '*')) {
-            $copyWhichTablesArray = array_flip(GeneralUtility::trimExplode(',', $this->copyWhichTables . ',pages'));
-            foreach ($copyTablesArray as $k => $table) {
-                // Pages are always going...
-                if (!$table || !isset($copyWhichTablesArray[$table])) {
-                    unset($copyTablesArray[$k]);
-                }
-            }
-        }
-        $copyTablesArray = array_unique($copyTablesArray);
+
+        $copyTablesAlongWithPage = $this->getAllowedTablesToCopyWhenCopyingAPage();
         // Begin to copy pages if we're allowed to:
-        if ($this->admin || in_array('pages', $copyTablesArray, true)) {
-            // Copy this page we're on. And set first-flag (this will trigger that the record is hidden if that is configured)!
-            $theNewRootID = $this->copySpecificPage($uid, $destPid, $copyTablesArray, 1);
-            // If we're going to copy recursively...:
+        if ($this->admin || in_array('pages', $copyTablesAlongWithPage, true)) {
+            // Copy this page we're on. And set first-flag (this will trigger that the record is hidden if that is configured)
+            // This method also copies the localizations of a page
+            $theNewRootID = $this->copySpecificPage($uid, $destPid, $copyTablesAlongWithPage, true);
+            // If we're going to copy recursively
             if ($theNewRootID && $this->copyTree) {
                 // Get ALL subpages to copy (read-permissions are respected!):
                 $CPtable = $this->int_pageTreeInfo([], $uid, (int)$this->copyTree, $theNewRootID);
@@ -3503,7 +3505,7 @@ class DataHandler implements LoggerAwareInterface
                 foreach ($CPtable as $thePageUid => $thePagePid) {
                     $newPid = $this->copyMappingArray['pages'][$thePagePid];
                     if (isset($newPid)) {
-                        $this->copySpecificPage($thePageUid, $newPid, $copyTablesArray);
+                        $this->copySpecificPage($thePageUid, $newPid, $copyTablesAlongWithPage);
                     } else {
                         $this->log('pages', $uid, 5, 0, 1, 'Something went wrong during copying branch');
                         break;
@@ -3515,6 +3517,36 @@ class DataHandler implements LoggerAwareInterface
         }
     }
 
+    /**
+     * Compile a list of tables that should be copied along when a page is about to be copied.
+     *
+     * First, get the list that the user is allowed to modify (all if admin),
+     * and then check against a possible limitation within "DataHandler->copyWhichTables" if not set to "*"
+     * to limit the list further down
+     *
+     * @return array
+     */
+    protected function getAllowedTablesToCopyWhenCopyingAPage(): array
+    {
+        // Finding list of tables to copy.
+        // These are the tables, the user may modify
+        $copyTablesArray = $this->admin ? $this->compileAdminTables() : explode(',', $this->BE_USER->groupData['tables_modify']);
+        // If not all tables are allowed then make a list of allowed tables.
+        // That is the tables that figure in both allowed tables AND the copyTable-list
+        if (strpos($this->copyWhichTables, '*') === false) {
+            $definedTablesToCopy = GeneralUtility::trimExplode(',', $this->copyWhichTables, true);
+            // Pages are always allowed
+            $definedTablesToCopy[] = 'pages';
+            $definedTablesToCopy = array_flip($definedTablesToCopy);
+            foreach ($copyTablesArray as $k => $table) {
+                if (!$table || !isset($definedTablesToCopy[$table])) {
+                    unset($copyTablesArray[$k]);
+                }
+            }
+        }
+        $copyTablesArray = array_unique($copyTablesArray);
+        return $copyTablesArray;
+    }
     /**
      * Copying a single page ($uid) to $destPid and all tables in the array copyTablesArray.
      *
@@ -5287,7 +5319,7 @@ class DataHandler implements LoggerAwareInterface
     {
         $uid = (int)$uid;
         if ($uid) {
-            $tableNames = array_keys($GLOBALS['TCA']);
+            $tableNames = $this->compileAdminTables();
             foreach ($tableNames as $table) {
                 if ($table !== 'pages') {
                     $queryBuilder = GeneralUtility::makeInstance(ConnectionPool::class)
@@ -6712,31 +6744,32 @@ class DataHandler implements LoggerAwareInterface
             // Not a number. Probably a new page
             return false;
         }
-        $allowedTableList = isset($GLOBALS['PAGES_TYPES'][$doktype]['allowedTables']) ? $GLOBALS['PAGES_TYPES'][$doktype]['allowedTables'] : $GLOBALS['PAGES_TYPES']['default']['allowedTables'];
-        $allowedArray = GeneralUtility::trimExplode(',', $allowedTableList, true);
-        // If all tables is OK the return TRUE
-        if (strstr($allowedTableList, '*')) {
-            // OK...
+        $allowedTableList = $GLOBALS['PAGES_TYPES'][$doktype]['allowedTables'] ?? $GLOBALS['PAGES_TYPES']['default']['allowedTables'];
+        // If all tables are allowed, return early
+        if (strpos($allowedTableList, '*') !== false) {
             return false;
         }
+        $allowedArray = GeneralUtility::trimExplode(',', $allowedTableList, true);
         $tableList = [];
-        foreach ($GLOBALS['TCA'] as $table => $_) {
+        $allTableNames = $this->compileAdminTables();
+        foreach ($allTableNames as $table) {
             // If the table is not in the allowed list, check if there are records...
-            if (!in_array($table, $allowedArray, true)) {
-                $queryBuilder = GeneralUtility::makeInstance(ConnectionPool::class)->getQueryBuilderForTable($table);
-                $queryBuilder->getRestrictions()->removeAll();
-                $count = $queryBuilder
-                    ->count('uid')
-                    ->from($table)
-                    ->where($queryBuilder->expr()->eq(
-                        'pid',
-                        $queryBuilder->createNamedParameter($page_uid, \PDO::PARAM_INT)
-                    ))
-                    ->execute()
-                    ->fetchColumn(0);
-                if ($count) {
-                    $tableList[] = $table;
-                }
+            if (in_array($table, $allowedArray, true)) {
+                continue;
+            }
+            $queryBuilder = GeneralUtility::makeInstance(ConnectionPool::class)->getQueryBuilderForTable($table);
+            $queryBuilder->getRestrictions()->removeAll();
+            $count = $queryBuilder
+                ->count('uid')
+                ->from($table)
+                ->where($queryBuilder->expr()->eq(
+                    'pid',
+                    $queryBuilder->createNamedParameter($page_uid, \PDO::PARAM_INT)
+                ))
+                ->execute()
+                ->fetchColumn(0);
+            if ($count) {
+                $tableList[] = $table;
             }
         }
         return implode(',', $tableList);
@@ -8197,7 +8230,8 @@ class DataHandler implements LoggerAwareInterface
         }
 
         if (!empty($pageIds)) {
-            foreach ($GLOBALS['TCA'] as $table => $_) {
+            $tableNames = $this->compileAdminTables();
+            foreach ($tableNames as $table) {
                 $query = GeneralUtility::makeInstance(ConnectionPool::class)->getQueryBuilderForTable($table);
                 $query->getRestrictions()
                     ->removeAll()