/** * EGroupware eTemplate2 - Details WebComponent * * @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License * @package api * @link https://www.egroupware.org * @author Hadi Nategh */ import {Et2Widget} from "../../Et2Widget/Et2Widget"; import {css, html} from "lit"; import {SlDetails} from "@shoelace-style/shoelace"; import shoelace from "../../Styles/shoelace"; import {property} from "lit/decorators/property.js"; import {customElement} from "lit/decorators/custom-element.js"; import {classMap} from "lit/directives/class-map.js"; @customElement("et2-details") export class Et2Details extends Et2Widget(SlDetails) { static get styles() { return [ ...super.styles, shoelace, css` :host { display: block; } :host([align="right"]) > div { justify-content: flex-end; } :host([align="left"]) > div { justify-content: flex-start; } /* CSS for child elements */ ::slotted(*) { flex: 1 1 auto; } ::slotted(img), ::slotted(et2-image) { /* Stop images from growing. In general we want them to stay */ flex-grow: 0; } ::slotted([align="left"]) { margin-right: auto; order: -1; } ::slotted([align="right"]) { margin-left: auto; order: 1; } .details { border: var(--sl-panel-border-width) solid var(--sl-panel-border-color); margin: 0px; overflow: hidden; } .details.hoist { position: relative; overflow: visible; } .details__body { display: none; } .details--open .details__body { display: block; } .details:not(.hoist).details--open.details--overlay-summary { .details__summary { visibility: hidden; } .details__body { margin-top: calc(-1 * var(--sl-input-height-medium)); } .details__body.overlaySummaryRightAligned { padding-right: calc(3 * var(--sl-spacing-medium)); } .details__body.overlaySummaryLeftAligned { padding-left: calc(3 * var(--sl-spacing-medium)); } } .details.hoist .details__body { position: absolute; z-index: var(--sl-z-index-drawer); background: var(--sl-color-neutral-0); box-shadow: var(--sl-shadow-large); width: 100%; min-width: fit-content; border-radius: var(--sl-border-radius-small); border: var(--sl-panel-border-width) solid var(--sl-panel-border-color); max-height: 15em; overflow-y: auto; } .details.hoist .details__body.overlaySummaryLeftAligned { top: 0; left: 2em; width: calc(100% - 2em); } .details.hoist .details__body.overlaySummaryRightAligned { top: 0; } .details__summary-icon--left-aligned { order: -1; } `, ]; } /** * Toggle when hover over */ @property({type: Boolean}) toggleOnHover = false; /** * Makes details content fixed position to break out of the container */ @property({type: Boolean}) hoist = false; /** * set toggle alignment either to left or right. Default is right alignment. */ @property({type: String}) toggleAlign : "right" | "left" = "right"; /** * Overlay summary container with the details container when in open state */ @property({type: Boolean}) overlaySummaryOnOpen = false; /** * Group multiple details together so only one can be open at once * @type {string} */ @property({type: String}) accordionGroup : string; /** * List of properties that get translated * Done separately to not interfere with properties - if we re-define label property, * labels go missing. */ static get translate() { return { ...super.translate, summary: true } } constructor() { super(); this.handleAccordionOpen = this.handleAccordionOpen.bind(this); } connectedCallback() { super.connectedCallback(); if(this.accordionGroup) { window.document.addEventListener("sl-show", this.handleAccordionOpen); } this.updateComplete.then(() => { if (this.toggleOnHover) { this.addEventListener("mouseover", this.show); window.document.addEventListener('mouseout', this._mouseOutEvent.bind(this)); } }); } disconnectedCallback() { super.disconnectedCallback(); window.document.removeEventListener("sl-show", this.handleAccordionOpen); window.document.removeEventListener('mouseout', this._mouseOutEvent); } /** * Handle mouse out event for hiding out details * @param event */ _mouseOutEvent(event) { if (!this.getDOMNode().contains(event.relatedTarget)) this.hide(); } handleAccordionOpen(event) { if(event.target !== this && this.accordionGroup && event.target.accordionGroup == this.accordionGroup) { this.hide(); } } render() { const isRtl = this.matches(':dir(rtl)'); return html`
${this.summary}
`; } }