mirror of
https://github.com/EGroupware/egroupware.git
synced 2024-12-30 10:39:18 +01:00
211 lines
5.6 KiB
TypeScript
211 lines
5.6 KiB
TypeScript
/**
|
|
* EGroupware eTemplate2 - Dropdown Button widget
|
|
*
|
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
|
* @package etemplate
|
|
* @subpackage api
|
|
* @link https://www.egroupware.org
|
|
* @author Nathan Gray
|
|
*/
|
|
|
|
|
|
import {SlButtonGroup, SlDropdown} from "@shoelace-style/shoelace";
|
|
import {css, html, LitElement, TemplateResult} from "lit";
|
|
import {Et2WidgetWithSelectMixin} from "../Et2Select/Et2WidgetWithSelectMixin";
|
|
import {SelectOption} from "../Et2Select/FindSelectOptions";
|
|
import shoelace from "../Styles/shoelace";
|
|
import {ifDefined} from "lit/directives/if-defined.js";
|
|
import {property} from "lit/decorators/property.js";
|
|
|
|
/**
|
|
* A split button - a button with a dropdown list
|
|
*
|
|
* There are several parts to the button UI:
|
|
* - Container: This is what is percieved as the dropdown button, the whole package together
|
|
* - Button: The part on the left that can be clicked
|
|
* - Arrow: The button to display the choices
|
|
* - Menu: The list of choices
|
|
*
|
|
* Menu options are passed via the select_options. They are normally the same
|
|
* as for a select box, but the title can also be full HTML if needed.
|
|
*
|
|
*/
|
|
export class Et2DropdownButton extends Et2WidgetWithSelectMixin(LitElement)
|
|
{
|
|
|
|
static get styles()
|
|
{
|
|
return [
|
|
...super.styles,
|
|
shoelace,
|
|
css`
|
|
:host {
|
|
/* Avoid unwanted style overlap from button */
|
|
border: none;
|
|
background-color: none;
|
|
}
|
|
:host, sl-menu {
|
|
/**
|
|
Adapt shoelace color variables to what we want
|
|
Maybe some logical variables from etemplate2.css here?
|
|
*/
|
|
--sl-color-primary-50: rgb(244, 246, 247);
|
|
--sl-color-primary-100: var(--gray-10);
|
|
--sl-color-primary-300: var(--input-border-color);
|
|
--sl-color-primary-400: var(--input-border-color);
|
|
--sl-color-primary-600: var(--primary-background-color);
|
|
--sl-color-primary-700: #505050;
|
|
}
|
|
:host(:active), :host([active]) {
|
|
background-color: initial;
|
|
}
|
|
sl-button-group {
|
|
display: initial;
|
|
}
|
|
#main {
|
|
flex: 1 1 auto;
|
|
}
|
|
|
|
et2-image {
|
|
width: 1em;
|
|
}
|
|
`,
|
|
];
|
|
}
|
|
|
|
static get properties()
|
|
{
|
|
return {
|
|
...super.properties
|
|
};
|
|
}
|
|
|
|
@property()
|
|
placement:string = "bottom-end";
|
|
|
|
// Make sure imports stay
|
|
private _group : SlButtonGroup;
|
|
private _dropdow : SlDropdown;
|
|
|
|
constructor()
|
|
{
|
|
super();
|
|
|
|
// Bind handlers - parent already got click
|
|
this._handleSelect = this._handleSelect.bind(this);
|
|
}
|
|
|
|
connectedCallback()
|
|
{
|
|
super.connectedCallback();
|
|
|
|
// Rebind click to just the main button, not the whole thing
|
|
this.removeEventListener("click", this._handleClick);
|
|
}
|
|
|
|
protected _renderOptions()
|
|
{
|
|
// We have our own render, so we can handle it internally
|
|
}
|
|
|
|
|
|
render() : TemplateResult
|
|
{
|
|
if(this.readonly)
|
|
{
|
|
return html``;
|
|
}
|
|
return html`
|
|
<sl-button-group>
|
|
<sl-button size="${egwIsMobile() ? "large" : "medium"}" id="main" part="main"
|
|
?disabled=${this.disabled}
|
|
@click=${this._handleClick}
|
|
>
|
|
${this.label}
|
|
</sl-button>
|
|
<sl-dropdown placement=${this.placement} hoist part="dropdown">
|
|
<slot name="trigger" slot="trigger">
|
|
<sl-button part="trigger" size="${egwIsMobile() ? "large" : "medium"}" slot="trigger" caret
|
|
?disabled=${this.disabled}></sl-button>
|
|
</slot>
|
|
<sl-menu @sl-select=${this._handleSelect} part="menu">
|
|
${(this.select_options || []).map((option : SelectOption) => this._optionTemplate(option))}
|
|
<slot></slot>
|
|
</sl-menu>
|
|
</sl-dropdown>
|
|
</sl-button-group>
|
|
`;
|
|
}
|
|
|
|
_optionTemplate(option : SelectOption) : TemplateResult
|
|
{
|
|
let icon = option.icon ? html`
|
|
<et2-image slot="prefix" src=${option.icon} icon></et2-image>` : '';
|
|
|
|
return html`
|
|
<sl-menu-item
|
|
value="${option.value}"
|
|
type="${ifDefined(option.checkbox)}checkbox"
|
|
?checked=${option.checked}
|
|
title="${!option.title || this.noLang ? option.title : this.egw().lang(option.title)}"
|
|
>
|
|
${icon}
|
|
${this.noLang ? option.label : this.egw().lang(option.label)}
|
|
</sl-menu-item>`;
|
|
}
|
|
|
|
protected _handleSelect(ev)
|
|
{
|
|
this._value = ev.detail.item.value;
|
|
|
|
// Trigger a change event
|
|
this.dispatchEvent(new Event("change"));
|
|
|
|
// Let it bubble, if anyone else is interested
|
|
}
|
|
|
|
get value() : string
|
|
{
|
|
return this._value;
|
|
}
|
|
|
|
set value(new_value)
|
|
{
|
|
let oldValue = this.value;
|
|
this._value = new_value;
|
|
this.requestUpdate("value", oldValue);
|
|
}
|
|
|
|
get _optionTargetNode() : HTMLElement
|
|
{
|
|
return this.shadowRoot.querySelector("sl-menu");
|
|
}
|
|
|
|
get buttonNode()
|
|
{
|
|
return this.shadowRoot.querySelector("#main");
|
|
}
|
|
|
|
get triggerButtonNode()
|
|
{
|
|
return this.shadowRoot.querySelector("[slot='trigger']");
|
|
}
|
|
|
|
get dropdownNode()
|
|
{
|
|
return this.shadowRoot.querySelector("sl-dropdown");
|
|
}
|
|
|
|
blur()
|
|
{
|
|
this.shadowRoot.querySelector("sl-button-group")?.dispatchEvent(new Event('blur'));
|
|
}
|
|
|
|
focus()
|
|
{
|
|
this.shadowRoot.querySelector("sl-button-group")?.dispatchEvent(new Event('focus'));
|
|
}
|
|
}
|
|
|
|
// @ts-ignore TypeScript is not recognizing that Et2Button is a LitElement
|
|
customElements.define("et2-dropdown-button", Et2DropdownButton); |