import { find } from 'lodash'; import createFlash from '~/flash'; import axios from '~/lib/utils/axios_utils'; import { __ } from '~/locale'; import AccessDropdown from '~/projects/settings/access_dropdown'; import { initToggle } from '~/toggles'; import { ACCESS_LEVELS, LEVEL_TYPES } from './constants'; export default class ProtectedBranchEdit { constructor(options) { this.hasLicense = options.hasLicense; this.$wraps = {}; this.hasChanges = false; this.$wrap = options.$wrap; this.$allowedToMergeDropdown = this.$wrap.find('.js-allowed-to-merge'); this.$allowedToPushDropdown = this.$wrap.find('.js-allowed-to-push'); this.$wraps[ACCESS_LEVELS.MERGE] = this.$allowedToMergeDropdown.closest( `.${ACCESS_LEVELS.MERGE}-container`, ); this.$wraps[ACCESS_LEVELS.PUSH] = this.$allowedToPushDropdown.closest( `.${ACCESS_LEVELS.PUSH}-container`, ); this.buildDropdowns(); this.initToggles(); } initToggles() { const wrap = this.$wrap.get(0); const forcePushToggle = initToggle(wrap.querySelector('.js-force-push-toggle')); if (forcePushToggle) { forcePushToggle.$on('change', (value) => { forcePushToggle.isLoading = true; forcePushToggle.disabled = true; this.updateProtectedBranch( { allow_force_push: value, }, () => { forcePushToggle.isLoading = false; forcePushToggle.disabled = false; }, ); }); } if (this.hasLicense) { const codeOwnerToggle = initToggle(wrap.querySelector('.js-code-owner-toggle')); if (codeOwnerToggle) { codeOwnerToggle.$on('change', (value) => { codeOwnerToggle.isLoading = true; codeOwnerToggle.disabled = true; this.updateProtectedBranch( { code_owner_approval_required: value, }, () => { codeOwnerToggle.isLoading = false; codeOwnerToggle.disabled = false; }, ); }); } } } updateProtectedBranch(formData, callback) { axios .patch(this.$wrap.data('url'), { protected_branch: formData, }) .then(callback) .catch(() => { createFlash({ message: __('Failed to update branch!') }); }); } buildDropdowns() { // Allowed to merge dropdown this[`${ACCESS_LEVELS.MERGE}_dropdown`] = new AccessDropdown({ accessLevel: ACCESS_LEVELS.MERGE, accessLevelsData: gon.merge_access_levels, $dropdown: this.$allowedToMergeDropdown, onSelect: this.onSelectOption.bind(this), onHide: this.onDropdownHide.bind(this), hasLicense: this.hasLicense, }); // Allowed to push dropdown this[`${ACCESS_LEVELS.PUSH}_dropdown`] = new AccessDropdown({ accessLevel: ACCESS_LEVELS.PUSH, accessLevelsData: gon.push_access_levels, $dropdown: this.$allowedToPushDropdown, onSelect: this.onSelectOption.bind(this), onHide: this.onDropdownHide.bind(this), hasLicense: this.hasLicense, }); } onSelectOption() { this.hasChanges = true; } onDropdownHide() { if (!this.hasChanges) { return; } this.hasChanges = true; this.updatePermissions(); } updatePermissions() { const formData = Object.keys(ACCESS_LEVELS).reduce((acc, level) => { const accessLevelName = ACCESS_LEVELS[level]; const inputData = this[`${accessLevelName}_dropdown`].getInputData(accessLevelName); acc[`${accessLevelName}_attributes`] = inputData; return acc; }, {}); axios .patch(this.$wrap.data('url'), { protected_branch: formData, }) .then(({ data }) => { this.hasChanges = false; Object.keys(ACCESS_LEVELS).forEach((level) => { const accessLevelName = ACCESS_LEVELS[level]; // The data coming from server will be the new persisted *state* for each dropdown this.setSelectedItemsToDropdown(data[accessLevelName], `${accessLevelName}_dropdown`); }); this.$allowedToMergeDropdown.enable(); this.$allowedToPushDropdown.enable(); }) .catch(() => { this.$allowedToMergeDropdown.enable(); this.$allowedToPushDropdown.enable(); createFlash({ message: __('Failed to update branch!') }); }); } setSelectedItemsToDropdown(items = [], dropdownName) { const itemsToAdd = items.map((currentItem) => { if (currentItem.user_id) { // Do this only for users for now // get the current data for selected items const selectedItems = this[dropdownName].getSelectedItems(); const currentSelectedItem = find(selectedItems, { user_id: currentItem.user_id, }); return { id: currentItem.id, user_id: currentItem.user_id, type: LEVEL_TYPES.USER, persisted: true, name: currentSelectedItem.name, username: currentSelectedItem.username, avatar_url: currentSelectedItem.avatar_url, }; } else if (currentItem.group_id) { return { id: currentItem.id, group_id: currentItem.group_id, type: LEVEL_TYPES.GROUP, persisted: true, }; } return { id: currentItem.id, access_level: currentItem.access_level, type: LEVEL_TYPES.ROLE, persisted: true, }; }); this[dropdownName].setSelectedItems(itemsToAdd); } }