/** * @license * Copyright (C) 2015 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 '../../shared/gr-icons/gr-icons'; import '../../shared/gr-rest-api-interface/gr-rest-api-interface'; import '../gr-change-list/gr-change-list'; import '../gr-repo-header/gr-repo-header'; import '../gr-user-header/gr-user-header'; import '../../../styles/shared-styles'; import {GestureEventListeners} from '@polymer/polymer/lib/mixins/gesture-event-listeners'; import {LegacyElementMixin} from '@polymer/polymer/lib/legacy/legacy-element-mixin'; import {PolymerElement} from '@polymer/polymer/polymer-element'; import {htmlTemplate} from './gr-change-list-view_html'; import {page} from '../../../utils/page-wrapper-utils'; import {GerritNav, GerritView} from '../../core/gr-navigation/gr-navigation'; import {customElement, property} from '@polymer/decorators'; import {AppElementParams} from '../../gr-app-types'; import { AccountDetailInfo, AccountId, ChangeId, ChangeInfo, EmailAddress, PatchRange, PreferencesInput, } from '../../../types/common'; import {RestApiService} from '../../../services/services/gr-rest-api/gr-rest-api'; import {ChangeListToggleReviewedDetail} from '../gr-change-list-item/gr-change-list-item'; import {ChangeStarToggleStarDetail} from '../../shared/gr-change-star/gr-change-star'; import {hasOwnProperty} from '../../../utils/common-util'; import {DiffViewMode} from '../../../constants/constants'; const LookupQueryPatterns = { CHANGE_ID: /^\s*i?[0-9a-f]{7,40}\s*$/i, CHANGE_NUM: /^\s*[1-9][0-9]*\s*$/g, COMMIT: /[0-9a-f]{40}/, }; const USER_QUERY_PATTERN = /^owner:\s?("[^"]+"|[^ ]+)$/; const REPO_QUERY_PATTERN = /^project:\s?("[^"]+"|[^ ]+)(\sstatus\s?:(open|"open"))?$/; const LIMIT_OPERATOR_PATTERN = /\blimit:(\d+)/i; export interface ChangeListViewState { changeNum?: ChangeId; patchRange?: PatchRange; selectedFileIndex?: number; showReplyDialog?: boolean; showDownloadDialog?: boolean; diffMode?: DiffViewMode; numFilesShown?: number; scrollTop?: number; query?: string; offset?: number; } export interface GrChangeListView { $: { restAPI: RestApiService & Element; prevArrow: HTMLAnchorElement; nextArrow: HTMLAnchorElement; }; } @customElement('gr-change-list-view') export class GrChangeListView extends GestureEventListeners( LegacyElementMixin(PolymerElement) ) { static get template() { return htmlTemplate; } /** * Fired when the title of the page should change. * * @event title-change */ @property({type: Object, observer: '_paramsChanged'}) params?: AppElementParams; @property({type: Boolean, computed: '_computeLoggedIn(account)'}) _loggedIn?: boolean; @property({type: Object}) account: AccountDetailInfo | null = null; @property({type: Object, notify: true}) viewState: ChangeListViewState = {}; @property({type: Object}) preferences?: PreferencesInput; @property({type: Number}) _changesPerPage?: number; @property({type: String}) _query = ''; @property({type: Number}) _offset?: number; @property({type: Array, observer: '_changesChanged'}) _changes?: ChangeInfo[]; @property({type: Boolean}) _loading = true; @property({type: String}) _userId: AccountId | EmailAddress | null = null; @property({type: String}) _repo: string | null = null; /** @override */ created() { super.created(); this.addEventListener('next-page', () => this._handleNextPage()); this.addEventListener('previous-page', () => this._handlePreviousPage()); } /** @override */ attached() { super.attached(); this._loadPreferences(); } _paramsChanged(value: AppElementParams) { if (value.view !== GerritView.SEARCH) return; this._loading = true; this._query = value.query; const offset = Number(value.offset); this._offset = isNaN(offset) ? 0 : offset; if ( this.viewState.query !== this._query || this.viewState.offset !== this._offset ) { this.set('viewState.selectedChangeIndex', 0); this.set('viewState.query', this._query); this.set('viewState.offset', this._offset); } // NOTE: This method may be called before attachment. Fire title-change // in an async so that attachment to the DOM can take place first. this.async(() => this.dispatchEvent( new CustomEvent('title-change', { detail: {title: this._query}, composed: true, bubbles: true, }) ) ); this.$.restAPI .getPreferences() .then(prefs => { if (!prefs) { throw new Error('getPreferences returned undefined'); } this._changesPerPage = prefs.changes_per_page; return this._getChanges(); }) .then(changes => { changes = changes || []; if (this._query && changes.length === 1) { let query: keyof typeof LookupQueryPatterns; for (query in LookupQueryPatterns) { if ( hasOwnProperty(LookupQueryPatterns, query) && this._query.match(LookupQueryPatterns[query]) ) { // "Back"/"Forward" buttons work correctly only with // opt_redirect options GerritNav.navigateToChange( changes[0], undefined, undefined, undefined, true ); return; } } } this._changes = changes; this._loading = false; }); } _loadPreferences() { return this.$.restAPI.getLoggedIn().then(loggedIn => { if (loggedIn) { this.$.restAPI.getPreferences().then(preferences => { this.preferences = preferences; }); } else { this.preferences = {}; } }); } _getChanges() { return this.$.restAPI.getChanges( this._changesPerPage, this._query, this._offset ); } _limitFor(query: string, defaultLimit: number) { const match = query.match(LIMIT_OPERATOR_PATTERN); if (!match) { return defaultLimit; } return parseInt(match[1], 10); } _computeNavLink( query: string, offset: number | undefined, direction: number, changesPerPage: number ) { offset = offset ?? 0; const limit = this._limitFor(query, changesPerPage); const newOffset = Math.max(0, offset + limit * direction); return GerritNav.getUrlForSearchQuery(query, newOffset); } _computePrevArrowClass(offset?: number) { return offset === 0 ? 'hide' : ''; } _computeNextArrowClass(changes?: ChangeInfo[]) { const more = changes?.length && changes[changes.length - 1]._more_changes; return more ? '' : 'hide'; } _computeNavClass(loading?: boolean) { return loading || !this._changes || !this._changes.length ? 'hide' : ''; } _handleNextPage() { if (this.$.nextArrow.hidden || !this._changesPerPage) return; page.show( this._computeNavLink(this._query, this._offset, 1, this._changesPerPage) ); } _handlePreviousPage() { if (this.$.prevArrow.hidden || !this._changesPerPage) return; page.show( this._computeNavLink(this._query, this._offset, -1, this._changesPerPage) ); } _changesChanged(changes?: ChangeInfo[]) { this._userId = null; this._repo = null; if (!changes || !changes.length) { return; } if (USER_QUERY_PATTERN.test(this._query)) { const owner = changes[0].owner; const userId = owner._account_id ? owner._account_id : owner.email; if (userId) { this._userId = userId; return; } } if (REPO_QUERY_PATTERN.test(this._query)) { this._repo = changes[0].project; } } _computeHeaderClass(id?: string) { return id ? '' : 'hide'; } _computePage(offset?: number, changesPerPage?: number) { if (offset === undefined || changesPerPage === undefined) return; return offset / changesPerPage + 1; } _computeLoggedIn(account?: AccountDetailInfo) { return !!(account && Object.keys(account).length > 0); } _handleToggleStar(e: CustomEvent) { this.$.restAPI.saveChangeStarred(e.detail.change._number, e.detail.starred); } _handleToggleReviewed(e: CustomEvent) { this.$.restAPI.saveChangeReviewed( e.detail.change._number, e.detail.reviewed ); } } declare global { interface HTMLElementTagNameMap { 'gr-change-list-view': GrChangeListView; } }