
Due to event retargetting, in shadow DOM, event will be fired both from the first element and the shadowroot host. Bug: Issue 11322 Change-Id: I52fbc0fac9dfdaf945dce3da0a266bfe58b0bd38
346 lines
8.5 KiB
JavaScript
346 lines
8.5 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 DEFAULT_LINKS = [{
|
|
title: 'Changes',
|
|
links: [
|
|
{
|
|
url: '/q/status:open',
|
|
name: 'Open',
|
|
},
|
|
{
|
|
url: '/q/status:merged',
|
|
name: 'Merged',
|
|
},
|
|
{
|
|
url: '/q/status:abandoned',
|
|
name: 'Abandoned',
|
|
},
|
|
],
|
|
}];
|
|
|
|
const DOCUMENTATION_LINKS = [
|
|
{
|
|
url: '/index.html',
|
|
name: 'Table of Contents',
|
|
},
|
|
{
|
|
url: '/user-search.html',
|
|
name: 'Searching',
|
|
},
|
|
{
|
|
url: '/user-upload.html',
|
|
name: 'Uploading',
|
|
},
|
|
{
|
|
url: '/access-control.html',
|
|
name: 'Access Control',
|
|
},
|
|
{
|
|
url: '/rest-api.html',
|
|
name: 'REST API',
|
|
},
|
|
{
|
|
url: '/intro-project-owner.html',
|
|
name: 'Project Owner Guide',
|
|
},
|
|
];
|
|
|
|
// Set of authentication methods that can provide custom registration page.
|
|
const AUTH_TYPES_WITH_REGISTER_URL = new Set([
|
|
'LDAP',
|
|
'LDAP_BIND',
|
|
'CUSTOM_EXTENSION',
|
|
]);
|
|
|
|
Polymer({
|
|
is: 'gr-main-header',
|
|
_legacyUndefinedCheck: true,
|
|
|
|
hostAttributes: {
|
|
role: 'banner',
|
|
},
|
|
|
|
properties: {
|
|
searchQuery: {
|
|
type: String,
|
|
notify: true,
|
|
},
|
|
loggedIn: {
|
|
type: Boolean,
|
|
reflectToAttribute: true,
|
|
},
|
|
loading: {
|
|
type: Boolean,
|
|
reflectToAttribute: true,
|
|
},
|
|
|
|
/** @type {?Object} */
|
|
_account: Object,
|
|
_adminLinks: {
|
|
type: Array,
|
|
value() { return []; },
|
|
},
|
|
_defaultLinks: {
|
|
type: Array,
|
|
value() {
|
|
return DEFAULT_LINKS;
|
|
},
|
|
},
|
|
_docBaseUrl: {
|
|
type: String,
|
|
value: null,
|
|
},
|
|
_links: {
|
|
type: Array,
|
|
computed: '_computeLinks(_defaultLinks, _userLinks, _adminLinks, ' +
|
|
'_topMenus, _docBaseUrl)',
|
|
},
|
|
_loginURL: {
|
|
type: String,
|
|
value: '/login',
|
|
},
|
|
_userLinks: {
|
|
type: Array,
|
|
value() { return []; },
|
|
},
|
|
_topMenus: {
|
|
type: Array,
|
|
value() { return []; },
|
|
},
|
|
_registerText: {
|
|
type: String,
|
|
value: 'Sign up',
|
|
},
|
|
_registerURL: {
|
|
type: String,
|
|
value: null,
|
|
},
|
|
},
|
|
|
|
behaviors: [
|
|
Gerrit.AdminNavBehavior,
|
|
Gerrit.BaseUrlBehavior,
|
|
Gerrit.DocsUrlBehavior,
|
|
Gerrit.FireBehavior,
|
|
],
|
|
|
|
observers: [
|
|
'_accountLoaded(_account)',
|
|
],
|
|
|
|
attached() {
|
|
this._loadAccount();
|
|
this._loadConfig();
|
|
this.listen(window, 'location-change', '_handleLocationChange');
|
|
},
|
|
|
|
detached() {
|
|
this.unlisten(window, 'location-change', '_handleLocationChange');
|
|
},
|
|
|
|
reload() {
|
|
this._loadAccount();
|
|
},
|
|
|
|
_handleLocationChange(e) {
|
|
const baseUrl = this.getBaseUrl();
|
|
if (baseUrl) {
|
|
// Strip the canonical path from the path since needing canonical in
|
|
// the path is uneeded and breaks the url.
|
|
this._loginURL = baseUrl + '/login/' + encodeURIComponent(
|
|
'/' + window.location.pathname.substring(baseUrl.length) +
|
|
window.location.search +
|
|
window.location.hash);
|
|
} else {
|
|
this._loginURL = '/login/' + encodeURIComponent(
|
|
window.location.pathname +
|
|
window.location.search +
|
|
window.location.hash);
|
|
}
|
|
},
|
|
|
|
_computeRelativeURL(path) {
|
|
return '//' + window.location.host + this.getBaseUrl() + path;
|
|
},
|
|
|
|
_computeLinks(defaultLinks, userLinks, adminLinks, topMenus, docBaseUrl) {
|
|
// Polymer 2: check for undefined
|
|
if ([
|
|
defaultLinks,
|
|
userLinks,
|
|
adminLinks,
|
|
topMenus,
|
|
docBaseUrl,
|
|
].some(arg => arg === undefined)) {
|
|
return undefined;
|
|
}
|
|
|
|
const links = defaultLinks.map(menu => {
|
|
return {
|
|
title: menu.title,
|
|
links: menu.links.slice(),
|
|
};
|
|
});
|
|
if (userLinks && userLinks.length > 0) {
|
|
links.push({
|
|
title: 'Your',
|
|
links: userLinks.slice(),
|
|
});
|
|
}
|
|
const docLinks = this._getDocLinks(docBaseUrl, DOCUMENTATION_LINKS);
|
|
if (docLinks.length) {
|
|
links.push({
|
|
title: 'Documentation',
|
|
links: docLinks,
|
|
class: 'hideOnMobile',
|
|
});
|
|
}
|
|
links.push({
|
|
title: 'Browse',
|
|
links: adminLinks.slice(),
|
|
});
|
|
const topMenuLinks = [];
|
|
links.forEach(link => { topMenuLinks[link.title] = link.links; });
|
|
for (const m of topMenus) {
|
|
const items = m.items.map(this._fixCustomMenuItem).filter(link => {
|
|
// Ignore GWT project links
|
|
return !link.url.includes('${projectName}');
|
|
});
|
|
if (m.name in topMenuLinks) {
|
|
items.forEach(link => { topMenuLinks[m.name].push(link); });
|
|
} else {
|
|
links.push({
|
|
title: m.name,
|
|
links: topMenuLinks[m.name] = items,
|
|
});
|
|
}
|
|
}
|
|
return links;
|
|
},
|
|
|
|
_getDocLinks(docBaseUrl, docLinks) {
|
|
if (!docBaseUrl || !docLinks) {
|
|
return [];
|
|
}
|
|
return docLinks.map(link => {
|
|
let url = docBaseUrl;
|
|
if (url && url[url.length - 1] === '/') {
|
|
url = url.substring(0, url.length - 1);
|
|
}
|
|
return {
|
|
url: url + link.url,
|
|
name: link.name,
|
|
target: '_blank',
|
|
};
|
|
});
|
|
},
|
|
|
|
_loadAccount() {
|
|
this.loading = true;
|
|
const promises = [
|
|
this.$.restAPI.getAccount(),
|
|
this.$.restAPI.getTopMenus(),
|
|
Gerrit.awaitPluginsLoaded(),
|
|
];
|
|
|
|
return Promise.all(promises).then(result => {
|
|
const account = result[0];
|
|
this._account = account;
|
|
this.loggedIn = !!account;
|
|
this.loading = false;
|
|
this._topMenus = result[1];
|
|
|
|
return this.getAdminLinks(account,
|
|
this.$.restAPI.getAccountCapabilities.bind(this.$.restAPI),
|
|
this.$.jsAPI.getAdminMenuLinks.bind(this.$.jsAPI))
|
|
.then(res => {
|
|
this._adminLinks = res.links;
|
|
});
|
|
});
|
|
},
|
|
|
|
_loadConfig() {
|
|
this.$.restAPI.getConfig()
|
|
.then(config => {
|
|
this._retrieveRegisterURL(config);
|
|
return this.getDocsBaseUrl(config, this.$.restAPI);
|
|
})
|
|
.then(docBaseUrl => { this._docBaseUrl = docBaseUrl; });
|
|
},
|
|
|
|
_accountLoaded(account) {
|
|
if (!account) { return; }
|
|
|
|
this.$.restAPI.getPreferences().then(prefs => {
|
|
this._userLinks = prefs.my.map(this._fixCustomMenuItem);
|
|
});
|
|
},
|
|
|
|
_retrieveRegisterURL(config) {
|
|
if (AUTH_TYPES_WITH_REGISTER_URL.has(config.auth.auth_type)) {
|
|
this._registerURL = config.auth.register_url;
|
|
if (config.auth.register_text) {
|
|
this._registerText = config.auth.register_text;
|
|
}
|
|
}
|
|
},
|
|
|
|
_computeIsInvisible(registerURL) {
|
|
return registerURL ? '' : 'invisible';
|
|
},
|
|
|
|
_fixCustomMenuItem(linkObj) {
|
|
// Normalize all urls to PolyGerrit style.
|
|
if (linkObj.url.startsWith('#')) {
|
|
linkObj.url = linkObj.url.slice(1);
|
|
}
|
|
|
|
// Delete target property due to complications of
|
|
// https://bugs.chromium.org/p/gerrit/issues/detail?id=5888
|
|
//
|
|
// The server tries to guess whether URL is a view within the UI.
|
|
// If not, it sets target='_blank' on the menu item. The server
|
|
// makes assumptions that work for the GWT UI, but not PolyGerrit,
|
|
// so we'll just disable it altogether for now.
|
|
delete linkObj.target;
|
|
|
|
return linkObj;
|
|
},
|
|
|
|
_generateSettingsLink() {
|
|
return this.getBaseUrl() + '/settings/';
|
|
},
|
|
|
|
_onMobileSearchTap(e) {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
this.fire('mobile-search', null, {bubbles: false});
|
|
},
|
|
|
|
_computeLinkGroupClass(linkGroup) {
|
|
if (linkGroup && linkGroup.class) {
|
|
return linkGroup.class;
|
|
}
|
|
|
|
return '';
|
|
},
|
|
});
|
|
})();
|