blob: e67b838a9a683b57879ecbd3e0a5523daef7c18c [file] [log] [blame]
Dave Borowitz8cdc76b2018-03-26 10:04:27 -04001/**
2 * @license
Ben Rohlfs94fcbbc2022-05-27 10:45:03 +02003 * Copyright 2017 Google LLC
4 * SPDX-License-Identifier: Apache-2.0
Dave Borowitz8cdc76b2018-03-26 10:04:27 -04005 */
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +01006
Tao Zhou7d334b42020-08-31 10:56:33 +02007import '../../shared/gr-dialog/gr-dialog';
8import '../../shared/gr-list-view/gr-list-view';
9import '../../shared/gr-overlay/gr-overlay';
Tao Zhou7d334b42020-08-31 10:56:33 +020010import '../gr-create-group-dialog/gr-create-group-dialog';
Tao Zhou7d334b42020-08-31 10:56:33 +020011import {GerritNav} from '../../core/gr-navigation/gr-navigation';
Tao Zhou7d334b42020-08-31 10:56:33 +020012import {AppElementAdminParams} from '../../gr-app-types';
13import {GrOverlay} from '../../shared/gr-overlay/gr-overlay';
Dmitrii Filippov55616542020-10-05 19:07:27 +020014import {GroupId, GroupInfo, GroupName} from '../../../types/common';
Tao Zhou7d334b42020-08-31 10:56:33 +020015import {GrCreateGroupDialog} from '../gr-create-group-dialog/gr-create-group-dialog';
Milutin Kristofic60150132020-11-23 20:15:23 +010016import {fireTitleChange} from '../../../utils/event-util';
Chris Poucetc6e880b2021-11-15 19:57:06 +010017import {getAppContext} from '../../../services/app-context';
Frank Borden949a17d2021-09-28 11:30:41 +000018import {SHOWN_ITEMS_COUNT} from '../../../constants/constants';
Paladox nonea6629992021-11-14 22:05:24 +000019import {tableStyles} from '../../../styles/gr-table-styles';
20import {sharedStyles} from '../../../styles/shared-styles';
21import {LitElement, PropertyValues, html} from 'lit';
22import {customElement, query, property, state} from 'lit/decorators';
23import {assertIsDefined} from '../../../utils/common-util';
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +010024
Tao Zhou7d334b42020-08-31 10:56:33 +020025declare global {
26 interface HTMLElementTagNameMap {
27 'gr-admin-group-list': GrAdminGroupList;
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +010028 }
Tao Zhou7d334b42020-08-31 10:56:33 +020029}
30
Tao Zhou7d334b42020-08-31 10:56:33 +020031@customElement('gr-admin-group-list')
Paladox nonea6629992021-11-14 22:05:24 +000032export class GrAdminGroupList extends LitElement {
33 readonly path = '/admin/groups';
34
35 @query('#createOverlay') private createOverlay?: GrOverlay;
36
37 @query('#createNewModal') private createNewModal?: GrCreateGroupDialog;
Tao Zhou7d334b42020-08-31 10:56:33 +020038
39 @property({type: Object})
40 params?: AppElementAdminParams;
41
42 /**
43 * Offset of currently visible query results.
44 */
Paladox nonea6629992021-11-14 22:05:24 +000045 @state() private offset = 0;
Tao Zhou7d334b42020-08-31 10:56:33 +020046
Paladox nonea6629992021-11-14 22:05:24 +000047 @state() private hasNewGroupName = false;
Tao Zhou7d334b42020-08-31 10:56:33 +020048
Paladox nonea6629992021-11-14 22:05:24 +000049 @state() private createNewCapability = false;
Tao Zhou7d334b42020-08-31 10:56:33 +020050
Paladox nonea6629992021-11-14 22:05:24 +000051 // private but used in test
52 @state() groups: GroupInfo[] = [];
Tao Zhou7d334b42020-08-31 10:56:33 +020053
Paladox nonea6629992021-11-14 22:05:24 +000054 @state() private groupsPerPage = 25;
Tao Zhou7d334b42020-08-31 10:56:33 +020055
Paladox nonea6629992021-11-14 22:05:24 +000056 // private but used in test
57 @state() loading = true;
Tao Zhou7d334b42020-08-31 10:56:33 +020058
Paladox nonea6629992021-11-14 22:05:24 +000059 @state() private filter = '';
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +010060
Chris Poucetc6e880b2021-11-15 19:57:06 +010061 private readonly restApiService = getAppContext().restApiService;
Ben Rohlfs43935a42020-12-01 19:14:09 +010062
Chris Poucet59dad572021-08-20 15:25:36 +000063 override connectedCallback() {
Ben Rohlfs5f520da2021-03-10 14:58:43 +010064 super.connectedCallback();
Paladox nonea6629992021-11-14 22:05:24 +000065 this.getCreateGroupCapability();
Milutin Kristofic60150132020-11-23 20:15:23 +010066 fireTitleChange(this, 'Groups');
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +010067 }
68
Paladox nonea6629992021-11-14 22:05:24 +000069 static override get styles() {
70 return [tableStyles, sharedStyles];
71 }
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +010072
Paladox nonea6629992021-11-14 22:05:24 +000073 override render() {
74 return html`
75 <gr-list-view
76 .createNew=${this.createNewCapability}
77 .filter=${this.filter}
78 .items=${this.groups}
79 .itemsPerPage=${this.groupsPerPage}
80 .loading=${this.loading}
81 .offset=${this.offset}
82 .path=${this.path}
83 @create-clicked=${() => this.handleCreateClicked()}
84 >
85 <table id="list" class="genericList">
86 <tbody>
87 <tr class="headerRow">
88 <th class="name topHeader">Group Name</th>
89 <th class="description topHeader">Group Description</th>
90 <th class="visibleToAll topHeader">Visible To All</th>
91 </tr>
92 <tr
93 id="loading"
94 class="loadingMsg ${this.loading ? 'loading' : ''}"
95 >
96 <td>Loading...</td>
97 </tr>
98 </tbody>
99 <tbody class=${this.loading ? 'loading' : ''}>
100 ${this.groups
101 .slice(0, SHOWN_ITEMS_COUNT)
102 .map(group => this.renderGroupList(group))}
103 </tbody>
104 </table>
105 </gr-list-view>
106 <gr-overlay id="createOverlay" with-backdrop>
107 <gr-dialog
108 id="createDialog"
109 class="confirmDialog"
110 ?disabled=${!this.hasNewGroupName}
111 confirm-label="Create"
112 confirm-on-enter
113 @confirm=${() => this.handleCreateGroup()}
114 @cancel=${() => this.handleCloseCreate()}
115 >
116 <div class="header" slot="header">Create Group</div>
117 <div class="main" slot="main">
118 <gr-create-group-dialog
119 id="createNewModal"
120 @has-new-group-name=${this.handleHasNewGroupName}
121 ></gr-create-group-dialog>
122 </div>
123 </gr-dialog>
124 </gr-overlay>
125 `;
126 }
127
128 private renderGroupList(group: GroupInfo) {
129 return html`
130 <tr class="table">
131 <td class="name">
132 <a href=${this.computeGroupUrl(group.id)}>${group.name}</a>
133 </td>
134 <td class="description">${group.description}</td>
135 <td class="visibleToAll">
136 ${group.options?.visible_to_all === true ? 'Y' : 'N'}
137 </td>
138 </tr>
139 `;
140 }
141
142 override willUpdate(changedProperties: PropertyValues) {
143 if (changedProperties.has('params')) {
144 this.paramsChanged();
145 }
146 }
147
148 // private but used in test
149 paramsChanged() {
150 this.filter = this.params?.filter ?? '';
151 this.offset = Number(this.params?.offset ?? 0);
152 this.maybeOpenCreateOverlay(this.params);
153
154 return this.getGroups(this.filter, this.groupsPerPage, this.offset);
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100155 }
Paladox nonefea8d952017-05-28 15:48:40 +0000156
Dmitrii Filippov3fd2b102019-11-15 16:16:46 +0100157 /**
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100158 * Opens the create overlay if the route has a hash 'create'
Paladox nonea6629992021-11-14 22:05:24 +0000159 *
160 * private but used in test
Tao Zhou9a076812019-12-17 09:59:28 +0100161 */
Paladox nonea6629992021-11-14 22:05:24 +0000162 maybeOpenCreateOverlay(params?: AppElementAdminParams) {
Tao Zhou7d334b42020-08-31 10:56:33 +0200163 if (params?.openCreateModal) {
Paladox nonea6629992021-11-14 22:05:24 +0000164 assertIsDefined(this.createOverlay, 'createOverlay');
165 this.createOverlay.open();
Dmitrii Filippov3fd2b102019-11-15 16:16:46 +0100166 }
Dmitrii Filippov3fd2b102019-11-15 16:16:46 +0100167 }
168
David Pursehousef7d7df42020-05-21 14:28:11 +0900169 /**
170 * Generates groups link (/admin/groups/<uuid>)
Paladox nonea6629992021-11-14 22:05:24 +0000171 *
172 * private but used in test
David Pursehousef7d7df42020-05-21 14:28:11 +0900173 */
Paladox nonea6629992021-11-14 22:05:24 +0000174 computeGroupUrl(id: string) {
Tao Zhou7d334b42020-08-31 10:56:33 +0200175 return GerritNav.getUrlForGroup(decodeURIComponent(id) as GroupId);
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100176 }
177
Paladox nonea6629992021-11-14 22:05:24 +0000178 private getCreateGroupCapability() {
Ben Rohlfs43935a42020-12-01 19:14:09 +0100179 return this.restApiService.getAccount().then(account => {
Paladox nonea6629992021-11-14 22:05:24 +0000180 if (!account) return;
Ben Rohlfs43935a42020-12-01 19:14:09 +0100181 return this.restApiService
Tao Zhou7d334b42020-08-31 10:56:33 +0200182 .getAccountCapabilities(['createGroup'])
183 .then(capabilities => {
184 if (capabilities?.createGroup) {
Paladox nonea6629992021-11-14 22:05:24 +0000185 this.createNewCapability = true;
Tao Zhou7d334b42020-08-31 10:56:33 +0200186 }
187 });
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100188 });
189 }
190
Paladox nonea6629992021-11-14 22:05:24 +0000191 private getGroups(filter: string, groupsPerPage: number, offset?: number) {
192 this.groups = [];
193 this.loading = true;
Ben Rohlfs43935a42020-12-01 19:14:09 +0100194 return this.restApiService
Tao Zhou7d334b42020-08-31 10:56:33 +0200195 .getGroups(filter, groupsPerPage, offset)
196 .then(groups => {
Paladox nonea6629992021-11-14 22:05:24 +0000197 if (!groups) return;
198 this.groups = Object.keys(groups).map(key => {
Tao Zhou7d334b42020-08-31 10:56:33 +0200199 const group = groups[key];
Dmitrii Filippov55616542020-10-05 19:07:27 +0200200 group.name = key as GroupName;
Tao Zhou7d334b42020-08-31 10:56:33 +0200201 return group;
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100202 });
Paladox nonea6629992021-11-14 22:05:24 +0000203 })
204 .finally(() => {
205 this.loading = false;
Tao Zhou7d334b42020-08-31 10:56:33 +0200206 });
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100207 }
208
Paladox nonea6629992021-11-14 22:05:24 +0000209 private refreshGroupsList() {
Ben Rohlfs43935a42020-12-01 19:14:09 +0100210 this.restApiService.invalidateGroupsCache();
Paladox nonea6629992021-11-14 22:05:24 +0000211 return this.getGroups(this.filter, this.groupsPerPage, this.offset);
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100212 }
213
Paladox nonea6629992021-11-14 22:05:24 +0000214 // private but used in test
215 handleCreateGroup() {
216 assertIsDefined(this.createNewModal, 'createNewModal');
217 this.createNewModal.handleCreateGroup().then(() => {
218 this.refreshGroupsList();
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100219 });
220 }
221
Paladox nonea6629992021-11-14 22:05:24 +0000222 // private but used in test
223 handleCloseCreate() {
224 assertIsDefined(this.createOverlay, 'createOverlay');
225 this.createOverlay.close();
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100226 }
227
Paladox nonea6629992021-11-14 22:05:24 +0000228 // private but used in test
229 handleCreateClicked() {
230 assertIsDefined(this.createOverlay, 'createOverlay');
231 this.createOverlay.open().then(() => {
232 assertIsDefined(this.createNewModal, 'createNewModal');
233 this.createNewModal.focus();
Dhruv Srivastava26320752021-02-02 12:43:27 +0100234 });
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100235 }
236
Paladox nonea6629992021-11-14 22:05:24 +0000237 private handleHasNewGroupName() {
238 assertIsDefined(this.createNewModal, 'createNewModal');
239 this.hasNewGroupName = !!this.createNewModal.name;
Paladox none65d320c2021-11-14 15:20:09 +0000240 }
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100241}