diff --git a/typo3/sysext/core/Documentation/Changelog/master/Feature-69439-EnhanceSQLQueryReductionInPageTreeInWorkspaces.rst b/typo3/sysext/core/Documentation/Changelog/master/Feature-69439-EnhanceSQLQueryReductionInPageTreeInWorkspaces.rst
new file mode 100644
index 0000000000000000000000000000000000000000..4f8739249191b1fe286a34984213f03527e0962a
--- /dev/null
+++ b/typo3/sysext/core/Documentation/Changelog/master/Feature-69439-EnhanceSQLQueryReductionInPageTreeInWorkspaces.rst
@@ -0,0 +1,28 @@
+========================================================================
+Feature: #69439 - Enhance SQL query reduction in page tree in workspaces
+========================================================================
+
+Description
+===========
+
+The process of determining whether a page has workspace versions can be
+extended by custom application code utilizing hooks. This way, the meaning
+of having versions can be modified by hooks further. For instance the
+default behavior of the TYPO3 core is to create a workspace version
+record on persisting the same record in the backend - without any
+actual changes to the data model.
+
++ $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['TYPO3\\CMS\\Workspaces\\Service\\WorkspaceService']['hasPageRecordVersions']
+  + $parameters['workspaceId']: The submitted workspace ID
+  + $parameters['pageId']: The submitted page ID
+  + $parameters['versionsOnPageCache']: Reference to the state array
++ $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['TYPO3\\CMS\\Workspaces\\Service\\WorkspaceService']['fetchPagesWithVersionsInTable']
+  + $parameters['workspaceId']: The submitted workspace ID
+  + $parameters['pageId']: The submitted page ID
+  + $parameters['pagesWithVersionsInTable']: Reference to the state array
+
+
+Impact
+======
+
+The hooks introduce the possibility to modify the determined results - only if those hooks are used.
\ No newline at end of file
diff --git a/typo3/sysext/workspaces/Classes/Service/WorkspaceService.php b/typo3/sysext/workspaces/Classes/Service/WorkspaceService.php
index 812cf1736d5f0941e13c8263cd7b5258d28aa21d..204abfd02f054393452d39f47025e0adfff01124 100644
--- a/typo3/sysext/workspaces/Classes/Service/WorkspaceService.php
+++ b/typo3/sysext/workspaces/Classes/Service/WorkspaceService.php
@@ -35,6 +35,11 @@ class WorkspaceService implements SingletonInterface
      */
     protected $versionsOnPageCache = array();
 
+    /**
+     * @var array
+     */
+    protected $pagesWithVersionsInTable = array();
+
     const TABLE_WORKSPACE = 'sys_workspace';
     const SELECT_ALL_WORKSPACES = -98;
     const LIVE_WORKSPACE_ID = 0;
@@ -719,56 +724,145 @@ class WorkspaceService implements SingletonInterface
     }
 
     /**
-     * Checks if a page has record versions according to a given workspace
+     * Determines whether a page has workspace versions.
      *
-     * @param int $workspace
+     * @param int $workspaceId
      * @param int $pageId
      * @return bool
      */
-    public function hasPageRecordVersions($workspace, $pageId)
+    public function hasPageRecordVersions($workspaceId, $pageId)
     {
-        $workspace = (int)$workspace;
-        $pageId = (int)$pageId;
-        if ($workspace === 0) {
+        if ((int)$workspaceId === 0 || (int)$pageId === 0) {
             return false;
         }
 
-        if (isset($this->versionsOnPageCache[$pageId][$workspace])) {
-            return $this->versionsOnPageCache[$pageId][$workspace];
+        if (isset($this->versionsOnPageCache[$workspaceId][$pageId])) {
+            return $this->versionsOnPageCache[$workspaceId][$pageId];
         }
 
-        if (!empty($this->versionsOnPageCache)) {
-            return false;
+        $this->versionsOnPageCache[$workspaceId][$pageId] = false;
+
+        foreach ($GLOBALS['TCA'] as $tableName => $tableConfiguration) {
+            if ($tableName === 'pages' || empty($tableConfiguration['ctrl']['versioningWS'])) {
+                continue;
+            }
+
+            $pages = $this->fetchPagesWithVersionsInTable($workspaceId, $tableName);
+            // Early break on first match
+            if (!empty($pages[(string)$pageId])) {
+                $this->versionsOnPageCache[$workspaceId][$pageId] = true;
+                break;
+            }
+        }
+
+        if (!empty($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['TYPO3\\CMS\\Workspaces\\Service\\WorkspaceService']['hasPageRecordVersions'])
+            && is_array($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['TYPO3\\CMS\\Workspaces\\Service\\WorkspaceService']['hasPageRecordVersions'])) {
+            $parameters = array(
+                'workspaceId' => $workspaceId,
+                'pageId' => $pageId,
+                'versionsOnPageCache' => &$this->versionsOnPageCache,
+            );
+            foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['TYPO3\\CMS\\Workspaces\\Service\\WorkspaceService']['hasPageRecordVersions'] as $hookFunction) {
+                GeneralUtility::callUserFunction($hookFunction, $parameters, $this);
+            }
         }
 
-        $this->versionsOnPageCache[$pageId][$workspace] = false;
+        return $this->versionsOnPageCache[$workspaceId][$pageId];
+    }
+
+    /**
+     * Gets all pages that have workspace versions per table.
+     *
+     * Result:
+     * [
+     *   'tt_content' => [
+     *     1 => 1,
+     *     11 => 11,
+     *     13 => 13,
+     *     15 => 15
+     *   ],
+     *   'tx_something => [
+     *     15 => 15,
+     *     11 => 11,
+     *     21 => 21
+     *   ],
+     * ]
+     *
+     * @param int $workspaceId
+     * @return array
+     */
+    public function getPagesWithVersionsInTable($workspaceId)
+    {
         foreach ($GLOBALS['TCA'] as $tableName => $tableConfiguration) {
             if ($tableName === 'pages' || empty($tableConfiguration['ctrl']['versioningWS'])) {
                 continue;
             }
+
+            $this->fetchPagesWithVersionsInTable($workspaceId, $tableName);
+        }
+
+        return $this->pagesWithVersionsInTable[$workspaceId];
+    }
+
+    /**
+     * Gets all pages that have workspace versions in a particular table.
+     *
+     * Result:
+     * [
+     *   1 => 1,
+     *   11 => 11,
+     *   13 => 13,
+     *   15 => 15
+     * ],
+     *
+     * @param int $workspaceId
+     * @param string $tableName
+     * @return array
+     */
+    protected function fetchPagesWithVersionsInTable($workspaceId, $tableName)
+    {
+        if ((int)$workspaceId === 0) {
+            return array();
+        }
+
+        if (!isset($this->pagesWithVersionsInTable[$workspaceId])) {
+            $this->pagesWithVersionsInTable[$workspaceId] = array();
+        }
+
+        if (!isset($this->pagesWithVersionsInTable[$workspaceId][$tableName])) {
+            $this->pagesWithVersionsInTable[$workspaceId][$tableName] = array();
+
             // Consider records that are moved to a different page
             $movePointer = new VersionState(VersionState::MOVE_POINTER);
             $joinStatement = '(A.t3ver_oid=B.uid AND A.t3ver_state<>' . $movePointer
                 . ' OR A.t3ver_oid=B.t3ver_move_id AND A.t3ver_state=' . $movePointer . ')';
 
-            // Select all records from this table in the database from the workspace
-            // This joins the online version with the offline version as tables A and B
-            $records = $this->getDatabaseConnection()->exec_SELECTgetRows(
-                'B.uid as live_uid, B.pid as live_pid, A.uid as offline_uid',
+
+            $pageIds = $this->getDatabaseConnection()->exec_SELECTgetRows(
+                'B.pid AS pageId',
                 $tableName . ' A,' . $tableName . ' B',
-                'A.pid=-1 AND A.t3ver_wsid=' . $workspace . ' AND ' . $joinStatement .
-                BackendUtility::deleteClause($tableName, 'A') . BackendUtility::deleteClause($tableName, 'B'),
-                'live_pid'
+                'A.pid=-1 AND A.t3ver_wsid=' . (int)$workspaceId . ' AND ' . $joinStatement
+                    . BackendUtility::deleteClause($tableName, 'A') . BackendUtility::deleteClause($tableName, 'B'),
+                'pageId', '', '',
+                'pageId'
             );
 
-            if (!empty($records)) {
-                foreach ($records as $record) {
-                    $this->versionsOnPageCache[$record['live_pid']][$workspace] = true;
+            $this->pagesWithVersionsInTable[$workspaceId][$tableName] = $pageIds;
+
+            if (!empty($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['TYPO3\\CMS\\Workspaces\\Service\\WorkspaceService']['fetchPagesWithVersionsInTable'])
+                && is_array($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['TYPO3\\CMS\\Workspaces\\Service\\WorkspaceService']['fetchPagesWithVersionsInTable'])) {
+                $parameters = array(
+                    'workspaceId' => $workspaceId,
+                    'tableName' => $tableName,
+                    'pagesWithVersionsInTable' => &$this->pagesWithVersionsInTable,
+                );
+                foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['TYPO3\\CMS\\Workspaces\\Service\\WorkspaceService']['fetchPagesWithVersionsInTable'] as $hookFunction) {
+                    GeneralUtility::callUserFunction($hookFunction, $parameters, $this);
                 }
             }
         }
 
-        return $this->versionsOnPageCache[$pageId][$workspace];
+        return $this->pagesWithVersionsInTable[$workspaceId][$tableName];
     }
 
     /**