blob: 67acf00ffe108251194579f8e9f40884d8188fb6 [file] [log] [blame]
/**
* @license
* Copyright (C) 2017 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import '@polymer/iron-input/iron-input';
import '../../../styles/gr-font-styles';
import '../../../styles/gr-form-styles';
import '../../../styles/shared-styles';
import '../../shared/gr-button/gr-button';
import '../../shared/gr-icons/gr-icons';
import '../gr-permission/gr-permission';
import {PolymerElement} from '@polymer/polymer/polymer-element';
import {htmlTemplate} from './gr-access-section_html';
import {
AccessPermissions,
PermissionArray,
PermissionArrayItem,
toSortedPermissionsArray,
} from '../../../utils/access-util';
import {customElement, property} from '@polymer/decorators';
import {
EditablePermissionInfo,
PermissionAccessSection,
EditableProjectAccessGroups,
} from '../gr-repo-access/gr-repo-access-interfaces';
import {
CapabilityInfoMap,
GitRef,
LabelNameToLabelTypeInfoMap,
RepoName,
} from '../../../types/common';
import {PolymerDomRepeatEvent} from '../../../types/types';
import {fireEvent} from '../../../utils/event-util';
/**
* Fired when the section has been modified or removed.
*
* @event access-modified
*/
/**
* Fired when a section that was previously added was removed.
*
* @event added-section-removed
*/
const GLOBAL_NAME = 'GLOBAL_CAPABILITIES';
// The name that gets automatically input when a new reference is added.
const NEW_NAME = 'refs/heads/*';
const REFS_NAME = 'refs/';
const ON_BEHALF_OF = '(On Behalf Of)';
const LABEL = 'Label';
export interface GrAccessSection {
$: {
permissionSelect: HTMLSelectElement;
};
}
@customElement('gr-access-section')
export class GrAccessSection extends PolymerElement {
static get template() {
return htmlTemplate;
}
@property({type: String})
repo?: RepoName;
@property({type: Object})
capabilities?: CapabilityInfoMap;
@property({type: Object, notify: true, observer: '_updateSection'})
section?: PermissionAccessSection;
@property({type: Object})
groups?: EditableProjectAccessGroups;
@property({type: Object})
labels?: LabelNameToLabelTypeInfoMap;
@property({type: Boolean, observer: '_handleEditingChanged'})
editing = false;
@property({type: Boolean})
canUpload?: boolean;
@property({type: Array})
ownerOf?: GitRef[];
@property({type: String})
_originalId?: GitRef;
@property({type: Boolean})
_editingRef = false;
@property({type: Boolean})
_deleted = false;
@property({type: Array})
_permissions?: PermissionArray<EditablePermissionInfo>;
constructor() {
super();
this.addEventListener('access-saved', () => this._handleAccessSaved());
}
_updateSection(section: PermissionAccessSection) {
this._permissions = toSortedPermissionsArray(section.value.permissions);
this._originalId = section.id;
}
_handleAccessSaved() {
if (!this.section) {
return;
}
// Set a new 'original' value to keep track of after the value has been
// saved.
this._updateSection(this.section);
}
_handleValueChange() {
if (!this.section) {
return;
}
if (!this.section.value.added) {
this.section.value.modified = this.section.id !== this._originalId;
// Allows overall access page to know a change has been made.
// For a new section, this is not fired because new permissions and
// rules have to be added in order to save, modifying the ref is not
// enough.
fireEvent(this, 'access-modified');
}
this.section.value.updatedId = this.section.id;
}
_handleEditingChanged(editing: boolean, editingOld: boolean) {
// Ignore when editing gets set initially.
if (!editingOld) {
return;
}
if (!this.section || !this._permissions) {
return;
}
// Restore original values if no longer editing.
if (!editing) {
this._editingRef = false;
this._deleted = false;
delete this.section.value.deleted;
// Restore section ref.
this.set(['section', 'id'], this._originalId);
// Remove any unsaved but added permissions.
this._permissions = this._permissions.filter(p => !p.value.added);
for (const key of Object.keys(this.section.value.permissions)) {
if (this.section.value.permissions[key].added) {
delete this.section.value.permissions[key];
}
}
}
}
_computePermissions(
name: string,
capabilities?: CapabilityInfoMap,
labels?: LabelNameToLabelTypeInfoMap,
// This is just for triggering re-computation. We don't use the value.
_?: unknown
) {
let allPermissions;
const section = this.section;
if (!section || !section.value) {
return [];
}
if (name === GLOBAL_NAME) {
allPermissions = toSortedPermissionsArray(capabilities);
} else {
const labelOptions = this._computeLabelOptions(labels);
allPermissions = labelOptions.concat(
toSortedPermissionsArray(AccessPermissions)
);
}
return allPermissions.filter(
permission => !section.value.permissions[permission.id]
);
}
_handleAddedPermissionRemoved(e: PolymerDomRepeatEvent) {
if (!this._permissions) {
return;
}
const index = e.model.index;
this._permissions = this._permissions
.slice(0, index)
.concat(this._permissions.slice(index + 1, this._permissions.length));
}
_computeLabelOptions(labels?: LabelNameToLabelTypeInfoMap) {
const labelOptions = [];
if (!labels) {
return [];
}
for (const labelName of Object.keys(labels)) {
labelOptions.push({
id: 'label-' + labelName,
value: {
name: `${LABEL} ${labelName}`,
id: 'label-' + labelName,
},
});
labelOptions.push({
id: 'labelAs-' + labelName,
value: {
name: `${LABEL} ${labelName} ${ON_BEHALF_OF}`,
id: 'labelAs-' + labelName,
},
});
}
return labelOptions;
}
_computePermissionName(
name: string,
permission: PermissionArrayItem<EditablePermissionInfo>,
capabilities?: CapabilityInfoMap
): string | undefined {
if (name === GLOBAL_NAME) {
return capabilities?.[permission.id]?.name;
} else if (AccessPermissions[permission.id]) {
return AccessPermissions[permission.id]?.name;
} else if (permission.value.label) {
let behalfOf = '';
if (permission.id.startsWith('labelAs-')) {
behalfOf = ON_BEHALF_OF;
}
return `${LABEL} ${permission.value.label}${behalfOf}`;
}
return undefined;
}
_computeSectionName(name: string) {
// When a new section is created, it doesn't yet have a ref. Set into
// edit mode so that the user can input one.
if (!name) {
this._editingRef = true;
// Needed for the title value. This is the same default as GWT.
name = NEW_NAME;
// Needed for the input field value.
this.set('section.id', name);
}
if (name === GLOBAL_NAME) {
return 'Global Capabilities';
} else if (name.startsWith(REFS_NAME)) {
return `Reference: ${name}`;
}
return name;
}
_handleRemoveReference() {
if (!this.section) {
return;
}
if (this.section.value.added) {
fireEvent(this, 'added-section-removed');
}
this._deleted = true;
this.section.value.deleted = true;
fireEvent(this, 'access-modified');
}
_handleUndoRemove() {
if (!this.section) {
return;
}
this._deleted = false;
delete this.section.value.deleted;
}
editRefInput() {
return this.root!.querySelector(
PolymerElement
? 'iron-input.editRefInput'
: 'input[is=iron-input].editRefInput'
) as HTMLInputElement;
}
editReference() {
this._editingRef = true;
this.editRefInput().focus();
}
_isEditEnabled(
canUpload: boolean | undefined,
ownerOf: GitRef[] | undefined,
sectionId: GitRef
) {
return canUpload || (ownerOf && ownerOf.indexOf(sectionId) >= 0);
}
_computeSectionClass(
editing: boolean,
canUpload: boolean | undefined,
ownerOf: GitRef[] | undefined,
editingRef: boolean,
deleted: boolean
) {
const classList = [];
if (
editing &&
this.section &&
this._isEditEnabled(canUpload, ownerOf, this.section.id)
) {
classList.push('editing');
}
if (editingRef) {
classList.push('editingRef');
}
if (deleted) {
classList.push('deleted');
}
return classList.join(' ');
}
_computeEditBtnClass(name: string) {
return name === GLOBAL_NAME ? 'global' : '';
}
_handleAddPermission() {
const value = this.$.permissionSelect.value as GitRef;
const permission: PermissionArrayItem<EditablePermissionInfo> = {
id: value,
value: {rules: {}, added: true},
};
// This is needed to update the 'label' property of the
// 'label-<label-name>' permission.
//
// The value from the add permission dropdown will either be
// label-<label-name> or labelAs-<labelName>.
// But, the format of the API response is as such:
// "permissions": {
// "label-Code-Review": {
// "label": "Code-Review",
// "rules": {...}
// }
// }
// }
// When we add a new item, we have to push the new permission in the same
// format as the ones that have been returned by the API.
if (value.startsWith('label')) {
permission.value.label = value
.replace('label-', '')
.replace('labelAs-', '');
}
// Add to the end of the array (used in dom-repeat) and also to the
// section object that is two way bound with its parent element.
this.push('_permissions', permission);
this.set(['section.value.permissions', permission.id], permission.value);
}
}
declare global {
interface HTMLElementTagNameMap {
'gr-access-section': GrAccessSection;
}
}