
In case the 'g' keyup event is lost, disregard 'g' keydowns that are at least 1 second old. Also filter 'g' keydown handler to when no modifier is pressed. Bug: Issue 7734 Change-Id: I960926a3b8d42938b23e358dd7c0e707562e92cb
287 lines
7.8 KiB
JavaScript
287 lines
7.8 KiB
JavaScript
// 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';
|
|
|
|
// The maximum age of a keydown event to be used in a jump navigation. This is
|
|
// only for cases when the keyup event is lost.
|
|
const G_KEY_TIMEOUT_MS = 1000;
|
|
|
|
// Eagerly render Polymer components when backgrounded. (Skips
|
|
// requestAnimationFrame.)
|
|
// @see https://github.com/Polymer/polymer/issues/3851
|
|
// TODO: Reassess after Polymer 2.0 upgrade.
|
|
// @see Issue 4699
|
|
Polymer.RenderStatus._makeReady();
|
|
|
|
Polymer({
|
|
is: 'gr-app',
|
|
|
|
/**
|
|
* Fired when the URL location changes.
|
|
*
|
|
* @event location-change
|
|
*/
|
|
|
|
properties: {
|
|
/**
|
|
* @type {{ query: string, view: string }}
|
|
*/
|
|
params: Object,
|
|
keyEventTarget: {
|
|
type: Object,
|
|
value() { return document.body; },
|
|
},
|
|
|
|
_account: {
|
|
type: Object,
|
|
observer: '_accountChanged',
|
|
},
|
|
|
|
/**
|
|
* The last time the g key was pressed in milliseconds (or a keydown event
|
|
* was handled if the key is held down).
|
|
* @type {number|null}
|
|
*/
|
|
_lastGKeyPressTimestamp: {
|
|
type: Number,
|
|
value: null,
|
|
},
|
|
|
|
/**
|
|
* @type {{ plugin: Object }}
|
|
*/
|
|
_serverConfig: Object,
|
|
_version: String,
|
|
_showChangeListView: Boolean,
|
|
_showDashboardView: Boolean,
|
|
_showChangeView: Boolean,
|
|
_showDiffView: Boolean,
|
|
_showSettingsView: Boolean,
|
|
_showAdminView: Boolean,
|
|
_showCLAView: Boolean,
|
|
_showEditorView: Boolean,
|
|
/** @type {?} */
|
|
_viewState: Object,
|
|
/** @type {?} */
|
|
_lastError: Object,
|
|
_lastSearchPage: String,
|
|
_path: String,
|
|
_isShadowDom: Boolean,
|
|
},
|
|
|
|
listeners: {
|
|
'page-error': '_handlePageError',
|
|
'title-change': '_handleTitleChange',
|
|
'location-change': '_handleLocationChange',
|
|
},
|
|
|
|
observers: [
|
|
'_viewChanged(params.view)',
|
|
],
|
|
|
|
behaviors: [
|
|
Gerrit.BaseUrlBehavior,
|
|
Gerrit.KeyboardShortcutBehavior,
|
|
],
|
|
|
|
keyBindings: {
|
|
'?': '_showKeyboardShortcuts',
|
|
'g:keydown': '_gKeyDown',
|
|
'g:keyup': '_gKeyUp',
|
|
'a m o': '_jumpKeyPressed',
|
|
},
|
|
|
|
ready() {
|
|
this._isShadowDom = Polymer.Settings.useShadow;
|
|
this.$.router.start();
|
|
|
|
this.$.restAPI.getAccount().then(account => {
|
|
this._account = account;
|
|
});
|
|
this.$.restAPI.getConfig().then(config => {
|
|
this._serverConfig = config;
|
|
});
|
|
this.$.restAPI.getVersion().then(version => {
|
|
this._version = version;
|
|
});
|
|
|
|
this.$.reporting.appStarted();
|
|
this._viewState = {
|
|
changeView: {
|
|
changeNum: null,
|
|
patchRange: null,
|
|
selectedFileIndex: 0,
|
|
showReplyDialog: false,
|
|
diffMode: null,
|
|
numFilesShown: null,
|
|
scrollTop: 0,
|
|
},
|
|
changeListView: {
|
|
query: null,
|
|
offset: 0,
|
|
selectedChangeIndex: 0,
|
|
},
|
|
dashboardView: {
|
|
selectedChangeIndex: 0,
|
|
},
|
|
};
|
|
},
|
|
|
|
_accountChanged(account) {
|
|
if (!account) { return; }
|
|
|
|
// Preferences are cached when a user is logged in; warm them.
|
|
this.$.restAPI.getPreferences();
|
|
this.$.restAPI.getDiffPreferences();
|
|
this.$.errorManager.knownAccountId =
|
|
this._account && this._account._account_id || null;
|
|
},
|
|
|
|
_viewChanged(view) {
|
|
this.$.errorView.classList.remove('show');
|
|
this.set('_showChangeListView', view === Gerrit.Nav.View.SEARCH);
|
|
this.set('_showDashboardView', view === Gerrit.Nav.View.DASHBOARD);
|
|
this.set('_showChangeView', view === Gerrit.Nav.View.CHANGE);
|
|
this.set('_showDiffView', view === Gerrit.Nav.View.DIFF);
|
|
this.set('_showSettingsView', view === Gerrit.Nav.View.SETTINGS);
|
|
this.set('_showAdminView', view === Gerrit.Nav.View.ADMIN ||
|
|
view === Gerrit.Nav.View.GROUP);
|
|
this.set('_showCLAView', view === Gerrit.Nav.View.AGREEMENTS);
|
|
this.set('_showEditorView', view === Gerrit.Nav.View.EDIT);
|
|
if (this.params.justRegistered) {
|
|
this.$.registration.open();
|
|
}
|
|
this.$.header.unfloat();
|
|
},
|
|
|
|
// Argument used for binding update only.
|
|
_computeLoggedIn(account) {
|
|
return !!(account && Object.keys(account).length > 0);
|
|
},
|
|
|
|
_computeShowGwtUiLink(config) {
|
|
return config.gerrit.web_uis && config.gerrit.web_uis.includes('GWT');
|
|
},
|
|
|
|
_handlePageError(e) {
|
|
const props = [
|
|
'_showChangeListView',
|
|
'_showDashboardView',
|
|
'_showChangeView',
|
|
'_showDiffView',
|
|
'_showSettingsView',
|
|
];
|
|
for (const showProp of props) {
|
|
this.set(showProp, false);
|
|
}
|
|
|
|
this.$.errorView.classList.add('show');
|
|
const response = e.detail.response;
|
|
const err = {text: [response.status, response.statusText].join(' ')};
|
|
if (response.status === 404) {
|
|
err.emoji = '¯\\_(ツ)_/¯';
|
|
this._lastError = err;
|
|
} else {
|
|
err.emoji = 'o_O';
|
|
response.text().then(text => {
|
|
err.moreInfo = text;
|
|
this._lastError = err;
|
|
});
|
|
}
|
|
},
|
|
|
|
_handleLocationChange(e) {
|
|
const hash = e.detail.hash.substring(1);
|
|
let pathname = e.detail.pathname;
|
|
if (pathname.startsWith('/c/') && parseInt(hash, 10) > 0) {
|
|
pathname += '@' + hash;
|
|
}
|
|
this.set('_path', pathname);
|
|
this._handleSearchPageChange();
|
|
},
|
|
|
|
_handleSearchPageChange() {
|
|
if (!this.params) {
|
|
return;
|
|
}
|
|
const viewsToCheck = [Gerrit.Nav.View.SEARCH, Gerrit.Nav.View.DASHBOARD];
|
|
if (viewsToCheck.includes(this.params.view)) {
|
|
this.set('_lastSearchPage', location.pathname);
|
|
}
|
|
},
|
|
|
|
_handleTitleChange(e) {
|
|
if (e.detail.title) {
|
|
document.title = e.detail.title + ' · Gerrit Code Review';
|
|
} else {
|
|
document.title = '';
|
|
}
|
|
},
|
|
|
|
_showKeyboardShortcuts(e) {
|
|
if (this.shouldSuppressKeyboardShortcut(e)) { return; }
|
|
this.$.keyboardShortcuts.open();
|
|
},
|
|
|
|
_handleKeyboardShortcutDialogClose() {
|
|
this.$.keyboardShortcuts.close();
|
|
},
|
|
|
|
_handleAccountDetailUpdate(e) {
|
|
this.$.mainHeader.reload();
|
|
if (this.params.view === Gerrit.Nav.View.SETTINGS) {
|
|
this.$$('gr-settings-view').reloadAccountDetail();
|
|
}
|
|
},
|
|
|
|
_handleRegistrationDialogClose(e) {
|
|
this.params.justRegistered = false;
|
|
this.$.registration.close();
|
|
},
|
|
|
|
_computeShadowClass(isShadowDom) {
|
|
return isShadowDom ? 'shadow' : '';
|
|
},
|
|
|
|
_gKeyDown(e) {
|
|
if (this.modifierPressed(e)) { return; }
|
|
this._lastGKeyPressTimestamp = Date.now();
|
|
},
|
|
|
|
_gKeyUp() {
|
|
this._lastGKeyPressTimestamp = null;
|
|
},
|
|
|
|
_jumpKeyPressed(e) {
|
|
if (!this._lastGKeyPressTimestamp ||
|
|
(Date.now() - this._lastGKeyPressTimestamp > G_KEY_TIMEOUT_MS) ||
|
|
this.shouldSuppressKeyboardShortcut(e)) { return; }
|
|
e.preventDefault();
|
|
|
|
let status = null;
|
|
if (e.detail.key === 'a') {
|
|
status = 'abandoned';
|
|
} else if (e.detail.key === 'm') {
|
|
status = 'merged';
|
|
} else if (e.detail.key === 'o') {
|
|
status = 'open';
|
|
}
|
|
if (status !== null) {
|
|
Gerrit.Nav.navigateToStatusSearch(status);
|
|
}
|
|
},
|
|
});
|
|
})();
|