/** * 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} 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"; @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); } .details.hoist { position: relative; } .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; } `, ]; } /** * 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; /** * 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 } } connectedCallback() { super.connectedCallback(); this.updateComplete.then(() => { if (this.toggleOnHover) { this.addEventListener("mouseover", this.show); window.document.addEventListener('mouseout', this._mouseOutEvent.bind(this)); } if (this.hoist) { this.shadowRoot.querySelector('.details').classList.add('hoist'); } if (this.toggleAlign === 'left') { this.shadowRoot.querySelector('.details__summary-icon').style.order = -1; } if (this.overlaySummaryOnOpen) { this.shadowRoot.querySelector('.details__body').classList.add(this.toggleAlign === 'left' ? 'overlaySummaryLeftAligend' : 'overlaySummaryRightAligned'); } }); } disconnectedCallback() { super.disconnectedCallback(); 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(); } }