import { Component, ComponentInterface, Element, Event, EventEmitter, Prop, State, Watch } from '@stencil/core'; import { Mode } from '../../interface'; const SPLIT_PANE_MAIN = 'split-pane-main'; const SPLIT_PANE_SIDE = 'split-pane-side'; const QUERY: { [key: string]: string } = { 'xs': '(min-width: 0px)', 'sm': '(min-width: 576px)', 'md': '(min-width: 768px)', 'lg': '(min-width: 992px)', 'xl': '(min-width: 1200px)', 'never': '' }; @Component({ tag: 'ion-split-pane', styleUrls: { ios: 'split-pane.ios.scss', md: 'split-pane.md.scss' } }) export class SplitPane implements ComponentInterface { private rmL: any; mode!: Mode; @Element() el!: HTMLElement; @State() visible = false; @Prop({ context: 'isServer' }) isServer!: boolean; @Prop({ context: 'window' }) win!: Window; /** * The content `id` of the split-pane's main content. * This property can be used instead of the `[main]` attribute to select the `main` * content of the split-pane. */ @Prop() contentId?: string; /** * If `true`, the split pane will be hidden. */ @Prop() disabled = false; /** * When the split-pane should be shown. * Can be a CSS media query expression, or a shortcut expression. * Can also be a boolean expression. */ @Prop() when: string | boolean = QUERY['lg']; /** * Expression to be called when the split-pane visibility has changed */ @Event() ionSplitPaneVisible!: EventEmitter<{visible: boolean}>; @Watch('visible') visibleChanged(visible: boolean) { const detail = { visible, isPane: this.isPane.bind(this) }; this.ionSplitPaneVisible.emit(detail); } componentDidLoad() { this.styleChildren(); this.updateState(); } componentDidUnload() { if (this.rmL) { this.rmL(); this.rmL = undefined; } } @Watch('disabled') @Watch('when') protected updateState() { if (this.isServer) { return; } if (this.rmL) { this.rmL(); this.rmL = undefined; } // Check if the split-pane is disabled if (this.disabled) { this.visible = false; return; } // When query is a boolean const query = this.when; if (typeof query === 'boolean') { this.visible = query; return; } // When query is a string, let's find first if it is a shortcut const mediaQuery = QUERY[query] || query; // Media query is empty or null, we hide it if (mediaQuery.length === 0) { this.visible = false; return; } if ((this.win as any).matchMedia) { // Listen on media query const callback = (q: MediaQueryList) => { this.visible = q.matches; }; const mediaList = this.win.matchMedia(mediaQuery); (mediaList as any).addListener(callback as any); this.rmL = () => (mediaList as any).removeListener(callback as any); this.visible = mediaList.matches; } } private isPane(element: HTMLElement): boolean { if (!this.visible) { return false; } return element.parentElement === this.el && element.classList.contains(SPLIT_PANE_SIDE); } private styleChildren() { if (this.isServer) { return; } const contentId = this.contentId; const children = this.el.children; const nu = this.el.childElementCount; let foundMain = false; for (let i = 0; i < nu; i++) { const child = children[i] as HTMLElement; const isMain = contentId !== undefined ? child.id === contentId : child.hasAttribute('main'); if (isMain) { if (foundMain) { console.warn('split pane cannot have more than one main node'); return; } foundMain = true; } setPaneClass(child, isMain); } if (!foundMain) { console.warn('split pane does not have a specified main node'); } } hostData() { return { class: { [`${this.mode}`]: true, // Used internally for styling [`split-pane-${this.mode}`]: true, 'split-pane-visible': this.visible } }; } } function setPaneClass(el: HTMLElement, isMain: boolean) { let toAdd; let toRemove; if (isMain) { toAdd = SPLIT_PANE_MAIN; toRemove = SPLIT_PANE_SIDE; } else { toAdd = SPLIT_PANE_SIDE; toRemove = SPLIT_PANE_MAIN; } const classList = el.classList; classList.add(toAdd); classList.remove(toRemove); }