blob: 88e8fc20e6c2f92218fe16f134c06ae3dd5e1549 [file] [log] [blame]
/**
* @license
* Copyright 2016 Google LLC
* SPDX-License-Identifier: Apache-2.0
*/
import '@polymer/iron-dropdown/iron-dropdown';
import '../gr-button/gr-button';
import {GrButton} from '../gr-button/gr-button';
import '../gr-cursor-manager/gr-cursor-manager';
import '../gr-tooltip-content/gr-tooltip-content';
import '../../../styles/shared-styles';
import {getBaseUrl} from '../../../utils/url-util';
import {IronDropdownElement} from '@polymer/iron-dropdown/iron-dropdown';
import {GrCursorManager} from '../gr-cursor-manager/gr-cursor-manager';
import {property, customElement, query, state} from 'lit/decorators.js';
import {addShortcut, Key} from '../../../utils/dom-util';
import {css, html, LitElement, nothing, PropertyValues} from 'lit';
import {sharedStyles} from '../../../styles/shared-styles';
import {ifDefined} from 'lit/directives/if-defined.js';
import {fire} from '../../../utils/event-util';
import {ValueChangedEvent} from '../../../types/events';
import {assertIsDefined} from '../../../utils/common-util';
const REL_NOOPENER = 'noopener';
const REL_EXTERNAL = 'external';
declare global {
interface HTMLElementEventMap {
'opened-changed': ValueChangedEvent<boolean>;
}
interface HTMLElementTagNameMap {
'gr-dropdown': GrDropdown;
}
}
export interface DropdownLink {
url?: string;
name?: string;
external?: boolean;
target?: string | null;
download?: boolean;
id?: string;
tooltip?: string;
}
export interface DropdownContent {
text: string;
bold?: boolean;
}
@customElement('gr-dropdown')
export class GrDropdown extends LitElement {
@query('#dropdown')
dropdown?: IronDropdownElement;
@query('#trigger')
trigger?: GrButton;
static override get styles() {
return [
sharedStyles,
css`
:host {
display: inline-block;
}
.dropdown-trigger {
text-decoration: none;
width: 100%;
}
.dropdown-content {
background-color: var(--dropdown-background-color);
box-shadow: var(--elevation-level-2);
min-width: 112px;
max-width: 280px;
}
gr-button {
vertical-align: top;
}
gr-avatar {
height: 2em;
width: 2em;
vertical-align: middle;
}
gr-button[link]:focus {
outline: 5px auto -webkit-focus-ring-color;
}
ul {
list-style: none;
}
.topContent,
li {
border-bottom: 1px solid var(--border-color);
}
li:last-of-type {
border: none;
}
li .itemAction {
cursor: pointer;
display: block;
padding: var(--spacing-m) var(--spacing-l);
}
li .itemAction {
color: var(--gr-dropdown-item-color);
background-color: var(--gr-dropdown-item-background-color);
border: var(--gr-dropdown-item-border);
text-transform: var(--gr-dropdown-item-text-transform);
}
li .itemAction.disabled {
color: var(--deemphasized-text-color);
cursor: default;
}
li .itemAction:link,
li .itemAction:visited {
text-decoration: none;
}
li .itemAction:not(.disabled):hover {
background-color: var(--hover-background-color);
}
li:focus,
li.selected {
background-color: var(--selection-background-color);
outline: none;
}
li:focus .itemAction,
li.selected .itemAction {
background-color: transparent;
}
.topContent {
display: block;
padding: var(--spacing-m) var(--spacing-l);
color: var(--gr-dropdown-item-color);
background-color: var(--gr-dropdown-item-background-color);
border: var(--gr-dropdown-item-border);
text-transform: var(--gr-dropdown-item-text-transform);
}
.bold-text {
font-weight: var(--font-weight-bold);
}
`,
];
}
/**
* Fired when a non-link dropdown item with the given ID is tapped.
*
* @event tap-item-<id>
*/
/**
* Fired when a non-link dropdown item is tapped.
*
* @event tap-item
*/
@property({type: Array})
items?: DropdownLink[];
@property({type: Boolean, attribute: 'down-arrow'})
downArrow = false;
@property({type: Array})
topContent?: DropdownContent[];
@property({type: String, attribute: 'horizontal-align'})
horizontalAlign = 'left';
/**
* Style the dropdown trigger as a link (rather than a button).
*/
@property({type: Boolean})
link = false;
@property({type: Number, attribute: 'vertical-offset'})
verticalOffset = 40;
@state()
private opened = false;
/**
* List the IDs of dropdown buttons to be disabled. (Note this only
* disables buttons and not link entries.)
*/
@property({type: Array})
disabledIds: string[] = [];
/** Called in disconnectedCallback. */
private cleanups: (() => void)[] = [];
// Used within the tests so needs to be non-private.
cursor = new GrCursorManager();
constructor() {
super();
this.cursor.cursorTargetClass = 'selected';
this.cursor.focusOnMove = true;
}
override connectedCallback() {
super.connectedCallback();
this.cleanups.push(addShortcut(this, {key: Key.UP}, () => this.handleUp()));
this.cleanups.push(
addShortcut(this, {key: Key.DOWN}, () => this.handleDown())
);
this.cleanups.push(
addShortcut(this, {key: Key.ENTER}, () => this.handleEnter())
);
this.cleanups.push(
addShortcut(this, {key: Key.SPACE}, () => this.handleEnter())
);
}
override disconnectedCallback() {
this.cursor.unsetCursor();
for (const cleanup of this.cleanups) cleanup();
this.cleanups = [];
super.disconnectedCallback();
}
override willUpdate(changedProperties: PropertyValues) {
if (changedProperties.has('opened')) {
fire(this, 'opened-changed', {value: this.opened});
}
}
override updated(changedProperties: PropertyValues) {
if (changedProperties.has('items')) {
this.resetCursorStops();
}
if (changedProperties.has('opened') && this.opened) {
this.resetCursorStops();
this.cursor.setCursorAtIndex(0);
if (this.cursor.target !== null) this.cursor.target.focus();
}
}
override render() {
return html` <gr-button
?link=${this.link}
class="dropdown-trigger"
id="trigger"
?down-arrow=${this.downArrow}
@click=${this.dropdownTriggerTapHandler}
>
<slot></slot>
</gr-button>
<iron-dropdown
id="dropdown"
.verticalAlign=${'top'}
.verticalOffset=${this.verticalOffset}
allowOutsideScroll
.horizontalAlign=${this.horizontalAlign}
@click=${() => this.close()}
@opened-changed=${(e: CustomEvent) => (this.opened = e.detail.value)}
>
${this.renderDropdownContent()}
</iron-dropdown>`;
}
private renderDropdownContent() {
return html` <div class="dropdown-content" slot="dropdown-content">
<ul>
${this.renderTopContent()}
${(this.items ?? []).map(link => this.renderDropdownLink(link))}
</ul>
</div>`;
}
private renderTopContent() {
if (!this.topContent) return nothing;
return html`
<div class="topContent">
${(this.topContent ?? []).map(item => this.renderTopContentItem(item))}
</div>
`;
}
private renderTopContentItem(item: DropdownContent) {
return html`
<div class="${this.getClassIfBold(item.bold)} top-item" tabindex="-1">
${item.text}
</div>
`;
}
private renderDropdownLink(link: DropdownLink) {
const disabledClass = this.computeDisabledClass(link.id);
return html`
<li tabindex="-1">
<gr-tooltip-content
?has-tooltip=${!!link.tooltip}
title=${ifDefined(link.tooltip)}
>
<span
class="itemAction ${disabledClass}"
data-id=${ifDefined(link.id)}
@click=${this.handleItemTap}
?hidden=${!!link.url}
tabindex="-1"
>${link.name}</span
>
<a
class="itemAction"
href=${this.computeLinkURL(link)}
?download=${!!link.download}
rel=${ifDefined(this.computeLinkRel(link) ?? undefined)}
target=${ifDefined(link.target ?? undefined)}
?hidden=${!link.url}
tabindex="-1"
>${link.name}</a
>
</gr-tooltip-content>
</li>
`;
}
/**
* Handle the up key.
*/
private handleUp() {
assertIsDefined(this.dropdown);
if (this.dropdown.opened) {
this.cursor.previous();
} else {
this.open();
}
}
/**
* Handle the down key.
*/
private handleDown() {
assertIsDefined(this.dropdown);
if (this.dropdown.opened) {
this.cursor.next();
} else {
this.open();
}
}
/**
* Handle the enter key.
*/
private handleEnter() {
assertIsDefined(this.dropdown);
if (this.dropdown.opened) {
// TODO(milutin): This solution is not particularly robust in general.
// Since gr-tooltip-content click on shadow dom is not propagated down,
// we have to target `a` inside it.
if (this.cursor.target !== null) {
const el = this.cursor.target.querySelector(':not([hidden]) a');
if (el) {
(el as HTMLElement).click();
}
}
} else {
this.open();
}
}
/**
* Handle a click on the button to open the dropdown.
*/
private dropdownTriggerTapHandler(e: MouseEvent) {
assertIsDefined(this.dropdown);
e.preventDefault();
e.stopPropagation();
if (this.dropdown.opened) {
this.close();
} else {
this.open();
}
}
/**
* Open the dropdown and initialize the cursor.
* Private but used in tests.
*/
open() {
assertIsDefined(this.dropdown);
this.dropdown.open();
}
// Private but used in tests.
close() {
// async is needed so that that the click event is fired before the
// dropdown closes (This was a bug for touch devices).
setTimeout(() => {
this.dropdown?.close();
}, 1);
}
/**
* Get the class for a top-content item based on the given boolean.
*
* @param bold Whether the item is bold.
* @return The class for the top-content item.
*
* Private but used in tests.
*/
getClassIfBold(bold?: boolean) {
return bold ? 'bold-text' : '';
}
/**
* Build a URL for the given host and path. The base URL will be only added,
* if it is not already included in the path.
*
* @return The scheme-relative URL.
*/
_computeURLHelper(host: string, path: string) {
const base = path.startsWith(getBaseUrl()) ? '' : getBaseUrl();
return '//' + host + base + path;
}
/**
* Build a scheme-relative URL for the current host. Will include the base
* URL if one is present. Note: the URL will be scheme-relative but absolute
* with regard to the host.
*
* @param path The path for the URL.
* @return The scheme-relative URL.
*/
private computeRelativeURL(path: string) {
const host = window.location.host;
return this._computeURLHelper(host, path);
}
/**
* Compute the URL for a link object.
*
* Private but used in tests.
*/
computeLinkURL(link: DropdownLink) {
if (typeof link.url === 'undefined') {
return '';
}
if (link.target || !link.url.startsWith('/')) {
return link.url;
}
return this.computeRelativeURL(link.url);
}
/**
* Compute the value for the rel attribute of an anchor for the given link
* object. If the link has a target value, then the rel must be "noopener"
* for security reasons.
* Private but used in tests.
*/
computeLinkRel(link: DropdownLink) {
// Note: noopener takes precedence over external.
if (link.target) {
return REL_NOOPENER;
}
if (link.external) {
return REL_EXTERNAL;
}
return null;
}
/**
* Handle a click on an item of the dropdown.
*/
private handleItemTap(e: MouseEvent) {
if (e.target === null || !this.items) {
return;
}
const id = (e.target as Element).getAttribute('data-id');
const item = this.items.find(item => item.id === id);
if (id && !this.disabledIds.includes(id)) {
if (item) {
this.dispatchEvent(
new CustomEvent('tap-item', {
detail: item,
bubbles: true,
composed: true,
})
);
}
this.dispatchEvent(new CustomEvent('tap-item-' + id));
}
}
/**
* Recompute the stops for the dropdown item cursor.
*/
private resetCursorStops() {
assertIsDefined(this.dropdown);
if (this.items && this.items.length > 0 && this.dropdown?.opened) {
this.cursor.stops = Array.from(
this.shadowRoot?.querySelectorAll('li') ?? []
);
}
}
/**
* If a dropdown item is shown as a button, get the class for the button.
*
* @return The class for the item button.
*/
private computeDisabledClass(id?: string) {
return id && this.disabledIds.includes(id) ? 'disabled' : '';
}
}