From 1c66f7ecaf555e499663c4e81b96d63d803322f7 Mon Sep 17 00:00:00 2001
From: Alex Pott <alex.a.pott@googlemail.com>
Date: Sat, 20 Apr 2024 16:53:47 +0100
Subject: [PATCH] Issue #3422600 by smustgrave, Spokje, quietone, catch,
 larowlan, lauriii, Wim Leers: Remove Tour module

---
 core/.phpstan-baseline.php                    |  10 -
 core/MAINTAINERS.txt                          |   3 -
 .../optional/tour.tour.block-layout.yml       |  39 --
 .../optional/tour.tour.language-add.yml       |  31 --
 .../optional/tour.tour.language-edit.yml      |  45 --
 .../config/optional/tour.tour.language.yml    |  52 ---
 .../tour/config/optional/tour.tour.locale.yml |  66 ---
 .../config/optional/tour.tour.views-ui.yml    |  85 ----
 .../tour/config/schema/tour.schema.yml        |  67 ---
 core/modules/tour/css/tour.module.css         | 153 -------
 .../tour/help_topics/tour.overview.html.twig  |  16 -
 core/modules/tour/js/tour.js                  | 414 ------------------
 core/modules/tour/src/Annotation/Tip.php      |  39 --
 core/modules/tour/src/Entity/Tour.php         | 189 --------
 .../Plugin/HelpSection/TourHelpSection.php    | 133 ------
 .../src/Plugin/tour/tip/TipPluginText.php     |  72 ---
 core/modules/tour/src/TipPluginBase.php       | 112 -----
 core/modules/tour/src/TipPluginInterface.php  | 101 -----
 core/modules/tour/src/TipPluginManager.php    |  37 --
 .../modules/tour/src/TipsPluginCollection.php |  26 --
 .../tour/src/TourAccessControlHandler.php     |  28 --
 core/modules/tour/src/TourInterface.php       |  65 ---
 .../tour/src/TourTipPluginInterface.php       |  20 -
 core/modules/tour/src/TourViewBuilder.php     | 139 ------
 .../Functional/Block/BlockLayoutTourTest.php  |  54 ---
 .../tour/tests/src/Functional/GenericTest.php |  15 -
 .../tests/src/Functional/Jsonapi/TourTest.php | 148 -------
 .../Functional/Language/LanguageTourTest.php  |  73 ---
 .../Locale/LocaleTranslateStringTourTest.php  |  63 ---
 .../src/Functional/Rest/TourJsonAnonTest.php  |  32 --
 .../Functional/Rest/TourJsonBasicAuthTest.php |  42 --
 .../Functional/Rest/TourJsonCookieTest.php    |  37 --
 .../Functional/Rest/TourResourceTestBase.php  | 118 -----
 .../src/Functional/Rest/TourXmlAnonTest.php   |  34 --
 .../Functional/Rest/TourXmlBasicAuthTest.php  |  44 --
 .../src/Functional/Rest/TourXmlCookieTest.php |  39 --
 .../src/Functional/TourCacheTagsTest.php      |  84 ----
 .../tests/src/Functional/TourHelpPageTest.php | 155 -------
 .../tour/tests/src/Functional/TourTest.php    | 311 -------------
 .../tests/src/Functional/TourTestBase.php     |  79 ----
 .../tests/src/Functional/TourTestBasic.php    |  75 ----
 .../Functional/ViewsUi/ViewsUITourTest.php    | 123 ------
 .../TourJavascriptTest.php                    | 134 ------
 .../tour/tests/src/Kernel/TourPluginTest.php  |  48 --
 .../tests/src/Kernel/TourTipLegacyTest.php    |  25 --
 .../tests/src/Kernel/TourValidationTest.php   |  51 ---
 .../tour/tests/src/Unit/Entity/TourTest.php   | 142 ------
 .../tour/tests/src/Unit/TipPluginBaseTest.php |  33 --
 .../language/it/tour.tour.tour-test.yml       |   7 -
 .../config/install/tour.tour.tour-test-2.yml  |  16 -
 .../config/install/tour.tour.tour-test.yml    |  39 --
 .../config/schema/tour_test.schema.yml        |   9 -
 .../src/Controller/TourTestController.php     |  74 ----
 .../src/Plugin/tour/tip/TipPluginImage.php    |  87 ----
 .../Plugin/tour/tip/TipPluginImageLegacy.php  |  84 ----
 .../Plugin/tour/tip/TipPluginTextLegacy.php   |  96 ----
 .../tour/tests/tour_test/tour_test.info.yml   |   7 -
 .../tour_test/tour_test.links.action.yml      |   5 -
 .../tour/tests/tour_test/tour_test.module     |  37 --
 .../tests/tour_test/tour_test.routing.yml     |  30 --
 core/modules/tour/tour.api.php                |  61 ---
 core/modules/tour/tour.info.yml               |   7 -
 core/modules/tour/tour.libraries.yml          |  18 -
 core/modules/tour/tour.module                 | 117 -----
 core/modules/tour/tour.permissions.yml        |   2 -
 core/modules/tour/tour.services.yml           |   4 -
 core/themes/claro/claro.info.yml              |   2 -
 core/themes/claro/claro.libraries.yml         |   6 -
 core/themes/claro/css/theme/tour.theme.css    | 104 -----
 .../claro/css/theme/tour.theme.pcss.css       |  88 ----
 .../olivero/css/components/site-header.css    |   2 +-
 .../css/components/site-header.pcss.css       |   2 +-
 core/themes/stable9/css/tour/tour.module.css  | 173 --------
 core/themes/stable9/js/tour.js                | 139 ------
 core/themes/stable9/stable9.info.yml          |   9 -
 core/themes/stable9/stable9.libraries.yml     |   5 -
 76 files changed, 2 insertions(+), 5129 deletions(-)
 delete mode 100644 core/modules/tour/config/optional/tour.tour.block-layout.yml
 delete mode 100644 core/modules/tour/config/optional/tour.tour.language-add.yml
 delete mode 100644 core/modules/tour/config/optional/tour.tour.language-edit.yml
 delete mode 100644 core/modules/tour/config/optional/tour.tour.language.yml
 delete mode 100644 core/modules/tour/config/optional/tour.tour.locale.yml
 delete mode 100644 core/modules/tour/config/optional/tour.tour.views-ui.yml
 delete mode 100644 core/modules/tour/config/schema/tour.schema.yml
 delete mode 100644 core/modules/tour/css/tour.module.css
 delete mode 100644 core/modules/tour/help_topics/tour.overview.html.twig
 delete mode 100644 core/modules/tour/js/tour.js
 delete mode 100644 core/modules/tour/src/Annotation/Tip.php
 delete mode 100644 core/modules/tour/src/Entity/Tour.php
 delete mode 100644 core/modules/tour/src/Plugin/HelpSection/TourHelpSection.php
 delete mode 100644 core/modules/tour/src/Plugin/tour/tip/TipPluginText.php
 delete mode 100644 core/modules/tour/src/TipPluginBase.php
 delete mode 100644 core/modules/tour/src/TipPluginInterface.php
 delete mode 100644 core/modules/tour/src/TipPluginManager.php
 delete mode 100644 core/modules/tour/src/TipsPluginCollection.php
 delete mode 100644 core/modules/tour/src/TourAccessControlHandler.php
 delete mode 100644 core/modules/tour/src/TourInterface.php
 delete mode 100644 core/modules/tour/src/TourTipPluginInterface.php
 delete mode 100644 core/modules/tour/src/TourViewBuilder.php
 delete mode 100644 core/modules/tour/tests/src/Functional/Block/BlockLayoutTourTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/GenericTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/Jsonapi/TourTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/Language/LanguageTourTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/Locale/LocaleTranslateStringTourTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/Rest/TourJsonAnonTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/Rest/TourJsonBasicAuthTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/Rest/TourJsonCookieTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/Rest/TourResourceTestBase.php
 delete mode 100644 core/modules/tour/tests/src/Functional/Rest/TourXmlAnonTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/Rest/TourXmlBasicAuthTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/Rest/TourXmlCookieTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/TourCacheTagsTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/TourHelpPageTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/TourTest.php
 delete mode 100644 core/modules/tour/tests/src/Functional/TourTestBase.php
 delete mode 100644 core/modules/tour/tests/src/Functional/TourTestBasic.php
 delete mode 100644 core/modules/tour/tests/src/Functional/ViewsUi/ViewsUITourTest.php
 delete mode 100644 core/modules/tour/tests/src/FunctionalJavascript/TourJavascriptTest.php
 delete mode 100644 core/modules/tour/tests/src/Kernel/TourPluginTest.php
 delete mode 100644 core/modules/tour/tests/src/Kernel/TourTipLegacyTest.php
 delete mode 100644 core/modules/tour/tests/src/Kernel/TourValidationTest.php
 delete mode 100644 core/modules/tour/tests/src/Unit/Entity/TourTest.php
 delete mode 100644 core/modules/tour/tests/src/Unit/TipPluginBaseTest.php
 delete mode 100644 core/modules/tour/tests/tour_test/config/install/language/it/tour.tour.tour-test.yml
 delete mode 100644 core/modules/tour/tests/tour_test/config/install/tour.tour.tour-test-2.yml
 delete mode 100644 core/modules/tour/tests/tour_test/config/install/tour.tour.tour-test.yml
 delete mode 100644 core/modules/tour/tests/tour_test/config/schema/tour_test.schema.yml
 delete mode 100644 core/modules/tour/tests/tour_test/src/Controller/TourTestController.php
 delete mode 100644 core/modules/tour/tests/tour_test/src/Plugin/tour/tip/TipPluginImage.php
 delete mode 100644 core/modules/tour/tests/tour_test/src/Plugin/tour/tip/TipPluginImageLegacy.php
 delete mode 100644 core/modules/tour/tests/tour_test/src/Plugin/tour/tip/TipPluginTextLegacy.php
 delete mode 100644 core/modules/tour/tests/tour_test/tour_test.info.yml
 delete mode 100644 core/modules/tour/tests/tour_test/tour_test.links.action.yml
 delete mode 100644 core/modules/tour/tests/tour_test/tour_test.module
 delete mode 100644 core/modules/tour/tests/tour_test/tour_test.routing.yml
 delete mode 100644 core/modules/tour/tour.api.php
 delete mode 100644 core/modules/tour/tour.info.yml
 delete mode 100644 core/modules/tour/tour.libraries.yml
 delete mode 100644 core/modules/tour/tour.module
 delete mode 100644 core/modules/tour/tour.permissions.yml
 delete mode 100644 core/modules/tour/tour.services.yml
 delete mode 100644 core/themes/claro/css/theme/tour.theme.css
 delete mode 100644 core/themes/claro/css/theme/tour.theme.pcss.css
 delete mode 100644 core/themes/stable9/css/tour/tour.module.css
 delete mode 100644 core/themes/stable9/js/tour.js

diff --git a/core/.phpstan-baseline.php b/core/.phpstan-baseline.php
index 2094ecfc6e36..3b5796557a70 100644
--- a/core/.phpstan-baseline.php
+++ b/core/.phpstan-baseline.php
@@ -1563,16 +1563,6 @@
 	'count' => 1,
 	'path' => __DIR__ . '/modules/taxonomy/src/VocabularyForm.php',
 ];
-$ignoreErrors[] = [
-	'message' => '#^Method Drupal\\\\tour\\\\TipPluginBase\\:\\:get\\(\\) should return string but return statement is missing\\.$#',
-	'count' => 1,
-	'path' => __DIR__ . '/modules/tour/src/TipPluginBase.php',
-];
-$ignoreErrors[] = [
-	'message' => '#^Variable \\$location on left side of \\?\\? always exists and is not nullable\\.$#',
-	'count' => 1,
-	'path' => __DIR__ . '/modules/tour/src/TipPluginBase.php',
-];
 $ignoreErrors[] = [
 	'message' => '#^Variable \\$violation_messages might not be defined\\.$#',
 	'count' => 1,
diff --git a/core/MAINTAINERS.txt b/core/MAINTAINERS.txt
index 3330080db449..bf39b43399f5 100644
--- a/core/MAINTAINERS.txt
+++ b/core/MAINTAINERS.txt
@@ -401,9 +401,6 @@ Token
 Toolbar
 - Théodore Biadala 'nod_' https://www.drupal.org/u/nod_
 
-Tour
-- Nick Schuch 'nick_schuch' https://www.drupal.org/u/nick_schuch
-
 Transliteration
 - Andrei Mateescu 'amateescu' https://www.drupal.org/u/amateescu
 
diff --git a/core/modules/tour/config/optional/tour.tour.block-layout.yml b/core/modules/tour/config/optional/tour.tour.block-layout.yml
deleted file mode 100644
index 8cfdfb316518..000000000000
--- a/core/modules/tour/config/optional/tour.tour.block-layout.yml
+++ /dev/null
@@ -1,39 +0,0 @@
-langcode: en
-status: true
-dependencies:
-  module:
-    - block
-id: block-layout
-label: 'Block Layout Page'
-module: block
-routes:
-  -
-    route_name: block.admin_display
-tips:
-  block-layout:
-    id: block-layout
-    plugin: text
-    label: 'Block Layout'
-    weight: 1
-    body: 'Blocks are boxes of content rendered into an area, or region, of a web page that can be displayed in regions (such as footer or sidebar) on your page.'
-  place-block:
-    id: place-block
-    plugin: text
-    label: 'Place Blocks'
-    weight: 2
-    selector: .button--small
-    body: 'Any custom or contributed block can be added to a particular region by clicking on a button Place block. A new block can also be created by clicking on Place Block'
-  block-region:
-    id: block-region
-    plugin: text
-    label: 'Block Region'
-    weight: 3
-    selector: .block-region-select
-    body: 'Assign or change the region of a block by clicking here. A dropdown list with all the regions will appear. You can place one block in multiple regions.'
-  configure-block:
-    id: configure-block
-    plugin: text
-    label: 'Configure Block'
-    weight: 4
-    selector: .dropbutton-widget
-    body: 'By Clicking on "Configure" you can go ahead and edit the contents of the block, deal with the visibility settings and even change the placement of where it is on your theme.'
diff --git a/core/modules/tour/config/optional/tour.tour.language-add.yml b/core/modules/tour/config/optional/tour.tour.language-add.yml
deleted file mode 100644
index faa333d93111..000000000000
--- a/core/modules/tour/config/optional/tour.tour.language-add.yml
+++ /dev/null
@@ -1,31 +0,0 @@
-langcode: en
-status: true
-dependencies:
-  module:
-    - language
-id: language-add
-label: 'Adding languages'
-module: language
-routes:
-  -
-    route_name: language.add
-tips:
-  language-add-overview:
-    id: language-add-overview
-    plugin: text
-    label: 'Adding languages'
-    weight: 1
-    body: '<p>This page provides the ability to add common languages to your site.</p><p>If the desired language is not available, you can add a custom language.</p>'
-  language-add-choose:
-    id: language-add-choose
-    plugin: text
-    label: 'Select language'
-    weight: 2
-    selector: '#edit-predefined-langcode'
-    body: '<p>Choose a language from the list, or choose "Custom language..." at the end of the list.</p><p>Click the "Add language" button when you are done choosing your language.</p><p>When adding a custom language, you will get an additional form where you can provide the name, code, and direction of the language.</p>'
-  language-add-continue:
-    id: language-add-continue
-    plugin: text
-    label: 'Continuing on'
-    weight: 3
-    body: '<p>Now that you have an overview of the "Add languages" feature, you can continue by:<ul><li>Adding a language</li><li>Adding a custom language</li><li><a href="[site:url]admin/config/regional/language">Viewing configured languages</a></li></ul></p>'
diff --git a/core/modules/tour/config/optional/tour.tour.language-edit.yml b/core/modules/tour/config/optional/tour.tour.language-edit.yml
deleted file mode 100644
index 2a50b89e3a8d..000000000000
--- a/core/modules/tour/config/optional/tour.tour.language-edit.yml
+++ /dev/null
@@ -1,45 +0,0 @@
-langcode: en
-status: true
-dependencies:
-  module:
-    - language
-id: language-edit
-label: 'Editing languages'
-module: language
-routes:
-  -
-    route_name: entity.configurable_language.edit_form
-tips:
-  language-edit-overview:
-    id: language-edit-overview
-    plugin: text
-    label: 'Editing languages'
-    weight: 1
-    body: '<p>This page provides the ability to edit a language on your site, including custom languages.</p>'
-  language-edit-langcode:
-    id: language-edit-langcode
-    plugin: text
-    label: 'Language code'
-    weight: 2
-    selector: '#edit-langcode-view'
-    body: '<p>You cannot change the code of a language on the site, since it is used by the system to keep track of the language.</p>'
-  language-edit-label:
-    id: language-edit-label
-    plugin: text
-    label: 'Language name'
-    weight: 3
-    selector: '#edit-label'
-    body: '<p>The language name is used throughout the site for all users and is written in English. Names of built-in languages can be translated using the Interface Translation module, and names of both built-in and custom languages can be translated using the Configuration Translation module.</p>'
-  language-edit-direction:
-    id: language-edit-direction
-    plugin: text
-    label: 'Language direction'
-    weight: 4
-    selector: '#edit-direction--wrapper--description'
-    body: '<p>Choose if the language is a "Left to right" or "Right to left" language.</p><p>Note that not all themes support "Right to left" layouts, so test your theme if you are using "Right to left".</p>'
-  language-edit-continue:
-    id: language-edit-continue
-    plugin: text
-    label: 'Continuing on'
-    weight: 5
-    body: '<p>Now that you have an overview of the "Edit language" feature, you can continue by:<ul><li>Editing a language</li><li><a href="[site:url]admin/config/regional/language">Viewing configured languages</a></li></ul></p>'
diff --git a/core/modules/tour/config/optional/tour.tour.language.yml b/core/modules/tour/config/optional/tour.tour.language.yml
deleted file mode 100644
index 08f2ede4ff48..000000000000
--- a/core/modules/tour/config/optional/tour.tour.language.yml
+++ /dev/null
@@ -1,52 +0,0 @@
-langcode: en
-status: true
-dependencies:
-  module:
-    - language
-id: language
-label: Language
-module: language
-routes:
-  -
-    route_name: entity.configurable_language.collection
-tips:
-  language-overview:
-    id: language-overview
-    plugin: text
-    label: Languages
-    weight: 1
-    body: '<p>The "Languages" page allows you to add, edit, delete, and reorder languages for the site.</p>'
-  language-add:
-    id: language-add
-    plugin: text
-    label: 'Adding languages'
-    weight: 2
-    selector: .button-action
-    body: '<p>To add more languages to your site, click the "Add language" button.</p><p>Added languages will be displayed in the language list and can then be edited or deleted.</p>'
-  language-reorder:
-    id: language-reorder
-    plugin: text
-    label: 'Reordering languages'
-    weight: 3
-    selector: .draggable
-    body: '<p>To reorder the languages on your site, use the drag icons next to each language.</p><p>The order shown here is the display order for language lists on the site such as in the language switcher blocks provided by the Interface Translation and Content Translation modules.</p><p>When you are done with reordering the languages, click the "Save configuration" button for the changes to take effect.</p>'
-  language-default:
-    id: language-default
-    plugin: text
-    label: 'Set a language as default'
-    weight: 4
-    selector: .js-form-item-site-default-language
-    body: '<p>You can change the default language of the site by choosing one of your configured languages as default. The site will use the default language in situations where no choice is made but a language should be set, for example as the language of the displayed interface.</p>'
-  language-operations:
-    id: language-operations
-    plugin: text
-    label: 'Modifying languages'
-    weight: 5
-    selector: .dropbutton-wrapper
-    body: '<p>Operations are provided for editing and deleting your languages.</p><p>You can edit the name and the direction of the language.</p><p>Deleted languages can be added back at a later time. Deleting a language will remove all interface translations associated with it, and content in this language will be set to be language neutral. Note that you cannot delete the default language of the site.</p>'
-  language-continue:
-    id: language-continue
-    plugin: text
-    label: 'Continuing on'
-    weight: 6
-    body: '<p>Now that you have an overview of the "Languages" page, you can continue by:<ul><li><a href="[site:url]admin/config/regional/language/add">Adding a language</a></li><li>Reordering languages</li><li>Editing a language</li><li>Deleting a language</li></ul></p>'
diff --git a/core/modules/tour/config/optional/tour.tour.locale.yml b/core/modules/tour/config/optional/tour.tour.locale.yml
deleted file mode 100644
index e79a088cfb04..000000000000
--- a/core/modules/tour/config/optional/tour.tour.locale.yml
+++ /dev/null
@@ -1,66 +0,0 @@
-langcode: en
-status: true
-dependencies:
-  module:
-    - locale
-id: locale
-label: Translation
-module: locale
-routes:
-  -
-    route_name: locale.translate_page
-tips:
-  locale-overview:
-    id: locale-overview
-    plugin: text
-    label: 'User interface translation'
-    weight: 1
-    body: 'This page allows you to translate the user interface or modify existing translations. If you have installed your site initially in English, you must first add another language on the <a href="[site:url]admin/config/regional/language">Languages page</a>, in order to use this page.'
-  locale-language:
-    id: locale-language
-    plugin: text
-    label: 'Translation language'
-    weight: 2
-    selector: '#edit-langcode'
-    body: 'Choose the language you want to translate.'
-  locale-search:
-    id: locale-search
-    plugin: text
-    label: Search
-    weight: 3
-    selector: '#edit-string'
-    body: 'Enter the specific word or sentence you want to translate, you can also write just a part of a word.'
-  locale-filter:
-    id: locale-filter
-    plugin: text
-    label: 'Filter the search'
-    weight: 4
-    selector: '#edit-translation'
-    body: 'You can search for untranslated strings if you want to translate something that isn''t translated yet. If you want to modify an existing translation, you might want to search only for translated strings.'
-  locale-submit:
-    id: locale-submit
-    plugin: text
-    label: 'Apply your search criteria'
-    weight: 5
-    selector: '#edit-submit'
-    body: 'To apply your search criteria, click on the <em>Filter</em> button.'
-  locale-translate:
-    id: locale-translate
-    plugin: text
-    label: Translate
-    weight: 6
-    selector: .js-form-type-textarea
-    body: 'You can write your own translation in the text fields of the right column. Try to figure out in which context the text will be used in order to translate it in the appropriate way.'
-  locale-validate:
-    id: locale-validate
-    plugin: text
-    label: 'Validate the translation'
-    weight: 7
-    selector: '#edit-submit--2'
-    body: 'When you have finished your translations, click on the <em>Save translations</em> button. You must save your translations, each time before changing the page or making a new search.'
-  locale-continue:
-    id: locale-continue
-    plugin: text
-    label: 'Continuing on'
-    weight: 8
-    body: 'The translations you have made here will be used on your site''s user interface. If you want to use them on another site or modify them on an external translation editor, you can <a href="[site:url]admin/config/regional/translate/export">export them</a> to a .po file and <a href="[site:url]admin/config/regional/translate/import">import them</a> later.'
diff --git a/core/modules/tour/config/optional/tour.tour.views-ui.yml b/core/modules/tour/config/optional/tour.tour.views-ui.yml
deleted file mode 100644
index 644b51f03659..000000000000
--- a/core/modules/tour/config/optional/tour.tour.views-ui.yml
+++ /dev/null
@@ -1,85 +0,0 @@
-langcode: en
-status: true
-dependencies:
-  module:
-    - views_ui
-id: views-ui
-label: 'View edit page'
-module: views_ui
-routes:
-  -
-    route_name: entity.view.edit_form
-  -
-    route_name: entity.view.edit_display_form
-tips:
-  views-main:
-    id: views-main
-    plugin: text
-    label: 'Manage view settings'
-    weight: 1
-    body: 'View or edit the configuration.'
-  views-ui-displays:
-    id: views-ui-displays
-    plugin: text
-    label: 'Displays in this view'
-    weight: 2
-    selector: '#views-display-top'
-    body: 'A display is a way of outputting the results, e.g., as a page or a block. A view can contain multiple displays, which are listed here. The active display is highlighted.'
-  views-ui-view-admin:
-    id: views-ui-view-admin
-    plugin: text
-    label: 'View administration'
-    weight: 3
-    position: right
-    selector: '#views-display-extra-actions'
-    body: 'Perform administrative tasks, including adding a description and creating a clone. Click the drop-down button to view the available options.'
-  views-ui-format:
-    id: views-ui-format
-    plugin: text
-    label: 'Output format'
-    weight: 4
-    selector: .views-ui-display-tab-bucket.format
-    body: 'Choose how to output results. E.g., choose <em>Content</em> to output each item completely, using your configured display settings. Or choose <em>Fields</em>, which allows you to output only specific fields for each result. Additional formats can be added by installing modules to <em>extend</em> Drupal''s base functionality.'
-  views-ui-fields:
-    id: views-ui-fields
-    plugin: text
-    label: Fields
-    weight: 5
-    selector: .views-ui-display-tab-bucket.field
-    body: 'If this view uses fields, they are listed here. You can click on a field to configure it.'
-  views-ui-filter:
-    id: views-ui-filter
-    plugin: text
-    label: 'Filter your view'
-    weight: 6
-    selector: .views-ui-display-tab-bucket.filter
-    body: 'Add filters to limit the results in the output. E.g., to only show content that is <em>published</em>, you would add a filter for <em>Published</em> and select <em>Yes</em>.'
-  views-ui-filter-operations:
-    id: views-ui-filter-operations
-    plugin: text
-    label: 'Filter actions'
-    weight: 7
-    selector: '.views-ui-display-tab-bucket.filter .dropbutton-widget'
-    body: 'Add, rearrange or remove filters.'
-  views-ui-sorts:
-    id: views-ui-sorts
-    plugin: text
-    label: 'Sort Criteria'
-    weight: 8
-    selector: .views-ui-display-tab-bucket.sort
-    body: 'Control the order in which the results are output. Click on an active sort rule to configure it.'
-  views-ui-sorts-operations:
-    id: views-ui-sorts-operations
-    plugin: text
-    label: 'Sort actions'
-    weight: 9
-    selector: '.views-ui-display-tab-bucket.sort .dropbutton-widget'
-    body: 'Add, rearrange or remove sorting rules.'
-  views-ui-preview:
-    id: views-ui-preview
-    plugin: text
-    label: Preview
-    weight: 10
-    position: right
-    selector: '#preview-submit'
-    body: 'Show a preview of the view output.'
diff --git a/core/modules/tour/config/schema/tour.schema.yml b/core/modules/tour/config/schema/tour.schema.yml
deleted file mode 100644
index a51874b28848..000000000000
--- a/core/modules/tour/config/schema/tour.schema.yml
+++ /dev/null
@@ -1,67 +0,0 @@
-# Schema for the configuration files of the Tour module.
-
-tour.tour.*:
-  type: config_entity
-  label: 'Tour settings'
-  mapping:
-    id:
-      type: machine_name
-      label: 'ID'
-      constraints:
-        # Tour IDs also allow dashes.
-        Regex:
-          pattern: '/^[a-z0-9_-]+$/'
-          message: "The %value machine name is not valid."
-    label:
-      type: required_label
-      label: 'Label'
-    module:
-      type: string
-      label: 'Providing module'
-    routes:
-      type: sequence
-      label: 'Route settings'
-      sequence:
-        type: route
-        label: 'Route'
-    tips:
-      type: sequence
-      label: 'Tips'
-      sequence:
-        type: tour.tip.[plugin]
-        label: 'Tour tip'
-
-tour.tip:
-  type: mapping
-  label: 'Tour tip'
-  mapping:
-    id:
-      type: string
-      label: 'ID'
-    plugin:
-      type: string
-      label: 'Plugin'
-      constraints:
-        PluginExists:
-          manager: plugin.manager.tour.tip
-          interface: '\Drupal\tour\TipPluginInterface'
-    label:
-      type: required_label
-      label: 'Label'
-    weight:
-      type: weight
-      label: 'Weight'
-    position:
-      type: string
-      label: 'Position'
-    selector:
-      type: string
-      label: 'Selector'
-
-tour.tip.text:
-  type: tour.tip
-  label: 'Textual tour tip'
-  mapping:
-    body:
-      type: text
-      label: 'Body'
diff --git a/core/modules/tour/css/tour.module.css b/core/modules/tour/css/tour.module.css
deleted file mode 100644
index 50eb62aaba3d..000000000000
--- a/core/modules/tour/css/tour.module.css
+++ /dev/null
@@ -1,153 +0,0 @@
-/**
- * @file
- * Styling for tour module.
- */
-
-/* Tab appearance. */
-.toolbar .toolbar-bar .tour-toolbar-tab.toolbar-tab {
-  float: right; /* LTR */
-}
-[dir="rtl"] .toolbar .toolbar-bar .tour-toolbar-tab.toolbar-tab {
-  float: left;
-}
-
-/* Style the tour progress indicator. */
-.tour-progress {
-  position: absolute;
-  right: 20px; /* LTR */
-  bottom: 20px;
-}
-[dir="rtl"] .tour-progress {
-  right: auto;
-  left: 20px;
-}
-
-/**
- * The following are largely Shepherd's default styles, with a few modifications
- * to facilitate a graceful transition from Joyride, the library used prior to
- * Shepherd.
- */
-.shepherd-footer {
-  display: flex;
-  justify-content: flex-start;
-  padding: 0 20px 20px;
-}
-
-.shepherd-footer .shepherd-button:last-child {
-  margin-right: 0;
-}
-
-.shepherd-cancel-icon {
-  position: absolute;
-  top: 20px;
-  right: 20px;
-  margin: 0;
-  padding: 0;
-  cursor: pointer;
-  border: none;
-  background: transparent;
-  line-height: 1em;
-}
-
-.shepherd-title {
-  margin: 0;
-  padding: 0;
-}
-
-.shepherd-header {
-  position: relative;
-  margin-bottom: 10px;
-  padding: 20px 50px 0 20px;
-}
-
-.shepherd-text {
-  padding: 0 20px;
-}
-
-.shepherd-text p {
-  margin: 0 0 1.4em;
-}
-
-.shepherd-element {
-  z-index: 110;
-  width: 300px;
-  background: #fff;
-}
-
-@media only screen and (max-width: 767px) {
-  .shepherd-element {
-    left: 2.5%;
-    width: 85%;
-  }
-}
-
-.shepherd-enabled.shepherd-element {
-  opacity: 1;
-}
-
-.shepherd-element[data-popper-reference-hidden]:not(.shepherd-centered) {
-  opacity: 0;
-}
-
-.shepherd-element,
-.shepherd-element *,
-.shepherd-element ::after,
-.shepherd-element ::before {
-  box-sizing: border-box;
-}
-
-.shepherd-arrow,
-.shepherd-arrow::before {
-  position: absolute;
-  width: 16px;
-  height: 16px;
-}
-
-.shepherd-arrow::before {
-  content: "";
-  transform: rotate(45deg);
-  background: #fff;
-}
-
-.shepherd-element[data-popper-placement^=top] > .shepherd-arrow {
-  bottom: -8px;
-}
-
-.shepherd-element[data-popper-placement^=bottom] > .shepherd-arrow {
-  top: -8px;
-}
-
-.shepherd-element[data-popper-placement^=left] > .shepherd-arrow {
-  right: -8px;
-}
-
-.shepherd-element[data-popper-placement^=right] > .shepherd-arrow {
-  left: -8px;
-}
-
-.shepherd-target-click-disabled.shepherd-enabled.shepherd-target,
-.shepherd-target-click-disabled.shepherd-enabled.shepherd-target * {
-  pointer-events: none;
-}
-
-.shepherd-modal-overlay-container {
-  position: fixed;
-  z-index: 105;
-  top: 0;
-  left: 0;
-  overflow: hidden;
-  width: 100vw;
-  height: 0;
-  pointer-events: none;
-  opacity: 0;
-  fill-rule: evenodd;
-}
-
-.shepherd-modal-overlay-container.shepherd-modal-is-visible {
-  height: 100vh;
-  opacity: 0.5;
-}
-
-.shepherd-modal-overlay-container.shepherd-modal-is-visible path {
-  pointer-events: all;
-}
diff --git a/core/modules/tour/help_topics/tour.overview.html.twig b/core/modules/tour/help_topics/tour.overview.html.twig
deleted file mode 100644
index af2a19760d17..000000000000
--- a/core/modules/tour/help_topics/tour.overview.html.twig
+++ /dev/null
@@ -1,16 +0,0 @@
----
-label: 'Taking tours of administrative pages'
-related:
-  - core.ui_components
----
-<h2>{% trans %}Goal{% endtrans %}</h2>
-<p>{% trans %}Take a tour of an administrative page.{% endtrans %}</p>
-<h2>{% trans %}What are tours?{% endtrans %}</h2>
-<p>{% trans %}The core Tour module provides users with <em>tours</em>, which are guided tours of the administrative interface. Each tour starts on a particular administrative page, and consists of one or more <em>tips</em> that highlight elements of the page, guide you through a workflow, or explain key concepts. Users need <em>Access tour</em> permission to view tours, and JavaScript must be enabled in their browsers.{% endtrans %}</p>
-<h2>{% trans %}Steps{% endtrans %}</h2>
-<ol>
-  <li>{% trans %}Make sure that the core Tour module is installed, and that you have a role with the <em>Access tour</em> permission. Also, make sure that a toolbar module is installed (either the core Toolbar module or a contributed module replacement).{% endtrans %}</li>
-  <li>{% trans %}Visit an administrative page that has a tour, such as the edit view page provided by the core Views UI module.{% endtrans %}</li>
-  <li>{% trans %}Click the <em>Tour</em> button at the right end of the toolbar (left end for right-to-left languages). The first tip of the tour should appear.{% endtrans %}</li>
-  <li>{% trans %}Click the <em>Next</em> button to advance to the next tip, and <em>End tour</em> at the end to close the tour.{% endtrans %}</li>
-</ol>
diff --git a/core/modules/tour/js/tour.js b/core/modules/tour/js/tour.js
deleted file mode 100644
index 43858e4e2a71..000000000000
--- a/core/modules/tour/js/tour.js
+++ /dev/null
@@ -1,414 +0,0 @@
-/**
- * @file
- * Attaches behaviors for the Tour module's toolbar tab.
- */
-
-(($, Backbone, Drupal, settings, document, Shepherd) => {
-  const queryString = decodeURI(window.location.search);
-
-  /**
-   * Attaches the tour's toolbar tab behavior.
-   *
-   * It uses the query string for:
-   * - tour: When ?tour=1 is present, the tour will start automatically after
-   *   the page has loaded.
-   * - tips: Pass ?tips=class in the url to filter the available tips to the
-   *   subset which match the given class.
-   *
-   * @example
-   * http://example.com/foo?tour=1&tips=bar
-   *
-   * @type {Drupal~behavior}
-   *
-   * @prop {Drupal~behaviorAttach} attach
-   *   Attach tour functionality on `tour` events.
-   */
-  Drupal.behaviors.tour = {
-    attach(context) {
-      once('tour', 'body').forEach(() => {
-        const model = new Drupal.tour.models.StateModel();
-        // eslint-disable-next-line no-new
-        new Drupal.tour.views.ToggleTourView({
-          el: $(context).find('#toolbar-tab-tour'),
-          model,
-        });
-
-        model
-          // Allow other scripts to respond to tour events.
-          .on('change:isActive', (tourModel, isActive) => {
-            $(document).trigger(
-              isActive ? 'drupalTourStarted' : 'drupalTourStopped',
-            );
-          });
-        // Initialization: check whether a tour is available on the current
-        // page.
-        if (settings._tour_internal) {
-          model.set('tour', settings._tour_internal);
-        }
-        // Start the tour immediately if toggled via query string.
-        if (/tour=?/i.test(queryString)) {
-          model.set('isActive', true);
-        }
-      });
-    },
-  };
-
-  /**
-   * @namespace
-   */
-  Drupal.tour = Drupal.tour || {
-    /**
-     * @namespace Drupal.tour.models
-     */
-    models: {},
-
-    /**
-     * @namespace Drupal.tour.views
-     */
-    views: {},
-  };
-
-  /**
-   * Backbone Model for tours.
-   *
-   * @constructor
-   *
-   * @augments Backbone.Model
-   */
-  Drupal.tour.models.StateModel = Backbone.Model.extend(
-    /** @lends Drupal.tour.models.StateModel# */ {
-      /**
-       * @type {object}
-       */
-      defaults: /** @lends Drupal.tour.models.StateModel# */ {
-        /**
-         * Indicates whether the Drupal root window has a tour.
-         *
-         * @type {Array}
-         */
-        tour: [],
-
-        /**
-         * Indicates whether the tour is currently running.
-         *
-         * @type {boolean}
-         */
-        isActive: false,
-
-        /**
-         * Indicates which tour is the active one (necessary to cleanly stop).
-         *
-         * @type {Array}
-         */
-        activeTour: [],
-      },
-    },
-  );
-
-  Drupal.tour.views.ToggleTourView = Backbone.View.extend(
-    /** @lends Drupal.tour.views.ToggleTourView# */ {
-      /**
-       * @type {object}
-       */
-      events: { click: 'onClick' },
-
-      /**
-       * Handles edit mode toggle interactions.
-       *
-       * @constructs
-       *
-       * @augments Backbone.View
-       */
-      initialize() {
-        this.listenTo(this.model, 'change:tour change:isActive', this.render);
-        this.listenTo(this.model, 'change:isActive', this.toggleTour);
-      },
-
-      /**
-       * {@inheritdoc}
-       *
-       * @return {Drupal.tour.views.ToggleTourView}
-       *   The `ToggleTourView` view.
-       */
-      render() {
-        // Render the visibility.
-        this.$el.toggleClass('hidden', this._getTour().length === 0);
-        // Render the state.
-        const isActive = this.model.get('isActive');
-        this.$el
-          .find('button')
-          .toggleClass('is-active', isActive)
-          .attr('aria-pressed', isActive);
-        return this;
-      },
-
-      /**
-       * Model change handler; starts or stops the tour.
-       */
-      toggleTour() {
-        if (this.model.get('isActive')) {
-          this._removeIrrelevantTourItems(this._getTour());
-          const tourItems = this.model.get('tour');
-          const that = this;
-
-          if (tourItems.length) {
-            // If Joyride is positioned relative to the top or bottom of an
-            // element, and its secondary position is right or left, then the
-            // arrow is also positioned right or left. Shepherd defaults to
-            // center positioning the arrow.
-            //
-            // In most cases, this arrow positioning difference has
-            // little impact. However, tours built with Joyride may have tips
-            // using a higher level selector than the element the tip is
-            // expected to point to, and relied on Joyride's arrow positioning
-            // to align the arrow with the expected reference element. Joyride's
-            // arrow positioning behavior is replicated here to prevent those
-            // use cases from causing UI regressions.
-            //
-            // This modifier is provided here instead of TourViewBuilder (where
-            // most position modifications are) because it includes adding a
-            // JavaScript callback function.
-            settings.tourShepherdConfig.defaultStepOptions.popperOptions.modifiers.push(
-              {
-                name: 'moveArrowJoyridePosition',
-                enabled: true,
-                phase: 'write',
-                fn({ state }) {
-                  const { arrow } = state.elements;
-                  const { placement } = state;
-                  if (
-                    arrow &&
-                    /^top|bottom/.test(placement) &&
-                    /-start|-end$/.test(placement)
-                  ) {
-                    const horizontalPosition = placement.split('-')[1];
-                    const offset =
-                      horizontalPosition === 'start'
-                        ? 28
-                        : state.elements.popper.clientWidth - 56;
-                    arrow.style.transform = `translate3d(${offset}px, 0px, 0px)`;
-                  }
-                },
-              },
-            );
-            const shepherdTour = new Shepherd.Tour(settings.tourShepherdConfig);
-            shepherdTour.on('cancel', () => {
-              that.model.set('isActive', false);
-            });
-            shepherdTour.on('complete', () => {
-              that.model.set('isActive', false);
-            });
-
-            tourItems.forEach((tourStepConfig, index) => {
-              // Create the configuration for a given tour step by using values
-              // defined in TourViewBuilder.
-              // @see \Drupal\tour\TourViewBuilder::viewMultiple()
-              const tourItemOptions = {
-                title: tourStepConfig.title
-                  ? Drupal.checkPlain(tourStepConfig.title)
-                  : null,
-                text: () => Drupal.theme('tourItemContent', tourStepConfig),
-                attachTo: tourStepConfig.attachTo,
-                buttons: [Drupal.tour.nextButton(shepherdTour, tourStepConfig)],
-                classes: tourStepConfig.classes,
-                index,
-              };
-
-              tourItemOptions.when = {
-                show() {
-                  const nextButton =
-                    shepherdTour.currentStep.el.querySelector('footer button');
-
-                  // Drupal disables Shepherd's built in focus after item
-                  // creation functionality due to focus being set on the tour
-                  // item container after every scroll and resize event. In its
-                  // place, the 'next' button is focused here.
-                  nextButton.focus();
-
-                  // When Stable 9 is part of the active theme, the
-                  // Drupal.tour.convertToJoyrideMarkup() function is available.
-                  // This function converts Shepherd markup to Joyride markup,
-                  // facilitating the use of the Shepherd library that is
-                  // backwards compatible with customizations intended for
-                  // Joyride.
-                  // The Drupal.tour.convertToJoyrideMarkup() function is
-                  // internal, and will eventually be removed from Drupal core.
-                  if (Drupal.tour.hasOwnProperty('convertToJoyrideMarkup')) {
-                    Drupal.tour.convertToJoyrideMarkup(shepherdTour);
-                  }
-                },
-              };
-
-              shepherdTour.addStep(tourItemOptions);
-            });
-            shepherdTour.start();
-            this.model.set({ isActive: true, activeTour: shepherdTour });
-          }
-        } else {
-          this.model.get('activeTour').cancel();
-          this.model.set({ isActive: false, activeTour: [] });
-        }
-      },
-
-      /**
-       * Toolbar tab click event handler; toggles isActive.
-       *
-       * @param {jQuery.Event} event
-       *   The click event.
-       */
-      onClick(event) {
-        this.model.set('isActive', !this.model.get('isActive'));
-        event.preventDefault();
-        event.stopPropagation();
-      },
-
-      /**
-       * Gets the tour.
-       *
-       * @return {array}
-       *   An array of Shepherd tour item objects.
-       */
-      _getTour() {
-        return this.model.get('tour');
-      },
-
-      /**
-       * Removes tour items for elements that don't have matching page elements.
-       *
-       * Or that are explicitly filtered out via the 'tips' query string.
-       *
-       * @example
-       * <caption>This will filter out tips that do not have a matching
-       * page element or don't have the "bar" class.</caption>
-       * http://example.com/foo?tips=bar
-       *
-       * @param {Object[]} tourItems
-       *   An array containing tour Step config objects.
-       *   The object properties relevant to this function:
-       *   - classes {string}: A string of classes to be added to the tour step
-       *     when rendered.
-       *   - selector {string}: The selector a tour step is associated with.
-       */
-      _removeIrrelevantTourItems(tourItems) {
-        const tips = /tips=([^&]+)/.exec(queryString);
-        const filteredTour = tourItems.filter((tourItem) => {
-          // If the query parameter 'tips' is set, remove all tips that don't
-          // have the matching class. The `tourItem` variable is a step config
-          // object, and the 'classes' property is a ShepherdJS Step() config
-          // option that provides a string.
-          if (
-            tips &&
-            tourItem.hasOwnProperty('classes') &&
-            tourItem.classes.indexOf(tips[1]) === -1
-          ) {
-            return false;
-          }
-
-          // If a selector is configured but there isn't a matching element,
-          // return false.
-          return !(
-            tourItem.selector && !document.querySelector(tourItem.selector)
-          );
-        });
-
-        // If there are tours filtered, we'll have to update model.
-        if (tourItems.length !== filteredTour.length) {
-          filteredTour.forEach((filteredTourItem, filteredTourItemId) => {
-            filteredTour[filteredTourItemId].counter = Drupal.t(
-              '!tour_item of !total',
-              {
-                '!tour_item': filteredTourItemId + 1,
-                '!total': filteredTour.length,
-              },
-            );
-
-            if (filteredTourItemId === filteredTour.length - 1) {
-              filteredTour[filteredTourItemId].cancelText =
-                Drupal.t('End tour');
-            }
-          });
-          this.model.set('tour', filteredTour);
-        }
-      },
-    },
-  );
-
-  /**
-   * Provides an object that will become the tour item's 'next' button.
-   *
-   * Similar to a theme function, themes can override this function to customize
-   * the resulting button. Unlike a theme function, it returns an object instead
-   * of a string, which is why it is not part of Drupal.theme.
-   *
-   * @param {Tour} shepherdTour
-   *  A class representing a Shepherd site tour.
-   * @param {Object} tourStepConfig
-   *   An object generated in TourViewBuilder used for creating the options
-   *   passed to `Tour.addStep(options)`.
-   *   Contains the following properties:
-   *   - id {string}: The tour.tip ID specified by its config
-   *   - selector {string|null}: The selector of the element the tour step is
-   *     attaching to.
-   *   - module {string}: The module providing the tip plugin used by this step.
-   *   - counter {string}: A string indicating which tour step this is out of
-   *     how many total steps.
-   *   - attachTo {Object} This is directly mapped to the `attachTo` Step()
-   *     option. It has two properties:
-   *     - element {string}: The selector of the element the step attaches to.
-   *     - on {string}: a PopperJS compatible string to specify step position.
-   *   - classes {string}: Will be added to the class attribute of the step.
-   *   - body {string}: Markup that is mapped to the `text` Step() option. Will
-   *     become the step content.
-   *   - title {string}: is mapped to the `title` Step() option.
-   *
-   * @return {{classes: string, action: string, text: string}}
-   *    An object structured in the manner Shepherd requires to create the
-   *    'next' button.
-   *
-   * @see https://shepherdjs.dev/docs/Tour.html
-   * @see \Drupal\tour\TourViewBuilder::viewMultiple()
-   * @see https://shepherdjs.dev/docs/Step.html
-   */
-  Drupal.tour.nextButton = (shepherdTour, tourStepConfig) => {
-    return {
-      classes: 'button button--primary',
-      text: tourStepConfig.cancelText
-        ? tourStepConfig.cancelText
-        : Drupal.t('Next'),
-      action: tourStepConfig.cancelText
-        ? shepherdTour.cancel
-        : shepherdTour.next,
-    };
-  };
-
-  /**
-   * Theme function for tour item content.
-   *
-   * @param {Object} tourStepConfig
-   *   An object generated in TourViewBuilder used for creating the options
-   *   passed to `Tour.addStep(options)`.
-   *   Contains the following properties:
-   *   - id {string}: The tour.tip ID specified by its config
-   *   - selector {string|null}: The selector of the element the tour step is
-   *     attaching to.
-   *   - module {string}: The module providing the tip plugin used by this step.
-   *   - counter {string}: A string indicating which tour step this is out of
-   *     how many total steps.
-   *   - attachTo {Object} This is directly mapped to the `attachTo` Step()
-   *     option. It has two properties:
-   *     - element {string}: The selector of the element the step attaches to.
-   *     - on {string}: a PopperJS compatible string to specify step position.
-   *   - classes {string}: Will be added to the class attribute of the step.
-   *   - body {string}: Markup that is mapped to the `text` Step() option. Will
-   *     become the step content.
-   *   - title {string}: is mapped to the `title` Step() option.
-   *
-   * @return {string}
-   *   The tour item content markup.
-   *
-   * @see \Drupal\tour\TourViewBuilder::viewMultiple()
-   * @see https://shepherdjs.dev/docs/Step.html
-   */
-  Drupal.theme.tourItemContent = (tourStepConfig) =>
-    `${tourStepConfig.body}<div class="tour-progress">${tourStepConfig.counter}</div>`;
-})(jQuery, Backbone, Drupal, drupalSettings, document, window.Shepherd);
diff --git a/core/modules/tour/src/Annotation/Tip.php b/core/modules/tour/src/Annotation/Tip.php
deleted file mode 100644
index d1daeb75ee90..000000000000
--- a/core/modules/tour/src/Annotation/Tip.php
+++ /dev/null
@@ -1,39 +0,0 @@
-<?php
-
-namespace Drupal\tour\Annotation;
-
-use Drupal\Component\Annotation\Plugin;
-
-/**
- * Defines a tour item annotation object.
- *
- * Plugin Namespace: Plugin\tour\tip
- *
- * For a working example, see \Drupal\tour\Plugin\tour\tip\TipPluginText
- *
- * @see \Drupal\tour\TipPluginBase
- * @see \Drupal\tour\TipPluginInterface
- * @see \Drupal\tour\TipPluginManager
- * @see plugin_api
- *
- * @Annotation
- */
-class Tip extends Plugin {
-
-  /**
-   * The plugin ID.
-   *
-   * @var string
-   */
-  public $id;
-
-  /**
-   * The title of the plugin.
-   *
-   * @var \Drupal\Core\Annotation\Translation
-   *
-   * @ingroup plugin_translatable
-   */
-  public $title;
-
-}
diff --git a/core/modules/tour/src/Entity/Tour.php b/core/modules/tour/src/Entity/Tour.php
deleted file mode 100644
index 8e168fe0a816..000000000000
--- a/core/modules/tour/src/Entity/Tour.php
+++ /dev/null
@@ -1,189 +0,0 @@
-<?php
-
-namespace Drupal\tour\Entity;
-
-use Drupal\Core\Config\Entity\ConfigEntityBase;
-use Drupal\tour\TipsPluginCollection;
-use Drupal\tour\TourInterface;
-
-/**
- * Defines the configured tour entity.
- *
- * @ConfigEntityType(
- *   id = "tour",
- *   label = @Translation("Tour"),
- *   label_collection = @Translation("Tours"),
- *   label_singular = @Translation("tour"),
- *   label_plural = @Translation("tours"),
- *   label_count = @PluralTranslation(
- *     singular = "@count tour",
- *     plural = "@count tours",
- *   ),
- *   handlers = {
- *     "view_builder" = "Drupal\tour\TourViewBuilder",
- *     "access" = "Drupal\tour\TourAccessControlHandler",
- *   },
- *   admin_permission = "administer site configuration",
- *   entity_keys = {
- *     "id" = "id",
- *     "label" = "label"
- *   },
- *   config_export = {
- *     "id",
- *     "label",
- *     "module",
- *     "routes",
- *     "tips",
- *   },
- *   lookup_keys = {
- *     "routes.*.route_name"
- *   }
- * )
- */
-class Tour extends ConfigEntityBase implements TourInterface {
-
-  /**
-   * The name (plugin ID) of the tour.
-   *
-   * @var string
-   */
-  protected $id;
-
-  /**
-   * The module which this tour is assigned to.
-   *
-   * @var string
-   */
-  protected $module;
-
-  /**
-   * The label of the tour.
-   *
-   * @var string
-   */
-  protected $label;
-
-  /**
-   * The routes on which this tour should be displayed.
-   *
-   * @var array
-   */
-  protected $routes = [];
-
-  /**
-   * The routes on which this tour should be displayed, keyed by route id.
-   *
-   * @var array
-   */
-  protected $keyedRoutes;
-
-  /**
-   * Holds the collection of tips that are attached to this tour.
-   *
-   * @var \Drupal\tour\TipsPluginCollection
-   */
-  protected $tipsCollection;
-
-  /**
-   * The array of plugin config, only used for export and to populate the $tipsCollection.
-   *
-   * @var array
-   */
-  protected $tips = [];
-
-  /**
-   * {@inheritdoc}
-   */
-  public function __construct(array $values, $entity_type) {
-    parent::__construct($values, $entity_type);
-
-    $this->tipsCollection = new TipsPluginCollection(\Drupal::service('plugin.manager.tour.tip'), $this->tips);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getRoutes() {
-    return $this->routes;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getTip($id) {
-    return $this->tipsCollection->get($id);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getTips() {
-    $tips = [];
-    foreach ($this->tips as $id => $tip) {
-      $tips[] = $this->getTip($id);
-    }
-    uasort($tips, function ($a, $b) {
-      return $a->getWeight() <=> $b->getWeight();
-    });
-
-    \Drupal::moduleHandler()->alter('tour_tips', $tips, $this);
-    return array_values($tips);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getModule() {
-    return $this->module;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function hasMatchingRoute($route_name, $route_params) {
-    if (!isset($this->keyedRoutes)) {
-      $this->keyedRoutes = [];
-      foreach ($this->getRoutes() as $route) {
-        $this->keyedRoutes[$route['route_name']] = $route['route_params'] ?? [];
-      }
-    }
-    if (!isset($this->keyedRoutes[$route_name])) {
-      // We don't know about this route.
-      return FALSE;
-    }
-    if (empty($this->keyedRoutes[$route_name])) {
-      // We don't need to worry about route params, the route name is enough.
-      return TRUE;
-    }
-    foreach ($this->keyedRoutes[$route_name] as $key => $value) {
-      // If a required param is missing or doesn't match, return FALSE.
-      if (empty($route_params[$key]) || $route_params[$key] !== $value) {
-        return FALSE;
-      }
-    }
-    return TRUE;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function resetKeyedRoutes() {
-    unset($this->keyedRoutes);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function calculateDependencies() {
-    parent::calculateDependencies();
-
-    foreach ($this->tipsCollection as $instance) {
-      $definition = $instance->getPluginDefinition();
-      $this->addDependency('module', $definition['provider']);
-    }
-
-    $this->addDependency('module', $this->module);
-    return $this;
-  }
-
-}
diff --git a/core/modules/tour/src/Plugin/HelpSection/TourHelpSection.php b/core/modules/tour/src/Plugin/HelpSection/TourHelpSection.php
deleted file mode 100644
index 4716af427131..000000000000
--- a/core/modules/tour/src/Plugin/HelpSection/TourHelpSection.php
+++ /dev/null
@@ -1,133 +0,0 @@
-<?php
-
-namespace Drupal\tour\Plugin\HelpSection;
-
-use Drupal\Core\Entity\EntityTypeManagerInterface;
-use Drupal\Core\Link;
-use Drupal\Core\Plugin\ContainerFactoryPluginInterface;
-use Drupal\Core\Url;
-use Drupal\help\Plugin\HelpSection\HelpSectionPluginBase;
-use Symfony\Component\DependencyInjection\ContainerInterface;
-
-/**
- * Provides the tours list section for the help page.
- *
- * @HelpSection(
- *   id = "tour",
- *   title = @Translation("Tours"),
- *   weight = 10,
- *   description = @Translation("Tours guide you through workflows or explain concepts on various user interface pages. The tours with links in this list are on user interface landing pages; the tours without links will show on individual pages (such as when editing a View using the Views UI module). Available tours:"),
- *   permission = "access tour"
- * )
- */
-class TourHelpSection extends HelpSectionPluginBase implements ContainerFactoryPluginInterface {
-
-  /**
-   * The entity type manager.
-   *
-   * @var \Drupal\Core\Entity\EntityTypeManagerInterface
-   */
-  protected $entityTypeManager;
-
-  /**
-   * Constructs a TourHelpSection object.
-   *
-   * @param array $configuration
-   *   A configuration array containing information about the plugin instance.
-   * @param string $plugin_id
-   *   The plugin_id for the plugin instance.
-   * @param mixed $plugin_definition
-   *   The plugin implementation definition.
-   * @param \Drupal\Core\Entity\EntityTypeManagerInterface $entity_type_manager
-   *   The entity type manager service.
-   */
-  public function __construct(array $configuration, $plugin_id, $plugin_definition, EntityTypeManagerInterface $entity_type_manager) {
-    parent::__construct($configuration, $plugin_id, $plugin_definition);
-    $this->entityTypeManager = $entity_type_manager;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
-    return new static(
-      $configuration,
-      $plugin_id,
-      $plugin_definition,
-      $container->get('entity_type.manager')
-    );
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getCacheMaxAge() {
-    // The calculation of which URL (if any) gets put on which tour depends
-    // on a route access check. This can have a lot of inputs, including user
-    // permissions and other factors. Rather than doing a complicated
-    // accounting of the cache metadata for all of these possible factors, set
-    // the max age of the cache to zero to prevent using incorrect cached
-    // information.
-    return 0;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function listTopics() {
-    /** @var \Drupal\tour\TourInterface[] $tours */
-    $tours = $this->entityTypeManager->getStorage('tour')->loadMultiple();
-    // Sort in the manner defined by Tour.
-    uasort($tours, ['Drupal\tour\Entity\Tour', 'sort']);
-
-    // Make a link to each tour, using the first of its routes that can
-    // be linked to by this user, if any.
-    $topics = [];
-    foreach ($tours as $tour) {
-      $title = $tour->label();
-      $id = $tour->id();
-      $routes = $tour->getRoutes();
-      $made_link = FALSE;
-      foreach ($routes as $route) {
-        // Some tours are for routes with parameters. For instance, there is
-        // currently a tour in the Language module for the language edit page,
-        // which appears on all pages with URLs like:
-        // /admin/config/regional/language/edit/LANGCODE.
-        // There is no way to make a link to the page that displays the tour,
-        // because it is a set of pages. The easiest way to detect this is to
-        // use a try/catch exception -- try to make a link, and it will error
-        // out with a missing parameter exception if the route leads to a set
-        // of pages instead of a single page.
-        try {
-          $params = $route['route_params'] ?? [];
-          $url = Url::fromRoute($route['route_name'], $params);
-          // Skip this route if the current user cannot access it.
-          if (!$url->access()) {
-            continue;
-          }
-
-          // Generate the link HTML directly, using toString(), to catch
-          // missing parameter exceptions now instead of at render time.
-          $topics[$id] = Link::fromTextAndUrl($title, $url)->toString();
-          // If the line above didn't generate an exception, we have a good
-          // link that the user can access.
-          $made_link = TRUE;
-          break;
-        }
-        catch (\Exception $e) {
-          // Exceptions are normally due to routes that need parameters. If
-          // there is an exception, just try the next route and see if we can
-          // find one that will work for us.
-        }
-      }
-      if (!$made_link) {
-        // None of the routes worked to make a link, so at least display the
-        // tour title.
-        $topics[$id] = $title;
-      }
-    }
-
-    return $topics;
-  }
-
-}
diff --git a/core/modules/tour/src/Plugin/tour/tip/TipPluginText.php b/core/modules/tour/src/Plugin/tour/tip/TipPluginText.php
deleted file mode 100644
index 93963bfb5573..000000000000
--- a/core/modules/tour/src/Plugin/tour/tip/TipPluginText.php
+++ /dev/null
@@ -1,72 +0,0 @@
-<?php
-
-namespace Drupal\tour\Plugin\tour\tip;
-
-use Drupal\Core\Plugin\ContainerFactoryPluginInterface;
-use Drupal\Core\Utility\Token;
-use Drupal\tour\TipPluginBase;
-use Symfony\Component\DependencyInjection\ContainerInterface;
-
-/**
- * Displays some text as a tip.
- *
- * @Tip(
- *   id = "text",
- *   title = @Translation("Text")
- * )
- */
-class TipPluginText extends TipPluginBase implements ContainerFactoryPluginInterface {
-
-  /**
-   * The body text which is used for render of this Text Tip.
-   *
-   * @var string
-   */
-  protected $body;
-
-  /**
-   * Token service.
-   *
-   * @var \Drupal\Core\Utility\Token
-   */
-  protected $token;
-
-  /**
-   * Constructs a \Drupal\tour\Plugin\tour\tip\TipPluginText object.
-   *
-   * @param array $configuration
-   *   A configuration array containing information about the plugin instance.
-   * @param string $plugin_id
-   *   The plugin_id for the plugin instance.
-   * @param mixed $plugin_definition
-   *   The plugin implementation definition.
-   * @param \Drupal\Core\Utility\Token $token
-   *   The token service.
-   */
-  public function __construct(array $configuration, $plugin_id, $plugin_definition, Token $token) {
-    parent::__construct($configuration, $plugin_id, $plugin_definition);
-    $this->token = $token;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
-    return new static($configuration, $plugin_id, $plugin_definition, $container->get('token'));
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getBody(): array {
-    return [
-      '#type' => 'html_tag',
-      '#tag' => 'p',
-      '#value' => $this->token->replace($this->get('body')),
-      '#attributes' => [
-        'class' => ['tour-tip-body'],
-      ],
-    ];
-  }
-
-}
diff --git a/core/modules/tour/src/TipPluginBase.php b/core/modules/tour/src/TipPluginBase.php
deleted file mode 100644
index bb24470c87ed..000000000000
--- a/core/modules/tour/src/TipPluginBase.php
+++ /dev/null
@@ -1,112 +0,0 @@
-<?php
-
-namespace Drupal\tour;
-
-use Drupal\Core\Plugin\PluginBase;
-
-/**
- * Defines a base tour item implementation.
- *
- * @see \Drupal\tour\Annotation\Tip
- * @see \Drupal\tour\TipPluginInterface
- * @see \Drupal\tour\TipPluginManager
- * @see plugin_api
- */
-abstract class TipPluginBase extends PluginBase implements TipPluginInterface {
-
-  /**
-   * The label which is used for render of this tip.
-   *
-   * @var string
-   */
-  protected $label;
-
-  /**
-   * Allows tips to take more priority that others.
-   *
-   * @var string
-   */
-  protected $weight;
-
-  /**
-   * {@inheritdoc}
-   */
-  public function id() {
-    return $this->get('id');
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getLabel() {
-    return $this->get('label');
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getWeight() {
-    return $this->get('weight');
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function get($key) {
-    if (!empty($this->configuration[$key])) {
-      return $this->configuration[$key];
-    }
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function set($key, $value) {
-    $this->configuration[$key] = $value;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getLocation(): ?string {
-    $location = $this->get('position');
-
-    // The location values accepted by PopperJS, the library used for
-    // positioning the tip.
-    assert(in_array(trim($location ?? ''), [
-      'auto',
-      'auto-start',
-      'auto-end',
-      'top',
-      'top-start',
-      'top-end',
-      'bottom',
-      'bottom-start',
-      'bottom-end',
-      'right',
-      'right-start',
-      'right-end',
-      'left',
-      'left-start',
-      'left-end',
-      '',
-    ], TRUE), "$location is not a valid Tour Tip position value");
-
-    return $location;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getSelector(): ?string {
-    return $this->get('selector');
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getBody(): array {
-    return [];
-  }
-
-}
diff --git a/core/modules/tour/src/TipPluginInterface.php b/core/modules/tour/src/TipPluginInterface.php
deleted file mode 100644
index 6fd4b59e2e0c..000000000000
--- a/core/modules/tour/src/TipPluginInterface.php
+++ /dev/null
@@ -1,101 +0,0 @@
-<?php
-
-namespace Drupal\tour;
-
-/**
- * Defines an interface for tour items.
- *
- * @see \Drupal\tour\Annotation\Tip
- * @see \Drupal\tour\TipPluginBase
- * @see \Drupal\tour\TipPluginManager
- * @see plugin_api
- */
-interface TipPluginInterface {
-
-  /**
-   * Returns id of the tip.
-   *
-   * @return string
-   *   The id of the tip.
-   */
-  public function id();
-
-  /**
-   * Returns label of the tip.
-   *
-   * @return string
-   *   The label of the tip.
-   */
-  public function getLabel();
-
-  /**
-   * Returns weight of the tip.
-   *
-   * @return string
-   *   The weight of the tip.
-   */
-  public function getWeight();
-
-  /**
-   * Used for returning values by key.
-   *
-   * @var string
-   *   Key of the value.
-   *
-   * @return string
-   *   Value of the key.
-   */
-  public function get($key);
-
-  /**
-   * Returns the selector the tour tip will attach to.
-   *
-   * This typically maps to the Shepherd Step options `attachTo.element`
-   * property.
-   *
-   * @return null|string
-   *   A selector string, or null for an unattached tip.
-   *
-   * @see https://shepherdjs.dev/docs/Step.html
-   */
-  public function getSelector(): ?string;
-
-  /**
-   * Returns the body content of the tooltip.
-   *
-   * This typically maps to the Shepherd Step options `text` property.
-   *
-   * @return array
-   *   A render array.
-   *
-   * @see https://shepherdjs.dev/docs/Step.html
-   */
-  public function getBody(): array;
-
-  /**
-   * Returns the configured placement of the tip relative to the element.
-   *
-   * If null, the tip will automatically determine the best position based on
-   * the element's position in the viewport.
-   *
-   * This typically maps to the Shepherd Step options `attachTo.on` property.
-   *
-   * @return string|null
-   *   The tip placement relative to the element.
-   *
-   * @see https://shepherdjs.dev/docs/Step.html
-   */
-  public function getLocation(): ?string;
-
-  /**
-   * Used for returning values by key.
-   *
-   * @var string
-   *   Key of the value.
-   *
-   * @var string
-   *   Value of the key.
-   */
-  public function set($key, $value);
-
-}
diff --git a/core/modules/tour/src/TipPluginManager.php b/core/modules/tour/src/TipPluginManager.php
deleted file mode 100644
index 4675fd803b4d..000000000000
--- a/core/modules/tour/src/TipPluginManager.php
+++ /dev/null
@@ -1,37 +0,0 @@
-<?php
-
-namespace Drupal\tour;
-
-use Drupal\Core\Cache\CacheBackendInterface;
-use Drupal\Core\Extension\ModuleHandlerInterface;
-use Drupal\Core\Plugin\DefaultPluginManager;
-
-/**
- * Provides a plugin manager for tour items.
- *
- * @see \Drupal\tour\Annotation\Tip
- * @see \Drupal\tour\TipPluginBase
- * @see \Drupal\tour\TipPluginInterface
- * @see plugin_api
- */
-class TipPluginManager extends DefaultPluginManager {
-
-  /**
-   * Constructs a new TipPluginManager.
-   *
-   * @param \Traversable $namespaces
-   *   An object that implements \Traversable which contains the root paths
-   *   keyed by the corresponding namespace to look for plugin implementations,
-   * @param \Drupal\Core\Cache\CacheBackendInterface $cache_backend
-   *   Cache backend instance to use.
-   * @param \Drupal\Core\Extension\ModuleHandlerInterface $module_handler
-   *   The module handler to invoke the alter hook with.
-   */
-  public function __construct(\Traversable $namespaces, CacheBackendInterface $cache_backend, ModuleHandlerInterface $module_handler) {
-    parent::__construct('Plugin/tour/tip', $namespaces, $module_handler, 'Drupal\tour\TipPluginInterface', 'Drupal\tour\Annotation\Tip');
-
-    $this->alterInfo('tour_tips_info');
-    $this->setCacheBackend($cache_backend, 'tour_plugins');
-  }
-
-}
diff --git a/core/modules/tour/src/TipsPluginCollection.php b/core/modules/tour/src/TipsPluginCollection.php
deleted file mode 100644
index 5480f10dcd72..000000000000
--- a/core/modules/tour/src/TipsPluginCollection.php
+++ /dev/null
@@ -1,26 +0,0 @@
-<?php
-
-namespace Drupal\tour;
-
-use Drupal\Core\Plugin\DefaultLazyPluginCollection;
-
-/**
- * A collection of tips.
- */
-class TipsPluginCollection extends DefaultLazyPluginCollection {
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $pluginKey = 'plugin';
-
-  /**
-   * {@inheritdoc}
-   *
-   * @return \Drupal\tour\TipPluginInterface
-   */
-  public function &get($instance_id) {
-    return parent::get($instance_id);
-  }
-
-}
diff --git a/core/modules/tour/src/TourAccessControlHandler.php b/core/modules/tour/src/TourAccessControlHandler.php
deleted file mode 100644
index a0d427a0b128..000000000000
--- a/core/modules/tour/src/TourAccessControlHandler.php
+++ /dev/null
@@ -1,28 +0,0 @@
-<?php
-
-namespace Drupal\tour;
-
-use Drupal\Core\Access\AccessResult;
-use Drupal\Core\Entity\EntityAccessControlHandler;
-use Drupal\Core\Entity\EntityInterface;
-use Drupal\Core\Session\AccountInterface;
-
-/**
- * Defines the access control handler for the tour entity type.
- *
- * @see \Drupal\tour\Entity\Tour
- */
-class TourAccessControlHandler extends EntityAccessControlHandler {
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function checkAccess(EntityInterface $entity, $operation, AccountInterface $account) {
-    if ($operation === 'view') {
-      return AccessResult::allowedIfHasPermissions($account, ['access tour', 'administer site configuration'], 'OR');
-    }
-
-    return parent::checkAccess($entity, $operation, $account);
-  }
-
-}
diff --git a/core/modules/tour/src/TourInterface.php b/core/modules/tour/src/TourInterface.php
deleted file mode 100644
index 31590713fe67..000000000000
--- a/core/modules/tour/src/TourInterface.php
+++ /dev/null
@@ -1,65 +0,0 @@
-<?php
-
-namespace Drupal\tour;
-
-use Drupal\Core\Config\Entity\ConfigEntityInterface;
-
-/**
- * Provides an interface defining a tour entity.
- */
-interface TourInterface extends ConfigEntityInterface {
-
-  /**
-   * The routes that this tour will appear on.
-   *
-   * @return array
-   *   Returns array of routes for the tour.
-   */
-  public function getRoutes();
-
-  /**
-   * Whether the tour matches a given set of route parameters.
-   *
-   * @param string $route_name
-   *   The route name the parameters are for.
-   * @param array $route_params
-   *   Associative array of raw route params.
-   *
-   * @return bool
-   *   TRUE if the tour matches the route parameters.
-   */
-  public function hasMatchingRoute($route_name, $route_params);
-
-  /**
-   * Returns tip plugin.
-   *
-   * @param string $id
-   *   The identifier of the tip.
-   *
-   * @return \Drupal\tour\TipPluginInterface
-   *   The tip plugin.
-   */
-  public function getTip($id);
-
-  /**
-   * Returns the tips for this tour.
-   *
-   * @return array
-   *   An array of tip plugins.
-   */
-  public function getTips();
-
-  /**
-   * Gets the module this tour belongs to.
-   *
-   * @return string
-   *   The module this tour belongs to.
-   */
-  public function getModule();
-
-  /**
-   * Resets the statically cached keyed routes.
-   */
-  public function resetKeyedRoutes();
-
-}
diff --git a/core/modules/tour/src/TourTipPluginInterface.php b/core/modules/tour/src/TourTipPluginInterface.php
deleted file mode 100644
index e486383e9099..000000000000
--- a/core/modules/tour/src/TourTipPluginInterface.php
+++ /dev/null
@@ -1,20 +0,0 @@
-<?php
-
-namespace Drupal\tour;
-
-@trigger_error('The ' . __NAMESPACE__ . '\TourTipPluginInterface is deprecated in drupal:10.1.0 and is removed from drupal:11.0.0. Implement ' . __NAMESPACE__ . '\TipPluginInterface instead. See https://www.drupal.org/node/3340701', E_USER_DEPRECATED);
-
-/**
- * Defines an interface for tour items.
- *
- * @see \Drupal\tour\Annotation\Tip
- * @see \Drupal\tour\TipPluginBase
- * @see \Drupal\tour\TipPluginManager
- * @see plugin_api
- *
- * @deprecated in drupal:10.1.0 and is removed from drupal:11.0.0. Implements
- *   TipPluginInterface instead.
- *
- * @see https://www.drupal.org/node/3340701
- */
-interface TourTipPluginInterface extends TipPluginInterface {}
diff --git a/core/modules/tour/src/TourViewBuilder.php b/core/modules/tour/src/TourViewBuilder.php
deleted file mode 100644
index b41f846be6a7..000000000000
--- a/core/modules/tour/src/TourViewBuilder.php
+++ /dev/null
@@ -1,139 +0,0 @@
-<?php
-
-namespace Drupal\tour;
-
-use Drupal\Core\Cache\Cache;
-use Drupal\Core\Entity\EntityViewBuilder;
-use Drupal\Component\Utility\Html;
-
-/**
- * Provides a Tour view builder.
- *
- * Note: Does not invoke any alter hooks. In other view
- * builders, the view alter hooks are run later in the process
- */
-class TourViewBuilder extends EntityViewBuilder {
-
-  /**
-   * {@inheritdoc}
-   */
-  public function viewMultiple(array $entities = [], $view_mode = 'full', $langcode = NULL) {
-    /** @var \Drupal\tour\TourInterface[] $entities */
-    $tour = [];
-    $cache_tags = [];
-    $total_tips = 0;
-    foreach ($entities as $entity_id => $entity) {
-      $tour[$entity_id] = $entity->getTips();
-      $total_tips += count($tour[$entity_id]);
-      $cache_tags = Cache::mergeTags($cache_tags, $entity->getCacheTags());
-    }
-
-    $items = [];
-    foreach ($tour as $tour_id => $tips) {
-      $tourEntity = $entities[$tour_id];
-
-      foreach ($tips as $index => $tip) {
-        $classes = [
-          'tip-module-' . Html::getClass($tourEntity->getModule()),
-          'tip-type-' . Html::getClass($tip->getPluginId()),
-          'tip-' . Html::getClass($tip->id()),
-        ];
-
-        $selector = $tip->getSelector();
-        $location = $tip->getLocation();
-
-        $body_render_array = $tip->getBody();
-        $body = (string) \Drupal::service('renderer')->renderInIsolation($body_render_array);
-        $output = [
-          'body' => $body,
-          'title' => $tip->getLabel(),
-        ];
-
-        $selector = $tip->getSelector();
-
-        if ($output) {
-          $items[] = [
-            'id' => $tip->id(),
-            'selector' => $selector,
-            'module' => $tourEntity->getModule(),
-            'type' => $tip->getPluginId(),
-            'counter' => $this->t('@tour_item of @total', [
-              '@tour_item' => $index + 1,
-              '@total' => $total_tips,
-            ]),
-            'attachTo' => [
-              'element' => $selector,
-              'on' => $location ?? 'bottom-start',
-            ],
-            // Shepherd expects classes to be provided as a string.
-            'classes' => implode(' ', $classes),
-          ] + $output;
-        }
-      }
-    }
-
-    // If there is at least one tour item, build the tour.
-    if ($items) {
-      $key = array_key_last($items);
-      $items[$key]['cancelText'] = t('End tour');
-    }
-
-    $build = [
-      '#cache' => [
-        'tags' => $cache_tags,
-      ],
-    ];
-
-    // If at least one tour was built, attach tips and the tour library.
-    if ($items) {
-      $build['#attached']['drupalSettings']['tourShepherdConfig'] = [
-        'defaultStepOptions' => [
-          'classes' => 'drupal-tour',
-          'cancelIcon' => [
-            'enabled' => TRUE,
-            'label' => $this->t('Close'),
-          ],
-          'modalOverlayOpeningPadding' => 3,
-          'scrollTo' => [
-            'behavior' => 'smooth',
-            'block' => 'center',
-          ],
-          'popperOptions' => [
-            'modifiers' => [
-              // Prevent overlap with the element being highlighted.
-              [
-                'name' => 'offset',
-                'options' => [
-                  'offset' => [-10, 20],
-                ],
-              ],
-              // Pad the arrows so they don't hit the edge of rounded corners.
-              [
-                'name' => 'arrow',
-                'options' => [
-                  'padding' => 12,
-                ],
-              ],
-              // Disable Shepherd's focusAfterRender modifier, which results in
-              // the tour item container being focused on any scroll or resize
-              // event.
-              [
-                'name' => 'focusAfterRender',
-                'enabled' => FALSE,
-              ],
-
-            ],
-          ],
-        ],
-        'useModalOverlay' => TRUE,
-      ];
-      // This property is used for storing the tour items. It may change without
-      // notice and should not be extended or modified in contrib.
-      // see: https://www.drupal.org/project/drupal/issues/3214593
-      $build['#attached']['drupalSettings']['_tour_internal'] = $items;
-      $build['#attached']['library'][] = 'tour/tour';
-    }
-    return $build;
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Functional/Block/BlockLayoutTourTest.php b/core/modules/tour/tests/src/Functional/Block/BlockLayoutTourTest.php
deleted file mode 100644
index 6b7cc2fa4bb0..000000000000
--- a/core/modules/tour/tests/src/Functional/Block/BlockLayoutTourTest.php
+++ /dev/null
@@ -1,54 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional\Block;
-
-use Drupal\Tests\tour\Functional\TourTestBase;
-
-/**
- * Tests the Block Layout tour.
- *
- * @group tour
- * @group legacy
- */
-class BlockLayoutTourTest extends TourTestBase {
-
-  /**
-   * An admin user with administrative permissions for Blocks.
-   *
-   * @var \Drupal\user\UserInterface
-   */
-  protected $adminUser;
-
-  /**
-   * Modules to enable.
-   *
-   * @var array
-   */
-  protected static $modules = ['block', 'tour'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    parent::setUp();
-    $this->adminUser = $this->drupalCreateUser(['administer blocks', 'access tour']);
-    $this->drupalLogin($this->adminUser);
-    $this->drupalPlaceBlock('local_actions_block');
-  }
-
-  /**
-   * Tests Block Layout tour tip availability.
-   */
-  public function testBlockLayoutTourTips() {
-    $this->drupalGet('admin/structure/block');
-    $this->assertTourTips();
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Functional/GenericTest.php b/core/modules/tour/tests/src/Functional/GenericTest.php
deleted file mode 100644
index ee621552d181..000000000000
--- a/core/modules/tour/tests/src/Functional/GenericTest.php
+++ /dev/null
@@ -1,15 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional;
-
-use Drupal\Tests\system\Functional\Module\GenericModuleTestBase;
-
-/**
- * Generic module test for tour.
- *
- * @group tour
- * @group legacy
- */
-class GenericTest extends GenericModuleTestBase {}
diff --git a/core/modules/tour/tests/src/Functional/Jsonapi/TourTest.php b/core/modules/tour/tests/src/Functional/Jsonapi/TourTest.php
deleted file mode 100644
index ca0f810741d7..000000000000
--- a/core/modules/tour/tests/src/Functional/Jsonapi/TourTest.php
+++ /dev/null
@@ -1,148 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional\Jsonapi;
-
-use Drupal\Core\Url;
-use Drupal\Tests\jsonapi\Functional\ConfigEntityResourceTestBase;
-use Drupal\tour\Entity\Tour;
-
-/**
- * JSON:API integration test for the "Tour" config entity type.
- *
- * @group tour
- * @group legacy
- */
-class TourTest extends ConfigEntityResourceTestBase {
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $modules = ['tour'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $entityTypeId = 'tour';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $resourceTypeName = 'tour--tour';
-
-  /**
-   * {@inheritdoc}
-   *
-   * @var \Drupal\tour\TourInterface
-   */
-  protected $entity;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUpAuthorization($method) {
-    $this->grantPermissionsToTestedRole(['access tour']);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function createEntity() {
-    $tour = Tour::create([
-      'id' => 'tour-llama',
-      'label' => 'Llama tour',
-      'langcode' => 'en',
-      'module' => 'tour',
-      'routes' => [
-        [
-          'route_name' => '<front>',
-        ],
-      ],
-      'tips' => [
-        'tour-llama-1' => [
-          'id' => 'tour-llama-1',
-          'plugin' => 'text',
-          'label' => 'Llama',
-          'body' => 'Who handle the awesomeness of llamas?',
-          'weight' => 100,
-          'selector' => '#tour-llama-1',
-        ],
-      ],
-    ]);
-    $tour->save();
-
-    return $tour;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function getExpectedDocument() {
-    $self_url = Url::fromUri('base:/jsonapi/tour/tour/' . $this->entity->uuid())->setAbsolute()->toString(TRUE)->getGeneratedUrl();
-    return [
-      'jsonapi' => [
-        'meta' => [
-          'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
-          ],
-        ],
-        'version' => '1.0',
-      ],
-      'links' => [
-        'self' => ['href' => $self_url],
-      ],
-      'data' => [
-        'id' => $this->entity->uuid(),
-        'type' => 'tour--tour',
-        'links' => [
-          'self' => ['href' => $self_url],
-        ],
-        'attributes' => [
-          'dependencies' => [],
-          'label' => 'Llama tour',
-          'langcode' => 'en',
-          'module' => 'tour',
-          'routes' => [
-            [
-              'route_name' => '<front>',
-            ],
-          ],
-          'status' => TRUE,
-          'tips' => [
-            'tour-llama-1' => [
-              'id' => 'tour-llama-1',
-              'plugin' => 'text',
-              'label' => 'Llama',
-              'body' => 'Who handle the awesomeness of llamas?',
-              'weight' => 100,
-              'selector' => '#tour-llama-1',
-            ],
-          ],
-          'drupal_internal__id' => 'tour-llama',
-        ],
-      ],
-    ];
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function getPostDocument() {
-    // @todo Update in https://www.drupal.org/node/2300677.
-    return [];
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function getExpectedUnauthorizedAccessMessage($method) {
-    return "The following permissions are required: 'access tour' OR 'administer site configuration'.";
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Functional/Language/LanguageTourTest.php b/core/modules/tour/tests/src/Functional/Language/LanguageTourTest.php
deleted file mode 100644
index 7de1c814584d..000000000000
--- a/core/modules/tour/tests/src/Functional/Language/LanguageTourTest.php
+++ /dev/null
@@ -1,73 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional\Language;
-
-use Drupal\Tests\tour\Functional\TourTestBase;
-
-/**
- * Tests tour functionality.
- *
- * @group tour
- * @group legacy
- */
-class LanguageTourTest extends TourTestBase {
-
-  /**
-   * An admin user with administrative permissions for views.
-   *
-   * @var \Drupal\user\UserInterface
-   */
-  protected $adminUser;
-
-  /**
-   * Modules to enable.
-   *
-   * @var array
-   */
-  protected static $modules = ['block', 'language', 'tour'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    parent::setUp();
-    $this->adminUser = $this->drupalCreateUser([
-      'administer languages',
-      'access tour',
-    ]);
-    $this->drupalLogin($this->adminUser);
-    $this->drupalPlaceBlock('local_actions_block');
-  }
-
-  /**
-   * Tests language tour tip availability.
-   */
-  public function testLanguageTour() {
-    $this->drupalGet('admin/config/regional/language');
-    $this->assertTourTips();
-  }
-
-  /**
-   * Go to add language page and check the tour tooltips.
-   */
-  public function testLanguageAddTour() {
-    $this->drupalGet('admin/config/regional/language/add');
-    $this->assertTourTips();
-  }
-
-  /**
-   * Go to edit language page and check the tour tooltips.
-   */
-  public function testLanguageEditTour() {
-    $this->drupalGet('admin/config/regional/language/edit/en');
-    $this->assertTourTips();
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Functional/Locale/LocaleTranslateStringTourTest.php b/core/modules/tour/tests/src/Functional/Locale/LocaleTranslateStringTourTest.php
deleted file mode 100644
index cb3d3f6d1fe2..000000000000
--- a/core/modules/tour/tests/src/Functional/Locale/LocaleTranslateStringTourTest.php
+++ /dev/null
@@ -1,63 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional\Locale;
-
-use Drupal\Tests\tour\Functional\TourTestBase;
-
-/**
- * Tests the Translate Interface tour.
- *
- * @group tour
- * @group legacy
- */
-class LocaleTranslateStringTourTest extends TourTestBase {
-
-  /**
-   * An admin user with administrative permissions to translate.
-   *
-   * @var \Drupal\user\UserInterface
-   */
-  protected $adminUser;
-
-  /**
-   * Modules to enable.
-   *
-   * @var array
-   */
-  protected static $modules = ['locale', 'tour'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    parent::setUp();
-    $this->adminUser = $this->drupalCreateUser([
-      'translate interface',
-      'access tour',
-      'administer languages',
-    ]);
-    $this->drupalLogin($this->adminUser);
-  }
-
-  /**
-   * Tests locale tour tip availability.
-   */
-  public function testTranslateStringTourTips() {
-    // Add another language so there are no missing form items.
-    $edit = [];
-    $edit['predefined_langcode'] = 'es';
-    $this->drupalGet('admin/config/regional/language/add');
-    $this->submitForm($edit, 'Add language');
-
-    $this->drupalGet('admin/config/regional/translate');
-    $this->assertTourTips();
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Functional/Rest/TourJsonAnonTest.php b/core/modules/tour/tests/src/Functional/Rest/TourJsonAnonTest.php
deleted file mode 100644
index 3e3b5545d3c9..000000000000
--- a/core/modules/tour/tests/src/Functional/Rest/TourJsonAnonTest.php
+++ /dev/null
@@ -1,32 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional\Rest;
-
-use Drupal\Tests\rest\Functional\AnonResourceTestTrait;
-
-/**
- * @group rest
- * @group legacy
- */
-class TourJsonAnonTest extends TourResourceTestBase {
-
-  use AnonResourceTestTrait;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $format = 'json';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $mimeType = 'application/json';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-}
diff --git a/core/modules/tour/tests/src/Functional/Rest/TourJsonBasicAuthTest.php b/core/modules/tour/tests/src/Functional/Rest/TourJsonBasicAuthTest.php
deleted file mode 100644
index 3a1bbe71ebb9..000000000000
--- a/core/modules/tour/tests/src/Functional/Rest/TourJsonBasicAuthTest.php
+++ /dev/null
@@ -1,42 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional\Rest;
-
-use Drupal\Tests\rest\Functional\BasicAuthResourceTestTrait;
-
-/**
- * @group rest
- * @group legacy
- */
-class TourJsonBasicAuthTest extends TourResourceTestBase {
-
-  use BasicAuthResourceTestTrait;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $modules = ['basic_auth'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $format = 'json';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $mimeType = 'application/json';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $auth = 'basic_auth';
-
-}
diff --git a/core/modules/tour/tests/src/Functional/Rest/TourJsonCookieTest.php b/core/modules/tour/tests/src/Functional/Rest/TourJsonCookieTest.php
deleted file mode 100644
index 5ac348713832..000000000000
--- a/core/modules/tour/tests/src/Functional/Rest/TourJsonCookieTest.php
+++ /dev/null
@@ -1,37 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional\Rest;
-
-use Drupal\Tests\rest\Functional\CookieResourceTestTrait;
-
-/**
- * @group rest
- * @group legacy
- */
-class TourJsonCookieTest extends TourResourceTestBase {
-
-  use CookieResourceTestTrait;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $format = 'json';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $mimeType = 'application/json';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $auth = 'cookie';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-}
diff --git a/core/modules/tour/tests/src/Functional/Rest/TourResourceTestBase.php b/core/modules/tour/tests/src/Functional/Rest/TourResourceTestBase.php
deleted file mode 100644
index b15d40a2dee0..000000000000
--- a/core/modules/tour/tests/src/Functional/Rest/TourResourceTestBase.php
+++ /dev/null
@@ -1,118 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional\Rest;
-
-use Drupal\Tests\rest\Functional\EntityResource\ConfigEntityResourceTestBase;
-use Drupal\tour\Entity\Tour;
-
-abstract class TourResourceTestBase extends ConfigEntityResourceTestBase {
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $modules = ['tour'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $entityTypeId = 'tour';
-
-  /**
-   * @var \Drupal\tour\TourInterface
-   */
-  protected $entity;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUpAuthorization($method) {
-    $this->grantPermissionsToTestedRole(['access tour']);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function createEntity() {
-    $tour = Tour::create([
-      'id' => 'tour-llama',
-      'label' => 'Llama tour',
-      'langcode' => 'en',
-      'module' => 'tour',
-      'routes' => [
-        [
-          'route_name' => '<front>',
-        ],
-      ],
-      'tips' => [
-        'tour-llama-1' => [
-          'id' => 'tour-llama-1',
-          'plugin' => 'text',
-          'label' => 'Llama',
-          'body' => 'Who handle the awesomeness of llamas?',
-          'weight' => 100,
-          'selector' => '#tour-llama-1',
-        ],
-      ],
-    ]);
-    $tour->save();
-
-    return $tour;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function getExpectedNormalizedEntity() {
-    return [
-      'dependencies' => [],
-      'id' => 'tour-llama',
-      'label' => 'Llama tour',
-      'langcode' => 'en',
-      'module' => 'tour',
-      'routes' => [
-        [
-          'route_name' => '<front>',
-        ],
-      ],
-      'status' => TRUE,
-      'tips' => [
-        'tour-llama-1' => [
-          'id' => 'tour-llama-1',
-          'plugin' => 'text',
-          'label' => 'Llama',
-          'body' => 'Who handle the awesomeness of llamas?',
-          'weight' => 100,
-          'selector' => '#tour-llama-1',
-        ],
-      ],
-      'uuid' => $this->entity->uuid(),
-    ];
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function getNormalizedPostEntity() {
-    // @todo Update in https://www.drupal.org/node/2300677.
-    return [];
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function getExpectedCacheContexts() {
-    return [
-      'user.permissions',
-    ];
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function getExpectedUnauthorizedAccessMessage($method) {
-    return "The following permissions are required: 'access tour' OR 'administer site configuration'.";
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Functional/Rest/TourXmlAnonTest.php b/core/modules/tour/tests/src/Functional/Rest/TourXmlAnonTest.php
deleted file mode 100644
index 621679deb201..000000000000
--- a/core/modules/tour/tests/src/Functional/Rest/TourXmlAnonTest.php
+++ /dev/null
@@ -1,34 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional\Rest;
-
-use Drupal\Tests\rest\Functional\AnonResourceTestTrait;
-use Drupal\Tests\rest\Functional\EntityResource\XmlEntityNormalizationQuirksTrait;
-
-/**
- * @group rest
- * @group legacy
- */
-class TourXmlAnonTest extends TourResourceTestBase {
-
-  use AnonResourceTestTrait;
-  use XmlEntityNormalizationQuirksTrait;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $format = 'xml';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $mimeType = 'text/xml; charset=UTF-8';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-}
diff --git a/core/modules/tour/tests/src/Functional/Rest/TourXmlBasicAuthTest.php b/core/modules/tour/tests/src/Functional/Rest/TourXmlBasicAuthTest.php
deleted file mode 100644
index d71501e0a8b0..000000000000
--- a/core/modules/tour/tests/src/Functional/Rest/TourXmlBasicAuthTest.php
+++ /dev/null
@@ -1,44 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional\Rest;
-
-use Drupal\Tests\rest\Functional\BasicAuthResourceTestTrait;
-use Drupal\Tests\rest\Functional\EntityResource\XmlEntityNormalizationQuirksTrait;
-
-/**
- * @group rest
- * @group legacy
- */
-class TourXmlBasicAuthTest extends TourResourceTestBase {
-
-  use BasicAuthResourceTestTrait;
-  use XmlEntityNormalizationQuirksTrait;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $modules = ['basic_auth'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $format = 'xml';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $mimeType = 'text/xml; charset=UTF-8';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $auth = 'basic_auth';
-
-}
diff --git a/core/modules/tour/tests/src/Functional/Rest/TourXmlCookieTest.php b/core/modules/tour/tests/src/Functional/Rest/TourXmlCookieTest.php
deleted file mode 100644
index 2fe2656f80d5..000000000000
--- a/core/modules/tour/tests/src/Functional/Rest/TourXmlCookieTest.php
+++ /dev/null
@@ -1,39 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional\Rest;
-
-use Drupal\Tests\rest\Functional\CookieResourceTestTrait;
-use Drupal\Tests\rest\Functional\EntityResource\XmlEntityNormalizationQuirksTrait;
-
-/**
- * @group rest
- * @group legacy
- */
-class TourXmlCookieTest extends TourResourceTestBase {
-
-  use CookieResourceTestTrait;
-  use XmlEntityNormalizationQuirksTrait;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $format = 'xml';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $mimeType = 'text/xml; charset=UTF-8';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $auth = 'cookie';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-}
diff --git a/core/modules/tour/tests/src/Functional/TourCacheTagsTest.php b/core/modules/tour/tests/src/Functional/TourCacheTagsTest.php
deleted file mode 100644
index 7bb039acef55..000000000000
--- a/core/modules/tour/tests/src/Functional/TourCacheTagsTest.php
+++ /dev/null
@@ -1,84 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional;
-
-use Drupal\Core\Url;
-use Drupal\Tests\system\Functional\Cache\PageCacheTagsTestBase;
-use Drupal\tour\Entity\Tour;
-use Drupal\user\Entity\Role;
-use Drupal\user\RoleInterface;
-
-/**
- * Tests the Tour entity's cache tags.
- *
- * @group tour
- * @group legacy
- */
-class TourCacheTagsTest extends PageCacheTagsTestBase {
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $modules = ['tour', 'tour_test'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    parent::setUp();
-
-    // Give anonymous users permission to view nodes, so that we can verify the
-    // cache tags of cached versions of node pages.
-    Role::load(RoleInterface::ANONYMOUS_ID)->grantPermission('access tour')
-      ->save();
-  }
-
-  /**
-   * Tests cache tags presence and invalidation of the Tour entity.
-   *
-   * Tests the following cache tags:
-   * - 'tour:<tour ID>'
-   */
-  public function testRenderedTour() {
-    $url = Url::fromRoute('tour_test.1');
-
-    // Prime the page cache.
-    $this->verifyPageCache($url, 'MISS');
-
-    // Verify a cache hit, but also the presence of the correct cache tags.
-    $expected_tags = [
-      'config:tour.tour.tour-test',
-      'config:user.role.anonymous',
-      'http_response',
-      'rendered',
-    ];
-    $this->verifyPageCache($url, 'HIT', $expected_tags);
-
-    // Verify that after modifying the tour, there is a cache miss.
-    Tour::load('tour-test')->save();
-    $this->verifyPageCache($url, 'MISS');
-
-    // Verify a cache hit.
-    $this->verifyPageCache($url, 'HIT', $expected_tags);
-
-    // Verify that after deleting the tour, there is a cache miss.
-    Tour::load('tour-test')->delete();
-    $this->verifyPageCache($url, 'MISS');
-
-    // Verify a cache hit.
-    $expected_tags = [
-      'config:user.role.anonymous',
-      'http_response',
-      'rendered',
-    ];
-    $this->verifyPageCache($url, 'HIT', $expected_tags);
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Functional/TourHelpPageTest.php b/core/modules/tour/tests/src/Functional/TourHelpPageTest.php
deleted file mode 100644
index 13ccab55b1f8..000000000000
--- a/core/modules/tour/tests/src/Functional/TourHelpPageTest.php
+++ /dev/null
@@ -1,155 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional;
-
-use Drupal\Tests\BrowserTestBase;
-
-/**
- * Verifies help page display of tours.
- *
- * @group help
- * @group legacy
- */
-class TourHelpPageTest extends BrowserTestBase {
-
-  /**
-   * Modules to enable, including some providing tours.
-   *
-   * @var array
-   */
-  protected static $modules = ['help', 'tour', 'locale', 'language'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-  /**
-   * User that can access tours and help.
-   *
-   * @var \Drupal\user\UserInterface
-   */
-  protected $tourUser;
-
-  /**
-   * A user who can access help but not tours.
-   *
-   * @var \Drupal\user\UserInterface
-   */
-  protected $noTourUser;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    parent::setUp();
-
-    // Create users. For the Tour user, include permissions for the language
-    // tours' parent pages, but not the translation tour's parent page. See
-    // self:getTourList().
-    $this->tourUser = $this->drupalCreateUser([
-      'access help pages',
-      'access tour',
-      'administer languages',
-    ]);
-    $this->noTourUser = $this->drupalCreateUser([
-      'access help pages',
-    ]);
-  }
-
-  /**
-   * Logs in users, tests help pages.
-   */
-  public function testHelp() {
-    $this->drupalLogin($this->tourUser);
-    $this->verifyHelp();
-
-    $this->drupalLogin($this->noTourUser);
-    $this->verifyHelp(FALSE);
-  }
-
-  /**
-   * Verifies the logged in user has access to the help properly.
-   *
-   * @param bool $tours_ok
-   *   (optional) TRUE (default) if the user should see tours, FALSE if not.
-   */
-  protected function verifyHelp($tours_ok = TRUE) {
-    $this->drupalGet('admin/help');
-
-    // All users should be able to see the module section.
-    $this->assertSession()->pageTextContains('Module overviews are provided by modules');
-    foreach ($this->getModuleList() as $name) {
-      $this->assertSession()->linkExists($name);
-    }
-
-    // Some users should be able to see the tour section.
-    if ($tours_ok) {
-      $this->assertSession()->pageTextContains('Tours guide you through workflows');
-    }
-    else {
-      $this->assertSession()->pageTextNotContains('Tours guide you through workflows');
-    }
-
-    $titles = $this->getTourList();
-
-    // Test the titles that should be links.
-    foreach ($titles[0] as $title) {
-      if ($tours_ok) {
-        $this->assertSession()->linkExists($title);
-      }
-      else {
-        $this->assertSession()->linkNotExists($title);
-        // Just test the first item in the list of links that should not
-        // be there, because the second matches the name of a module that is
-        // in the Module overviews section, so the link will be there and
-        // this test will fail. Testing one should be sufficient to verify
-        // the page is working correctly.
-        break;
-      }
-    }
-
-    // Test the titles that should not be links.
-    foreach ($titles[1] as $title) {
-      if ($tours_ok) {
-        $this->assertSession()->pageTextContains($title);
-        $this->assertSession()->linkNotExistsExact($title);
-      }
-      else {
-        $this->assertSession()->pageTextNotContains($title);
-        // Just test the first item in the list of text that should not
-        // be there, because the second matches part of the name of a module
-        // that is in the Module overviews section, so the text will be there
-        // and this test will fail. Testing one should be sufficient to verify
-        // the page is working correctly.
-        break;
-      }
-    }
-  }
-
-  /**
-   * Gets a list of modules to test for hook_help() pages.
-   *
-   * @return array
-   *   A list of module names to test.
-   */
-  protected function getModuleList() {
-    return ['Help', 'Tour'];
-  }
-
-  /**
-   * Gets a list of tours to test.
-   *
-   * @return array
-   *   A list of tour titles to test. The first array element is a list of tours
-   *   with links, and the second is a list of tours without links. Assumes
-   *   that the user being tested has 'administer languages' permission but
-   *   not 'translate interface'.
-   */
-  protected function getTourList() {
-    return [['Adding languages', 'Language'], ['Editing languages', 'Translation']];
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Functional/TourTest.php b/core/modules/tour/tests/src/Functional/TourTest.php
deleted file mode 100644
index 27ad89854985..000000000000
--- a/core/modules/tour/tests/src/Functional/TourTest.php
+++ /dev/null
@@ -1,311 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional;
-
-use Drupal\Core\Url;
-use Drupal\language\Entity\ConfigurableLanguage;
-use Drupal\tour\Entity\Tour;
-
-// cspell:ignore pioggia spagna
-
-/**
- * Tests the functionality of tour tips.
- *
- * @group tour
- * @group legacy
- */
-class TourTest extends TourTestBasic {
-
-  /**
-   * Modules to enable.
-   *
-   * @var array
-   */
-  protected static $modules = [
-    'block',
-    'tour',
-    'locale',
-    'language',
-    'tour_test',
-  ];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-  /**
-   * The permissions required for a logged in user to test tour tips.
-   *
-   * @var array
-   *   A list of permissions.
-   */
-  protected $permissions = ['access tour', 'administer languages'];
-
-  /**
-   * Tour tip attributes to be tested. Keyed by the path.
-   *
-   * @var array
-   *   An array of tip attributes, keyed by path.
-   */
-  protected $tips = [
-    'tour-test-1' => [],
-  ];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    parent::setUp();
-
-    $this->drupalPlaceBlock('local_actions_block', [
-      'theme' => 'claro',
-      'region' => 'content',
-    ]);
-  }
-
-  /**
-   * Tests tour functionality.
-   */
-  public function testTourFunctionality() {
-    // Navigate to tour-test-1 and verify the tour_test_1 tip is found with appropriate classes.
-    $this->drupalGet('tour-test-1');
-
-    // Test the TourTestBase class assertTourTips() method.
-    $tips = [];
-    $tips[] = ['data-id' => 'tour-test-1'];
-    $tips[] = ['data-class' => 'tour-test-5'];
-    $this->assertTourTips($tips);
-    $this->assertTourTips();
-
-    $tips = $this->getTourTips();
-
-    $href = Url::fromRoute('<front>', [], ['absolute' => TRUE])->toString();
-    $elements = [];
-    foreach ($tips as $tip) {
-      if ($tip['id'] == 'tour-test-1' && $tip['module'] == 'tour_test' && $tip['type'] == 'text' && str_contains($tip['body'], $href) && str_contains($tip['body'], 'Drupal')) {
-        $elements[] = $tip;
-      }
-    }
-    $this->assertCount(1, $elements, 'Found Token replacement.');
-
-    $elements = $this->findTip([
-      'id' => 'tour-test-1',
-      'title' => 'The first tip',
-    ]);
-    $this->assertCount(1, $elements, 'Found English variant of tip 1.');
-
-    $elements = $this->findTip([
-      'id' => 'tour-test-2',
-      'title' => 'The quick brown fox',
-    ]);
-    $this->assertNotCount(1, $elements, 'Did not find English variant of tip 2.');
-
-    $elements = $this->findTip([
-      'id' => 'tour-test-1',
-      'title' => 'La pioggia cade in spagna',
-    ]);
-    $this->assertNotCount(1, $elements, 'Did not find Italian variant of tip 1.');
-
-    // Ensure that plugins work.
-    $elements = [];
-    foreach ($tips as $tip) {
-      if (str_contains($tip['body'], 'http://local/image.png')) {
-        $elements[] = $tip;
-      }
-    }
-    $this->assertCount(1, $elements, 'Image plugin tip found.');
-
-    // Navigate to tour-test-2/subpath and verify the tour_test_2 tip is found.
-    $this->drupalGet('tour-test-2/subpath');
-
-    $elements = $this->findTip([
-      'id' => 'tour-test-2',
-      'title' => 'The quick brown fox',
-    ]);
-    $this->assertCount(1, $elements, 'Found English variant of tip 2.');
-
-    $elements = $this->findTip([
-      'id' => 'tour-test-1',
-      'title' => 'The first tip',
-    ]);
-    $this->assertNotCount(1, $elements, 'Did not find English variant of tip 1.');
-
-    // Enable Italian language and navigate to it/tour-test1 and verify italian
-    // version of tip is found.
-    ConfigurableLanguage::createFromLangcode('it')->save();
-    $this->drupalGet('it/tour-test-1');
-
-    $elements = $this->findTip([
-      'id' => 'tour-test-1',
-      'title' => 'La pioggia cade in spagna',
-    ]);
-    $this->assertCount(1, $elements, 'Found Italian variant of tip 1.');
-
-    $elements = $this->findTip([
-      'id' => 'tour-test-2',
-      'title' => 'The quick brown fox',
-    ]);
-    $this->assertNotCount(1, $elements, 'Did not find English variant of tip 1.');
-
-    // Programmatically create a tour for use through the remainder of the test.
-    $tour = Tour::create([
-      'id' => 'tour-entity-create-test-en',
-      'label' => 'Tour test english',
-      'langcode' => 'en',
-      'module' => 'system',
-      'routes' => [
-        ['route_name' => 'tour_test.1'],
-      ],
-      'tips' => [
-        'tour-test-1' => [
-          'id' => 'tour-code-test-1',
-          'plugin' => 'text',
-          'label' => 'The rain in spain is <strong>strong</strong>',
-          'body' => 'Falls mostly on the plain.',
-          'weight' => '100',
-          'selector' => '#tour-code-test-1',
-        ],
-        'tour-code-test-2' => [
-          'id' => 'tour-code-test-2',
-          'plugin' => 'image',
-          'label' => 'The awesome image',
-          'url' => 'http://local/image.png',
-          'weight' => 1,
-          'selector' => '#tour-code-test-2',
-        ],
-      ],
-    ]);
-    $tour->save();
-
-    // Ensure that a tour entity has the expected dependencies based on plugin
-    // providers and the module named in the configuration entity.
-    $dependencies = $tour->calculateDependencies()->getDependencies();
-    $this->assertEquals(['system', 'tour_test'], $dependencies['module']);
-
-    $this->drupalGet('tour-test-1');
-
-    // Load it back from the database and verify storage worked.
-    $entity_save_tip = Tour::load('tour-entity-create-test-en');
-    // Verify that hook_ENTITY_TYPE_load() integration worked.
-    $this->assertEquals('Load hooks work', $entity_save_tip->loaded);
-    // Verify that hook_ENTITY_TYPE_presave() integration worked.
-    $this->assertEquals('Tour test english alter', $entity_save_tip->label());
-
-    // Navigate to tour-test-1 and verify the new tip is found.
-    $this->drupalGet('tour-test-1');
-
-    $elements = $this->findTip([
-      'id' => 'tour-code-test-1',
-      'title' => 'The rain in spain is <strong>strong</strong>',
-    ]);
-    $this->assertCount(1, $elements, 'Found the required tip markup for tip 4');
-
-    // Verify that the weight sorting works by ensuring the lower weight item
-    // (tip 4) has the 'End tour' button.
-    $elements = $this->findTip([
-      'id' => 'tour-code-test-1',
-      'text' => 'End tour',
-    ]);
-    $this->assertCount(1, $elements, 'Found code tip was weighted last and had "End tour".');
-
-    // Test hook_tour_alter().
-    $this->assertSession()->responseContains('Altered by hook_tour_tips_alter');
-
-    // Navigate to tour-test-3 and verify the tour_test_1 tip is found with
-    // appropriate classes.
-    $this->drupalGet('tour-test-3/foo');
-
-    $elements = $this->findTip([
-      'id' => 'tour-test-1',
-      'module' => 'tour_test',
-      'type' => 'text',
-      'title' => 'The first tip',
-    ]);
-    $this->assertCount(1, $elements, 'Found English variant of tip 1.');
-
-    // Navigate to tour-test-3 and verify the tour_test_1 tip is not found with
-    // appropriate classes.
-    $this->drupalGet('tour-test-3/bar');
-
-    $elements = $this->findTip([
-      'id' => 'tour-test-1',
-      'module' => 'tour_test',
-      'type' => 'text',
-      'title' => 'The first tip',
-    ]);
-    $this->assertCount(0, $elements, 'Did not find English variant of tip 1.');
-  }
-
-  /**
-   * Tests enabling and disabling the tour tip functionality.
-   */
-  public function testStatus() {
-    // Set tour tip status as enabled.
-    $tour = Tour::load('tour-test');
-    $tour->setStatus(TRUE);
-    $tour->save();
-
-    $this->drupalGet('tour-test-1');
-    $this->assertSession()->statusCodeEquals(200);
-
-    // Tour tips should be visible on the page.
-    $this->assertTourTips();
-
-    $tour->setStatus(FALSE);
-    $tour->save();
-
-    // Navigate and verify the tour_test_1 tip is not found with
-    // appropriate classes.
-    $this->drupalGet('tour-test-1');
-    $this->assertSession()->statusCodeEquals(200);
-
-    // No tips expected as tour is disabled.
-    $this->assertTourTips(expectEmpty: TRUE);
-  }
-
-  /**
-   * Gets tour tips from the JavaScript drupalSettings variable.
-   *
-   * @return array
-   *   A list of tips and their data.
-   */
-  protected function getTourTips() {
-    $tips = [];
-    $drupalSettings = $this->getDrupalSettings();
-    if (isset($drupalSettings['_tour_internal'])) {
-      foreach ($drupalSettings['_tour_internal'] as $tip) {
-        $tips[] = $tip;
-      }
-    }
-
-    return $tips;
-  }
-
-  /**
-   * Find specific tips by their parameters in the list of tips.
-   *
-   * @param array $params
-   *   The list of search parameters and their values.
-   *
-   * @return array
-   *   A list of tips which match the parameters.
-   */
-  protected function findTip(array $params) {
-    $tips = $this->getTourTips();
-    $elements = [];
-    foreach ($tips as $tip) {
-      foreach ($params as $param => $value) {
-        if (isset($tip[$param]) && $tip[$param] != $value) {
-          continue 2;
-        }
-      }
-      $elements[] = $tip;
-    }
-
-    return $elements;
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Functional/TourTestBase.php b/core/modules/tour/tests/src/Functional/TourTestBase.php
deleted file mode 100644
index 0a01fa2de18d..000000000000
--- a/core/modules/tour/tests/src/Functional/TourTestBase.php
+++ /dev/null
@@ -1,79 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional;
-
-use Drupal\Tests\BrowserTestBase;
-
-/**
- * Base class for testing Tour functionality.
- */
-abstract class TourTestBase extends BrowserTestBase {
-
-  /**
-   * Asserts the presence of page elements for tour tips.
-   *
-   * @code
-   * // Basic example.
-   * $this->assertTourTips();
-   *
-   * // Advanced example. The following would be used for multi-page or
-   * // targeting a specific subset of tips.
-   * $tips = array();
-   * $tips[] = array('data-id' => 'foo');
-   * $tips[] = array('data-id' => 'bar');
-   * $tips[] = array('data-class' => 'baz');
-   * $this->assertTourTips($tips);
-   * @endcode
-   *
-   * @param array $tips
-   *   A list of tips which provide either a "data-id" or "data-class".
-   * @param bool $expectEmpty
-   *   Whether or not the field is expected to be Empty.
-   */
-  public function assertTourTips(array $tips = [], bool $expectEmpty = FALSE) {
-    // Get the rendered tips and their data-id and data-class attributes.
-    if (empty($tips)) {
-      // Tips are rendered as drupalSettings values.
-      $drupalSettings = $this->getDrupalSettings();
-      if (isset($drupalSettings['_tour_internal'])) {
-        foreach ($drupalSettings['_tour_internal'] as $tip) {
-          $tips[] = [
-            'selector' => $tip['selector'] ?? NULL,
-          ];
-        }
-      }
-    }
-
-    $tip_count = count($tips);
-    if ($tip_count === 0 && $expectEmpty) {
-      // No tips found as expected.
-      return;
-    }
-    if ($tip_count > 0 && $expectEmpty) {
-      $this->fail("No tips were expected but $tip_count were found");
-    }
-    $this->assertGreaterThan(0, $tip_count);
-
-    // Check for corresponding page elements.
-    $total = 0;
-    $modals = 0;
-    foreach ($tips as $tip) {
-      if (!empty($tip['data-id'])) {
-        $elements = $this->getSession()->getPage()->findAll('css', '#' . $tip['data-id']);
-        $this->assertCount(1, $elements, sprintf('Found corresponding page element for tour tip with id #%s', $tip['data-id']));
-      }
-      elseif (!empty($tip['data-class'])) {
-        $elements = $this->getSession()->getPage()->findAll('css', '.' . $tip['data-class']);
-        $this->assertNotEmpty($elements, sprintf("Page element for tour tip with class .%s should be present", $tip['data-class']));
-      }
-      else {
-        // It's a modal.
-        $modals++;
-      }
-      $total++;
-    }
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Functional/TourTestBasic.php b/core/modules/tour/tests/src/Functional/TourTestBasic.php
deleted file mode 100644
index de9e0793feff..000000000000
--- a/core/modules/tour/tests/src/Functional/TourTestBasic.php
+++ /dev/null
@@ -1,75 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional;
-
-/**
- * Simple tour tips test base.
- */
-abstract class TourTestBasic extends TourTestBase {
-
-  /**
-   * Tour tip attributes to be tested. Keyed by the path.
-   *
-   * @var array
-   *   An array of tip attributes, keyed by path.
-   *
-   * @code
-   * protected $tips = array(
-   *   '/foo/bar' => array(
-   *     array('data-id' => 'foo'),
-   *     array('data-class' => 'bar'),
-   *   ),
-   * );
-   * @endcode
-   */
-  protected $tips = [];
-
-  /**
-   * An admin user with administrative permissions for tour.
-   *
-   * @var \Drupal\user\UserInterface
-   */
-  protected $adminUser;
-
-  /**
-   * The permissions required for a logged in user to test tour tips.
-   *
-   * @var array
-   *   A list of permissions.
-   */
-  protected $permissions = ['access tour'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    parent::setUp();
-
-    // Make sure we are using distinct default and administrative themes for
-    // the duration of these tests.
-    $this->container->get('theme_installer')->install(['olivero', 'claro']);
-    $this->config('system.theme')
-      ->set('default', 'olivero')
-      ->set('admin', 'claro')
-      ->save();
-
-    $this->permissions[] = 'view the administration theme';
-
-    // Create an admin user to view tour tips.
-    $this->adminUser = $this->drupalCreateUser($this->permissions);
-    $this->drupalLogin($this->adminUser);
-  }
-
-  /**
-   * A simple tip test.
-   */
-  public function testTips() {
-    foreach ($this->tips as $path => $attributes) {
-      $this->drupalGet($path);
-      $this->assertTourTips($attributes);
-    }
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Functional/ViewsUi/ViewsUITourTest.php b/core/modules/tour/tests/src/Functional/ViewsUi/ViewsUITourTest.php
deleted file mode 100644
index 6834a5631db4..000000000000
--- a/core/modules/tour/tests/src/Functional/ViewsUi/ViewsUITourTest.php
+++ /dev/null
@@ -1,123 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Functional\ViewsUi;
-
-use Drupal\Tests\tour\Functional\TourTestBase;
-use Drupal\language\Entity\ConfigurableLanguage;
-
-/**
- * Tests the Views UI tour.
- *
- * @group tour
- * @group legacy
- */
-class ViewsUITourTest extends TourTestBase {
-
-  /**
-   * An admin user with administrative permissions for views.
-   *
-   * @var \Drupal\user\UserInterface
-   */
-  protected $adminUser;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-  /**
-   * String translation storage object.
-   *
-   * @var \Drupal\locale\StringStorageInterface
-   */
-  protected $localeStorage;
-
-  /**
-   * Modules to enable.
-   *
-   * @var array
-   */
-  protected static $modules = ['views_ui', 'tour', 'language', 'locale'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    parent::setUp();
-    $this->adminUser = $this->drupalCreateUser([
-      'administer views',
-      'access tour',
-    ]);
-    $this->drupalLogin($this->adminUser);
-  }
-
-  /**
-   * Tests views_ui tour tip availability.
-   */
-  public function testViewsUiTourTips() {
-    // Create a basic view that shows all content, with a page and a block
-    // display.
-    $view['label'] = $this->randomMachineName(16);
-    $view['id'] = $this->randomMachineName(16);
-    $view['page[create]'] = 1;
-    $view['page[path]'] = $this->randomMachineName(16);
-    $this->drupalGet('admin/structure/views/add');
-    $this->submitForm($view, 'Save and edit');
-    $this->assertTourTips();
-  }
-
-  /**
-   * Tests views_ui tour tip availability in a different language.
-   */
-  public function testViewsUiTourTipsTranslated() {
-    $langcode = 'nl';
-
-    // Add a default locale storage for this test.
-    $this->localeStorage = $this->container->get('locale.storage');
-
-    // Add Dutch language programmatically.
-    ConfigurableLanguage::createFromLangcode($langcode)->save();
-
-    // Handler titles that need translations.
-    $handler_titles = [
-      'Format',
-      'Fields',
-      'Sort criteria',
-      'Filter criteria',
-    ];
-
-    foreach ($handler_titles as $handler_title) {
-      // Create source string.
-      $source = $this->localeStorage->createString([
-        'source' => $handler_title,
-      ]);
-      $source->save();
-      $this->createTranslation($source, $langcode);
-    }
-
-    // Create a basic view that shows all content, with a page and a block
-    // display.
-    $view['label'] = $this->randomMachineName(16);
-    $view['id'] = $this->randomMachineName(16);
-    $view['page[create]'] = 1;
-    $view['page[path]'] = $this->randomMachineName(16);
-    // Load the page in dutch.
-    $this->drupalGet($langcode . '/admin/structure/views/add');
-    $this->submitForm($view, 'Save and edit');
-    $this->assertTourTips();
-  }
-
-  /**
-   * Creates single translation for source string.
-   */
-  public function createTranslation($source, $langcode) {
-    return $this->localeStorage->createTranslation([
-      'lid' => $source->lid,
-      'language' => $langcode,
-      'translation' => $this->randomMachineName(100),
-    ])->save();
-  }
-
-}
diff --git a/core/modules/tour/tests/src/FunctionalJavascript/TourJavascriptTest.php b/core/modules/tour/tests/src/FunctionalJavascript/TourJavascriptTest.php
deleted file mode 100644
index abfe8af37907..000000000000
--- a/core/modules/tour/tests/src/FunctionalJavascript/TourJavascriptTest.php
+++ /dev/null
@@ -1,134 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\FunctionalJavascript;
-
-use Drupal\FunctionalJavascriptTests\WebDriverTestBase;
-
-/**
- * General Tour tests that require JavaScript.
- *
- * @group tour
- * @group legacy
- */
-class TourJavascriptTest extends WebDriverTestBase {
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $modules = [
-    'tour',
-    'tour_test',
-    'toolbar',
-  ];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    parent::setUp();
-
-    $admin_user = $this->drupalCreateUser([
-      'access toolbar',
-      'access tour',
-    ]);
-    $this->drupalLogin($admin_user);
-  }
-
-  /**
-   * Confirm the 'tips' and 'tour 'query arguments.
-   */
-  public function testQueryArg() {
-    $assert_session = $this->assertSession();
-
-    $this->drupalGet('tour-test-1');
-    $assert_session->assertNoElementAfterWait('css', '.tip-tour-test-1');
-    $assert_session->pageTextContains('Where does the rain in Spain fail?');
-    $assert_session->pageTextNotContains('Im all these things');
-    $assert_session->pageTextNotContains('The first tip');
-
-    $this->drupalGet('tour-test-1', [
-      'query' => [
-        'tips' => 'tip-tour-test-6',
-      ],
-    ]);
-    $this->assertNotNull($assert_session->waitForElementVisible('css', '.tip-tour-test-6'));
-    $assert_session->pageTextContains('Im all these things');
-
-    $this->drupalGet('tour-test-1', [
-      'query' => [
-        'tour' => '1',
-      ],
-    ]);
-    $this->assertNotNull($assert_session->waitForElementVisible('css', '.tip-tour-test-1'));
-    $assert_session->pageTextContains('The first tip');
-  }
-
-  /**
-   * Tests stepping through a tour.
-   */
-  public function testGeneralTourUse() {
-    $page = $this->getSession()->getPage();
-    $assert_session = $this->assertSession();
-
-    $this->drupalGet('tour-test-1');
-
-    $assert_session->assertNoElementAfterWait('css', '.tip-tour-test-1');
-
-    // Open the tour.
-    $page->find('css', '#toolbar-tab-tour button')->press();
-
-    // Confirm the tour can be cancelled.
-    $tip_to_close = $assert_session->waitForElementVisible('css', '.shepherd-enabled.tip-tour-test-1');
-    $this->assertNotNull($tip_to_close);
-    $tip_text = $tip_to_close->getText();
-    $this->assertStringContainsString('always the best dressed', $tip_text);
-    $this->assertStringContainsString('1 of 3', $tip_text);
-    $this->assertStringNotContainsString('End tour', $tip_text);
-
-    // Cancel the tour.
-    $tip_to_close->find('css', '.shepherd-cancel-icon')->press();
-    $assert_session->assertNoElementAfterWait('css', '.tip-tour-test-1');
-    $assert_session->assertNoElementAfterWait('css', '.shepherd-enabled');
-
-    // Navigate through the three steps of the tour.
-    $page->find('css', '#toolbar-tab-tour button')->press();
-    $tip1 = $assert_session->waitForElementVisible('css', '.shepherd-enabled.tip-tour-test-1');
-    $this->assertNotNull($tip1);
-
-    // Click the next button.
-    $tip1->find('css', '.button--primary:contains("Next")')->press();
-
-    // The second tour tip should appear, confirm it has the expected content.
-    $tip2 = $assert_session->waitForElementVisible('css', '.shepherd-enabled.tip-tour-test-3');
-    $assert_session->pageTextNotContains('always the best dressed');
-    $tip_text = $tip2->getText();
-    $this->assertStringContainsString('The awesome image', $tip_text);
-    $this->assertStringContainsString('2 of 3', $tip_text);
-    $this->assertStringNotContainsString('End tour', $tip_text);
-
-    // Click the next button.
-    $tip2->find('css', '.button--primary:contains("Next")')->press();
-
-    // The third tour tip should appear, confirm it has the expected content.
-    $tip3 = $assert_session->waitForElementVisible('css', '.shepherd-enabled.tip-tour-test-6');
-    $assert_session->pageTextNotContains('The awesome image');
-    $tip_text = $tip3->getText();
-    $this->assertStringContainsString('Im all these things', $tip_text);
-    $this->assertStringContainsString('3 of 3', $tip_text);
-    $this->assertStringNotContainsString('Next', $tip_text);
-
-    // The final tip should have a button to end the tour. Press and confirm all
-    // tips removed.
-    $tip3->find('css', '.button--primary:contains("End tour")')->press();
-    $assert_session->assertNoElementAfterWait('css', '.shepherd-enabled');
-    $assert_session->pageTextNotContains('The awesome image');
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Kernel/TourPluginTest.php b/core/modules/tour/tests/src/Kernel/TourPluginTest.php
deleted file mode 100644
index 53384dc5b756..000000000000
--- a/core/modules/tour/tests/src/Kernel/TourPluginTest.php
+++ /dev/null
@@ -1,48 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Kernel;
-
-use Drupal\KernelTests\KernelTestBase;
-
-/**
- * Tests the functionality of tour plugins.
- *
- * @group tour
- * @group legacy
- */
-class TourPluginTest extends KernelTestBase {
-
-  /**
-   * Modules to enable.
-   *
-   * @var array
-   */
-  protected static $modules = ['tour'];
-
-  /**
-   * Stores the tour plugin manager.
-   *
-   * @var \Drupal\tour\TipPluginManager
-   */
-  protected $pluginManager;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    parent::setUp();
-
-    $this->installConfig(['tour']);
-    $this->pluginManager = $this->container->get('plugin.manager.tour.tip');
-  }
-
-  /**
-   * Tests tour plugins.
-   */
-  public function testTourPlugins() {
-    $this->assertCount(1, $this->pluginManager->getDefinitions(), 'Only tour plugins for the enabled modules were returned.');
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Kernel/TourTipLegacyTest.php b/core/modules/tour/tests/src/Kernel/TourTipLegacyTest.php
deleted file mode 100644
index 144c234577ad..000000000000
--- a/core/modules/tour/tests/src/Kernel/TourTipLegacyTest.php
+++ /dev/null
@@ -1,25 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Kernel;
-
-use Drupal\tour\TourTipPluginInterface;
-use PHPUnit\Framework\TestCase;
-use Symfony\Bridge\PhpUnit\ExpectDeprecationTrait;
-
-/**
- * @coversDefaultClass \Drupal\tour\TourTipPluginInterface
- * @group tour
- * @group legacy
- */
-class TourTipLegacyTest extends TestCase {
-  use ExpectDeprecationTrait;
-
-  public function testPluginHelperDeprecation(): void {
-    $this->expectDeprecation('The Drupal\tour\TourTipPluginInterface is deprecated in drupal:10.1.0 and is removed from drupal:11.0.0. Implement Drupal\tour\TipPluginInterface instead. See https://www.drupal.org/node/3340701');
-    $plugin = $this->createMock(TourTipPluginInterface::class);
-    $this->assertInstanceOf(TourTipPluginInterface::class, $plugin);
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Kernel/TourValidationTest.php b/core/modules/tour/tests/src/Kernel/TourValidationTest.php
deleted file mode 100644
index 4d62061c1bd6..000000000000
--- a/core/modules/tour/tests/src/Kernel/TourValidationTest.php
+++ /dev/null
@@ -1,51 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Kernel;
-
-use Drupal\KernelTests\Core\Config\ConfigEntityValidationTestBase;
-use Drupal\tour\Entity\Tour;
-
-/**
- * Tests validation of tour entities.
- *
- * @group tour
- * @group legacy
- */
-class TourValidationTest extends ConfigEntityValidationTestBase {
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $modules = ['tour'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    parent::setUp();
-
-    $this->entity = Tour::create([
-      'id' => 'test',
-      'label' => 'Test',
-      'module' => 'system',
-    ]);
-    $this->entity->save();
-  }
-
-  /**
-   * Tour IDs are atypical in that they allow dashes in the machine name.
-   */
-  public static function providerInvalidMachineNameCharacters(): array {
-    $cases = parent::providerInvalidMachineNameCharacters();
-    // Remove the existing test case that verifies a machine name containing
-    // periods is invalid.
-    self::assertSame(['dash-separated', FALSE], $cases['INVALID: dash separated']);
-    unset($cases['INVALID: dash separated']);
-    // And instead add a test case that verifies it is allowed for tours.
-    $cases['VALID: dash separated'] = ['dash-separated', TRUE];
-    return $cases;
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Unit/Entity/TourTest.php b/core/modules/tour/tests/src/Unit/Entity/TourTest.php
deleted file mode 100644
index 8a674644b9fe..000000000000
--- a/core/modules/tour/tests/src/Unit/Entity/TourTest.php
+++ /dev/null
@@ -1,142 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Unit\Entity;
-
-use Drupal\Tests\UnitTestCase;
-
-/**
- * @coversDefaultClass \Drupal\tour\Entity\Tour
- *
- * @group tour
- * @group legacy
- */
-class TourTest extends UnitTestCase {
-
-  /**
-   * Tests \Drupal\tour\Entity\Tour::hasMatchingRoute().
-   *
-   * @param array $routes
-   *   Array of routes as per the Tour::routes property.
-   * @param string $route_name
-   *   The route name to match.
-   * @param array $route_params
-   *   Array of route params.
-   * @param bool $result
-   *   Expected result.
-   *
-   * @covers ::hasMatchingRoute
-   *
-   * @dataProvider routeProvider
-   */
-  public function testHasMatchingRoute($routes, $route_name, $route_params, $result) {
-    $tour = $this->getMockBuilder('\Drupal\tour\Entity\Tour')
-      ->disableOriginalConstructor()
-      ->onlyMethods(['getRoutes'])
-      ->getMock();
-
-    $tour->expects($this->any())
-      ->method('getRoutes')
-      ->willReturn($routes);
-
-    $this->assertSame($result, $tour->hasMatchingRoute($route_name, $route_params));
-
-    $tour->resetKeyedRoutes();
-  }
-
-  /**
-   * Provides sample routes for testing.
-   */
-  public static function routeProvider() {
-    return [
-      // Simple match.
-      [
-        [
-          ['route_name' => 'some.route'],
-        ],
-        'some.route',
-        [],
-        TRUE,
-      ],
-      // Simple non-match.
-      [
-        [
-          ['route_name' => 'another.route'],
-        ],
-        'some.route',
-        [],
-        FALSE,
-      ],
-      // Empty params.
-      [
-        [
-          [
-            'route_name' => 'some.route',
-            'route_params' => ['foo' => 'bar'],
-          ],
-        ],
-        'some.route',
-        [],
-        FALSE,
-      ],
-      // Match on params.
-      [
-        [
-          [
-            'route_name' => 'some.route',
-            'route_params' => ['foo' => 'bar'],
-          ],
-        ],
-        'some.route',
-        ['foo' => 'bar'],
-        TRUE,
-      ],
-      // Non-matching params.
-      [
-        [
-          [
-            'route_name' => 'some.route',
-            'route_params' => ['foo' => 'bar'],
-          ],
-        ],
-        'some.route',
-        ['bar' => 'foo'],
-        FALSE,
-      ],
-      // One matching, one not.
-      [
-        [
-          [
-            'route_name' => 'some.route',
-            'route_params' => ['foo' => 'bar'],
-          ],
-          [
-            'route_name' => 'some.route',
-            'route_params' => ['bar' => 'foo'],
-          ],
-        ],
-        'some.route',
-        ['bar' => 'foo'],
-        TRUE,
-      ],
-      // One matching, one not.
-      [
-        [
-          [
-            'route_name' => 'some.route',
-            'route_params' => ['foo' => 'bar'],
-          ],
-          [
-            'route_name' => 'some.route',
-            'route_params' => ['foo' => 'baz'],
-          ],
-        ],
-        'some.route',
-        ['foo' => 'baz'],
-        TRUE,
-      ],
-    ];
-  }
-
-}
diff --git a/core/modules/tour/tests/src/Unit/TipPluginBaseTest.php b/core/modules/tour/tests/src/Unit/TipPluginBaseTest.php
deleted file mode 100644
index 86367c99e98e..000000000000
--- a/core/modules/tour/tests/src/Unit/TipPluginBaseTest.php
+++ /dev/null
@@ -1,33 +0,0 @@
-<?php
-
-declare(strict_types=1);
-
-namespace Drupal\Tests\tour\Unit;
-
-use Drupal\Tests\UnitTestCase;
-use Drupal\tour\TipPluginBase;
-
-/**
- * @coversDefaultClass \Drupal\tour\TipPluginBase
- *
- * @group tour
- * @group legacy
- */
-class TipPluginBaseTest extends UnitTestCase {
-
-  /**
-   * @covers ::getLocation
-   */
-  public function testGetLocationAssertion() {
-    $base_plugin = $this->getMockForAbstractClass(TipPluginBase::class, [], '', FALSE);
-
-    $base_plugin->set('position', 'right');
-    $this->assertSame('right', $base_plugin->getLocation());
-
-    $base_plugin->set('position', 'not_valid');
-    $this->expectException(\AssertionError::class);
-    $this->expectExceptionMessage('not_valid is not a valid Tour Tip position value');
-    $base_plugin->getLocation();
-  }
-
-}
diff --git a/core/modules/tour/tests/tour_test/config/install/language/it/tour.tour.tour-test.yml b/core/modules/tour/tests/tour_test/config/install/language/it/tour.tour.tour-test.yml
deleted file mode 100644
index 6e449139c82f..000000000000
--- a/core/modules/tour/tests/tour_test/config/install/language/it/tour.tour.tour-test.yml
+++ /dev/null
@@ -1,7 +0,0 @@
-label: Tour test italian
-tips:
-  tour-test-1:
-    # cspell:disable
-    label: La pioggia cade in spagna
-    body: Per lo più in pianura.
-    # cspell:disable
diff --git a/core/modules/tour/tests/tour_test/config/install/tour.tour.tour-test-2.yml b/core/modules/tour/tests/tour_test/config/install/tour.tour.tour-test-2.yml
deleted file mode 100644
index 657aee3d7a8c..000000000000
--- a/core/modules/tour/tests/tour_test/config/install/tour.tour.tour-test-2.yml
+++ /dev/null
@@ -1,16 +0,0 @@
-langcode: en
-id: tour-test-2
-label: 'Tour test english'
-module: tour_test
-routes:
-  -
-    route_name: tour_test.2
-tips:
-  tour-test-2:
-    id: tour-test-2
-    plugin: text
-    label: 'The quick brown fox'
-    weight: 2
-    selector: '#tour-test-2'
-    # cspell:disable-next-line
-    body: 'Per lo più in pianura.'
diff --git a/core/modules/tour/tests/tour_test/config/install/tour.tour.tour-test.yml b/core/modules/tour/tests/tour_test/config/install/tour.tour.tour-test.yml
deleted file mode 100644
index 5ca1fa48998b..000000000000
--- a/core/modules/tour/tests/tour_test/config/install/tour.tour.tour-test.yml
+++ /dev/null
@@ -1,39 +0,0 @@
-langcode: en
-id: tour-test
-label: 'Tour test english'
-module: tour_test
-routes:
-  -
-    route_name: tour_test.1
-  -
-    route_name: tour_test.3
-    route_params:
-      locale: foo
-tips:
-  tour-test-1:
-    id: tour-test-1
-    plugin: text
-    label: 'The first tip'
-    weight: 1
-    selector: '#tour-test-1'
-    body: 'Is <a href="[site:url]">[site:name]</a> always the best dressed?'
-  tour-test-action:
-    id: tour-test-3
-    plugin: text
-    label: 'The action'
-    weight: 2
-    selector: .button-action
-    body: 'The action button of awesome'
-  tour-test-3:
-    id: tour-test-3
-    plugin: image
-    label: 'The awesome image'
-    url: 'http://local/image.png'
-    weight: 1
-  tour-test-6:
-    id: tour-test-6
-    plugin: text
-    label: 'Im a list'
-    weight: 6
-    selector: '#tour-test-3'
-    body: '<p>Im all these things:</p><ul><li>Modal</li><li>Awesome</li></ul>'
diff --git a/core/modules/tour/tests/tour_test/config/schema/tour_test.schema.yml b/core/modules/tour/tests/tour_test/config/schema/tour_test.schema.yml
deleted file mode 100644
index a43cc2682f56..000000000000
--- a/core/modules/tour/tests/tour_test/config/schema/tour_test.schema.yml
+++ /dev/null
@@ -1,9 +0,0 @@
-# Schema for the configuration files of the Tour Test module.
-
-tour.tip.image:
-  type: tour.tip
-  label: 'Image tour tip'
-  mapping:
-    url:
-      type: uri
-      label: 'Image URL'
diff --git a/core/modules/tour/tests/tour_test/src/Controller/TourTestController.php b/core/modules/tour/tests/tour_test/src/Controller/TourTestController.php
deleted file mode 100644
index 56505aa0a0fe..000000000000
--- a/core/modules/tour/tests/tour_test/src/Controller/TourTestController.php
+++ /dev/null
@@ -1,74 +0,0 @@
-<?php
-
-namespace Drupal\tour_test\Controller;
-
-/**
- * Controller routines for tour_test routes.
- */
-class TourTestController {
-
-  /**
-   * Outputs some content for testing tours.
-   *
-   * @param string $locale
-   *   (optional) Dummy locale variable for testing routing parameters. Defaults
-   *   to 'foo'.
-   *
-   * @return array
-   *   Array of markup.
-   */
-  public function tourTest1($locale = 'foo') {
-    return [
-      'tip-1' => [
-        '#type' => 'container',
-        '#attributes' => [
-          'id' => 'tour-test-1',
-        ],
-        '#children' => t('Where does the rain in Spain fail?'),
-      ],
-      'tip-3' => [
-        '#type' => 'container',
-        '#attributes' => [
-          'id' => 'tour-test-3',
-        ],
-        '#children' => t('Tip created now?'),
-      ],
-      'tip-4' => [
-        '#type' => 'container',
-        '#attributes' => [
-          'id' => 'tour-test-4',
-        ],
-        '#children' => t('Tip created later?'),
-      ],
-      'tip-5' => [
-        '#type' => 'container',
-        '#attributes' => [
-          'class' => ['tour-test-5'],
-        ],
-        '#children' => t('Tip created later?'),
-      ],
-      'code-tip-1' => [
-        '#type' => 'container',
-        '#attributes' => [
-          'id' => 'tour-code-test-1',
-        ],
-        '#children' => t('Tip created now?'),
-      ],
-    ];
-  }
-
-  /**
-   * Outputs some content for testing tours.
-   */
-  public function tourTest2() {
-    return [
-      '#type' => 'container',
-      '#attributes' => [
-        'id' => 'tour-test-2',
-      ],
-      '#children' => t('Pangram example'),
-    ];
-
-  }
-
-}
diff --git a/core/modules/tour/tests/tour_test/src/Plugin/tour/tip/TipPluginImage.php b/core/modules/tour/tests/tour_test/src/Plugin/tour/tip/TipPluginImage.php
deleted file mode 100644
index ed4a14b2ff10..000000000000
--- a/core/modules/tour/tests/tour_test/src/Plugin/tour/tip/TipPluginImage.php
+++ /dev/null
@@ -1,87 +0,0 @@
-<?php
-
-namespace Drupal\tour_test\Plugin\tour\tip;
-
-use Drupal\Core\Plugin\ContainerFactoryPluginInterface;
-use Drupal\Core\Utility\Token;
-use Drupal\tour\TipPluginBase;
-use Symfony\Component\DependencyInjection\ContainerInterface;
-
-/**
- * Displays an image as a tip.
- *
- * @Tip(
- *   id = "image",
- *   title = @Translation("Image")
- * )
- */
-class TipPluginImage extends TipPluginBase implements ContainerFactoryPluginInterface {
-
-  /**
-   * The URL which is used for the image in this Tip.
-   *
-   * @var string
-   *   A URL used for the image.
-   */
-  protected $url;
-
-  /**
-   * The alt text which is used for the image in this Tip.
-   *
-   * @var string
-   *   An alt text used for the image.
-   */
-  protected $alt;
-
-  /**
-   * Token service.
-   *
-   * @var \Drupal\Core\Utility\Token
-   */
-  protected $token;
-
-  /**
-   * Constructs a \Drupal\tour\Plugin\tour\tip\TipPluginText object.
-   *
-   * @param array $configuration
-   *   A configuration array containing information about the plugin instance.
-   * @param string $plugin_id
-   *   The plugin_id for the plugin instance.
-   * @param mixed $plugin_definition
-   *   The plugin implementation definition.
-   * @param \Drupal\Core\Utility\Token $token
-   *   The token service.
-   */
-  public function __construct(array $configuration, $plugin_id, $plugin_definition, Token $token) {
-    parent::__construct($configuration, $plugin_id, $plugin_definition);
-    $this->token = $token;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
-    return new static($configuration, $plugin_id, $plugin_definition, $container->get('token'));
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getBody(): array {
-    $image = [
-      '#theme' => 'image',
-      '#uri' => $this->get('url'),
-      '#alt' => $this->get('alt'),
-    ];
-
-    return [
-      '#type' => 'html_tag',
-      '#tag' => 'p',
-      '#attributes' => [
-        'class' => ['tour-tip-image'],
-      ],
-      'image' => $image,
-    ];
-  }
-
-}
diff --git a/core/modules/tour/tests/tour_test/src/Plugin/tour/tip/TipPluginImageLegacy.php b/core/modules/tour/tests/tour_test/src/Plugin/tour/tip/TipPluginImageLegacy.php
deleted file mode 100644
index d1582fba46a1..000000000000
--- a/core/modules/tour/tests/tour_test/src/Plugin/tour/tip/TipPluginImageLegacy.php
+++ /dev/null
@@ -1,84 +0,0 @@
-<?php
-
-namespace Drupal\tour_test\Plugin\tour\tip;
-
-use Drupal\Component\Utility\Html;
-use Drupal\Core\Plugin\ContainerFactoryPluginInterface;
-use Drupal\Core\Utility\Token;
-use Drupal\tour\TipPluginBase;
-use Symfony\Component\DependencyInjection\ContainerInterface;
-
-/**
- * Displays an image as a tip.
- *
- * @Tip(
- *   id = "image_legacy",
- *   title = @Translation("Image Legacy")
- * )
- */
-class TipPluginImageLegacy extends TipPluginBase implements ContainerFactoryPluginInterface {
-
-  /**
-   * The URL which is used for the image in this Tip.
-   *
-   * @var string
-   *   A URL used for the image.
-   */
-  protected $url;
-
-  /**
-   * The alt text which is used for the image in this Tip.
-   *
-   * @var string
-   *   An alt text used for the image.
-   */
-  protected $alt;
-
-  /**
-   * Token service.
-   *
-   * @var \Drupal\Core\Utility\Token
-   */
-  protected $token;
-
-  /**
-   * Constructs a TipPluginImageLegacy object.
-   *
-   * @param array $configuration
-   *   A configuration array containing information about the plugin instance.
-   * @param string $plugin_id
-   *   The plugin_id for the plugin instance.
-   * @param mixed $plugin_definition
-   *   The plugin implementation definition.
-   * @param \Drupal\Core\Utility\Token $token
-   *   The token service.
-   */
-  public function __construct(array $configuration, $plugin_id, $plugin_definition, Token $token) {
-    parent::__construct($configuration, $plugin_id, $plugin_definition);
-    $this->token = $token;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
-    return new static($configuration, $plugin_id, $plugin_definition, $container->get('token'));
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getConfigurationOrNot() {
-    $image = [
-      '#theme' => 'image',
-      '#uri' => $this->get('url'),
-      '#alt' => $this->get('alt'),
-    ];
-
-    return [
-      'title' => Html::escape($this->get('label')),
-      'body' => $this->token->replace(\Drupal::service('renderer')->renderInIsolation($image)),
-    ];
-  }
-
-}
diff --git a/core/modules/tour/tests/tour_test/src/Plugin/tour/tip/TipPluginTextLegacy.php b/core/modules/tour/tests/tour_test/src/Plugin/tour/tip/TipPluginTextLegacy.php
deleted file mode 100644
index 9cfc2dba311e..000000000000
--- a/core/modules/tour/tests/tour_test/src/Plugin/tour/tip/TipPluginTextLegacy.php
+++ /dev/null
@@ -1,96 +0,0 @@
-<?php
-
-namespace Drupal\tour_test\Plugin\tour\tip;
-
-use Drupal\Component\Utility\Html;
-use Drupal\Core\Plugin\ContainerFactoryPluginInterface;
-use Drupal\Core\Utility\Token;
-use Drupal\tour\TipPluginBase;
-use Symfony\Component\DependencyInjection\ContainerInterface;
-
-/**
- * Displays some text as a tip.
- *
- * @Tip(
- *   id = "text_legacy",
- *   title = @Translation("Text Legacy")
- * )
- */
-class TipPluginTextLegacy extends TipPluginBase implements ContainerFactoryPluginInterface {
-
-  /**
-   * The body text which is used for render of this Text Tip.
-   *
-   * @var string
-   */
-  protected $body;
-
-  /**
-   * Token service.
-   *
-   * @var \Drupal\Core\Utility\Token
-   */
-  protected $token;
-
-  /**
-   * The forced position of where the tip will be located.
-   *
-   * @var string
-   */
-  protected $location;
-
-  /**
-   * Unique aria-id.
-   *
-   * @var string
-   */
-  protected $ariaId;
-
-  /**
-   * Constructs a TipPluginTextLegacy object.
-   *
-   * @param array $configuration
-   *   A configuration array containing information about the plugin instance.
-   * @param string $plugin_id
-   *   The plugin_id for the plugin instance.
-   * @param mixed $plugin_definition
-   *   The plugin implementation definition.
-   * @param \Drupal\Core\Utility\Token $token
-   *   The token service.
-   */
-  public function __construct(array $configuration, $plugin_id, $plugin_definition, Token $token) {
-    parent::__construct($configuration, $plugin_id, $plugin_definition);
-    $this->token = $token;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
-    return new static($configuration, $plugin_id, $plugin_definition, $container->get('token'));
-  }
-
-  /**
-   * Returns an ID that is guaranteed uniqueness.
-   *
-   * @return string
-   *   A unique id to be used to generate aria attributes.
-   */
-  public function getAriaId() {
-    if (!$this->ariaId) {
-      $this->ariaId = Html::getUniqueId($this->get('id'));
-    }
-    return $this->ariaId;
-  }
-
-  /**
-   * Returns body of the text tip.
-   *
-   * @return array
-   *   The tip body.
-   */
-  public function getBody(): array {
-    return [$this->get('body')];
-  }
-
-}
diff --git a/core/modules/tour/tests/tour_test/tour_test.info.yml b/core/modules/tour/tests/tour_test/tour_test.info.yml
deleted file mode 100644
index 7111abf8323d..000000000000
--- a/core/modules/tour/tests/tour_test/tour_test.info.yml
+++ /dev/null
@@ -1,7 +0,0 @@
-name: Tour module tests
-type: module
-description: Tests module for tour module.
-package: Testing
-version: VERSION
-dependencies:
-  - drupal:tour
diff --git a/core/modules/tour/tests/tour_test/tour_test.links.action.yml b/core/modules/tour/tests/tour_test/tour_test.links.action.yml
deleted file mode 100644
index db52cb580610..000000000000
--- a/core/modules/tour/tests/tour_test/tour_test.links.action.yml
+++ /dev/null
@@ -1,5 +0,0 @@
-tour_test_action:
-  route_name: tour_test.1_action
-  title: 'Tour test action'
-  appears_on:
-    - tour_test.1
diff --git a/core/modules/tour/tests/tour_test/tour_test.module b/core/modules/tour/tests/tour_test/tour_test.module
deleted file mode 100644
index a299ec88b8ab..000000000000
--- a/core/modules/tour/tests/tour_test/tour_test.module
+++ /dev/null
@@ -1,37 +0,0 @@
-<?php
-
-/**
- * @file
- * Provides tests for tour module.
- */
-
-use Drupal\Core\Entity\EntityInterface;
-
-/**
- * Implements hook_ENTITY_TYPE_load() for tour.
- */
-function tour_test_tour_load($entities) {
-  if (isset($entities['tour-entity-create-test-en'])) {
-    $entities['tour-entity-create-test-en']->loaded = 'Load hooks work';
-  }
-}
-
-/**
- * Implements hook_ENTITY_TYPE_presave() for tour.
- */
-function tour_test_tour_presave($entity) {
-  if ($entity->id() == 'tour-entity-create-test-en') {
-    $entity->set('label', $entity->label() . ' alter');
-  }
-}
-
-/**
- * Implements hook_tour_tips_alter().
- */
-function tour_test_tour_tips_alter(array &$tour_tips, EntityInterface $entity) {
-  foreach ($tour_tips as $tour_tip) {
-    if ($tour_tip->get('id') == 'tour-code-test-1') {
-      $tour_tip->set('body', 'Altered by hook_tour_tips_alter');
-    }
-  }
-}
diff --git a/core/modules/tour/tests/tour_test/tour_test.routing.yml b/core/modules/tour/tests/tour_test/tour_test.routing.yml
deleted file mode 100644
index 86ffeacb56d9..000000000000
--- a/core/modules/tour/tests/tour_test/tour_test.routing.yml
+++ /dev/null
@@ -1,30 +0,0 @@
-tour_test.1:
-  path: '/tour-test-1'
-  defaults:
-    _controller: '\Drupal\tour_test\Controller\TourTestController::tourTest1'
-  options:
-    _admin_route: TRUE
-  requirements:
-    _access: 'TRUE'
-
-tour_test.1_action:
-  path: '/tour-test-1/action'
-  defaults:
-    _controller: '\Drupal\tour_test\Controller\TourTestController::tourTest1'
-  requirements:
-    _access: 'TRUE'
-
-tour_test.2:
-  path: '/tour-test-2/subpath'
-  defaults:
-    _controller: '\Drupal\tour_test\Controller\TourTestController::tourTest2'
-  requirements:
-    _access: 'TRUE'
-
-tour_test.3:
-  path: '/tour-test-3/{locale}'
-  defaults:
-    locale: 'foo'
-    _controller: '\Drupal\tour_test\Controller\TourTestController::tourTest1'
-  requirements:
-    _access: 'TRUE'
diff --git a/core/modules/tour/tour.api.php b/core/modules/tour/tour.api.php
deleted file mode 100644
index 9c582c025a0b..000000000000
--- a/core/modules/tour/tour.api.php
+++ /dev/null
@@ -1,61 +0,0 @@
-<?php
-
-/**
- * @file
- * Describes API functions for tour module.
- */
-
-/**
- * @defgroup help_docs Help and documentation
- * @{
- * Documenting modules, themes, and install profiles
- *
- * @section sec_tour Tours
- * Modules can provide tours of administrative pages by creating tour config
- * files and placing them in their config/optional subdirectory. See
- * @link https://www.drupal.org/docs/8/api/tour-api/overview Tour API overview @endlink
- * for more information. The contributed
- * @link https://www.drupal.org/project/tour_ui Tour UI module @endlink
- * can also be used to create tour config files.
- * @}
- */
-
-/**
- * @addtogroup hooks
- * @{
- */
-
-/**
- * Allow modules to alter tour items before render.
- *
- * @param array $tour_tips
- *   Array of \Drupal\tour\TipPluginInterface items.
- * @param \Drupal\Core\Entity\EntityInterface $entity
- *   The tour which contains the $tour_tips.
- */
-function hook_tour_tips_alter(array &$tour_tips, \Drupal\Core\Entity\EntityInterface $entity) {
-  foreach ($tour_tips as $tour_tip) {
-    if ($tour_tip->get('id') == 'tour-code-test-1') {
-      $tour_tip->set('body', 'Altered by hook_tour_tips_alter');
-    }
-  }
-}
-
-/**
- * Allow modules to alter tip plugin definitions.
- *
- * @param array $info
- *   The array of tip plugin definitions, keyed by plugin ID.
- *
- * @see \Drupal\tour\Annotation\Tip
- */
-function hook_tour_tips_info_alter(&$info) {
-  // Swap out the class used for this tip plugin.
-  if (isset($info['text'])) {
-    $info['class'] = 'Drupal\my_module\Plugin\tour\tip\MyCustomTipPlugin';
-  }
-}
-
-/**
- * @} End of "addtogroup hooks".
- */
diff --git a/core/modules/tour/tour.info.yml b/core/modules/tour/tour.info.yml
deleted file mode 100644
index 70ad4e81e735..000000000000
--- a/core/modules/tour/tour.info.yml
+++ /dev/null
@@ -1,7 +0,0 @@
-name: Tour
-type: module
-description: 'Displays guided tours of the site interface.'
-package: Core
-lifecycle: deprecated
-lifecycle_link: https://www.drupal.org/docs/core-modules-and-themes/deprecated-and-obsolete-modules-and-themes#s-tour
-version: VERSION
diff --git a/core/modules/tour/tour.libraries.yml b/core/modules/tour/tour.libraries.yml
deleted file mode 100644
index 52cbc50573d8..000000000000
--- a/core/modules/tour/tour.libraries.yml
+++ /dev/null
@@ -1,18 +0,0 @@
-tour:
-  version: VERSION
-  js:
-    js/tour.js: {}
-  dependencies:
-    - core/jquery
-    - core/once
-    - core/drupal
-    # @todo Remove this in https://www.drupal.org/project/drupal/issues/3204011
-    - core/internal.backbone
-    - core/internal.shepherd
-    - tour/tour-styling
-
-tour-styling:
-  version: VERSION
-  css:
-    component:
-      css/tour.module.css: { media: screen }
diff --git a/core/modules/tour/tour.module b/core/modules/tour/tour.module
deleted file mode 100644
index da432ef9ed7f..000000000000
--- a/core/modules/tour/tour.module
+++ /dev/null
@@ -1,117 +0,0 @@
-<?php
-
-/**
- * @file
- * Main functions of the module.
- */
-
-use Drupal\Core\Routing\RouteMatchInterface;
-use Drupal\tour\Entity\Tour;
-
-/**
- * Implements hook_help().
- */
-function tour_help($route_name, RouteMatchInterface $route_match) {
-  switch ($route_name) {
-    case 'help.page.tour':
-      $output = '';
-      $output .= '<h2>' . t('About') . '</h2>';
-      $output .= '<p>' . t("The Tour module provides users with guided tours of the site interface. Each tour consists of several tips that highlight elements of the user interface, guide the user through a workflow, or explain key concepts of the website. For more information, see the <a href=':tour'>online documentation for the Tour module</a>.", [':tour' => 'https://www.drupal.org/documentation/modules/tour']) . '</p>';
-      $output .= '<h2>' . t('Uses') . '</h2>';
-      $output .= '<dl>';
-      $output .= '<dt>' . t('Viewing tours') . '</dt>';
-      $output .= '<dd>' . t("If a tour is available on a page, a <em>Tour</em> button will be visible in the toolbar. If you click this button the first tip of the tour will appear. The tour continues after clicking the <em>Next</em> button in the tip. To see a tour users must have the permission <em>Access tour</em> and JavaScript must be enabled in the browser") . '</dd>';
-      $output .= '<dt>' . t('Creating tours') . '</dt>';
-      $output .= '<dd>' . t("Tours can be written as YAML-documents with a text editor, or using the contributed <a href=':tour_ui'>Tour UI</a> module. For more information, see <a href=':doc_url'>the online documentation for writing tours</a>.", [':doc_url' => 'https://www.drupal.org/developing/api/tour', ':tour_ui' => 'https://www.drupal.org/project/tour_ui']) . '</dd>';
-      $output .= '</dl>';
-      return $output;
-  }
-}
-
-/**
- * Implements hook_toolbar().
- */
-function tour_toolbar() {
-  $items = [];
-  $items['tour'] = [
-    '#cache' => [
-      'contexts' => [
-        'user.permissions',
-      ],
-    ],
-  ];
-
-  if (!\Drupal::currentUser()->hasPermission('access tour')) {
-    return $items;
-  }
-
-  $items['tour'] += [
-    '#type' => 'toolbar_item',
-    'tab' => [
-      '#type' => 'html_tag',
-      '#tag' => 'button',
-      '#value' => t('Tour'),
-      '#attributes' => [
-        'class' => ['toolbar-icon', 'toolbar-icon-help'],
-        'aria-pressed' => 'false',
-        'type' => 'button',
-      ],
-    ],
-    '#wrapper_attributes' => [
-      'class' => ['tour-toolbar-tab', 'hidden'],
-      'id' => 'toolbar-tab-tour',
-    ],
-    '#attached' => [
-      'library' => [
-        'tour/tour',
-      ],
-    ],
-  ];
-
-  return $items;
-}
-
-/**
- * Implements hook_page_bottom().
- */
-function tour_page_bottom(array &$page_bottom) {
-  if (!\Drupal::currentUser()->hasPermission('access tour')) {
-    return;
-  }
-
-  // Load all of the items and match on route name.
-  $route_match = \Drupal::routeMatch();
-  $route_name = $route_match->getRouteName();
-
-  $results = \Drupal::entityQuery('tour')
-    ->condition('routes.*.route_name', $route_name)
-    ->condition('status', TRUE)
-    ->execute();
-  if (!empty($results) && $tours = Tour::loadMultiple(array_keys($results))) {
-    foreach ($tours as $id => $tour) {
-      // Match on params.
-      if (!$tour->hasMatchingRoute($route_name, $route_match->getRawParameters()->all())) {
-        unset($tours[$id]);
-      }
-    }
-    if (!empty($tours)) {
-      $page_bottom['tour'] = \Drupal::entityTypeManager()
-        ->getViewBuilder('tour')
-        ->viewMultiple($tours, 'full');
-    }
-  }
-}
-
-/**
- * Implements hook_ENTITY_TYPE_insert() for tour entities.
- */
-function tour_tour_insert($entity) {
-  \Drupal::service('plugin.manager.tour.tip')->clearCachedDefinitions();
-}
-
-/**
- * Implements hook_ENTITY_TYPE_update() for tour entities.
- */
-function tour_tour_update($entity) {
-  \Drupal::service('plugin.manager.tour.tip')->clearCachedDefinitions();
-}
diff --git a/core/modules/tour/tour.permissions.yml b/core/modules/tour/tour.permissions.yml
deleted file mode 100644
index d5a4ebcc95ad..000000000000
--- a/core/modules/tour/tour.permissions.yml
+++ /dev/null
@@ -1,2 +0,0 @@
-access tour:
-  title: 'Access tours'
diff --git a/core/modules/tour/tour.services.yml b/core/modules/tour/tour.services.yml
deleted file mode 100644
index 38f310e9975e..000000000000
--- a/core/modules/tour/tour.services.yml
+++ /dev/null
@@ -1,4 +0,0 @@
-services:
-  plugin.manager.tour.tip:
-    class: Drupal\tour\TipPluginManager
-    parent: default_plugin_manager
diff --git a/core/themes/claro/claro.info.yml b/core/themes/claro/claro.info.yml
index dc9b06eada01..daf248154b29 100644
--- a/core/themes/claro/claro.info.yml
+++ b/core/themes/claro/claro.info.yml
@@ -140,8 +140,6 @@ libraries-extend:
     - claro/system.admin
   core/drupal.autocomplete:
     - claro/autocomplete
-  tour/tour-styling:
-    - claro/tour-styling
   shortcut/drupal.shortcut:
     - claro/drupal.shortcut
   core/drupal.ajax:
diff --git a/core/themes/claro/claro.libraries.yml b/core/themes/claro/claro.libraries.yml
index a4eb33cc637a..0c362bb8ed23 100644
--- a/core/themes/claro/claro.libraries.yml
+++ b/core/themes/claro/claro.libraries.yml
@@ -138,12 +138,6 @@ drupal.node.preview:
     theme:
       css/components/node-preview.css: {}
 
-tour-styling:
-  version: VERSION
-  css:
-    theme:
-      css/theme/tour.theme.css: {}
-
 media-form:
   version: VERSION
   css:
diff --git a/core/themes/claro/css/theme/tour.theme.css b/core/themes/claro/css/theme/tour.theme.css
deleted file mode 100644
index 72d900b27087..000000000000
--- a/core/themes/claro/css/theme/tour.theme.css
+++ /dev/null
@@ -1,104 +0,0 @@
-/*
- * DO NOT EDIT THIS FILE.
- * See the following change record for more information,
- * https://www.drupal.org/node/3084859
- * @preserve
- */
-
-/**
- * @file
- * Styles for Tour theme.
- */
-
-/* Default styles for the container */
-
-.shepherd-element {
-  color: #fff;
-  border-radius: 0.3125rem;
-  background-color: #000;
-  background-color: rgba(0, 0, 0, 0.8);
-}
-
-.shepherd-element:focus {
-  outline: 2px dotted transparent;
-  box-shadow:
-    0 0 0 2px var(--color-white),
-    0 0 0 5px var(--color-focus);
-}
-
-/* Mobile */
-
-@media only screen and (max-width: 47.9375rem) {
-  .shepherd-element {
-    border-radius: 0;
-  }
-}
-
-.shepherd-arrow::before {
-  background: none;
-}
-
-/* Add a little css triangle pip, older browser just miss out on the fanciness of it. */
-
-.shepherd-arrow {
-  border: solid 14px rgba(0, 0, 0, 0.8);
-}
-
-.shepherd-element[data-popper-placement^="top"] > .shepherd-arrow {
-  bottom: -1.75rem;
-  border-right-color: transparent;
-  border-bottom-color: transparent;
-  border-left-color: transparent;
-}
-
-.shepherd-element[data-popper-placement^="bottom"] > .shepherd-arrow {
-  top: -1.75rem;
-  border-top-color: transparent;
-  border-right-color: transparent;
-  border-left-color: transparent;
-}
-
-.shepherd-element[data-popper-placement^="left"] > .shepherd-arrow {
-  right: -1.75rem;
-  border-top-color: transparent;
-  border-right-color: transparent; /* LTR */
-  border-bottom-color: transparent;
-}
-
-.shepherd-element[data-popper-placement^="right"] > .shepherd-arrow {
-  left: -1.75rem;
-  border-top-color: transparent;
-  border-bottom-color: transparent;
-  border-left-color: transparent; /* LTR */
-}
-
-.shepherd-text p {
-  line-height: 1.385em;
-}
-
-.shepherd-text a {
-  color: var(--color-sunglow);
-}
-
-.shepherd-cancel-icon {
-  color: #fff;
-  font-size: 1.4em;
-  font-weight: bold;
-  line-height: 1;
-}
-
-.shepherd-cancel-icon:hover,
-.shepherd-cancel-icon:focus {
-  color: rgba(255, 255, 255, 0.9);
-}
-
-.shepherd-button {
-  margin: 0;
-}
-
-.shepherd-content *:focus {
-  outline: 2px dotted transparent;
-  box-shadow:
-    0 0 0 2px var(--color-white),
-    0 0 0 5px var(--color-focus);
-}
diff --git a/core/themes/claro/css/theme/tour.theme.pcss.css b/core/themes/claro/css/theme/tour.theme.pcss.css
deleted file mode 100644
index e1f6dd080530..000000000000
--- a/core/themes/claro/css/theme/tour.theme.pcss.css
+++ /dev/null
@@ -1,88 +0,0 @@
-/**
- * @file
- * Styles for Tour theme.
- */
-
-/* Default styles for the container */
-.shepherd-element {
-  color: #fff;
-  border-radius: 5px;
-  background-color: #000;
-  background-color: rgba(0, 0, 0, 0.8);
-}
-
-.shepherd-element:focus {
-  outline: 2px dotted transparent;
-  box-shadow: 0 0 0 2px var(--color-white), 0 0 0 5px var(--color-focus);
-}
-
-/* Mobile */
-@media only screen and (max-width: 767px) {
-  .shepherd-element {
-    border-radius: 0;
-  }
-}
-
-.shepherd-arrow::before {
-  background: none;
-}
-
-/* Add a little css triangle pip, older browser just miss out on the fanciness of it. */
-.shepherd-arrow {
-  border: solid 14px rgba(0, 0, 0, 0.8);
-}
-
-.shepherd-element[data-popper-placement^=top] > .shepherd-arrow {
-  bottom: -28px;
-  border-right-color: transparent;
-  border-bottom-color: transparent;
-  border-left-color: transparent;
-}
-
-.shepherd-element[data-popper-placement^=bottom] > .shepherd-arrow {
-  top: -28px;
-  border-top-color: transparent;
-  border-right-color: transparent;
-  border-left-color: transparent;
-}
-
-.shepherd-element[data-popper-placement^=left] > .shepherd-arrow {
-  right: -28px;
-  border-top-color: transparent;
-  border-right-color: transparent; /* LTR */
-  border-bottom-color: transparent;
-}
-
-.shepherd-element[data-popper-placement^=right] > .shepherd-arrow {
-  left: -28px;
-  border-top-color: transparent;
-  border-bottom-color: transparent;
-  border-left-color: transparent; /* LTR */
-}
-
-.shepherd-text p {
-  line-height: 1.385em;
-}
-.shepherd-text a {
-  color: var(--color-sunglow);
-}
-
-.shepherd-cancel-icon {
-  color: #fff;
-  font-size: 1.4em;
-  font-weight: bold;
-  line-height: 1;
-}
-.shepherd-cancel-icon:hover,
-.shepherd-cancel-icon:focus {
-  color: rgba(255, 255, 255, 0.9);
-}
-
-.shepherd-button {
-  margin: 0;
-}
-
-.shepherd-content *:focus {
-  outline: 2px dotted transparent;
-  box-shadow: 0 0 0 2px var(--color-white), 0 0 0 5px var(--color-focus);
-}
diff --git a/core/themes/olivero/css/components/site-header.css b/core/themes/olivero/css/components/site-header.css
index 6dc7c5162ac3..ecbdc1ea9cc7 100644
--- a/core/themes/olivero/css/components/site-header.css
+++ b/core/themes/olivero/css/components/site-header.css
@@ -14,7 +14,7 @@
   position: relative;
   /**
    * Ensure mobile site header is always above other elements including
-   * contextual links, and Tour.
+   * contextual links.
    */
   z-index: 101;
 }
diff --git a/core/themes/olivero/css/components/site-header.pcss.css b/core/themes/olivero/css/components/site-header.pcss.css
index 1ab7251194de..ee6822646357 100644
--- a/core/themes/olivero/css/components/site-header.pcss.css
+++ b/core/themes/olivero/css/components/site-header.pcss.css
@@ -9,7 +9,7 @@
   position: relative;
   /**
    * Ensure mobile site header is always above other elements including
-   * contextual links, and Tour.
+   * contextual links.
    */
   z-index: 101;
 
diff --git a/core/themes/stable9/css/tour/tour.module.css b/core/themes/stable9/css/tour/tour.module.css
deleted file mode 100644
index b5de43f199e8..000000000000
--- a/core/themes/stable9/css/tour/tour.module.css
+++ /dev/null
@@ -1,173 +0,0 @@
-/**
- * @file
- * Styling for tour module.
- */
-
-/* Tab appearance. */
-.toolbar .toolbar-bar .tour-toolbar-tab.toolbar-tab {
-  float: right; /* LTR */
-}
-[dir="rtl"] .toolbar .toolbar-bar .tour-toolbar-tab.toolbar-tab {
-  float: left;
-}
-
-/* Override placement of the tour progress indicator. */
-.tour-progress {
-  position: absolute;
-  right: 20px; /* LTR */
-  bottom: 20px;
-}
-[dir="rtl"] .tour-progress {
-  right: auto;
-  left: 20px;
-}
-
-/* Default styles for the container */
-.joyride-tip-guide {
-  position: absolute;
-  z-index: 101;
-  top: 0;
-  left: 0;
-  display: none;
-  width: 300px;
-  background: #fff;
-}
-
-@media only screen and (max-width: 767px) {
-  .joyride-tip-guide {
-    left: 2.5%;
-    width: 85%;
-  }
-}
-
-.joyride-content-wrapper {
-  position: relative;
-  padding: 20px 50px 20px 20px;  /* LTR */
-}
-[dir="rtl"] .joyride-content-wrapper {
-  padding: 20px 20px 20px 50px;
-}
-
-/* Add a little css triangle pip, older browser just miss out on the fanciness of it. */
-.joyride-tip-guide .joyride-nub {
-  position: absolute;
-  left: 22px;
-  display: block;
-  width: 0;
-  height: 0;
-}
-
-.joyride-tip-guide .joyride-nub.top {
-  top: -28px;
-  bottom: auto;
-}
-
-.joyride-tip-guide .joyride-nub.bottom {
-  bottom: -28px;
-}
-
-.joyride-tip-guide .joyride-nub.right {
-  top: 22px;
-  right: -28px;
-  bottom: auto;
-  left: auto;
-}
-
-.joyride-tip-guide .joyride-nub.left {
-  top: 22px;
-  right: auto;
-  bottom: auto;
-  left: -28px;
-}
-
-.joyride-tip-guide .joyride-nub.top-right {
-  top: -28px;
-  right: 28px;
-  bottom: auto;
-  left: auto;
-}
-
-.joyride-tip-guide .tour-tip-label {
-  margin-top: 0;
-}
-
-.joyride-tip-guide p {
-  margin: 0 0 1.4em;
-}
-
-.joyride-timer-indicator-wrap {
-  position: absolute;
-  right: 17px;
-  bottom: 16px;
-  width: 50px;
-  height: 3px;
-}
-.joyride-timer-indicator {
-  display: block;
-  width: 0;
-  height: inherit;
-}
-
-.joyride-close-tip {
-  position: absolute;
-  top: 20px;
-  right: 20px; /* LTR */
-  line-height: 1em;
-}
-[dir="rtl"] .joyride-close-tip {
-  right: auto;
-  left: 20px;
-}
-
-.joyride-modal-bg {
-  position: fixed;
-  z-index: 100;
-  top: 0;
-  left: 0;
-  display: none;
-  width: 100%;
-  height: 100%;
-  cursor: pointer;
-}
-
-.joyride-expose-wrapper {
-  position: absolute;
-  z-index: 102;
-}
-
-.joyride-expose-cover {
-  position: absolute;
-  z-index: 10000;
-  top: 0;
-  left: 0;
-}
-
-/* In Joyride, display: block is added as an inline style. */
-.shepherd-element.joyride-tip-guide {
-  display: block;
-}
-.shepherd-element.joyride-tip-guide[hidden] {
-  display: none;
-}
-
-.shepherd-modal-overlay-container {
-  position: fixed;
-  z-index: 100;
-  top: 0;
-  left: 0;
-  overflow: hidden;
-  width: 100vw;
-  height: 0;
-  pointer-events: none;
-  opacity: 0;
-  fill-rule: evenodd;
-}
-
-.shepherd-modal-overlay-container.shepherd-modal-is-visible {
-  height: 100vh;
-  opacity: 0.5;
-}
-
-.shepherd-modal-overlay-container.shepherd-modal-is-visible path {
-  pointer-events: all;
-}
diff --git a/core/themes/stable9/js/tour.js b/core/themes/stable9/js/tour.js
deleted file mode 100644
index 3a565ee19107..000000000000
--- a/core/themes/stable9/js/tour.js
+++ /dev/null
@@ -1,139 +0,0 @@
-/**
- * @file
- * Provides backwards compatibility for Tours that no longer use Joyride.
- */
-
-((Drupal) => {
-  /**
-   * Converts the markup of a Shepherd tour tip to match Joyride.
-   *
-   * @param {Tour} shepherdTour
-   *   A ShepherdJS tour object.
-   *
-   * @internal
-   */
-  Drupal.tour.convertToJoyrideMarkup = (shepherdTour) => {
-    /**
-     * Changes the tag of an element.
-     *
-     * @param {HTMLElement} element
-     *  The element that will have its tag changed.
-     * @param {string} tag
-     *  The tag the element should be changed to.
-     */
-    const changeTag = (element, tag) => {
-      if (element) {
-        const newTagElement = document.createElement(tag);
-        [...element.attributes].forEach((attr) => {
-          newTagElement.setAttribute(attr.name, attr.value);
-        });
-        newTagElement.innerHTML = element.innerHTML;
-        element.parentNode.replaceChild(newTagElement, element);
-      }
-    };
-
-    // Create variables for the elements that will be rearranged.
-    const shepherdElement = shepherdTour.currentStep.el;
-    const shepherdContent = shepherdElement.querySelector('.shepherd-content');
-    const shepherdCancel = shepherdElement.querySelector(
-      '.shepherd-cancel-icon',
-    );
-    const shepherdTitle = shepherdElement.querySelector('.shepherd-title');
-    const shepherdText = shepherdElement.querySelector('.shepherd-text');
-    const shepherdNext = shepherdElement.querySelector('footer .button');
-    const tourProgress = shepherdElement.querySelector('.tour-progress');
-
-    // Add attributes to the elements so they match what they were when Joyride
-    // was providing Tour functionality.
-    shepherdElement.classList.add('joyride-tip-guide');
-    shepherdContent.classList.add('joyride-content-wrapper');
-    shepherdNext.classList.add('joyride-next-tip');
-    shepherdNext.setAttribute('href', '#');
-    shepherdNext.setAttribute('role', 'button');
-    shepherdNext.removeAttribute('type');
-    shepherdCancel.classList.add('joyride-close-tip');
-    shepherdCancel.removeAttribute('type');
-    shepherdCancel.setAttribute('href', '#');
-    shepherdCancel.setAttribute('role', 'button');
-    shepherdElement.setAttribute(
-      'data-index',
-      shepherdTour.currentStep.options.index,
-    );
-    shepherdElement.querySelector('footer').remove();
-
-    // Rearrange elements so their structure matches Joyride's.
-    shepherdContent.insertBefore(shepherdTitle, shepherdContent.firstChild);
-    shepherdContent.insertBefore(tourProgress, shepherdText.nextSibling);
-    shepherdContent.appendChild(shepherdCancel);
-    shepherdContent.querySelector('.shepherd-header').remove();
-    shepherdContent.insertBefore(shepherdNext, tourProgress.nextSibling);
-    shepherdCancel.innerHTML = '<span aria-hidden="true">×</span>';
-    shepherdTitle.classList.add('tour-tip-label');
-
-    // Convert elements to use the tags they used in Joyride.
-    changeTag(shepherdTitle, 'h2');
-
-    // Remove the wrapper Shepherd adds for tip content.
-    shepherdText.outerHTML = shepherdText.innerHTML;
-
-    // Convert the next and cancel buttons to links so they match Joyride's
-    // markup. They must be re-queried as they were potentially moved elsewhere
-    // in the DOM.
-    changeTag(shepherdElement.querySelector('.joyride-close-tip'), 'a');
-    changeTag(shepherdElement.querySelector('.joyride-next-tip'), 'a');
-
-    // The arrow protruding from a tip pointing to the element it references.
-    const shepherdArrow = shepherdElement.querySelector('.shepherd-arrow');
-
-    if (shepherdArrow) {
-      shepherdArrow.classList.add('joyride-nub');
-
-      if (shepherdTour.currentStep.options.attachTo.on) {
-        // Shepherd's positions are opposite of Joyride's as they specify the
-        // tip location relative to the corresponding element as opposed to
-        // their location on the tip itself.
-        const stepToTipPosition = {
-          bottom: 'top',
-          top: 'bottom',
-          left: 'right',
-          right: 'left',
-        };
-        shepherdArrow.classList.add(
-          // Split at '-' as shepherd positioning accommodates dash-delimited
-          // secondary axis positioning.
-          // shepherdTour.currentStep.options.attachTo.on.split('-')[0]
-          stepToTipPosition[
-            // Split at '-' as shepherd positioning accommodates dash-delimited
-            // secondary axis positioning.
-            shepherdTour.currentStep.options.attachTo.on.split('-')[0]
-          ],
-        );
-      }
-      changeTag(shepherdArrow, 'span');
-    } else {
-      // If there is no Shepherd arrow, there still needs to be markup for a
-      // non-displayed nub to match Joyride's markup.
-      const nub = document.createElement('span');
-      nub.classList.add('joyride-nub');
-      nub.setAttribute('style', 'display: none;');
-      shepherdElement.insertBefore(nub, shepherdElement.firstChild);
-    }
-
-    // When the next and cancel buttons were converted to links, they became
-    // new DOM elements that no longer have their associated event listeners.
-    // The events must be reintroduced here.
-    shepherdElement
-      .querySelector('.joyride-next-tip')
-      .addEventListener('click', (e) => {
-        e.preventDefault();
-        shepherdTour.next();
-      });
-    shepherdElement
-      .querySelector('.joyride-close-tip')
-      .addEventListener('click', (e) => {
-        e.preventDefault();
-        shepherdTour.cancel();
-      });
-    shepherdElement.querySelector('.joyride-next-tip').focus();
-  };
-})(Drupal);
diff --git a/core/themes/stable9/stable9.info.yml b/core/themes/stable9/stable9.info.yml
index dd81e741775d..cace78bf8a82 100644
--- a/core/themes/stable9/stable9.info.yml
+++ b/core/themes/stable9/stable9.info.yml
@@ -278,11 +278,6 @@ libraries-override:
       state:
         css/toolbar.menu.css: css/toolbar/toolbar.menu.css
 
-  tour/tour-styling:
-    css:
-      component:
-        css/tour.module.css: css/tour/tour.module.css
-
   update/drupal.update.admin:
     css:
       theme:
@@ -317,7 +312,3 @@ libraries-override:
     css:
       theme:
         css/field_ui_display_mode_table.css: css/field_ui/field_ui_display_mode_table.css
-
-libraries-extend:
-  tour/tour:
-    - stable9/tour
diff --git a/core/themes/stable9/stable9.libraries.yml b/core/themes/stable9/stable9.libraries.yml
index 828fdd4aa16d..98535a37153f 100644
--- a/core/themes/stable9/stable9.libraries.yml
+++ b/core/themes/stable9/stable9.libraries.yml
@@ -8,8 +8,3 @@ normalize:
   css:
     base:
       css/core/assets/vendor/normalize-css/normalize.css: { weight: -20 }
-
-tour:
-  version: VERSION
-  js:
-    js/tour.js: {}
-- 
GitLab