Commit 948e03df authored by Andrew Fontaine's avatar Andrew Fontaine

Merge branch 'cngo-migrate-project-settings-to-use-gl-toggle' into 'master'

Migrate project settings to use GlToggle

See merge request gitlab-org/gitlab!52729
parents fd226bfa 348f987b
<script> <script>
import { GlIcon } from '@gitlab/ui'; import { GlIcon, GlToggle } from '@gitlab/ui';
import projectFeatureToggle from '~/vue_shared/components/toggle_button.vue';
import { featureAccessLevelNone } from '../constants'; import { featureAccessLevelNone } from '../constants';
export default { export default {
components: { components: {
GlIcon, GlIcon,
projectFeatureToggle, GlToggle,
}, },
model: { model: {
prop: 'value', prop: 'value',
...@@ -78,11 +77,11 @@ export default { ...@@ -78,11 +77,11 @@ export default {
class="project-feature-controls gl-display-flex gl-align-items-center gl-my-3 gl-mx-0" class="project-feature-controls gl-display-flex gl-align-items-center gl-my-3 gl-mx-0"
> >
<input v-if="name" :name="name" :value="value" type="hidden" /> <input v-if="name" :name="name" :value="value" type="hidden" />
<project-feature-toggle <gl-toggle
v-if="showToggle" v-if="showToggle"
class="gl-flex-grow-0 gl-mr-3" class="gl-mr-3"
:value="featureEnabled" :value="featureEnabled"
:disabled-input="disabledInput" :disabled="disabledInput"
@change="toggleFeature" @change="toggleFeature"
/> />
<div class="select-wrapper gl-flex-fill-1"> <div class="select-wrapper gl-flex-fill-1">
......
<script> <script>
import { GlIcon, GlSprintf, GlLink, GlFormCheckbox } from '@gitlab/ui'; import { GlIcon, GlSprintf, GlLink, GlFormCheckbox, GlToggle } from '@gitlab/ui';
import settingsMixin from 'ee_else_ce/pages/projects/shared/permissions/mixins/settings_pannel_mixin'; import settingsMixin from 'ee_else_ce/pages/projects/shared/permissions/mixins/settings_pannel_mixin';
import { s__ } from '~/locale'; import { s__ } from '~/locale';
import projectFeatureToggle from '~/vue_shared/components/toggle_button.vue';
import glFeatureFlagsMixin from '~/vue_shared/mixins/gl_feature_flags_mixin'; import glFeatureFlagsMixin from '~/vue_shared/mixins/gl_feature_flags_mixin';
import { import {
visibilityOptions, visibilityOptions,
...@@ -22,12 +21,12 @@ const PAGE_FEATURE_ACCESS_LEVEL = s__('ProjectSettings|Everyone'); ...@@ -22,12 +21,12 @@ const PAGE_FEATURE_ACCESS_LEVEL = s__('ProjectSettings|Everyone');
export default { export default {
components: { components: {
projectFeatureSetting, projectFeatureSetting,
projectFeatureToggle,
projectSettingRow, projectSettingRow,
GlIcon, GlIcon,
GlSprintf, GlSprintf,
GlLink, GlLink,
GlFormCheckbox, GlFormCheckbox,
GlToggle,
}, },
mixins: [settingsMixin, glFeatureFlagsMixin()], mixins: [settingsMixin, glFeatureFlagsMixin()],
...@@ -483,9 +482,10 @@ export default { ...@@ -483,9 +482,10 @@ export default {
) )
}} }}
</div> </div>
<project-feature-toggle <gl-toggle
v-model="containerRegistryEnabled" v-model="containerRegistryEnabled"
:disabled-input="!repositoryEnabled" class="gl-my-2"
:disabled="!repositoryEnabled"
name="project[container_registry_enabled]" name="project[container_registry_enabled]"
/> />
</project-setting-row> </project-setting-row>
...@@ -498,9 +498,10 @@ export default { ...@@ -498,9 +498,10 @@ export default {
s__('ProjectSettings|Manages large files such as audio, video, and graphics files.') s__('ProjectSettings|Manages large files such as audio, video, and graphics files.')
" "
> >
<project-feature-toggle <gl-toggle
v-model="lfsEnabled" v-model="lfsEnabled"
:disabled-input="!repositoryEnabled" class="gl-my-2"
:disabled="!repositoryEnabled"
name="project[lfs_enabled]" name="project[lfs_enabled]"
/> />
<p v-if="!lfsEnabled && lfsObjectsExist"> <p v-if="!lfsEnabled && lfsObjectsExist">
...@@ -530,9 +531,10 @@ export default { ...@@ -530,9 +531,10 @@ export default {
s__('ProjectSettings|Every project can have its own space to store its packages.') s__('ProjectSettings|Every project can have its own space to store its packages.')
" "
> >
<project-feature-toggle <gl-toggle
v-model="packagesEnabled" v-model="packagesEnabled"
:disabled-input="!repositoryEnabled" class="gl-my-2"
:disabled="!repositoryEnabled"
name="project[packages_enabled]" name="project[packages_enabled]"
/> />
</project-setting-row> </project-setting-row>
......
---
title: Migrate project settings to use GlToggle
merge_request: 52729
author:
type: changed
...@@ -22,7 +22,7 @@ RSpec.describe 'Project settings > Issues', :js do ...@@ -22,7 +22,7 @@ RSpec.describe 'Project settings > Issues', :js do
expect(page).to have_content('Set a default description template to be used for new issues.') expect(page).to have_content('Set a default description template to be used for new issues.')
within('.sharing-permissions-form') do within('.sharing-permissions-form') do
find('.project-feature-controls[data-for="project[project_feature_attributes][issues_access_level]"] .project-feature-toggle').click find('.project-feature-controls[data-for="project[project_feature_attributes][issues_access_level]"] .gl-toggle').click
click_on('Save changes') click_on('Save changes')
end end
...@@ -41,7 +41,7 @@ RSpec.describe 'Project settings > Issues', :js do ...@@ -41,7 +41,7 @@ RSpec.describe 'Project settings > Issues', :js do
expect(page).not_to have_content('Set a default description template to be used for new issues.') expect(page).not_to have_content('Set a default description template to be used for new issues.')
within('.sharing-permissions-form') do within('.sharing-permissions-form') do
find('.project-feature-controls[data-for="project[project_feature_attributes][issues_access_level]"] .project-feature-toggle').click find('.project-feature-controls[data-for="project[project_feature_attributes][issues_access_level]"] .gl-toggle').click
click_on('Save changes') click_on('Save changes')
end end
......
...@@ -187,7 +187,7 @@ RSpec.describe 'Edit Project Settings' do ...@@ -187,7 +187,7 @@ RSpec.describe 'Edit Project Settings' do
click_button "Save changes" click_button "Save changes"
end end
expect(find(".sharing-permissions")).to have_selector(".project-feature-toggle.is-disabled", count: 4) expect(find(".sharing-permissions")).to have_selector(".gl-toggle.is-disabled", minimum: 4)
end end
it "shows empty features project homepage" do it "shows empty features project homepage" do
...@@ -282,10 +282,10 @@ RSpec.describe 'Edit Project Settings' do ...@@ -282,10 +282,10 @@ RSpec.describe 'Edit Project Settings' do
end end
def toggle_feature_off(feature_name) def toggle_feature_off(feature_name)
find(".project-feature-controls[data-for=\"#{feature_name}\"] .project-feature-toggle.is-checked").click find(".project-feature-controls[data-for=\"#{feature_name}\"] .gl-toggle.is-checked").click
end end
def toggle_feature_on(feature_name) def toggle_feature_on(feature_name)
find(".project-feature-controls[data-for=\"#{feature_name}\"] .project-feature-toggle:not(.is-checked)").click find(".project-feature-controls[data-for=\"#{feature_name}\"] .gl-toggle:not(.is-checked)").click
end end
end end
...@@ -39,7 +39,7 @@ RSpec.describe 'Projects settings' do ...@@ -39,7 +39,7 @@ RSpec.describe 'Projects settings' do
visit edit_project_path(project) visit edit_project_path(project)
forking_enabled_input = find('input[name="project[project_feature_attributes][forking_access_level]"]', visible: :hidden) forking_enabled_input = find('input[name="project[project_feature_attributes][forking_access_level]"]', visible: :hidden)
forking_enabled_button = find('input[name="project[project_feature_attributes][forking_access_level]"] + label > button') forking_enabled_button = find('[data-for="project[project_feature_attributes][forking_access_level]"] .gl-toggle')
expect(forking_enabled_input.value).to eq('20') expect(forking_enabled_input.value).to eq('20')
......
...@@ -51,7 +51,7 @@ RSpec.describe 'Projects > Settings > User manages merge request settings' do ...@@ -51,7 +51,7 @@ RSpec.describe 'Projects > Settings > User manages merge request settings' do
expect(page).to have_content 'All discussions must be resolved' expect(page).to have_content 'All discussions must be resolved'
within('.sharing-permissions-form') do within('.sharing-permissions-form') do
find('.project-feature-controls[data-for="project[project_feature_attributes][merge_requests_access_level]"] .project-feature-toggle').click find('.project-feature-controls[data-for="project[project_feature_attributes][merge_requests_access_level]"] .gl-toggle').click
find('input[value="Save changes"]').send_keys(:return) find('input[value="Save changes"]').send_keys(:return)
end end
...@@ -71,7 +71,7 @@ RSpec.describe 'Projects > Settings > User manages merge request settings' do ...@@ -71,7 +71,7 @@ RSpec.describe 'Projects > Settings > User manages merge request settings' do
expect(page).to have_content 'All discussions must be resolved' expect(page).to have_content 'All discussions must be resolved'
within('.sharing-permissions-form') do within('.sharing-permissions-form') do
find('.project-feature-controls[data-for="project[project_feature_attributes][builds_access_level]"] .project-feature-toggle').click find('.project-feature-controls[data-for="project[project_feature_attributes][builds_access_level]"] .gl-toggle').click
find('input[value="Save changes"]').send_keys(:return) find('input[value="Save changes"]').send_keys(:return)
end end
...@@ -92,7 +92,7 @@ RSpec.describe 'Projects > Settings > User manages merge request settings' do ...@@ -92,7 +92,7 @@ RSpec.describe 'Projects > Settings > User manages merge request settings' do
expect(page).not_to have_content 'All discussions must be resolved' expect(page).not_to have_content 'All discussions must be resolved'
within('.sharing-permissions-form') do within('.sharing-permissions-form') do
find('.project-feature-controls[data-for="project[project_feature_attributes][merge_requests_access_level]"] .project-feature-toggle').click find('.project-feature-controls[data-for="project[project_feature_attributes][merge_requests_access_level]"] .gl-toggle').click
find('input[value="Save changes"]').send_keys(:return) find('input[value="Save changes"]').send_keys(:return)
end end
......
...@@ -30,7 +30,7 @@ RSpec.describe 'Projects > Settings > Visibility settings', :js do ...@@ -30,7 +30,7 @@ RSpec.describe 'Projects > Settings > Visibility settings', :js do
context 'merge requests select' do context 'merge requests select' do
it 'hides merge requests section' do it 'hides merge requests section' do
find('.project-feature-controls[data-for="project[project_feature_attributes][merge_requests_access_level]"] .project-feature-toggle').click find('.project-feature-controls[data-for="project[project_feature_attributes][merge_requests_access_level]"] .gl-toggle').click
expect(page).to have_selector('.merge-requests-feature', visible: false) expect(page).to have_selector('.merge-requests-feature', visible: false)
end end
...@@ -46,7 +46,7 @@ RSpec.describe 'Projects > Settings > Visibility settings', :js do ...@@ -46,7 +46,7 @@ RSpec.describe 'Projects > Settings > Visibility settings', :js do
context 'builds select' do context 'builds select' do
it 'hides builds select section' do it 'hides builds select section' do
find('.project-feature-controls[data-for="project[project_feature_attributes][builds_access_level]"] .project-feature-toggle').click find('.project-feature-controls[data-for="project[project_feature_attributes][builds_access_level]"] .gl-toggle').click
expect(page).to have_selector('.builds-feature', visible: false) expect(page).to have_selector('.builds-feature', visible: false)
end end
......
import { mount, shallowMount } from '@vue/test-utils'; import { GlToggle } from '@gitlab/ui';
import { shallowMount } from '@vue/test-utils';
import projectFeatureSetting from '~/pages/projects/shared/permissions/components/project_feature_setting.vue'; import ProjectFeatureSetting from '~/pages/projects/shared/permissions/components/project_feature_setting.vue';
import projectFeatureToggle from '~/vue_shared/components/toggle_button.vue';
describe('Project Feature Settings', () => { describe('Project Feature Settings', () => {
const defaultProps = { const defaultProps = {
...@@ -19,81 +18,76 @@ describe('Project Feature Settings', () => { ...@@ -19,81 +18,76 @@ describe('Project Feature Settings', () => {
}; };
let wrapper; let wrapper;
const mountComponent = (customProps) => { const findHiddenInput = () => wrapper.find(`input[name=${defaultProps.name}]`);
const propsData = { ...defaultProps, ...customProps }; const findToggle = () => wrapper.findComponent(GlToggle);
return shallowMount(projectFeatureSetting, { propsData });
};
beforeEach(() => { const mountComponent = (customProps = {}) =>
wrapper = mountComponent(); shallowMount(ProjectFeatureSetting, {
}); propsData: {
...defaultProps,
...customProps,
},
});
afterEach(() => { afterEach(() => {
wrapper.destroy(); wrapper.destroy();
wrapper = null;
}); });
describe('Hidden name input', () => { describe('Hidden name input', () => {
it('should set the hidden name input if the name exists', () => { it('should set the hidden name input if the name exists', () => {
expect(wrapper.find(`input[name=${defaultProps.name}]`).attributes().value).toBe('1'); wrapper = mountComponent();
expect(findHiddenInput().attributes('value')).toBe('1');
}); });
it('should not set the hidden name input if the name does not exist', () => { it('should not set the hidden name input if the name does not exist', () => {
wrapper.setProps({ name: null }); wrapper = mountComponent({ name: null });
return wrapper.vm.$nextTick(() => { expect(findHiddenInput().exists()).toBe(false);
expect(wrapper.find(`input[name=${defaultProps.name}]`).exists()).toBe(false);
});
}); });
}); });
describe('Feature toggle', () => { describe('Feature toggle', () => {
it('should be hidden if "showToggle" is passed false', async () => { it('should be hidden if "showToggle" is passed false', () => {
wrapper.setProps({ showToggle: false }); wrapper = mountComponent({ showToggle: false });
await wrapper.vm.$nextTick();
expect(wrapper.find(projectFeatureToggle).element).toBeUndefined(); expect(findToggle().exists()).toBe(false);
}); });
it('should enable the feature toggle if the value is not 0', () => { it('should enable the feature toggle if the value is not 0', () => {
expect(wrapper.find(projectFeatureToggle).props().value).toBe(true); wrapper = mountComponent();
expect(findToggle().props('value')).toBe(true);
}); });
it('should enable the feature toggle if the value is less than 0', () => { it('should enable the feature toggle if the value is less than 0', () => {
wrapper.setProps({ value: -1 }); wrapper = mountComponent({ value: -1 });
return wrapper.vm.$nextTick(() => { expect(findToggle().props('value')).toBe(true);
expect(wrapper.find(projectFeatureToggle).props().value).toBe(true);
});
}); });
it('should disable the feature toggle if the value is 0', () => { it('should disable the feature toggle if the value is 0', () => {
wrapper.setProps({ value: 0 }); wrapper = mountComponent({ value: 0 });
return wrapper.vm.$nextTick(() => { expect(findToggle().props('value')).toBe(false);
expect(wrapper.find(projectFeatureToggle).props().value).toBe(false);
});
}); });
it('should disable the feature toggle if disabledInput is set', () => { it('should disable the feature toggle if disabledInput is set', () => {
wrapper.setProps({ disabledInput: true }); wrapper = mountComponent({ disabledInput: true });
return wrapper.vm.$nextTick(() => { expect(findToggle().props('disabled')).toBe(true);
expect(wrapper.find(projectFeatureToggle).props().disabledInput).toBe(true);
});
}); });
it('should emit a change event when the feature toggle changes', () => { it('should emit a change event when the feature toggle changes', () => {
// Needs to be fully mounted to be able to trigger the click event on the internal button wrapper = mountComponent({ propsData: defaultProps });
wrapper = mount(projectFeatureSetting, { propsData: defaultProps });
expect(wrapper.emitted('change')).toBeUndefined();
expect(wrapper.emitted().change).toBeUndefined(); findToggle().vm.$emit('change', false);
wrapper.find(projectFeatureToggle).find('button').trigger('click');
return wrapper.vm.$nextTick().then(() => { expect(wrapper.emitted('change')).toHaveLength(1);
expect(wrapper.emitted().change.length).toBe(1); expect(wrapper.emitted('change')[0]).toEqual([0]);
expect(wrapper.emitted().change[0]).toEqual([0]);
});
}); });
}); });
...@@ -108,26 +102,23 @@ describe('Project Feature Settings', () => { ...@@ -108,26 +102,23 @@ describe('Project Feature Settings', () => {
`( `(
'should set disabled to $isDisabled when disabledInput is $disabledInput, the value is $value and options are $options', 'should set disabled to $isDisabled when disabledInput is $disabledInput, the value is $value and options are $options',
({ disabledInput, value, options, isDisabled }) => { ({ disabledInput, value, options, isDisabled }) => {
wrapper.setProps({ disabledInput, value, options }); wrapper = mountComponent({ disabledInput, value, options });
return wrapper.vm.$nextTick(() => { const expected = isDisabled ? 'disabled' : undefined;
if (isDisabled) {
expect(wrapper.find('select').attributes().disabled).toEqual('disabled'); expect(wrapper.find('select').attributes('disabled')).toBe(expected);
} else {
expect(wrapper.find('select').attributes().disabled).toBeUndefined();
}
});
}, },
); );
it('should emit the change when a new option is selected', () => { it('should emit the change when a new option is selected', () => {
expect(wrapper.emitted().change).toBeUndefined(); wrapper = mountComponent();
expect(wrapper.emitted('change')).toBeUndefined();
wrapper.findAll('option').at(1).trigger('change'); wrapper.findAll('option').at(1).trigger('change');
return wrapper.vm.$nextTick().then(() => { expect(wrapper.emitted('change')).toHaveLength(1);
expect(wrapper.emitted().change.length).toBe(1); expect(wrapper.emitted('change')[0]).toEqual([2]);
expect(wrapper.emitted().change[0]).toEqual([2]);
});
}); });
}); });
}); });
import { GlToggle } from '@gitlab/ui';
import { shallowMount, mount } from '@vue/test-utils'; import { shallowMount, mount } from '@vue/test-utils';
import settingsPanel from '~/pages/projects/shared/permissions/components/settings_panel.vue'; import settingsPanel from '~/pages/projects/shared/permissions/components/settings_panel.vue';
import { import {
featureAccessLevel, featureAccessLevel,
...@@ -7,7 +7,6 @@ import { ...@@ -7,7 +7,6 @@ import {
visibilityOptions, visibilityOptions,
} from '~/pages/projects/shared/permissions/constants'; } from '~/pages/projects/shared/permissions/constants';
import projectFeatureSetting from '~/pages/projects/shared/permissions/components/project_feature_setting.vue'; import projectFeatureSetting from '~/pages/projects/shared/permissions/components/project_feature_setting.vue';
import projectFeatureToggle from '~/vue_shared/components/toggle_button.vue';
const defaultProps = { const defaultProps = {
currentSettings: { currentSettings: {
...@@ -69,57 +68,61 @@ describe('Settings Panel', () => { ...@@ -69,57 +68,61 @@ describe('Settings Panel', () => {
}); });
}; };
const overrideCurrentSettings = (
currentSettingsProps,
extraProps = {},
mountFn = shallowMount,
) => {
return mountComponent({ ...extraProps, currentSettings: currentSettingsProps }, mountFn);
};
const findLFSSettingsRow = () => wrapper.find({ ref: 'git-lfs-settings' }); const findLFSSettingsRow = () => wrapper.find({ ref: 'git-lfs-settings' });
const findLFSSettingsMessage = () => findLFSSettingsRow().find('p'); const findLFSSettingsMessage = () => findLFSSettingsRow().find('p');
const findLFSFeatureToggle = () => findLFSSettingsRow().find(projectFeatureToggle); const findLFSFeatureToggle = () => findLFSSettingsRow().find(GlToggle);
const findRepositoryFeatureProjectRow = () => wrapper.find({ ref: 'repository-settings' }); const findRepositoryFeatureProjectRow = () => wrapper.find({ ref: 'repository-settings' });
const findRepositoryFeatureSetting = () => const findRepositoryFeatureSetting = () =>
findRepositoryFeatureProjectRow().find(projectFeatureSetting); findRepositoryFeatureProjectRow().find(projectFeatureSetting);
const findProjectVisibilitySettings = () => wrapper.find({ ref: 'project-visibility-settings' });
const findAnalyticsRow = () => wrapper.find({ ref: 'analytics-settings' }); const findAnalyticsRow = () => wrapper.find({ ref: 'analytics-settings' });
const findProjectVisibilityLevelInput = () => wrapper.find('[name="project[visibility_level]"]');
beforeEach(() => { const findRequestAccessEnabledInput = () =>
wrapper = mountComponent(); wrapper.find('[name="project[request_access_enabled]"]');
}); const findMergeRequestsAccessLevelInput = () =>
wrapper.find('[name="project[project_feature_attributes][merge_requests_access_level]"]');
const findForkingAccessLevelInput = () =>
wrapper.find('[name="project[project_feature_attributes][forking_access_level]"]');
const findBuildsAccessLevelInput = () =>
wrapper.find('[name="project[project_feature_attributes][builds_access_level]"]');
const findContainerRegistrySettings = () => wrapper.find({ ref: 'container-registry-settings' });
const findContainerRegistryEnabledInput = () =>
wrapper.find('[name="project[container_registry_enabled]"]');
const findPackageSettings = () => wrapper.find({ ref: 'package-settings' });
const findPackagesEnabledInput = () => wrapper.find('[name="project[packages_enabled]"]');
const findPagesSettings = () => wrapper.find({ ref: 'pages-settings' });
const findEmailSettings = () => wrapper.find({ ref: 'email-settings' });
const findShowDefaultAwardEmojis = () =>
wrapper.find('input[name="project[project_setting_attributes][show_default_award_emojis]"]');
const findMetricsVisibilitySettings = () => wrapper.find({ ref: 'metrics-visibility-settings' });
const findAllowEditingCommitMessages = () =>
wrapper.find({ ref: 'allow-editing-commit-messages' }).exists();
const findOperationsSettings = () => wrapper.find({ ref: 'operations-settings' });
afterEach(() => { afterEach(() => {
wrapper.destroy(); wrapper.destroy();
wrapper = null;
}); });
describe('Project Visibility', () => { describe('Project Visibility', () => {
it('should set the project visibility help path', () => { it('should set the project visibility help path', () => {
expect(wrapper.find({ ref: 'project-visibility-settings' }).props().helpPath).toBe( wrapper = mountComponent();
expect(findProjectVisibilitySettings().props('helpPath')).toBe(
defaultProps.visibilityHelpPath, defaultProps.visibilityHelpPath,
); );
}); });
it('should not disable the visibility level dropdown', () => { it('should not disable the visibility level dropdown', () => {
wrapper.setProps({ canChangeVisibilityLevel: true }); wrapper = mountComponent({ canChangeVisibilityLevel: true });
return wrapper.vm.$nextTick(() => { expect(findProjectVisibilityLevelInput().attributes('disabled')).toBeUndefined();
expect(
wrapper.find('[name="project[visibility_level]"]').attributes().disabled,
).toBeUndefined();
});
}); });
it('should disable the visibility level dropdown', () => { it('should disable the visibility level dropdown', () => {
wrapper.setProps({ canChangeVisibilityLevel: false }); wrapper = mountComponent({ canChangeVisibilityLevel: false });
return wrapper.vm.$nextTick(() => { expect(findProjectVisibilityLevelInput().attributes('disabled')).toBe('disabled');
expect(wrapper.find('[name="project[visibility_level]"]').attributes().disabled).toBe(
'disabled',
);
});
}); });
it.each` it.each`
...@@ -133,56 +136,56 @@ describe('Settings Panel', () => { ...@@ -133,56 +136,56 @@ describe('Settings Panel', () => {
`( `(
'sets disabled to $disabled for the visibility option $option when given $allowedOptions', 'sets disabled to $disabled for the visibility option $option when given $allowedOptions',
({ option, allowedOptions, disabled }) => { ({ option, allowedOptions, disabled }) => {
wrapper.setProps({ allowedVisibilityOptions: allowedOptions }); wrapper = mountComponent({ allowedVisibilityOptions: allowedOptions });
return wrapper.vm.$nextTick(() => { const attributeValue = findProjectVisibilityLevelInput()
const attributeValue = wrapper .find(`option[value="${option}"]`)
.find(`[name="project[visibility_level]"] option[value="${option}"]`) .attributes('disabled');
.attributes().disabled;
const expected = disabled ? 'disabled' : undefined;
if (disabled) {
expect(attributeValue).toBe('disabled'); expect(attributeValue).toBe(expected);
} else {
expect(attributeValue).toBeUndefined();
}
});
}, },
); );
it('should set the visibility level description based upon the selected visibility level', () => { it('should set the visibility level description based upon the selected visibility level', () => {
wrapper.find('[name="project[visibility_level]"]').setValue(visibilityOptions.INTERNAL); wrapper = mountComponent();
findProjectVisibilityLevelInput().setValue(visibilityOptions.INTERNAL);
expect(wrapper.find({ ref: 'project-visibility-settings' }).text()).toContain( expect(findProjectVisibilitySettings().text()).toContain(
visibilityLevelDescriptions[visibilityOptions.INTERNAL], visibilityLevelDescriptions[visibilityOptions.INTERNAL],
); );
}); });
it('should show the request access checkbox if the visibility level is not private', () => { it('should show the request access checkbox if the visibility level is not private', () => {
wrapper = overrideCurrentSettings({ visibilityLevel: visibilityOptions.INTERNAL }); wrapper = mountComponent({
currentSettings: { visibilityLevel: visibilityOptions.INTERNAL },
});
expect(wrapper.find('[name="project[request_access_enabled]"]').exists()).toBe(true); expect(findRequestAccessEnabledInput().exists()).toBe(true);
}); });
it('should not show the request access checkbox if the visibility level is private', () => { it('should not show the request access checkbox if the visibility level is private', () => {
wrapper = overrideCurrentSettings({ visibilityLevel: visibilityOptions.PRIVATE }); wrapper = mountComponent({ currentSettings: { visibilityLevel: visibilityOptions.PRIVATE } });
expect(wrapper.find('[name="project[request_access_enabled]"]').exists()).toBe(false); expect(findRequestAccessEnabledInput().exists()).toBe(false);
}); });
}); });
describe('Repository', () => { describe('Repository', () => {
it('should set the repository help text when the visibility level is set to private', () => { it('should set the repository help text when the visibility level is set to private', () => {
wrapper = overrideCurrentSettings({ visibilityLevel: visibilityOptions.PRIVATE }); wrapper = mountComponent({ currentSettings: { visibilityLevel: visibilityOptions.PRIVATE } });
expect(findRepositoryFeatureProjectRow().props().helpText).toBe( expect(findRepositoryFeatureProjectRow().props('helpText')).toBe(
'View and edit files in this project.', 'View and edit files in this project.',
); );
}); });
it('should set the repository help text with a read access warning when the visibility level is set to non-private', () => { it('should set the repository help text with a read access warning when the visibility level is set to non-private', () => {
wrapper = overrideCurrentSettings({ visibilityLevel: visibilityOptions.PUBLIC }); wrapper = mountComponent({ currentSettings: { visibilityLevel: visibilityOptions.PUBLIC } });
expect(findRepositoryFeatureProjectRow().props().helpText).toBe( expect(findRepositoryFeatureProjectRow().props('helpText')).toBe(
'View and edit files in this project. Non-project members will only have read access.', 'View and edit files in this project. Non-project members will only have read access.',
); );
}); });
...@@ -190,175 +193,152 @@ describe('Settings Panel', () => { ...@@ -190,175 +193,152 @@ describe('Settings Panel', () => {
describe('Merge requests', () => { describe('Merge requests', () => {
it('should enable the merge requests access level input when the repository is enabled', () => { it('should enable the merge requests access level input when the repository is enabled', () => {
wrapper = overrideCurrentSettings({ repositoryAccessLevel: featureAccessLevel.EVERYONE }); wrapper = mountComponent({
currentSettings: { repositoryAccessLevel: featureAccessLevel.EVERYONE },
});
expect( expect(findMergeRequestsAccessLevelInput().props('disabledInput')).toBe(false);
wrapper
.find('[name="project[project_feature_attributes][merge_requests_access_level]"]')
.props().disabledInput,
).toBe(false);
}); });
it('should disable the merge requests access level input when the repository is disabled', () => { it('should disable the merge requests access level input when the repository is disabled', () => {
wrapper = overrideCurrentSettings({ repositoryAccessLevel: featureAccessLevel.NOT_ENABLED }); wrapper = mountComponent({
currentSettings: { repositoryAccessLevel: featureAccessLevel.NOT_ENABLED },
});
expect( expect(findMergeRequestsAccessLevelInput().props('disabledInput')).toBe(true);
wrapper
.find('[name="project[project_feature_attributes][merge_requests_access_level]"]')
.props().disabledInput,
).toBe(true);
}); });
}); });
describe('Forks', () => { describe('Forks', () => {
it('should enable the forking access level input when the repository is enabled', () => { it('should enable the forking access level input when the repository is enabled', () => {
wrapper = overrideCurrentSettings({ repositoryAccessLevel: featureAccessLevel.EVERYONE }); wrapper = mountComponent({
currentSettings: { repositoryAccessLevel: featureAccessLevel.EVERYONE },
});
expect( expect(findForkingAccessLevelInput().props('disabledInput')).toBe(false);
wrapper.find('[name="project[project_feature_attributes][forking_access_level]"]').props()
.disabledInput,
).toBe(false);
}); });
it('should disable the forking access level input when the repository is disabled', () => { it('should disable the forking access level input when the repository is disabled', () => {
wrapper = overrideCurrentSettings({ repositoryAccessLevel: featureAccessLevel.NOT_ENABLED }); wrapper = mountComponent({
currentSettings: { repositoryAccessLevel: featureAccessLevel.NOT_ENABLED },
});
expect( expect(findForkingAccessLevelInput().props('disabledInput')).toBe(true);
wrapper.find('[name="project[project_feature_attributes][forking_access_level]"]').props()
.disabledInput,
).toBe(true);
}); });
}); });
describe('Pipelines', () => { describe('Pipelines', () => {
it('should enable the builds access level input when the repository is enabled', () => { it('should enable the builds access level input when the repository is enabled', () => {
wrapper = overrideCurrentSettings({ repositoryAccessLevel: featureAccessLevel.EVERYONE }); wrapper = mountComponent({
currentSettings: { repositoryAccessLevel: featureAccessLevel.EVERYONE },
});
expect( expect(findBuildsAccessLevelInput().props('disabledInput')).toBe(false);
wrapper.find('[name="project[project_feature_attributes][builds_access_level]"]').props()
.disabledInput,
).toBe(false);
}); });
it('should disable the builds access level input when the repository is disabled', () => { it('should disable the builds access level input when the repository is disabled', () => {
wrapper = overrideCurrentSettings({ repositoryAccessLevel: featureAccessLevel.NOT_ENABLED }); wrapper = mountComponent({
currentSettings: { repositoryAccessLevel: featureAccessLevel.NOT_ENABLED },
});
expect( expect(findBuildsAccessLevelInput().props('disabledInput')).toBe(true);
wrapper.find('[name="project[project_feature_attributes][builds_access_level]"]').props()
.disabledInput,
).toBe(true);
}); });
}); });
describe('Container registry', () => { describe('Container registry', () => {
it('should show the container registry settings if the registry is available', () => { it('should show the container registry settings if the registry is available', () => {
wrapper.setProps({ registryAvailable: true }); wrapper = mountComponent({ registryAvailable: true });
return wrapper.vm.$nextTick(() => { expect(findContainerRegistrySettings().exists()).toBe(true);
expect(wrapper.find({ ref: 'container-registry-settings' }).exists()).toBe(true);
});
}); });
it('should hide the container registry settings if the registry is not available', () => { it('should hide the container registry settings if the registry is not available', () => {
wrapper.setProps({ registryAvailable: false }); wrapper = mountComponent({ registryAvailable: false });
return wrapper.vm.$nextTick(() => { expect(findContainerRegistrySettings().exists()).toBe(false);
expect(wrapper.find({ ref: 'container-registry-settings' }).exists()).toBe(false);
});
}); });
it('should set the container registry settings help path', () => { it('should set the container registry settings help path', () => {
wrapper.setProps({ registryAvailable: true }); wrapper = mountComponent({ registryAvailable: true });
return wrapper.vm.$nextTick(() => { expect(findContainerRegistrySettings().props('helpPath')).toBe(defaultProps.registryHelpPath);
expect(wrapper.find({ ref: 'container-registry-settings' }).props().helpPath).toBe(
defaultProps.registryHelpPath,
);
});
}); });
it('should show the container registry public note if the visibility level is public and the registry is available', () => { it('should show the container registry public note if the visibility level is public and the registry is available', () => {
wrapper = overrideCurrentSettings( wrapper = mountComponent({
{ visibilityLevel: visibilityOptions.PUBLIC }, currentSettings: { visibilityLevel: visibilityOptions.PUBLIC },
{ registryAvailable: true }, registryAvailable: true,
); });
expect(wrapper.find({ ref: 'container-registry-settings' }).text()).toContain( expect(findContainerRegistrySettings().text()).toContain(
'Note: the container registry is always visible when a project is public', 'Note: the container registry is always visible when a project is public',
); );
}); });
it('should hide the container registry public note if the visibility level is private and the registry is available', () => { it('should hide the container registry public note if the visibility level is private and the registry is available', () => {
wrapper = overrideCurrentSettings( wrapper = mountComponent({
{ visibilityLevel: visibilityOptions.PRIVATE }, currentSettings: { visibilityLevel: visibilityOptions.PRIVATE },
{ registryAvailable: true }, registryAvailable: true,
); });
expect(wrapper.find({ ref: 'container-registry-settings' }).text()).not.toContain( expect(findContainerRegistrySettings().text()).not.toContain(
'Note: the container registry is always visible when a project is public', 'Note: the container registry is always visible when a project is public',
); );
}); });
it('should enable the container registry input when the repository is enabled', () => { it('should enable the container registry input when the repository is enabled', () => {
wrapper = overrideCurrentSettings( wrapper = mountComponent({
{ repositoryAccessLevel: featureAccessLevel.EVERYONE }, currentSettings: { repositoryAccessLevel: featureAccessLevel.EVERYONE },
{ registryAvailable: true }, registryAvailable: true,
); });
expect( expect(findContainerRegistryEnabledInput().props('disabled')).toBe(false);
wrapper.find('[name="project[container_registry_enabled]"]').props().disabledInput,
).toBe(false);
}); });
it('should disable the container registry input when the repository is disabled', () => { it('should disable the container registry input when the repository is disabled', () => {
wrapper = overrideCurrentSettings( wrapper = mountComponent({
{ repositoryAccessLevel: featureAccessLevel.NOT_ENABLED }, currentSettings: { repositoryAccessLevel: featureAccessLevel.NOT_ENABLED },
{ registryAvailable: true }, registryAvailable: true,
); });
expect( expect(findContainerRegistryEnabledInput().props('disabled')).toBe(true);
wrapper.find('[name="project[container_registry_enabled]"]').props().disabledInput,
).toBe(true);
}); });
}); });
describe('Git Large File Storage', () => { describe('Git Large File Storage', () => {
it('should show the LFS settings if LFS is available', () => { it('should show the LFS settings if LFS is available', () => {
wrapper.setProps({ lfsAvailable: true }); wrapper = mountComponent({ lfsAvailable: true });
return wrapper.vm.$nextTick(() => { expect(findLFSSettingsRow().exists()).toBe(true);
expect(findLFSSettingsRow().exists()).toBe(true);
});
}); });
it('should hide the LFS settings if LFS is not available', () => { it('should hide the LFS settings if LFS is not available', () => {
wrapper.setProps({ lfsAvailable: false }); wrapper = mountComponent({ lfsAvailable: false });
return wrapper.vm.$nextTick(() => { expect(findLFSSettingsRow().exists()).toBe(false);
expect(findLFSSettingsRow().exists()).toBe(false);
});
}); });
it('should set the LFS settings help path', () => { it('should set the LFS settings help path', () => {
expect(findLFSSettingsRow().props().helpPath).toBe(defaultProps.lfsHelpPath); wrapper = mountComponent();
expect(findLFSSettingsRow().props('helpPath')).toBe(defaultProps.lfsHelpPath);
}); });
it('should enable the LFS input when the repository is enabled', () => { it('should enable the LFS input when the repository is enabled', () => {
wrapper = overrideCurrentSettings( wrapper = mountComponent({
{ repositoryAccessLevel: featureAccessLevel.EVERYONE }, currentSettings: { repositoryAccessLevel: featureAccessLevel.EVERYONE },
{ lfsAvailable: true }, lfsAvailable: true,
); });
expect(findLFSFeatureToggle().props().disabledInput).toBe(false); expect(findLFSFeatureToggle().props('disabled')).toBe(false);
}); });
it('should disable the LFS input when the repository is disabled', () => { it('should disable the LFS input when the repository is disabled', () => {
wrapper = overrideCurrentSettings( wrapper = mountComponent({
{ repositoryAccessLevel: featureAccessLevel.NOT_ENABLED }, currentSettings: { repositoryAccessLevel: featureAccessLevel.NOT_ENABLED },
{ lfsAvailable: true }, lfsAvailable: true,
); });
expect(findLFSFeatureToggle().props().disabledInput).toBe(true); expect(findLFSFeatureToggle().props('disabled')).toBe(true);
}); });
it('should not change lfsEnabled when disabling the repository', async () => { it('should not change lfsEnabled when disabling the repository', async () => {
...@@ -373,8 +353,7 @@ describe('Settings Panel', () => { ...@@ -373,8 +353,7 @@ describe('Settings Panel', () => {
expect(isToggleButtonChecked(lfsFeatureToggleButton)).toBe(true); expect(isToggleButtonChecked(lfsFeatureToggleButton)).toBe(true);
expect(isToggleButtonChecked(repositoryFeatureToggleButton)).toBe(true); expect(isToggleButtonChecked(repositoryFeatureToggleButton)).toBe(true);
repositoryFeatureToggleButton.trigger('click'); await repositoryFeatureToggleButton.trigger('click');
await wrapper.vm.$nextTick();
expect(isToggleButtonChecked(repositoryFeatureToggleButton)).toBe(false); expect(isToggleButtonChecked(repositoryFeatureToggleButton)).toBe(false);
// LFS toggle should still be checked // LFS toggle should still be checked
...@@ -418,47 +397,39 @@ describe('Settings Panel', () => { ...@@ -418,47 +397,39 @@ describe('Settings Panel', () => {
describe('Packages', () => { describe('Packages', () => {
it('should show the packages settings if packages are available', () => { it('should show the packages settings if packages are available', () => {
wrapper.setProps({ packagesAvailable: true }); wrapper = mountComponent({ packagesAvailable: true });
return wrapper.vm.$nextTick(() => { expect(findPackageSettings().exists()).toBe(true);
expect(wrapper.find({ ref: 'package-settings' }).exists()).toBe(true);
});
}); });
it('should hide the packages settings if packages are not available', () => { it('should hide the packages settings if packages are not available', () => {
wrapper.setProps({ packagesAvailable: false }); wrapper = mountComponent({ packagesAvailable: false });
return wrapper.vm.$nextTick(() => { expect(findPackageSettings().exists()).toBe(false);
expect(wrapper.find({ ref: 'package-settings' }).exists()).toBe(false);
});
}); });
it('should set the package settings help path', () => { it('should set the package settings help path', () => {
wrapper.setProps({ packagesAvailable: true }); wrapper = mountComponent({ packagesAvailable: true });
return wrapper.vm.$nextTick(() => { expect(findPackageSettings().props('helpPath')).toBe(defaultProps.packagesHelpPath);
expect(wrapper.find({ ref: 'package-settings' }).props().helpPath).toBe(
defaultProps.packagesHelpPath,
);
});
}); });
it('should enable the packages input when the repository is enabled', () => { it('should enable the packages input when the repository is enabled', () => {
wrapper = overrideCurrentSettings( wrapper = mountComponent({
{ repositoryAccessLevel: featureAccessLevel.EVERYONE }, currentSettings: { repositoryAccessLevel: featureAccessLevel.EVERYONE },
{ packagesAvailable: true }, packagesAvailable: true,
); });
expect(wrapper.find('[name="project[packages_enabled]"]').props().disabledInput).toBe(false); expect(findPackagesEnabledInput().props('disabled')).toBe(false);
}); });
it('should disable the packages input when the repository is disabled', () => { it('should disable the packages input when the repository is disabled', () => {
wrapper = overrideCurrentSettings( wrapper = mountComponent({
{ repositoryAccessLevel: featureAccessLevel.NOT_ENABLED }, currentSettings: { repositoryAccessLevel: featureAccessLevel.NOT_ENABLED },
{ packagesAvailable: true }, packagesAvailable: true,
); });
expect(wrapper.find('[name="project[packages_enabled]"]').props().disabledInput).toBe(true); expect(findPackagesEnabledInput().props('disabled')).toBe(true);
}); });
}); });
...@@ -472,78 +443,66 @@ describe('Settings Panel', () => { ...@@ -472,78 +443,66 @@ describe('Settings Panel', () => {
`( `(
'should $visibility the page settings if pagesAvailable is $pagesAvailable and pagesAccessControlEnabled is $pagesAccessControlEnabled', 'should $visibility the page settings if pagesAvailable is $pagesAvailable and pagesAccessControlEnabled is $pagesAccessControlEnabled',
({ pagesAvailable, pagesAccessControlEnabled, visibility }) => { ({ pagesAvailable, pagesAccessControlEnabled, visibility }) => {
wrapper.setProps({ pagesAvailable, pagesAccessControlEnabled }); wrapper = mountComponent({ pagesAvailable, pagesAccessControlEnabled });
return wrapper.vm.$nextTick(() => { expect(findPagesSettings().exists()).toBe(visibility === 'show');
expect(wrapper.find({ ref: 'pages-settings' }).exists()).toBe(visibility === 'show');
});
}, },
); );
it('should set the pages settings help path', () => { it('should set the pages settings help path', () => {
wrapper.setProps({ pagesAvailable: true, pagesAccessControlEnabled: true }); wrapper = mountComponent({ pagesAvailable: true, pagesAccessControlEnabled: true });
return wrapper.vm.$nextTick(() => { expect(findPagesSettings().props('helpPath')).toBe(defaultProps.pagesHelpPath);
expect(wrapper.find({ ref: 'pages-settings' }).props().helpPath).toBe(
defaultProps.pagesHelpPath,
);
});
}); });
}); });
describe('Email notifications', () => { describe('Email notifications', () => {
it('should show the disable email notifications input if emails an be disabled', () => { it('should show the disable email notifications input if emails an be disabled', () => {
wrapper.setProps({ canDisableEmails: true }); wrapper = mountComponent({ canDisableEmails: true });
return wrapper.vm.$nextTick(() => { expect(findEmailSettings().exists()).toBe(true);
expect(wrapper.find({ ref: 'email-settings' }).exists()).toBe(true);
});
}); });
it('should hide the disable email notifications input if emails cannot be disabled', () => { it('should hide the disable email notifications input if emails cannot be disabled', () => {
wrapper.setProps({ canDisableEmails: false }); wrapper = mountComponent({ canDisableEmails: false });
return wrapper.vm.$nextTick(() => { expect(findEmailSettings().exists()).toBe(false);
expect(wrapper.find({ ref: 'email-settings' }).exists()).toBe(false);
});
}); });
}); });
describe('Default award emojis', () => { describe('Default award emojis', () => {
it('should show the "Show default award emojis" input', () => { it('should show the "Show default award emojis" input', () => {
return wrapper.vm.$nextTick(() => { wrapper = mountComponent();
expect(
wrapper expect(findShowDefaultAwardEmojis().exists()).toBe(true);
.find('input[name="project[project_setting_attributes][show_default_award_emojis]"]')
.exists(),
).toBe(true);
});
}); });
}); });
describe('Metrics dashboard', () => { describe('Metrics dashboard', () => {
it('should show the metrics dashboard access toggle', () => { it('should show the metrics dashboard access toggle', () => {
return wrapper.vm.$nextTick(() => { wrapper = mountComponent();
expect(wrapper.find({ ref: 'metrics-visibility-settings' }).exists()).toBe(true);
}); expect(findMetricsVisibilitySettings().exists()).toBe(true);
}); });
it('should contain help text', () => { it('should contain help text', () => {
expect(wrapper.find({ ref: 'metrics-visibility-settings' }).props().helpText).toBe( wrapper = mountComponent();
expect(findMetricsVisibilitySettings().props('helpText')).toBe(
"Visualize the project's performance metrics.", "Visualize the project's performance metrics.",
); );
}); });
it.each` it.each`
scenario | selectedOption | selectedOptionLabel scenario | selectedOption | selectedOptionLabel
${{ visibilityLevel: visibilityOptions.PRIVATE }} | ${String(featureAccessLevel.PROJECT_MEMBERS)} | ${'Only Project Members'} ${{ currentSettings: { visibilityLevel: visibilityOptions.PRIVATE } }} | ${String(featureAccessLevel.PROJECT_MEMBERS)} | ${'Only Project Members'}
${{ operationsAccessLevel: featureAccessLevel.NOT_ENABLED }} | ${String(featureAccessLevel.NOT_ENABLED)} | ${'Enable feature to choose access level'} ${{ currentSettings: { operationsAccessLevel: featureAccessLevel.NOT_ENABLED } }} | ${String(featureAccessLevel.NOT_ENABLED)} | ${'Enable feature to choose access level'}
`( `(
'should disable the metrics visibility dropdown when #scenario', 'should disable the metrics visibility dropdown when #scenario',
({ scenario, selectedOption, selectedOptionLabel }) => { ({ scenario, selectedOption, selectedOptionLabel }) => {
wrapper = overrideCurrentSettings(scenario, {}, mount); wrapper = mountComponent(scenario, mount);
const select = wrapper.find({ ref: 'metrics-visibility-settings' }).find('select'); const select = findMetricsVisibilitySettings().find('select');
const option = select.find('option'); const option = select.find('option');
expect(select.attributes('disabled')).toBe('disabled'); expect(select.attributes('disabled')).toBe('disabled');
...@@ -556,31 +515,29 @@ describe('Settings Panel', () => { ...@@ -556,31 +515,29 @@ describe('Settings Panel', () => {
describe('Settings panel with feature flags', () => { describe('Settings panel with feature flags', () => {
describe('Allow edit of commit message', () => { describe('Allow edit of commit message', () => {
it('should show the allow editing of commit messages checkbox', async () => { it('should show the allow editing of commit messages checkbox', () => {
wrapper = mountComponent({ wrapper = mountComponent({
glFeatures: { allowEditingCommitMessages: true }, glFeatures: { allowEditingCommitMessages: true },
}); });
await wrapper.vm.$nextTick(); expect(findAllowEditingCommitMessages()).toBe(true);
expect(wrapper.find({ ref: 'allow-editing-commit-messages' }).exists()).toBe(true);
}); });
}); });
}); });
describe('Analytics', () => { describe('Analytics', () => {
it('should show the analytics toggle', async () => { it('should show the analytics toggle', () => {
await wrapper.vm.$nextTick(); wrapper = mountComponent();
expect(findAnalyticsRow().exists()).toBe(true); expect(findAnalyticsRow().exists()).toBe(true);
}); });
}); });
describe('Operations', () => { describe('Operations', () => {
it('should show the operations toggle', async () => { it('should show the operations toggle', () => {
await wrapper.vm.$nextTick(); wrapper = mountComponent();
expect(wrapper.find({ ref: 'operations-settings' }).exists()).toBe(true); expect(findOperationsSettings().exists()).toBe(true);
}); });
}); });
}); });
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