
These tags are preserved by the Closure compiler and vulcanize in order to serve the license notices embedded in the outputs. In a standalone Gerrit server, these license are also covered in the LICENSES.txt served with the documentation. When serving PG assets from a CDN, it's less obvious what the corresponding LICENSES.txt file is, since the CDN is not directly linked to a running Gerrit server. Safer to embed the licenses in the assets themselves. Change-Id: Id1add1451fad1baa7916882a6bda02c326ccc988
227 lines
6.8 KiB
JavaScript
227 lines
6.8 KiB
JavaScript
/**
|
|
* @license
|
|
* Copyright (C) 2016 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.
|
|
*/
|
|
(function() {
|
|
'use strict';
|
|
|
|
const PROJECT_PLACEHOLDER_PATTERN = /\$\{project\}/g;
|
|
const USER_PLACEHOLDER_PATTERN = /\$\{user\}/g;
|
|
|
|
// NOTE: These queries are tested in Java. Any changes made to definitions
|
|
// here require corresponding changes to:
|
|
// gerrit-server/src/test/java/com/google/gerrit/server/query/change/AbstractQueryChangesTest.java
|
|
const DEFAULT_SECTIONS = [
|
|
{
|
|
// WIP open changes owned by viewing user. This section is omitted when
|
|
// viewing other users, so we don't need to filter anything out.
|
|
name: 'Work in progress',
|
|
query: 'is:open owner:${user} is:wip',
|
|
selfOnly: true,
|
|
},
|
|
{
|
|
// Non-WIP open changes owned by viewed user. Filter out changes ignored
|
|
// by the viewing user.
|
|
name: 'Outgoing reviews',
|
|
query: 'is:open owner:${user} -is:wip -is:ignored',
|
|
},
|
|
{
|
|
// Non-WIP open changes not owned by the viewed user, that the viewed user
|
|
// is associated with (as either a reviewer or the assignee). Changes
|
|
// ignored by the viewing user are filtered out.
|
|
name: 'Incoming reviews',
|
|
query: 'is:open -owner:${user} -is:wip -is:ignored ' +
|
|
'(reviewer:${user} OR assignee:${user})',
|
|
},
|
|
{
|
|
// Open changes the viewed user is CCed on. Changes ignored by the viewing
|
|
// user are filtered out.
|
|
name: 'CCed on',
|
|
query: 'is:open -is:ignored cc:${user}',
|
|
},
|
|
{
|
|
name: 'Recently closed',
|
|
// Closed changes where viewed user is owner, reviewer, or assignee.
|
|
// Changes ignored by the viewing user are filtered out, and so are WIP
|
|
// changes not owned by the viewing user (the one instance of
|
|
// 'owner:self' is intentional and implements this logic).
|
|
query: 'is:closed -is:ignored (-is:wip OR owner:self) ' +
|
|
'(owner:${user} OR reviewer:${user} OR assignee:${user})',
|
|
suffixForDashboard: '-age:4w limit:10',
|
|
},
|
|
];
|
|
|
|
Polymer({
|
|
is: 'gr-dashboard-view',
|
|
|
|
/**
|
|
* Fired when the title of the page should change.
|
|
*
|
|
* @event title-change
|
|
*/
|
|
|
|
properties: {
|
|
account: {
|
|
type: Object,
|
|
value: null,
|
|
},
|
|
/** @type {{ selectedChangeIndex: number }} */
|
|
viewState: Object,
|
|
|
|
/** @type {{ user: string }} */
|
|
params: {
|
|
type: Object,
|
|
},
|
|
|
|
_results: Array,
|
|
_sectionMetadata: {
|
|
type: Array,
|
|
value() { return DEFAULT_SECTIONS; },
|
|
},
|
|
|
|
/**
|
|
* For showing a "loading..." string during ajax requests.
|
|
*/
|
|
_loading: {
|
|
type: Boolean,
|
|
value: true,
|
|
},
|
|
},
|
|
|
|
observers: [
|
|
'_paramsChanged(params.*)',
|
|
],
|
|
|
|
behaviors: [
|
|
Gerrit.RESTClientBehavior,
|
|
],
|
|
|
|
get options() {
|
|
return this.listChangesOptionsToHex(
|
|
this.ListChangesOption.LABELS,
|
|
this.ListChangesOption.DETAILED_ACCOUNTS,
|
|
this.ListChangesOption.REVIEWED
|
|
);
|
|
},
|
|
|
|
_getProjectDashboard(project, dashboard) {
|
|
const errFn = response => {
|
|
this.fire('page-error', {response});
|
|
};
|
|
return this.$.restAPI.getDashboard(
|
|
project, dashboard, errFn).then(response => {
|
|
if (!response) {
|
|
return;
|
|
}
|
|
return {
|
|
title: response.title,
|
|
sections: response.sections.map(section => {
|
|
const suffix = response.foreach ? ' ' + response.foreach : '';
|
|
return {
|
|
name: section.name,
|
|
query:
|
|
section.query.replace(
|
|
PROJECT_PLACEHOLDER_PATTERN, project) + suffix,
|
|
};
|
|
}),
|
|
};
|
|
});
|
|
},
|
|
|
|
_getUserDashboard(user, sections, title) {
|
|
sections = sections
|
|
.filter(section => (user === 'self' || !section.selfOnly))
|
|
.map(section => {
|
|
const dashboardSection = {
|
|
name: section.name,
|
|
query: section.query.replace(USER_PLACEHOLDER_PATTERN, user),
|
|
};
|
|
if (section.suffixForDashboard) {
|
|
dashboardSection.suffixForDashboard = section.suffixForDashboard;
|
|
}
|
|
return dashboardSection;
|
|
});
|
|
return Promise.resolve({title, sections});
|
|
},
|
|
|
|
_computeTitle(user) {
|
|
if (!user || user === 'self') {
|
|
return 'My Reviews';
|
|
}
|
|
return 'Dashboard for ' + user;
|
|
},
|
|
|
|
_isViewActive(params) {
|
|
return params.view === Gerrit.Nav.View.DASHBOARD;
|
|
},
|
|
|
|
_paramsChanged(paramsChangeRecord) {
|
|
const params = paramsChangeRecord.base;
|
|
|
|
if (!this._isViewActive(params)) {
|
|
return Promise.resolve();
|
|
}
|
|
|
|
const user = params.user || 'self';
|
|
|
|
// NOTE: This method may be called before attachment. Fire title-change
|
|
// in an async so that attachment to the DOM can take place first.
|
|
const title = params.title || this._computeTitle(user);
|
|
this.async(() => this.fire('title-change', {title}));
|
|
|
|
this._loading = true;
|
|
|
|
const dashboardPromise = params.project ?
|
|
this._getProjectDashboard(params.project, params.dashboard) :
|
|
this._getUserDashboard(
|
|
params.user || 'self',
|
|
params.sections || DEFAULT_SECTIONS,
|
|
params.title || this._computeTitle(params.user));
|
|
|
|
return dashboardPromise.then(dashboard => {
|
|
if (!dashboard) {
|
|
this._loading = false;
|
|
return;
|
|
}
|
|
const queries = dashboard.sections.map(section => {
|
|
if (section.suffixForDashboard) {
|
|
return section.query + ' ' + section.suffixForDashboard;
|
|
}
|
|
return section.query;
|
|
});
|
|
const req =
|
|
this.$.restAPI.getChanges(null, queries, null, this.options);
|
|
return req.then(response => {
|
|
this._loading = false;
|
|
this._results = response.map((results, i) => {
|
|
return {
|
|
sectionName: dashboard.sections[i].name,
|
|
query: dashboard.sections[i].query,
|
|
results,
|
|
};
|
|
});
|
|
});
|
|
}).catch(err => {
|
|
this._loading = false;
|
|
console.warn(err);
|
|
});
|
|
},
|
|
|
|
_computeUserHeaderClass(userParam) {
|
|
return userParam === 'self' ? 'hide' : '';
|
|
},
|
|
});
|
|
})();
|