Commit d3fc18ff authored by Miguel Rincon's avatar Miguel Rincon

Merge branch 'jnnkl-gl-modal-board-setting' into 'master'

Replace window.confirm with GlModal for board setting sidebar

See merge request gitlab-org/gitlab!78097
parents bbdf14ae 0eae96df
<script> <script>
import { GlButton, GlDrawer, GlLabel } from '@gitlab/ui'; import { GlButton, GlDrawer, GlLabel, GlModal, GlModalDirective } from '@gitlab/ui';
import { MountingPortal } from 'portal-vue'; import { MountingPortal } from 'portal-vue';
import { mapActions, mapState, mapGetters } from 'vuex'; import { mapActions, mapState, mapGetters } from 'vuex';
import { LIST, ListType, ListTypeTitles } from '~/boards/constants'; import { LIST, ListType, ListTypeTitles } from '~/boards/constants';
...@@ -11,8 +11,14 @@ import glFeatureFlagMixin from '~/vue_shared/mixins/gl_feature_flags_mixin'; ...@@ -11,8 +11,14 @@ import glFeatureFlagMixin from '~/vue_shared/mixins/gl_feature_flags_mixin';
export default { export default {
listSettingsText: __('List settings'), listSettingsText: __('List settings'),
i18n: {
modalAction: __('Remove list'),
modalCopy: __('Are you sure you want to remove this list?'),
modalCancel: __('Cancel'),
},
components: { components: {
GlButton, GlButton,
GlModal,
GlDrawer, GlDrawer,
GlLabel, GlLabel,
MountingPortal, MountingPortal,
...@@ -21,6 +27,9 @@ export default { ...@@ -21,6 +27,9 @@ export default {
BoardSettingsListTypes: () => BoardSettingsListTypes: () =>
import('ee_component/boards/components/board_settings_list_types.vue'), import('ee_component/boards/components/board_settings_list_types.vue'),
}, },
directives: {
GlModal: GlModalDirective,
},
mixins: [glFeatureFlagMixin(), Tracking.mixin()], mixins: [glFeatureFlagMixin(), Tracking.mixin()],
inject: ['canAdminList', 'scopedLabelsAvailable'], inject: ['canAdminList', 'scopedLabelsAvailable'],
inheritAttrs: false, inheritAttrs: false,
...@@ -29,6 +38,7 @@ export default { ...@@ -29,6 +38,7 @@ export default {
ListType, ListType,
}; };
}, },
modalId: 'board-settings-sidebar-modal',
computed: { computed: {
...mapGetters(['isSidebarOpen', 'isEpicBoard']), ...mapGetters(['isSidebarOpen', 'isEpicBoard']),
...mapState(['activeId', 'sidebarType', 'boardLists']), ...mapState(['activeId', 'sidebarType', 'boardLists']),
...@@ -59,16 +69,16 @@ export default { ...@@ -59,16 +69,16 @@ export default {
}, },
methods: { methods: {
...mapActions(['unsetActiveId', 'removeList']), ...mapActions(['unsetActiveId', 'removeList']),
handleModalPrimary() {
this.deleteBoard();
},
showScopedLabels(label) { showScopedLabels(label) {
return this.scopedLabelsAvailable && isScopedLabel(label); return this.scopedLabelsAvailable && isScopedLabel(label);
}, },
deleteBoard() { deleteBoard() {
// eslint-disable-next-line no-alert this.track('click_button', { label: 'remove_list' });
if (window.confirm(__('Are you sure you want to remove this list?'))) { this.removeList(this.activeId);
this.track('click_button', { label: 'remove_list' }); this.unsetActiveId();
this.removeList(this.activeId);
this.unsetActiveId();
}
}, },
}, },
}; };
...@@ -92,11 +102,10 @@ export default { ...@@ -92,11 +102,10 @@ export default {
<template #header> <template #header>
<div v-if="canAdminList && activeList.id" class="gl-mt-3"> <div v-if="canAdminList && activeList.id" class="gl-mt-3">
<gl-button <gl-button
v-gl-modal="$options.modalId"
variant="danger" variant="danger"
category="secondary" category="secondary"
size="small" size="small"
data-testid="remove-list"
@click.stop="deleteBoard"
>{{ __('Remove list') }} >{{ __('Remove list') }}
</gl-button> </gl-button>
</div> </div>
...@@ -122,5 +131,21 @@ export default { ...@@ -122,5 +131,21 @@ export default {
/> />
</template> </template>
</gl-drawer> </gl-drawer>
<gl-modal
:modal-id="$options.modalId"
:title="$options.i18n.modalAction"
size="sm"
:action-primary="{
text: $options.i18n.modalAction,
attributes: [{ variant: 'danger' }],
}"
:action-secondary="{
text: $options.i18n.modalCancel,
attributes: [{ variant: 'default' }],
}"
@primary="handleModalPrimary"
>
<p>{{ $options.i18n.modalCopy }}</p>
</gl-modal>
</mounting-portal> </mounting-portal>
</template> </template>
...@@ -153,7 +153,11 @@ RSpec.describe 'epic boards', :js do ...@@ -153,7 +153,11 @@ RSpec.describe 'epic boards', :js do
click_button 'List settings' click_button 'List settings'
end end
accept_confirm { click_button 'Remove list' } click_button 'Remove list'
page.within('.modal') do
click_button('Remove list', match: :first)
end
expect(page).not_to have_content(label.name) expect(page).not_to have_content(label.name)
end end
......
...@@ -583,7 +583,11 @@ RSpec.describe 'Project issue boards', :js do ...@@ -583,7 +583,11 @@ RSpec.describe 'Project issue boards', :js do
end end
page.within(find('.js-board-settings-sidebar')) do page.within(find('.js-board-settings-sidebar')) do
accept_confirm { find('[data-testid="remove-list"]').click } click_button 'Remove list'
end
page.within('.modal') do
click_button 'Remove list'
end end
end end
......
import { GlDrawer, GlLabel } from '@gitlab/ui'; import { GlDrawer, GlLabel, GlModal, GlButton } from '@gitlab/ui';
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import { MountingPortal } from 'portal-vue'; import { MountingPortal } from 'portal-vue';
import Vue, { nextTick } from 'vue'; import Vue, { nextTick } from 'vue';
import Vuex from 'vuex'; import Vuex from 'vuex';
import { createMockDirective, getBinding } from 'helpers/vue_mock_directive';
import { stubComponent } from 'helpers/stub_component'; import { stubComponent } from 'helpers/stub_component';
import { extendedWrapper } from 'helpers/vue_test_utils_helper'; import { extendedWrapper } from 'helpers/vue_test_utils_helper';
import BoardSettingsSidebar from '~/boards/components/board_settings_sidebar.vue'; import BoardSettingsSidebar from '~/boards/components/board_settings_sidebar.vue';
...@@ -20,8 +21,7 @@ describe('BoardSettingsSidebar', () => { ...@@ -20,8 +21,7 @@ describe('BoardSettingsSidebar', () => {
const labelTitle = mockLabelList.label.title; const labelTitle = mockLabelList.label.title;
const labelColor = mockLabelList.label.color; const labelColor = mockLabelList.label.color;
const listId = mockLabelList.id; const listId = mockLabelList.id;
const modalID = 'board-settings-sidebar-modal';
const findRemoveButton = () => wrapper.findByTestId('remove-list');
const createComponent = ({ const createComponent = ({
canAdminList = false, canAdminList = false,
...@@ -46,6 +46,9 @@ describe('BoardSettingsSidebar', () => { ...@@ -46,6 +46,9 @@ describe('BoardSettingsSidebar', () => {
canAdminList, canAdminList,
scopedLabelsAvailable: false, scopedLabelsAvailable: false,
}, },
directives: {
GlModal: createMockDirective(),
},
stubs: { stubs: {
GlDrawer: stubComponent(GlDrawer, { GlDrawer: stubComponent(GlDrawer, {
template: '<div><slot name="header"></slot><slot></slot></div>', template: '<div><slot name="header"></slot><slot></slot></div>',
...@@ -56,6 +59,8 @@ describe('BoardSettingsSidebar', () => { ...@@ -56,6 +59,8 @@ describe('BoardSettingsSidebar', () => {
}; };
const findLabel = () => wrapper.find(GlLabel); const findLabel = () => wrapper.find(GlLabel);
const findDrawer = () => wrapper.find(GlDrawer); const findDrawer = () => wrapper.find(GlDrawer);
const findModal = () => wrapper.find(GlModal);
const findRemoveButton = () => wrapper.find(GlButton);
afterEach(() => { afterEach(() => {
jest.restoreAllMocks(); jest.restoreAllMocks();
...@@ -161,5 +166,16 @@ describe('BoardSettingsSidebar', () => { ...@@ -161,5 +166,16 @@ describe('BoardSettingsSidebar', () => {
expect(findRemoveButton().exists()).toBe(true); expect(findRemoveButton().exists()).toBe(true);
}); });
it('has the correct ID on the button', () => {
createComponent({ canAdminList: true, activeId: listId, list: mockLabelList });
const binding = getBinding(findRemoveButton().element, 'gl-modal');
expect(binding.value).toBe(modalID);
});
it('has the correct ID on the modal', () => {
createComponent({ canAdminList: true, activeId: listId, list: mockLabelList });
expect(findModal().props('modalId')).toBe(modalID);
});
}); });
}); });
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment