From 95d8c46463a033c7df05460d81f1c7f30938ab16 Mon Sep 17 00:00:00 2001
From: Rudy Crespo <rcrespo@gitlab.com>
Date: Wed, 10 Apr 2024 17:48:18 +0000
Subject: [PATCH] Access standalone settings pages from Value Stream Analytics

As part of the migration of the new/edit value stream form from the
modal in the Value Stream Analytics page to their own standalone
settings pages, changes the flow of editing/creating value streams at
 the group level so that when the "Edit" button or "New Value Stream"
 option is selected, users are taken to their respective settings pages.
---
 .../group/value_stream_analytics/index.md     |  15 ++-
 .../create_value_stream_form/constants.js     |   2 +-
 .../components/value_stream_select.vue        |  39 ++++--
 .../analytics/cycle_analytics/index.js        |   9 +-
 .../value_stream_form_content_header.vue      |   6 +-
 .../cycle_analytics/request_params.rb         |   6 +-
 .../multiple_value_streams_spec.rb            |  92 ++++++++++----
 .../analytics/cycle_analytics_spec.rb         |   2 +-
 .../components/value_stream_select_spec.js    | 119 +++++++++++++++++-
 .../analytics/cycle_analytics/mock_data.js    |   1 +
 locale/gitlab.pot                             |   3 -
 qa/qa/ee/page/value_stream_analytics.rb       |   2 +-
 .../helpers/cycle_analytics_helpers.rb        |   4 +-
 13 files changed, 245 insertions(+), 55 deletions(-)

diff --git a/doc/user/group/value_stream_analytics/index.md b/doc/user/group/value_stream_analytics/index.md
index 697f83b1fece7..fe32840569765 100644
--- a/doc/user/group/value_stream_analytics/index.md
+++ b/doc/user/group/value_stream_analytics/index.md
@@ -461,6 +461,11 @@ DETAILS:
 **Tier:** Premium, Ultimate
 **Offering:** GitLab.com, Self-managed, GitLab Dedicated
 
+> - **New value stream** feature [changed](https://gitlab.com/gitlab-org/gitlab/-/issues/381002) from a dialog to a page in GitLab 16.11 [with a flag](../../../administration/feature_flags.md) named `vsa_standalone_settings_page`. Disabled by default.
+
+FLAG:
+On self-managed GitLab, by default the **New value stream** feature is not available. To make it available, an administrator can enable the [feature flag](../../../administration/feature_flags.md) named `vsa_standalone_settings_page`. On GitLab.com and GitLab Dedicated, this feature is not available. This feature is not ready for production use.
+
 ### Create a value stream with GitLab default stages
 
 > - [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/221202) in GitLab 13.3
@@ -479,7 +484,7 @@ create custom stages in addition to those provided in the default template.
 1. To add a custom stage, select **Add another stage**.
    - Enter a name for the stage.
    - Select a **Start event** and a **Stop event**.
-1. Select **Create value stream**.
+1. Select **New value stream**.
 
 NOTE:
 If you have recently upgraded to GitLab Premium, it can take up to 30 minutes for data to collect and display.
@@ -494,13 +499,13 @@ When you create a value stream, you can create and add custom stages that align
 
 1. On the left sidebar, select **Search or go to** and find your project or group.
 1. Select **Analyze > Value Stream analytics**.
-1. Select **Create value stream**.
+1. Select **New value stream**.
 1. For each stage:
    - Enter a name for the stage.
    - Select a **Start event** and a **Stop event**.
 1. To add another stage, select **Add another stage**.
 1. To re-order the stages, select the up or down arrows.
-1. Select **Create value stream**.
+1. Select **New value stream**.
 
 #### Label-based stages for custom value streams
 
@@ -527,6 +532,10 @@ DETAILS:
 **Offering:** GitLab.com, Self-managed, GitLab Dedicated
 
 > - [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/267537) in GitLab 13.10.
+> - **Edit value stream** feature [changed](https://gitlab.com/gitlab-org/gitlab/-/issues/381002) from a dialog to a page in GitLab 16.11 [with a flag](../../../administration/feature_flags.md) named `vsa_standalone_settings_page`. Disabled by default.
+
+FLAG:
+On self-managed GitLab, by default the **Edit value stream** feature is not available. To make it available, an administrator can enable the [feature flag](../../../administration/feature_flags.md) named `vsa_standalone_settings_page`. On GitLab.com and GitLab Dedicated, this feature is not available. This feature is not ready for production use.
 
 After you create a value stream, you can customize it to suit your purposes. To edit a value stream:
 
diff --git a/ee/app/assets/javascripts/analytics/cycle_analytics/components/create_value_stream_form/constants.js b/ee/app/assets/javascripts/analytics/cycle_analytics/components/create_value_stream_form/constants.js
index 18095de1af39d..15842d3ae9d1b 100644
--- a/ee/app/assets/javascripts/analytics/cycle_analytics/components/create_value_stream_form/constants.js
+++ b/ee/app/assets/javascripts/analytics/cycle_analytics/components/create_value_stream_form/constants.js
@@ -4,7 +4,7 @@ export const NAME_MAX_LENGTH = 100;
 export const NAME_MIN_LENGTH = 3;
 
 export const i18n = {
-  FORM_TITLE: s__('CreateValueStreamForm|Create value stream'),
+  FORM_TITLE: s__('CreateValueStreamForm|New value stream'),
   EDIT_FORM_TITLE: s__('CreateValueStreamForm|Edit value stream'),
   EDIT_FORM_ACTION: s__('CreateValueStreamForm|Save value stream'),
   FORM_CREATED: s__("CreateValueStreamForm|'%{name}' Value Stream created"),
diff --git a/ee/app/assets/javascripts/analytics/cycle_analytics/components/value_stream_select.vue b/ee/app/assets/javascripts/analytics/cycle_analytics/components/value_stream_select.vue
index ae9ddc29e1422..ef5e78785d3a7 100644
--- a/ee/app/assets/javascripts/analytics/cycle_analytics/components/value_stream_select.vue
+++ b/ee/app/assets/javascripts/analytics/cycle_analytics/components/value_stream_select.vue
@@ -12,6 +12,7 @@ import { mapState, mapActions } from 'vuex';
 import { slugifyWithUnderscore } from '~/lib/utils/text_utility';
 import { sprintf, __, s__ } from '~/locale';
 import Tracking from '~/tracking';
+import glFeatureFlagsMixin from '~/vue_shared/mixins/gl_feature_flags_mixin';
 import ValueStreamForm from './value_stream_form.vue';
 
 const i18n = {
@@ -38,7 +39,8 @@ export default {
   directives: {
     GlModalDirective,
   },
-  mixins: [Tracking.mixin()],
+  mixins: [Tracking.mixin(), glFeatureFlagsMixin()],
+  inject: ['newValueStreamPath', 'editValueStreamPath'],
   props: {
     canEdit: {
       type: Boolean,
@@ -81,6 +83,20 @@ export default {
         name: this.selectedValueStreamName,
       });
     },
+    isVSAStandaloneSettingsPageEnabled() {
+      return this.glFeatures?.vsaStandaloneSettingsPage;
+    },
+    createValueStreamButtonHref() {
+      return this.isVSAStandaloneSettingsPageEnabled ? this.newValueStreamPath : null;
+    },
+    editValueStreamButtonHref() {
+      if (!this.isVSAStandaloneSettingsPageEnabled || !this.selectedValueStreamId) return null;
+
+      return this.editValueStreamPath.replace(':id', this.selectedValueStreamId);
+    },
+    valueStreamFormModalId() {
+      return !this.isVSAStandaloneSettingsPageEnabled && 'value-stream-form-modal';
+    },
   },
   methods: {
     ...mapActions(['setSelectedValueStream', 'deleteValueStream']),
@@ -105,11 +121,17 @@ export default {
       });
     },
     onCreate() {
-      this.showForm = true;
+      if (!this.isVSAStandaloneSettingsPageEnabled) {
+        this.showForm = true;
+      }
+
       this.isEditing = false;
     },
     onEdit() {
-      this.showForm = true;
+      if (!this.isVSAStandaloneSettingsPageEnabled) {
+        this.showForm = true;
+      }
+
       this.isEditing = true;
     },
     slugify(valueStreamTitle) {
@@ -133,10 +155,11 @@ export default {
       <template v-if="canEdit" #footer>
         <div class="gl-border-t gl-p-2">
           <gl-button
-            v-gl-modal-directive="'value-stream-form-modal'"
+            v-gl-modal-directive="valueStreamFormModalId"
             class="gl-w-full gl-justify-content-start!"
             category="tertiary"
-            data-testid="create-value-stream"
+            :href="createValueStreamButtonHref"
+            data-testid="create-value-stream-option"
             data-track-action="click_dropdown"
             data-track-label="create_value_stream_form_open"
             @click="onCreate"
@@ -161,7 +184,8 @@ export default {
     </gl-collapsible-listbox>
     <gl-button
       v-if="isCustomValueStream && canEdit"
-      v-gl-modal-directive="'value-stream-form-modal'"
+      v-gl-modal-directive="valueStreamFormModalId"
+      :href="editValueStreamButtonHref"
       data-testid="edit-value-stream"
       data-track-action="click_button"
       data-track-label="edit_value_stream_form_open"
@@ -170,7 +194,8 @@ export default {
     >
     <gl-button
       v-if="!hasValueStreams"
-      v-gl-modal-directive="'value-stream-form-modal'"
+      v-gl-modal-directive="valueStreamFormModalId"
+      :href="createValueStreamButtonHref"
       data-testid="create-value-stream-button"
       data-track-action="click_button"
       data-track-label="create_value_stream_form_open"
diff --git a/ee/app/assets/javascripts/analytics/cycle_analytics/index.js b/ee/app/assets/javascripts/analytics/cycle_analytics/index.js
index ce97ae75a1786..11d1d1aa3ba89 100644
--- a/ee/app/assets/javascripts/analytics/cycle_analytics/index.js
+++ b/ee/app/assets/javascripts/analytics/cycle_analytics/index.js
@@ -19,7 +19,13 @@ const apolloProvider = new VueApollo({
 
 export default () => {
   const el = document.querySelector('#js-cycle-analytics');
-  const { emptyStateSvgPath, noDataSvgPath, noAccessSvgPath, newValueStreamPath } = el.dataset;
+  const {
+    emptyStateSvgPath,
+    noDataSvgPath,
+    noAccessSvgPath,
+    newValueStreamPath,
+    editValueStreamPath,
+  } = el.dataset;
   const initialData = buildCycleAnalyticsInitialData(el.dataset);
   const store = createStore();
 
@@ -47,6 +53,7 @@ export default () => {
     store,
     provide: {
       newValueStreamPath,
+      editValueStreamPath,
     },
     render: (createElement) =>
       createElement(CycleAnalytics, {
diff --git a/ee/app/assets/javascripts/analytics/cycle_analytics/vsa_settings/components/value_stream_form_content_header.vue b/ee/app/assets/javascripts/analytics/cycle_analytics/vsa_settings/components/value_stream_form_content_header.vue
index bad04e4d63470..7e7357f00504e 100644
--- a/ee/app/assets/javascripts/analytics/cycle_analytics/vsa_settings/components/value_stream_form_content_header.vue
+++ b/ee/app/assets/javascripts/analytics/cycle_analytics/vsa_settings/components/value_stream_form_content_header.vue
@@ -3,7 +3,7 @@ import { GlButton } from '@gitlab/ui';
 import { s__ } from '~/locale';
 import { i18n } from 'ee/analytics/cycle_analytics/components/create_value_stream_form/constants';
 
-const { EDIT_FORM_TITLE, EDIT_FORM_ACTION } = i18n;
+const { FORM_TITLE, EDIT_FORM_TITLE, EDIT_FORM_ACTION } = i18n;
 
 export default {
   name: 'ValueStreamFormContentHeader',
@@ -28,7 +28,7 @@ export default {
     },
   },
   i18n: {
-    newValueStream: s__('CreateValueStreamForm|New value stream'),
+    newValueStream: FORM_TITLE,
     editValueStreamTitle: EDIT_FORM_TITLE,
     saveValueStreamAction: EDIT_FORM_ACTION,
     viewValueStreamAction: s__('ValueStreamAnalytics|View value stream'),
@@ -56,7 +56,7 @@ export default {
       {{ formTitle }}
     </h1>
     <div
-      class="gl-display-flex gl-w-full gl-sm-w-auto gl-sm-flex-direction-row gl-flex-direction-column gl-gap-5"
+      class="gl-display-flex gl-w-full gl-sm-w-auto gl-sm-flex-direction-row gl-flex-direction-column gl-gap-3"
     >
       <gl-button
         v-if="isEditing"
diff --git a/ee/lib/ee/gitlab/analytics/cycle_analytics/request_params.rb b/ee/lib/ee/gitlab/analytics/cycle_analytics/request_params.rb
index 9e02de2d54c77..6f5bb63bd4471 100644
--- a/ee/lib/ee/gitlab/analytics/cycle_analytics/request_params.rb
+++ b/ee/lib/ee/gitlab/analytics/cycle_analytics/request_params.rb
@@ -41,7 +41,11 @@ def resource_paths
             paths.merge({
               milestones_path: url_helpers.group_milestones_path(group, format: :json),
               labels_path: url_helpers.group_labels_path(group, format: :json),
-              new_value_stream_path: url_helpers.new_group_analytics_cycle_analytics_value_stream_path(group)
+              new_value_stream_path: url_helpers.new_group_analytics_cycle_analytics_value_stream_path(group),
+              edit_value_stream_path: url_helpers.edit_group_analytics_cycle_analytics_value_stream_path(
+                group,
+                ':id'
+              )
             })
           end
 
diff --git a/ee/spec/features/groups/analytics/cycle_analytics/multiple_value_streams_spec.rb b/ee/spec/features/groups/analytics/cycle_analytics/multiple_value_streams_spec.rb
index 2a181877447b9..28c1f2a174c8c 100644
--- a/ee/spec/features/groups/analytics/cycle_analytics/multiple_value_streams_spec.rb
+++ b/ee/spec/features/groups/analytics/cycle_analytics/multiple_value_streams_spec.rb
@@ -38,23 +38,42 @@ def click_action_button(action, index)
     find_by_testid("stage-action-#{action}-#{index}").click
   end
 
+  def click_edit_button
+    click_link_or_button _('Edit')
+    wait_for_requests
+  end
+
+  def click_view_value_stream_button
+    click_link _('View value stream')
+    wait_for_requests
+  end
+
   def reload_value_stream
-    click_button 'Reload page'
+    click_button _('Reload page')
   end
 
-  def create_and_select_value_stream(name, with_aggregation = true)
+  def create_and_select_value_stream(name, with_aggregation = true, on_settings_page = true)
     create_custom_value_stream(name)
 
-    return unless with_aggregation
+    return unless with_aggregation && !on_settings_page
 
     reload_value_stream
+
     select_value_stream(name)
   end
 
-  shared_examples 'create a value stream' do |custom_value_stream_name|
+  def expect_successful_save(value_stream_name, expect_redirect)
+    if expect_redirect
+      expect(find_by_testid('dropdown-value-streams')).to have_text(value_stream_name)
+    else
+      expect(page).to have_text(_("'%{name}' Value Stream created") % { name: value_stream_name })
+    end
+  end
+
+  shared_examples 'create a value stream' do |custom_value_stream_name, on_settings_page|
     before do
       toggle_value_stream_dropdown
-      page.find_button(_('New Value Stream')).click
+      find_by_testid('create-value-stream-option').click
     end
 
     it 'includes additional form fields' do
@@ -64,7 +83,7 @@ def create_and_select_value_stream(name, with_aggregation = true)
     it 'can create a value stream' do
       save_value_stream(custom_value_stream_name)
 
-      expect(page).to have_text(_("'%{name}' Value Stream created") % { name: custom_value_stream_name })
+      expect_successful_save(custom_value_stream_name, on_settings_page)
     end
 
     it 'can create a value stream with only custom stages' do
@@ -73,7 +92,7 @@ def create_and_select_value_stream(name, with_aggregation = true)
       fill_in_custom_stage_fields
       save_value_stream(custom_value_stream_name)
 
-      expect(page).to have_text(_("'%{name}' Value Stream created") % { name: custom_value_stream_name })
+      expect_successful_save(custom_value_stream_name, on_settings_page)
     end
 
     it 'can create a value stream with a custom stage and hidden defaults' do
@@ -87,7 +106,8 @@ def create_and_select_value_stream(name, with_aggregation = true)
 
       save_value_stream(custom_value_stream_name)
 
-      expect(page).to have_text(_("'%{name}' Value Stream created") % { name: custom_value_stream_name })
+      expect_successful_save(custom_value_stream_name, on_settings_page)
+
       expect(path_nav_elem).to have_text("Cool custom stage - name")
     end
 
@@ -108,21 +128,19 @@ def create_and_select_value_stream(name, with_aggregation = true)
       fill_in_custom_stage_fields "Stage 2"
       save_value_stream(custom_value_stream_name)
 
-      expect(page).to have_text(_("'%{name}' Value Stream created") % { name: custom_value_stream_name })
+      expect_successful_save(custom_value_stream_name, on_settings_page)
     end
   end
 
-  shared_examples 'update a value stream' do |custom_value_stream_name, with_aggregation|
+  shared_examples 'update a value stream' do |custom_value_stream_name, with_aggregation, on_settings_page|
     before do
-      select_group(group)
-
-      create_and_select_value_stream(custom_value_stream_name, with_aggregation)
+      create_and_select_value_stream(custom_value_stream_name, with_aggregation, on_settings_page)
     end
 
     it 'can reorder stages' do
       expect(path_nav_stage_names_without_median).to eq(["Overview", "Issue", "Plan", "Code", "Test", "Review", "Staging", "Cool custom stage - name 7"])
 
-      page.find_button(_('Edit')).click
+      click_edit_button
       # Re-arrange a few stages
       page.all("[data-testid*='stage-action-move-down-']").first.click
       page.all("[data-testid*='stage-action-move-up-']").last.click
@@ -130,12 +148,14 @@ def create_and_select_value_stream(name, with_aggregation = true)
       click_save_value_stream_button
       wait_for_requests
 
+      click_view_value_stream_button if on_settings_page
+
       expect(path_nav_stage_names_without_median).to eq(["Overview", "Plan", "Issue", "Code", "Test", "Review", "Cool custom stage - name 7", "Staging"])
     end
 
     context 'updating' do
       before do
-        page.find_button(_('Edit')).click
+        click_edit_button
       end
 
       it 'includes additional form fields' do
@@ -160,9 +180,11 @@ def create_and_select_value_stream(name, with_aggregation = true)
         click_save_value_stream_button
         wait_for_requests
 
+        click_view_value_stream_button if on_settings_page
+
         expect(path_nav_elem).to have_text("Cool custom stage - name")
 
-        page.find_button(_('Edit')).click
+        click_edit_button
 
         # Delete the custom stages, delete the last one first since the list gets reordered after a deletion
         click_action_button('remove', 7)
@@ -175,6 +197,8 @@ def create_and_select_value_stream(name, with_aggregation = true)
         click_save_value_stream_button
         wait_for_requests
 
+        click_view_value_stream_button if on_settings_page
+
         expect(path_nav_elem).not_to have_text("Cool custom stage - name")
       end
 
@@ -187,17 +211,24 @@ def create_and_select_value_stream(name, with_aggregation = true)
         wait_for_requests
 
         expect(page).to have_text(_("'%{name}' Value Stream saved") % { name: custom_value_stream_name })
+
+        click_view_value_stream_button if on_settings_page
+
         expect(path_nav_elem).not_to have_text("Staging")
         expect(path_nav_elem).not_to have_text("Review")
         expect(path_nav_elem).not_to have_text("Test")
 
-        click_button(_('Edit'))
+        click_edit_button
+
         click_action_button('restore', 0)
 
         click_save_value_stream_button
         wait_for_requests
 
         expect(page).to have_text(_("'%{name}' Value Stream saved") % { name: custom_value_stream_name })
+
+        click_view_value_stream_button if on_settings_page
+
         expect(path_nav_elem).to have_text("Test")
       end
 
@@ -239,27 +270,27 @@ def create_and_select_value_stream(name, with_aggregation = true)
     end
   end
 
-  shared_examples 'create group value streams' do |with_aggregation|
+  shared_examples 'create group value streams' do |with_aggregation, on_settings_page|
     name = 'group value stream'
 
     before do
       select_group(group)
     end
 
-    it_behaves_like 'create a value stream', name
-    it_behaves_like 'update a value stream', name, with_aggregation
+    it_behaves_like 'create a value stream', name, on_settings_page
+    it_behaves_like 'update a value stream', name, with_aggregation, on_settings_page
     it_behaves_like 'delete a value stream', name
   end
 
-  shared_examples 'create sub group value streams' do |with_aggregation|
+  shared_examples 'create sub group value streams' do |with_aggregation, on_settings_page|
     name = 'sub group value stream'
 
     before do
       select_group(sub_group)
     end
 
-    it_behaves_like 'create a value stream', name
-    it_behaves_like 'update a value stream', name, with_aggregation
+    it_behaves_like 'create a value stream', name, on_settings_page
+    it_behaves_like 'update a value stream', name, with_aggregation, on_settings_page
     it_behaves_like 'delete a value stream', name
   end
 
@@ -305,7 +336,7 @@ def create_and_select_value_stream(name, with_aggregation = true)
         end
 
         it 'does not navigate to the new value stream settings page' do
-          expect(page).to have_current_path(group_analytics_cycle_analytics_path(group))
+          expect(page).to have_current_path(group_analytics_cycle_analytics_path(group), ignore_query: true)
         end
       end
     end
@@ -342,8 +373,17 @@ def create_and_select_value_stream(name, with_aggregation = true)
         create(:cycle_analytics_value_stream, namespace: sub_group, name: 'default')
       end
 
-      it_behaves_like 'create group value streams', true
-      it_behaves_like 'create sub group value streams', true
+      it_behaves_like 'create group value streams', true, true
+      it_behaves_like 'create sub group value streams', true, true
+
+      context 'when `vsa_standalone_settings_page` feature flag is disabled' do
+        before do
+          stub_feature_flags(vsa_standalone_settings_page: false)
+        end
+
+        it_behaves_like 'create group value streams', true
+        it_behaves_like 'create sub group value streams', true
+      end
     end
   end
 end
diff --git a/ee/spec/features/projects/analytics/cycle_analytics_spec.rb b/ee/spec/features/projects/analytics/cycle_analytics_spec.rb
index 4fc4477f62e41..4ba2ccac63289 100644
--- a/ee/spec/features/projects/analytics/cycle_analytics_spec.rb
+++ b/ee/spec/features/projects/analytics/cycle_analytics_spec.rb
@@ -71,7 +71,7 @@
         visit project_cycle_analytics_path(project)
         click_button('New value stream')
         fill_in('Value Stream name', with: 'foo stream')
-        click_button('Create value stream')
+        click_button('New value stream')
         # otherwise we get the "Data is collecting and loading"
         create_value_stream_aggregation(project_namespace)
         refresh
diff --git a/ee/spec/frontend/analytics/cycle_analytics/components/value_stream_select_spec.js b/ee/spec/frontend/analytics/cycle_analytics/components/value_stream_select_spec.js
index 4321457b2c032..af56493feb885 100644
--- a/ee/spec/frontend/analytics/cycle_analytics/components/value_stream_select_spec.js
+++ b/ee/spec/frontend/analytics/cycle_analytics/components/value_stream_select_spec.js
@@ -5,7 +5,13 @@ import Vuex from 'vuex';
 import ValueStreamSelect from 'ee/analytics/cycle_analytics/components/value_stream_select.vue';
 import { mockTracking, unmockTracking } from 'helpers/tracking_helper';
 import { shallowMountExtended, mountExtended } from 'helpers/vue_test_utils_helper';
-import { valueStreams, defaultStageConfig } from '../mock_data';
+import { createMockDirective, getBinding } from 'helpers/vue_mock_directive';
+import {
+  valueStreams,
+  defaultStageConfig,
+  newValueStreamPath,
+  editValueStreamPath,
+} from '../mock_data';
 
 Vue.use(Vuex);
 
@@ -19,6 +25,7 @@ describe('ValueStreamSelect', () => {
   const streamName = 'Cool stream';
   const selectedValueStream = valueStreams[0];
   const deleteValueStreamError = 'Cannot delete default value stream';
+  const editValueStreamPathWithId = editValueStreamPath.replace(':id', selectedValueStream.id);
 
   const fakeStore = ({ initialState = {} }) =>
     new Vuex.Store({
@@ -42,6 +49,7 @@ describe('ValueStreamSelect', () => {
     props = {},
     data = {},
     initialState = {},
+    provide = {},
     mountFn = shallowMountExtended,
   } = {}) =>
     mountFn(ValueStreamSelect, {
@@ -55,16 +63,28 @@ describe('ValueStreamSelect', () => {
         canEdit: true,
         ...props,
       },
+      provide: {
+        newValueStreamPath,
+        editValueStreamPath,
+        glFeatures: {
+          vsaStandaloneSettingsPage: true,
+        },
+        ...provide,
+      },
       mocks: {
         $toast: {
           show: mockToastShow,
         },
       },
+      directives: {
+        GlModalDirective: createMockDirective('gl-modal-directive'),
+      },
     });
 
   const findModal = (modal) => wrapper.findByTestId(`${modal}-value-stream-modal`);
   const submitModal = (modal) => findModal(modal).vm.$emit('primary', mockEvent);
   const findSelectValueStreamDropdown = () => wrapper.findComponent(GlCollapsibleListbox);
+  const findCreateValueStreamOption = () => wrapper.findByTestId('create-value-stream-option');
   const findCreateValueStreamButton = () => wrapper.findByTestId('create-value-stream-button');
   const findEditValueStreamButton = () => wrapper.findByTestId('edit-value-stream');
   const findDeleteValueStreamButton = () => wrapper.findByTestId('delete-value-stream');
@@ -128,9 +148,64 @@ describe('ValueStreamSelect', () => {
           expect(findDeleteValueStreamButton().exists()).toBe(true);
         });
 
-        it('renders an edit option for custom value streams', () => {
+        it('renders a create option for custom value streams', () => {
+          expect(findCreateValueStreamOption().exists()).toBe(true);
+          expect(findCreateValueStreamOption().text()).toBe('New Value Stream');
+          expect(findCreateValueStreamOption().attributes('href')).toBe(newValueStreamPath);
+        });
+
+        it('renders an edit button for custom value streams', () => {
           expect(findEditValueStreamButton().exists()).toBe(true);
           expect(findEditValueStreamButton().text()).toBe('Edit');
+          expect(findEditValueStreamButton().attributes('href')).toBe(editValueStreamPathWithId);
+        });
+
+        it('does not bind modal directive to edit button', () => {
+          const binding = getBinding(findEditValueStreamButton().element, 'gl-modal-directive');
+
+          expect(binding.value).toBe(false);
+        });
+
+        it('does not bind modal directive to create option', () => {
+          const binding = getBinding(findCreateValueStreamOption().element, 'gl-modal-directive');
+
+          expect(binding.value).toBe(false);
+        });
+
+        describe('vsaStandaloneSettingsPage = false', () => {
+          beforeEach(() => {
+            wrapper = createComponent({
+              mountFn: mountExtended,
+              initialState: {
+                valueStreams,
+                selectedValueStream: {
+                  ...selectedValueStream,
+                  isCustom: true,
+                },
+              },
+              provide: { glFeatures: { vsaStandaloneSettingsPage: false } },
+            });
+          });
+
+          it('renders create option without a link', () => {
+            expect(findCreateValueStreamOption().attributes('href')).toBe(undefined);
+          });
+
+          it('binds modal directive to create option', () => {
+            const binding = getBinding(findCreateValueStreamOption().element, 'gl-modal-directive');
+
+            expect(binding.value).toBe('value-stream-form-modal');
+          });
+
+          it('renders edit button without a link', () => {
+            expect(findEditValueStreamButton().attributes('href')).toBe(undefined);
+          });
+
+          it('binds modal directive to edit button', () => {
+            const binding = getBinding(findEditValueStreamButton().element, 'gl-modal-directive');
+
+            expect(binding.value).toBe('value-stream-form-modal');
+          });
         });
       });
 
@@ -150,11 +225,15 @@ describe('ValueStreamSelect', () => {
             },
           });
         });
+        it('does not render a create option for custom value streams', () => {
+          expect(findCreateValueStreamOption().exists()).toBe(false);
+        });
+
         it('does not render a delete option for custom value streams', () => {
           expect(findDeleteValueStreamButton().exists()).toBe(false);
         });
 
-        it('does not render an edit option for custom value streams', () => {
+        it('does not render an edit button for custom value streams', () => {
           expect(findEditValueStreamButton().exists()).toBe(false);
         });
       });
@@ -169,7 +248,7 @@ describe('ValueStreamSelect', () => {
         expect(findDeleteValueStreamButton().exists()).toBe(false);
       });
 
-      it('does not render an edit option for default value streams', () => {
+      it('does not render an edit button for default value streams', () => {
         expect(findEditValueStreamButton().exists()).toBe(false);
       });
     });
@@ -192,7 +271,7 @@ describe('ValueStreamSelect', () => {
       expect(findSelectValueStreamDropdown().exists()).toBe(true);
     });
 
-    it('does not render an edit option for default value streams', () => {
+    it('does not render an edit button for default value streams', () => {
       expect(findEditValueStreamButton().exists()).toBe(false);
     });
   });
@@ -208,15 +287,43 @@ describe('ValueStreamSelect', () => {
 
     it('displays the create value stream button', () => {
       expect(findCreateValueStreamButton().exists()).toBe(true);
+      expect(findCreateValueStreamButton().attributes('href')).toBe(newValueStreamPath);
+    });
+
+    it('does not bind modal directive to create value stream button', () => {
+      const binding = getBinding(findCreateValueStreamButton().element, 'gl-modal-directive');
+
+      expect(binding.value).toBe(false);
     });
 
     it('does not display the select value stream dropdown', () => {
       expect(findSelectValueStreamDropdown().exists()).toBe(false);
     });
 
-    it('does not render an edit option for default value streams', () => {
+    it('does not render an edit button for default value streams', () => {
       expect(findEditValueStreamButton().exists()).toBe(false);
     });
+
+    describe('vsaStandaloneSettingsPage = false', () => {
+      beforeEach(() => {
+        wrapper = createComponent({
+          initialState: {
+            valueStreams: [],
+          },
+          provide: { glFeatures: { vsaStandaloneSettingsPage: false } },
+        });
+      });
+
+      it('renders create value stream button without a link', () => {
+        expect(findCreateValueStreamButton().attributes('href')).toBe(undefined);
+      });
+
+      it('binds modal directive to create value stream button', () => {
+        const binding = getBinding(findCreateValueStreamButton().element, 'gl-modal-directive');
+
+        expect(binding.value).toBe('value-stream-form-modal');
+      });
+    });
   });
 
   describe('Delete value stream modal', () => {
diff --git a/ee/spec/frontend/analytics/cycle_analytics/mock_data.js b/ee/spec/frontend/analytics/cycle_analytics/mock_data.js
index 74c07b30bc548..15379ada6d3b7 100644
--- a/ee/spec/frontend/analytics/cycle_analytics/mock_data.js
+++ b/ee/spec/frontend/analytics/cycle_analytics/mock_data.js
@@ -72,6 +72,7 @@ export const valueStreams = [
 export const vsaPath = '/analytics/value_stream_analytics';
 export const valueStreamPath = `${vsaPath}?value_stream_id=${valueStreams[0].id}`;
 export const newValueStreamPath = `${vsaPath}/value_streams/new`;
+export const editValueStreamPath = `${vsaPath}/value_streams/:id/edit`;
 
 export const groupLabels = apiGroupLabels.map((l) =>
   convertObjectPropsToCamelCase({ ...l, title: l.name }),
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index d23bf27b680f9..91e119eaec104 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -15214,9 +15214,6 @@ msgstr ""
 msgid "CreateValueStreamForm|Create from no template"
 msgstr ""
 
-msgid "CreateValueStreamForm|Create value stream"
-msgstr ""
-
 msgid "CreateValueStreamForm|Default stages"
 msgstr ""
 
diff --git a/qa/qa/ee/page/value_stream_analytics.rb b/qa/qa/ee/page/value_stream_analytics.rb
index e7fc083c108df..ba0476143099c 100644
--- a/qa/qa/ee/page/value_stream_analytics.rb
+++ b/qa/qa/ee/page/value_stream_analytics.rb
@@ -146,7 +146,7 @@ def select_value_stream_type(value = 'default')
         def create_value_stream
           within_element('value-stream-form-modal') do
             # footer buttons are generic UI components from gitlab/ui
-            find_button("Create value stream").click
+            find_button("New value stream").click
           end
         end
 
diff --git a/spec/support/helpers/cycle_analytics_helpers.rb b/spec/support/helpers/cycle_analytics_helpers.rb
index 0847a57d92e12..82a8fbcb15186 100644
--- a/spec/support/helpers/cycle_analytics_helpers.rb
+++ b/spec/support/helpers/cycle_analytics_helpers.rb
@@ -63,7 +63,7 @@ def add_custom_label_stage_to_form
   def save_value_stream(custom_value_stream_name)
     fill_in 'create-value-stream-name', with: custom_value_stream_name
 
-    page.find_button(s_('CreateValueStreamForm|Create value stream')).click
+    click_button(_('New value stream'))
     wait_for_requests
   end
 
@@ -73,7 +73,7 @@ def click_save_value_stream_button
 
   def create_custom_value_stream(custom_value_stream_name)
     toggle_value_stream_dropdown
-    page.find_button(_('New Value Stream')).click
+    find_by_testid('create-value-stream-option').click
 
     add_custom_stage_to_form
     save_value_stream(custom_value_stream_name)
-- 
GitLab