Convert gr-label-scores to typescript
The change converts the following files to typescript: * elements/change/gr-label-scores/gr-label-scores.ts Change-Id: I3889a0eae05b22f0a718187c1ef5b6b41835d791
This commit is contained in:
@@ -14,70 +14,110 @@
|
|||||||
* See the License for the specific language governing permissions and
|
* See the License for the specific language governing permissions and
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
import '../../shared/gr-rest-api-interface/gr-rest-api-interface.js';
|
import '../../shared/gr-rest-api-interface/gr-rest-api-interface';
|
||||||
import '../gr-label-score-row/gr-label-score-row.js';
|
import '../gr-label-score-row/gr-label-score-row';
|
||||||
import '../../../styles/shared-styles.js';
|
import '../../../styles/shared-styles';
|
||||||
import {GestureEventListeners} from '@polymer/polymer/lib/mixins/gesture-event-listeners.js';
|
import {GestureEventListeners} from '@polymer/polymer/lib/mixins/gesture-event-listeners';
|
||||||
import {LegacyElementMixin} from '@polymer/polymer/lib/legacy/legacy-element-mixin.js';
|
import {LegacyElementMixin} from '@polymer/polymer/lib/legacy/legacy-element-mixin';
|
||||||
import {PolymerElement} from '@polymer/polymer/polymer-element.js';
|
import {PolymerElement} from '@polymer/polymer/polymer-element';
|
||||||
import {htmlTemplate} from './gr-label-scores_html.js';
|
import {htmlTemplate} from './gr-label-scores_html';
|
||||||
|
import {customElement, property} from '@polymer/decorators';
|
||||||
|
import {hasOwnProperty} from '../../../utils/common-util';
|
||||||
|
import {
|
||||||
|
LabelNameToValueMap,
|
||||||
|
ChangeInfo,
|
||||||
|
AccountInfo,
|
||||||
|
DetailedLabelInfo,
|
||||||
|
LabelNameToInfoMap,
|
||||||
|
} from '../../../types/common';
|
||||||
|
import {
|
||||||
|
GrLabelScoreRow,
|
||||||
|
LabelValuesMap,
|
||||||
|
} from '../gr-label-score-row/gr-label-score-row';
|
||||||
|
import {PolymerDeepPropertyChange} from '@polymer/polymer/interfaces';
|
||||||
|
|
||||||
/** @extends PolymerElement */
|
type Labels = {[label: string]: number};
|
||||||
|
@customElement('gr-label-scores')
|
||||||
class GrLabelScores extends GestureEventListeners(
|
class GrLabelScores extends GestureEventListeners(
|
||||||
LegacyElementMixin(
|
LegacyElementMixin(PolymerElement)
|
||||||
PolymerElement)) {
|
) {
|
||||||
static get template() { return htmlTemplate; }
|
static get template() {
|
||||||
|
return htmlTemplate;
|
||||||
static get is() { return 'gr-label-scores'; }
|
|
||||||
|
|
||||||
static get properties() {
|
|
||||||
return {
|
|
||||||
_labels: {
|
|
||||||
type: Array,
|
|
||||||
computed: '_computeLabels(change.labels.*, account)',
|
|
||||||
},
|
|
||||||
permittedLabels: {
|
|
||||||
type: Object,
|
|
||||||
observer: '_computeColumns',
|
|
||||||
},
|
|
||||||
/** @type {?} */
|
|
||||||
change: Object,
|
|
||||||
/** @type {?} */
|
|
||||||
account: Object,
|
|
||||||
|
|
||||||
_labelValues: Object,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
getLabelValues() {
|
@property({type: Array, computed: '_computeLabels(change.labels.*, account)'})
|
||||||
const labels = {};
|
_labels?: Labels;
|
||||||
for (const label in this.permittedLabels) {
|
|
||||||
if (!this.permittedLabels.hasOwnProperty(label)) { continue; }
|
|
||||||
|
|
||||||
const selectorEl = this.shadowRoot
|
@property({type: Object, observer: '_computeColumns'})
|
||||||
.querySelector(`gr-label-score-row[name="${label}"]`);
|
permittedLabels?: LabelNameToValueMap;
|
||||||
if (!selectorEl) { continue; }
|
|
||||||
|
@property({type: Object})
|
||||||
|
change?: ChangeInfo;
|
||||||
|
|
||||||
|
@property({type: Object})
|
||||||
|
account?: AccountInfo;
|
||||||
|
|
||||||
|
@property({type: Object})
|
||||||
|
_labelValues?: LabelValuesMap;
|
||||||
|
|
||||||
|
getLabelValues() {
|
||||||
|
const labels: Labels = {};
|
||||||
|
if (this.shadowRoot === null || !this.change) {
|
||||||
|
return labels;
|
||||||
|
}
|
||||||
|
for (const label in this.permittedLabels) {
|
||||||
|
if (!hasOwnProperty(this.permittedLabels, label)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const selectorEl = this.shadowRoot.querySelector(
|
||||||
|
`gr-label-score-row[name="${label}"]`
|
||||||
|
) as null | GrLabelScoreRow;
|
||||||
|
if (!selectorEl) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// The user may have not voted on this label.
|
// The user may have not voted on this label.
|
||||||
if (!selectorEl.selectedItem) { continue; }
|
if (!selectorEl.selectedItem) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
const selectedVal = parseInt(selectorEl.selectedValue, 10);
|
const selectedVal =
|
||||||
|
typeof selectorEl.selectedValue === 'string'
|
||||||
|
? parseInt(selectorEl.selectedValue, 10)
|
||||||
|
: selectorEl.selectedValue;
|
||||||
|
|
||||||
|
if (selectedVal === undefined) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// Only send the selection if the user changed it.
|
// Only send the selection if the user changed it.
|
||||||
let prevVal = this._getVoteForAccount(this.change.labels, label,
|
const prevVal = this._getVoteForAccount(
|
||||||
this.account);
|
this.change.labels,
|
||||||
if (prevVal !== null) {
|
label,
|
||||||
prevVal = parseInt(prevVal, 10);
|
this.account
|
||||||
|
);
|
||||||
|
|
||||||
|
let prevValNum: number | null | undefined;
|
||||||
|
if (typeof prevVal === 'string') {
|
||||||
|
prevValNum = parseInt(prevVal, 10);
|
||||||
|
} else {
|
||||||
|
prevValNum = prevVal;
|
||||||
}
|
}
|
||||||
if (selectedVal !== prevVal) {
|
|
||||||
|
if (selectedVal !== prevValNum) {
|
||||||
labels[label] = selectedVal;
|
labels[label] = selectedVal;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return labels;
|
return labels;
|
||||||
}
|
}
|
||||||
|
|
||||||
_getStringLabelValue(labels, labelName, numberValue) {
|
_getStringLabelValue(
|
||||||
for (const k in labels[labelName].values) {
|
labels: LabelNameToInfoMap,
|
||||||
|
labelName: string,
|
||||||
|
numberValue?: number
|
||||||
|
) {
|
||||||
|
for (const k in (labels[labelName] as DetailedLabelInfo).values) {
|
||||||
if (parseInt(k, 10) === numberValue) {
|
if (parseInt(k, 10) === numberValue) {
|
||||||
return k;
|
return k;
|
||||||
}
|
}
|
||||||
@@ -85,70 +125,95 @@ class GrLabelScores extends GestureEventListeners(
|
|||||||
return numberValue;
|
return numberValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
_getVoteForAccount(labels, labelName, account) {
|
_getVoteForAccount(
|
||||||
const votes = labels[labelName];
|
labels: LabelNameToInfoMap | undefined,
|
||||||
|
labelName: string,
|
||||||
|
account?: AccountInfo
|
||||||
|
) {
|
||||||
|
if (!labels) return null;
|
||||||
|
const votes = labels[labelName] as DetailedLabelInfo;
|
||||||
if (votes.all && votes.all.length > 0) {
|
if (votes.all && votes.all.length > 0) {
|
||||||
for (let i = 0; i < votes.all.length; i++) {
|
for (let i = 0; i < votes.all.length; i++) {
|
||||||
if (votes.all[i]._account_id == account._account_id) {
|
// TODO(TS): Replace == with === and check code can assign string to _account_id instead of number
|
||||||
|
// eslint-disable-next-line eqeqeq
|
||||||
|
if (account && votes.all[i]._account_id == account._account_id) {
|
||||||
return this._getStringLabelValue(
|
return this._getStringLabelValue(
|
||||||
labels, labelName, votes.all[i].value);
|
labels,
|
||||||
|
labelName,
|
||||||
|
votes.all[i].value
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
_computeLabels(labelRecord, account) {
|
_computeLabels(
|
||||||
|
labelRecord: PolymerDeepPropertyChange<
|
||||||
|
LabelNameToInfoMap,
|
||||||
|
LabelNameToInfoMap
|
||||||
|
>,
|
||||||
|
account?: AccountInfo
|
||||||
|
) {
|
||||||
// Polymer 2: check for undefined
|
// Polymer 2: check for undefined
|
||||||
if ([labelRecord, account].includes(undefined)) {
|
if ([labelRecord, account].includes(undefined)) {
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
const labelsObj = labelRecord.base;
|
const labelsObj = labelRecord.base;
|
||||||
if (!labelsObj) { return []; }
|
if (!labelsObj) {
|
||||||
return Object.keys(labelsObj).sort()
|
return [];
|
||||||
.map(key => {
|
}
|
||||||
return {
|
return Object.keys(labelsObj)
|
||||||
name: key,
|
.sort()
|
||||||
value: this._getVoteForAccount(labelsObj, key, this.account),
|
.map(key => {
|
||||||
};
|
return {
|
||||||
});
|
name: key,
|
||||||
|
value: this._getVoteForAccount(labelsObj, key, this.account),
|
||||||
|
};
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
_computeColumns(permittedLabels) {
|
_computeColumns(permittedLabels?: LabelNameToValueMap) {
|
||||||
|
if (!permittedLabels) return;
|
||||||
const labels = Object.keys(permittedLabels);
|
const labels = Object.keys(permittedLabels);
|
||||||
const values = {};
|
const values: Set<number> = new Set();
|
||||||
for (const label of labels) {
|
for (const label of labels) {
|
||||||
for (const value of permittedLabels[label]) {
|
for (const value of permittedLabels[label]) {
|
||||||
values[parseInt(value, 10)] = true;
|
values.add(parseInt(value, 10));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const orderedValues = Object.keys(values).sort((a, b) => a - b);
|
const orderedValues = Array.from(values.values()).sort((a, b) => a - b);
|
||||||
|
|
||||||
|
const labelValues: LabelValuesMap = {};
|
||||||
for (let i = 0; i < orderedValues.length; i++) {
|
for (let i = 0; i < orderedValues.length; i++) {
|
||||||
values[orderedValues[i]] = i;
|
labelValues[orderedValues[i]] = i;
|
||||||
}
|
}
|
||||||
this._labelValues = values;
|
this._labelValues = labelValues;
|
||||||
}
|
}
|
||||||
|
|
||||||
_changeIsMerged(changeStatus) {
|
_changeIsMerged(changeStatus: string) {
|
||||||
return changeStatus === 'MERGED';
|
return changeStatus === 'MERGED';
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
_computeLabelAccessClass(
|
||||||
* @param {string|undefined} label
|
label?: string,
|
||||||
* @param {Object|undefined} permittedLabels
|
permittedLabels?: LabelNameToValueMap
|
||||||
* @return {string}
|
) {
|
||||||
*/
|
if (!permittedLabels || !label) {
|
||||||
_computeLabelAccessClass(label, permittedLabels) {
|
|
||||||
if (label == null || permittedLabels == null) {
|
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
|
||||||
return permittedLabels.hasOwnProperty(label) &&
|
return hasOwnProperty(permittedLabels, label) &&
|
||||||
permittedLabels[label].length ? 'access' : 'no-access';
|
permittedLabels[label].length
|
||||||
|
? 'access'
|
||||||
|
: 'no-access';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define(GrLabelScores.is, GrLabelScores);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
'gr-label-scores': GrLabelScores;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -206,7 +206,7 @@ export interface ChangeInfo {
|
|||||||
owner: AccountInfo;
|
owner: AccountInfo;
|
||||||
actions?: ActionInfo[];
|
actions?: ActionInfo[];
|
||||||
requirements?: Requirement[];
|
requirements?: Requirement[];
|
||||||
labels?: LabelInfo[];
|
labels?: LabelNameToInfoMap;
|
||||||
permitted_labels?: LabelNameToValueMap;
|
permitted_labels?: LabelNameToValueMap;
|
||||||
removable_reviewers?: AccountInfo[];
|
removable_reviewers?: AccountInfo[];
|
||||||
reviewers?: AccountInfo[];
|
reviewers?: AccountInfo[];
|
||||||
@@ -460,7 +460,7 @@ export interface AttentionSetInfo {
|
|||||||
* https://gerrit-review.googlesource.com/Documentation/rest-api-changes.html#approval-info
|
* https://gerrit-review.googlesource.com/Documentation/rest-api-changes.html#approval-info
|
||||||
*/
|
*/
|
||||||
export interface ApprovalInfo extends AccountInfo {
|
export interface ApprovalInfo extends AccountInfo {
|
||||||
value?: string;
|
value?: number;
|
||||||
permitted_voting_range?: VotingRangeInfo;
|
permitted_voting_range?: VotingRangeInfo;
|
||||||
date?: Timestamp;
|
date?: Timestamp;
|
||||||
tag?: ReviewInputTag;
|
tag?: ReviewInputTag;
|
||||||
|
|||||||
Reference in New Issue
Block a user